流程引擎 Camunda工作流的退回起点 退回上一级

本文探讨了在Camunda工作流中实现任务退回的两种方法,包括退回起点和回退至上一级节点的问题。代码块1适用于退回起点但无法添加意见,而代码块2在测试环境中可行但在实际环境可能遇到困难。针对反复退回导致的历史数据问题,提出了使用LinkedHashSet来准确获取上一级节点。此外,还分享了如何判断任务状态和添加退回意见的方法。
摘要由CSDN通过智能技术生成

Camunda工作流的退回操作

话不多说,直接上代码

代码块 1:

/**
 * 退回到起点
 */
@Test
public void deny(){
    String processInstanceId = "37f9f9f4-9f72-11ec-8d1f-5254000c17ac";

    Task activeTask = taskService.createTaskQuery()
            .processInstanceId(processInstanceId)
            .active()
            .singleResult();
    HistoricTaskInstance taskInstance = historyService.createHistoricTaskInstanceQuery()
            .taskId(activeTask.getId())
            .singleResult();

    // 获取流程定义
    ProcessDefinitionEntity processDefinitionEntity = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService).getDeployedProcessDefinition(taskInstance.getProcessDefinitionId());

    // 获取当前活动
    ActivityImpl currentActivity = processDefinitionEntity.findActivity(taskInstance.getTaskDefinitionKey());

    // 获取起始活动
    List<HistoricActivityInstance> historicActivityInstances = historyService.createHistoricActivityInstanceQuery()
            .activityType("userTask")
            .processInstanceId(processInstanceId)
            .finished()
            .orderByHistoricActivityInstanceEndTime()
            .asc()
            .list();
    if(historicActivityInstances.size() == 0){
       return ;
    }
    ActivityImpl lastActivity = processDefinitionEntity.findActivity(historicActivityInstances.get(0).getActivityId());

    // 退回至起点
    runtimeService.createProcessInstanceModification(processInstanceId)
            .cancelAllForActivity(currentActivity.getActivityId())
            .startBeforeActivity(lastActivity.getActivityId())
            .setVariable("denyReason","就是搞你,就给你退回去")
            .execute();
}

这段代码是参考Camunda官网来写来的,具体地址: https://docs.camunda.org/manual/7.15/user-guide/process-engine/process-instance-modification/#process-instance-modification-in-junit-tests

还有一段代码是参照activiti的退回写的,代码如下:

代码块 2:

/**
 * 任务回退到开始
 */
@Test
public void rollBackToShangYiJi(){
    //String processInstanceId = "49b521ac-9f50-11ec-8d1f-5254000c17ac";
    String processInstanceId = "37f9f9f4-9f72-11ec-8d1f-5254000c17ac";
    // 根据流程实例 id 获取当前任务
    Task activeTask = taskService.createTaskQuery()
            .processInstanceId(processInstanceId)
            .active()
            .singleResult();
    HistoricTaskInstance taskInstance = historyService.createHistoricTaskInstanceQuery()
            .taskId(activeTask.getId())
            .singleResult();

    // 获取流程定义
    ProcessDefinitionEntity processDefinitionEntity = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService).getDeployedProcessDefinition(taskInstance.getProcessDefinitionId());

    // 获取当前活动
    ActivityImpl currentActivity = processDefinitionEntity.findActivity(taskInstance.getTaskDefinitionKey());

    //清除当前活动出口
    List<PvmTransition> originPvmTransitions = new ArrayList<>();
    List<PvmTransition> pvmTransitionList = currentActivity.getOutgoingTransitions();
    for (PvmTransition pvmTransition : pvmTransitionList) {
        originPvmTransitions.add(pvmTransition);
    }
    pvmTransitionList.clear();

    //查找最开始的user task节点
    List<HistoricActivityInstance> historicActivityInstances = historyService.createHistoricActivityInstanceQuery()
            .activityType("userTask")
            .processInstanceId(processInstanceId)
            .finished()
            .orderByHistoricActivityInstanceEndTime()
            .asc()
            .list();
    TransitionImpl transition = null;
    if(historicActivityInstances.size() > 0){
        ActivityImpl lastActivity = processDefinitionEntity.findActivity(historicActivityInstances.get(0).getActivityId());
        //把当前任务的新出口设置为最开始的userTask节点
        transition = currentActivity.createOutgoingTransition(lastActivity.getId());
        transition.setDestination(lastActivity);
    }
    // 完成任务
    List<Task> tasks = taskService.createTaskQuery()
            .processInstanceId(processInstanceId)
            .taskDefinitionKey(taskInstance.getTaskDefinitionKey())
            .list();

    for(Task task : tasks){
        taskService.complete(task.getId());
        historyService.deleteHistoricTaskInstance(task.getId());
    }
    //恢复活动方向
    currentActivity.getOutgoingTransitions().remove(transition);
    for (PvmTransition pvmTransition : originPvmTransitions) {
        pvmTransitionList.add(pvmTransition);
    }
}

