package org.dromara.workflow.service.impl; import cn.hutool.core.bean.BeanUtil; import cn.hutool.core.codec.Base64; import cn.hutool.core.collection.CollUtil; import cn.hutool.core.io.FileUtil; import cn.hutool.core.io.IoUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.StrUtil; import lombok.RequiredArgsConstructor; import lombok.SneakyThrows; import lombok.extern.slf4j.Slf4j; import org.dromara.common.core.exception.ServiceException; import org.dromara.common.core.utils.StreamUtils; import org.dromara.common.core.utils.StringUtils; import org.dromara.common.mybatis.core.page.PageQuery; import org.dromara.common.mybatis.core.page.TableDataInfo; import org.dromara.common.tenant.helper.TenantHelper; import org.dromara.workflow.common.constant.FlowConstant; import org.dromara.workflow.domain.WfCategory; import org.dromara.workflow.domain.WfDefinitionConfig; import org.dromara.workflow.domain.WfNodeConfig; import org.dromara.workflow.domain.bo.ProcessDefinitionBo; import org.dromara.workflow.domain.bo.WfDefinitionConfigBo; import org.dromara.workflow.domain.vo.ProcessDefinitionVo; import org.dromara.workflow.domain.vo.WfDefinitionConfigVo; import org.dromara.workflow.mapper.WfDefinitionConfigMapper; import org.dromara.workflow.service.IActProcessDefinitionService; import org.dromara.workflow.service.IWfCategoryService; import org.dromara.workflow.service.IWfDefinitionConfigService; import org.dromara.workflow.service.IWfNodeConfigService; import org.dromara.workflow.utils.ModelUtils; import org.dromara.workflow.utils.QueryUtils; import org.flowable.bpmn.model.UserTask; import org.flowable.engine.ProcessMigrationService; import org.flowable.engine.RepositoryService; import org.flowable.engine.history.HistoricProcessInstance; import org.flowable.engine.impl.bpmn.deployer.ResourceNameUtil; import org.flowable.engine.repository.*; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import org.springframework.web.multipart.MultipartFile; import java.io.IOException; import java.io.InputStream; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Set; import java.util.zip.ZipEntry; import java.util.zip.ZipInputStream; /** * 流程定义 服务层实现 * * @author may */ @Slf4j @RequiredArgsConstructor @Service public class ActProcessDefinitionServiceImpl implements IActProcessDefinitionService { @Autowired(required = false) private RepositoryService repositoryService; @Autowired(required = false) private ProcessMigrationService processMigrationService; private final IWfCategoryService wfCategoryService; private final IWfDefinitionConfigService wfDefinitionConfigService; private final WfDefinitionConfigMapper wfDefinitionConfigMapper; private final IWfNodeConfigService wfNodeConfigService; /** * 分页查询 * * @param bo 参数 * @return 返回分页列表 */ @Override public TableDataInfo page(ProcessDefinitionBo bo, PageQuery pageQuery) { ProcessDefinitionQuery query = QueryUtils.definitionQuery(); if (StringUtils.isNotEmpty(bo.getKey())) { query.processDefinitionKey(bo.getKey()); } if (StringUtils.isNotEmpty(bo.getCategoryCode())) { query.processDefinitionCategory(bo.getCategoryCode()); } if (StringUtils.isNotEmpty(bo.getName())) { query.processDefinitionNameLike("%" + bo.getName() + "%"); } query.orderByDeploymentId().desc(); // 分页查询 List processDefinitionVoList = new ArrayList<>(); List definitionList = query.latestVersion().listPage(pageQuery.getFirstNum(), pageQuery.getPageSize()); List deploymentList = null; if (CollUtil.isNotEmpty(definitionList)) { List deploymentIds = StreamUtils.toList(definitionList, ProcessDefinition::getDeploymentId); deploymentList = QueryUtils.deploymentQuery(deploymentIds).list(); } if (CollUtil.isNotEmpty(definitionList)) { List ids = StreamUtils.toList(definitionList, ProcessDefinition::getId); List wfDefinitionConfigVos = wfDefinitionConfigService.queryList(ids); for (ProcessDefinition processDefinition : definitionList) { ProcessDefinitionVo processDefinitionVo = BeanUtil.toBean(processDefinition, ProcessDefinitionVo.class); if (CollUtil.isNotEmpty(deploymentList)) { // 部署时间 deploymentList.stream().filter(e -> e.getId().equals(processDefinition.getDeploymentId())).findFirst().ifPresent(e -> { processDefinitionVo.setDeploymentTime(e.getDeploymentTime()); }); } if (CollUtil.isNotEmpty(wfDefinitionConfigVos)) { wfDefinitionConfigVos.stream().filter(e -> e.getDefinitionId().equals(processDefinition.getId())).findFirst().ifPresent(processDefinitionVo::setWfDefinitionConfigVo); } processDefinitionVoList.add(processDefinitionVo); } } // 总记录数 long total = query.count(); TableDataInfo build = TableDataInfo.build(); build.setRows(processDefinitionVoList); build.setTotal(total); return build; } /** * 查询历史流程定义列表 * * @param key 流程定义key */ @Override public List getListByKey(String key) { List processDefinitionVoList = new ArrayList<>(); ProcessDefinitionQuery query = QueryUtils.definitionQuery(); List definitionList = query.processDefinitionKey(key).list(); List deploymentList = null; if (CollUtil.isNotEmpty(definitionList)) { List deploymentIds = StreamUtils.toList(definitionList, ProcessDefinition::getDeploymentId); deploymentList = QueryUtils.deploymentQuery(deploymentIds).list(); } if (CollUtil.isNotEmpty(definitionList)) { List ids = StreamUtils.toList(definitionList, ProcessDefinition::getId); List wfDefinitionConfigVos = wfDefinitionConfigService.queryList(ids); for (ProcessDefinition processDefinition : definitionList) { ProcessDefinitionVo processDefinitionVo = BeanUtil.toBean(processDefinition, ProcessDefinitionVo.class); if (CollUtil.isNotEmpty(deploymentList)) { // 部署时间 deploymentList.stream().filter(e -> e.getId().equals(processDefinition.getDeploymentId())).findFirst().ifPresent(e -> { processDefinitionVo.setDeploymentTime(e.getDeploymentTime()); }); if (CollUtil.isNotEmpty(wfDefinitionConfigVos)) { wfDefinitionConfigVos.stream().filter(e -> e.getDefinitionId().equals(processDefinition.getId())).findFirst().ifPresent(processDefinitionVo::setWfDefinitionConfigVo); } } processDefinitionVoList.add(processDefinitionVo); } } return CollUtil.reverse(processDefinitionVoList); } /** * 查看流程定义图片 * * @param processDefinitionId 流程定义id */ @SneakyThrows @Override public String definitionImage(String processDefinitionId) { InputStream inputStream = repositoryService.getProcessDiagram(processDefinitionId); return Base64.encode(IoUtil.readBytes(inputStream)); } /** * 查看流程定义xml文件 * * @param processDefinitionId 流程定义id */ @Override public String definitionXml(String processDefinitionId) { StringBuilder xml = new StringBuilder(); ProcessDefinition processDefinition = repositoryService.getProcessDefinition(processDefinitionId); InputStream inputStream = repositoryService.getResourceAsStream(processDefinition.getDeploymentId(), processDefinition.getResourceName()); xml.append(IoUtil.read(inputStream, StandardCharsets.UTF_8)); return xml.toString(); } /** * 删除流程定义 * * @param deploymentIds 部署id * @param processDefinitionIds 流程定义id */ @Override @Transactional(rollbackFor = Exception.class) public boolean deleteDeployment(List deploymentIds, List processDefinitionIds) { try { List historicProcessInstances = QueryUtils.hisInstanceQuery().deploymentIdIn(deploymentIds).list(); if (CollUtil.isNotEmpty(historicProcessInstances)) { Set defIds = StreamUtils.toSet(historicProcessInstances, HistoricProcessInstance::getProcessDefinitionId); List processDefinitions = QueryUtils.definitionQuery().processDefinitionIds(defIds).list(); if (CollUtil.isNotEmpty(processDefinitions)) { Set keys = StreamUtils.toSet(processDefinitions, ProcessDefinition::getKey); throw new ServiceException("当前【" + String.join(",", keys) + "】流程定义已被使用不可删除!"); } } //删除流程定义 for (String deploymentId : deploymentIds) { repositoryService.deleteDeployment(deploymentId); } //删除流程定义配置 wfDefinitionConfigService.deleteByDefIds(processDefinitionIds); //删除节点配置 wfNodeConfigService.deleteByDefIds(processDefinitionIds); return true; } catch (Exception e) { log.error(e.getMessage(), e); throw new ServiceException(e.getMessage()); } } /** * 激活或者挂起流程定义 * * @param processDefinitionId 流程定义id */ @Override public boolean updateDefinitionState(String processDefinitionId) { try { ProcessDefinition processDefinition = QueryUtils.definitionQuery() .processDefinitionId(processDefinitionId).singleResult(); //将当前为挂起状态更新为激活状态 //参数说明:参数1:流程定义id,参数2:是否激活(true是否级联对应流程实例,激活了则对应流程实例都可以审批), //参数3:什么时候激活,如果为null则立即激活,如果为具体时间则到达此时间后激活 if (processDefinition.isSuspended()) { repositoryService.activateProcessDefinitionById(processDefinitionId, true, null); } else { repositoryService.suspendProcessDefinitionById(processDefinitionId, true, null); } return true; } catch (Exception e) { log.error(e.getMessage(), e); throw new ServiceException("操作失败:" + e.getMessage()); } } /** * 迁移流程定义 * * @param currentProcessDefinitionId 当前流程定义id * @param fromProcessDefinitionId 需要迁移到的流程定义id */ @Override public boolean migrationDefinition(String currentProcessDefinitionId, String fromProcessDefinitionId) { try { // 迁移验证 boolean migrationValid = processMigrationService.createProcessInstanceMigrationBuilder() .migrateToProcessDefinition(currentProcessDefinitionId) .validateMigrationOfProcessInstances(fromProcessDefinitionId) .isMigrationValid(); if (!migrationValid) { throw new ServiceException("流程定义差异过大无法迁移,请修改流程图"); } // 已结束的流程实例不会迁移 processMigrationService.createProcessInstanceMigrationBuilder() .migrateToProcessDefinition(currentProcessDefinitionId) .migrateProcessInstances(fromProcessDefinitionId); return true; } catch (Exception e) { log.error(e.getMessage(), e); throw new ServiceException(e.getMessage()); } } /** * 流程定义转换为模型 * * @param processDefinitionId 流程定义id */ @Override public boolean convertToModel(String processDefinitionId) { ProcessDefinition pd = QueryUtils.definitionQuery() .processDefinitionId(processDefinitionId).singleResult(); InputStream inputStream = repositoryService.getResourceAsStream(pd.getDeploymentId(), pd.getResourceName()); ModelQuery query = QueryUtils.modelQuery(); Model model = query.modelKey(pd.getKey()).singleResult(); try { if (ObjectUtil.isNotNull(model)) { repositoryService.addModelEditorSource(model.getId(), IoUtil.readBytes(inputStream)); } else { Model modelData = repositoryService.newModel(); modelData.setKey(pd.getKey()); modelData.setName(pd.getName()); modelData.setCategory(pd.getCategory()); modelData.setTenantId(pd.getTenantId()); repositoryService.saveModel(modelData); repositoryService.addModelEditorSource(modelData.getId(), IoUtil.readBytes(inputStream)); } return true; } catch (Exception e) { log.error(e.getMessage(), e); throw new ServiceException(e.getMessage()); } } /** * 通过zip或xml部署流程定义 * * @param file 文件 * @param categoryCode 分类 */ @SneakyThrows @Override @Transactional(rollbackFor = Exception.class) public void deployByFile(MultipartFile file, String categoryCode) { WfCategory wfCategory = wfCategoryService.queryByCategoryCode(categoryCode); if (wfCategory == null) { throw new ServiceException("流程分类不存在"); } // 文件后缀名 String suffix = FileUtil.extName(file.getOriginalFilename()); InputStream inputStream = file.getInputStream(); if (FlowConstant.ZIP.equalsIgnoreCase(suffix)) { ZipInputStream zipInputStream = null; try { zipInputStream = new ZipInputStream(inputStream); ZipEntry zipEntry; while ((zipEntry = zipInputStream.getNextEntry()) != null) { String filename = zipEntry.getName(); String[] splitFilename = filename.substring(0, filename.lastIndexOf(".")).split("-"); //流程名称 String processName = splitFilename[0]; //流程key String processKey = splitFilename[1]; ProcessDefinition oldProcessDefinition = QueryUtils.definitionQuery().processDefinitionKey(processKey).latestVersion().singleResult(); DeploymentBuilder builder = repositoryService.createDeployment(); Deployment deployment = builder.addInputStream(filename, zipInputStream) .tenantId(TenantHelper.getTenantId()) .name(processName).key(processKey).category(categoryCode).deploy(); ProcessDefinition definition = QueryUtils.definitionQuery().deploymentId(deployment.getId()).singleResult(); repositoryService.setProcessDefinitionCategory(definition.getId(), categoryCode); setWfConfig(oldProcessDefinition, definition); zipInputStream.closeEntry(); } } catch (IOException e) { throw new RuntimeException(e); } finally { if (zipInputStream != null) { zipInputStream.close(); } } //初始化配置数据(demo使用,不用可删除) initWfDefConfig(); } else { String originalFilename = file.getOriginalFilename(); if (StringUtils.containsAny(originalFilename, ResourceNameUtil.BPMN_RESOURCE_SUFFIXES)) { // 文件名 = 流程名称-流程key String[] splitFilename = originalFilename.substring(0, originalFilename.lastIndexOf(".")).split("-"); if (splitFilename.length < 2) { throw new ServiceException("文件名 = 流程名称-流程KEY"); } //流程名称 String processName = splitFilename[0]; //流程key String processKey = splitFilename[1]; ProcessDefinition oldProcessDefinition = QueryUtils.definitionQuery().processDefinitionKey(processKey).latestVersion().singleResult(); DeploymentBuilder builder = repositoryService.createDeployment(); Deployment deployment = builder.addInputStream(originalFilename, inputStream) .tenantId(TenantHelper.getTenantId()) .name(processName).key(processKey).category(categoryCode).deploy(); // 更新分类 ProcessDefinition definition = QueryUtils.definitionQuery().deploymentId(deployment.getId()).singleResult(); repositoryService.setProcessDefinitionCategory(definition.getId(), categoryCode); setWfConfig(oldProcessDefinition, definition); } else { throw new ServiceException("文件类型上传错误!"); } } } /** * 初始化配置数据(demo使用,不用可删除) */ private void initWfDefConfig() { List wfDefinitionConfigs = wfDefinitionConfigMapper.selectList(); if (CollUtil.isEmpty(wfDefinitionConfigs)) { ProcessDefinition processDefinition = QueryUtils.definitionQuery().processDefinitionKey("leave1").latestVersion().singleResult(); if (processDefinition != null) { WfDefinitionConfigBo wfDefinitionConfigBo = new WfDefinitionConfigBo(); wfDefinitionConfigBo.setDefinitionId(processDefinition.getId()); wfDefinitionConfigBo.setProcessKey(processDefinition.getKey()); wfDefinitionConfigBo.setTableName("test_leave"); wfDefinitionConfigBo.setVersion(processDefinition.getVersion()); wfDefinitionConfigService.saveOrUpdate(wfDefinitionConfigBo); } } } /** * 设置表单内容 * * @param oldProcessDefinition 部署前最新流程定义 * @param definition 部署后最新流程定义 */ private void setWfConfig(ProcessDefinition oldProcessDefinition, ProcessDefinition definition) { //更新流程定义表单 if (oldProcessDefinition != null) { WfDefinitionConfigVo definitionVo = wfDefinitionConfigService.getByDefId(oldProcessDefinition.getId()); if (definitionVo != null) { wfDefinitionConfigService.deleteByDefIds(Collections.singletonList(oldProcessDefinition.getId())); WfDefinitionConfigBo wfDefinitionConfigBo = new WfDefinitionConfigBo(); wfDefinitionConfigBo.setDefinitionId(definition.getId()); wfDefinitionConfigBo.setProcessKey(definition.getKey()); wfDefinitionConfigBo.setTableName(definitionVo.getTableName()); wfDefinitionConfigBo.setVersion(definition.getVersion()); wfDefinitionConfigBo.setRemark(definitionVo.getRemark()); wfDefinitionConfigService.saveOrUpdate(wfDefinitionConfigBo); } } //更新流程节点配置表单 List userTasks = ModelUtils.getUserTaskFlowElements(definition.getId()); UserTask applyUserTask = ModelUtils.getApplyUserTask(definition.getId()); List wfNodeConfigList = new ArrayList<>(); for (UserTask userTask : userTasks) { if (StringUtils.isNotBlank(userTask.getFormKey()) && userTask.getFormKey().contains(StrUtil.COLON)) { WfNodeConfig wfNodeConfig = new WfNodeConfig(); wfNodeConfig.setNodeId(userTask.getId()); wfNodeConfig.setNodeName(userTask.getName()); wfNodeConfig.setDefinitionId(definition.getId()); String[] split = userTask.getFormKey().split(StrUtil.COLON); wfNodeConfig.setFormType(split[0]); wfNodeConfig.setFormId(Long.valueOf(split[1])); wfNodeConfig.setApplyUserTask(applyUserTask.getId().equals(userTask.getId()) ? FlowConstant.TRUE : FlowConstant.FALSE); wfNodeConfigList.add(wfNodeConfig); } } if (CollUtil.isNotEmpty(wfNodeConfigList)) { wfNodeConfigService.saveOrUpdate(wfNodeConfigList); } } }