参考网址:https://blog.csdn.net/qq_41136963/article/details/108184871

目前存在的问题

代码块1:

无法添加退回意见

代码块2:

测试时可以反复退回
但在实际环境下,二次退回时,节点的namedOutgoingTransitions无法清空,导致无法使当前节点的出口设置为起始节点
博主在找不到解决方案的情况下选择了代码块1

关于退回到上一级

假如存在 o -> 1 -> 2 ->3 -> O 的流程
如果用户在2节点和3节点之间反复横跳,来回的同意和退回,就会导致出现一大堆的historicActivityInstances数据
如果还使用上面的两个代码块中的代码,仅仅通过调整数字下标是无法准确定位到当前节点的上一级节点的
如何准确的获取需要用到一个有序不重复的集合LinkedHashSet

// 新建一个有序不重复集合
LinkedHashSet linkedHashSet = new LinkedHashSet();
// 获取当前的任务节点
String activeTask = "Activity_0lj34xv";
String backTask = "";
historyService.createHistoricTaskInstanceQuery().processInstanceId(processInstanceId).orderByHistoricTaskInstanceEndTime().asc().list().stream().forEach(historicTaskInstance -> {
    linkedHashSet.add(historicTaskInstance.getTaskDefinitionKey());
});
if(linkedHashSet.size() == 0){
    return ;
}
// 遍历到当前的任务节点后,取上一次遍历的值
Iterator<String> iterator = linkedHashSet.iterator();
while(iterator.hasNext()){
    String nowTask = iterator.next();
    if(StrUtil.equals(nowTask,activeTask)){
        return backTask;
    }
    backTask = nowTask;
}
backTask.toString();

关于添加退回意见

在退回起点之前,直接选中当前任务节点添加comment即可

关于任务状态的分辨

源码里对任务状态的标识
在这里插入图片描述
可以根据自己需要做简单转换

int state = ((HistoricActivityInstanceEntity) historicActivityInstance).getActivityInstanceState();
if (state == ActivityInstanceState.ENDING.getStateCode()){
    activityInfoVo.setResult("同意");
}else if(state == ActivityInstanceState.CANCELED.getStateCode()){
    activityInfoVo.setResult("退回");
}else if(state == ActivityInstanceState.DEFAULT.getStateCode()){
    activityInfoVo.setResult("进行中");
}

最后,欢迎大家留言讨论, 共同进步,一起成长

  • 8
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 11
    评论
流程引擎(process engine)是整个camunda工作流的核心组件,扮演着重要的角色。它负责执行和控制整个工作流程中的各个环节和任务。流程引擎基于BPMN(Business Process Model and Notation)规范,将复杂的业务流程抽象为可执行的流程模型。 流程引擎具有以下特点和功能: 1. 流程定义和管理:流程引擎允许用户通过定义BPMN流程图来描述业务流程,并可以进行流程模型的部署、修改和版本控制等管理操作。 2. 流程实例控制:流程引擎可以根据流程定义创建流程实例,并对流程实例进行控制和管理。它能够保证流程按照预定的流转逻辑进行执行,并处理流程中的各种事件和异常情况。 3. 任务分配和执行:流程引擎负责将任务分配给具体的执行者,并跟踪和管理任务的执行情况。它可以定义任务的优先级、候选执行者和审批规则等,以及支持任务的委托、退回和撤销等操作。 4. 决策和业务规则:流程引擎可以集成决策引擎和规则引擎,用于处理流程中的决策和业务规则。它可以根据预定的规则和条件,自动执行相应的决策和规则,并控制流程的分支和流转。 5. 监控和统计:流程引擎提供了实时的监控和统计功能,可以跟踪流程的执行情况、任务的处理时间和负载情况等。它可以生成各种报表和图表,帮助用户了解和优化业务流程。 总之,流程引擎camunda工作流的大心脏,它负责执行和管控整个工作流程,并提供丰富的功能和工具,帮助用户简化和优化业务流程的管理和执行。
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值