在运行进程 - 实例上手动更改任务
-
21-12-2019 - |
题
在开发自定义BPM应用程序的顺序中,我们与另一个BPM引擎提供商一起使用的功能,也可以使用它与Camunda一起使用。 目标功能是将运行/重置运行进程实例设置为除当前活动之外的指定任务。从我们的角度来看,当例如:
- 创作进程 - 流程版本迁移引起的实例
- 解决事件
- 用户通过用户分辨出意外使用
最后我并没有真正找到一个简单的函数来做到这一点,而是解决了一些与某些限制的自定义代码。此代码中有一些弱点和不确定,因此我有以下问题:
我错过了实现这一目标的替代方法还是以下方法是正确的,或者甚至是目前完全不受支持的吗?
当前缺点imho:
- 首先和最重要的: 没有历史任务实例存储。这 导致它不可追溯谁甚至触发任务/激活/启动。 我在Camunda Google Group上找到了以下帖子( post )这说是 在这一点上纠正,因为它是流程定义范围的任务,但是 通过使用底层流程定义的任务定义我应该 是“范围”?
- 代码基于内部实现而不是官方接口 在此目的,必须手动完成大量的“bootstrap”/初始化 但作为用户(不是Camunda的开发人员),我并不完全清楚所需的内容 什么是可选的
- 某些零件如任务定义失败的解析表达式(见代码注释出) 但这可能是错误的使用引起的
这是代码(我们的Camunda服务门面的实验片段):
@Inject
protected HistoryService histService;
@Inject
protected TaskService taskService;
@Inject
protected ManagementService managementService;
@Inject
protected RuntimeService runtimeService;
@Inject
protected IdentityService identityService;
@Inject
protected RepositoryService repositoryService;
@Inject
protected FormService formService;
@Inject
protected ProcessEngine processEngine;
public void startTask(String processInstanceId, String taskKey) {
Collection<TaskDefinition> taskDefs = getAvailableTasks(
processInstanceId);
TaskEntity newTask = null;
TaskDefinition taskDef = null;
for (TaskDefinition taskDefinition : taskDefs) {
if (taskDefinition.getKey().equals(taskKey)) {
taskDef = taskDefinition;
break;
}
}
boolean taskDefExists = taskDef != null;
List<Task> runningTasksByKey = getTasksByKey(taskKey, processInstanceId);
boolean taskIsAlreadyRunning = runningTasksByKey != null
&& runningTasksByKey.size() > 0;
if (taskDefExists && !taskIsAlreadyRunning) {
newTask = (TaskEntity) taskService.newTask();
ProcessInstance procInst = getProcessInstance(processInstanceId);
ExecutionEntity procInstEntity = (ExecutionEntity) procInst;
String taskName = (String) taskDef.getNameExpression().
getExpressionText();
// String taskAssigne = (String) taskDef.getAssigneeExpression().
// getValue(
// procInstEntity);
// newTask.setAssignee(taskAssigne);
newTask.setTaskDefinitionKey(taskDef.getKey());
newTask.setProcessInstance(procInstEntity);
newTask.setTaskDefinition(taskDef);
newTask.setName(taskName);
newTask.setProcessInstanceId(processInstanceId);
newTask.setProcessDefinitionId(procInstEntity.
getProcessDefinitionId());
taskService.saveTask(newTask);
TaskServiceImpl taskServiceImpl = (TaskServiceImpl) BpmPlatform.
getProcessEngineService().getDefaultProcessEngine().
getTaskService();
CommandExecutor commandExecutor = taskServiceImpl.
getCommandExecutor();
ExecutionEntity executionEntity = commandExecutor.execute(
new SaveTaskActivityInstanceCmd(newTask,
procInstEntity));
// commandExecutor.execute(new `SaveTaskHistoricActivityInstanceCmd(executionEntity, newTask));`
}
}
public Collection<TaskDefinition> getAvailableTasks(String processInstanceId) {
Map<String, TaskDefinition> taskDefs = null;
Collection<TaskDefinition> taskDefObjects = null;
if (processInstanceId != null) {
ProcessInstanceQuery procInstQuery = runtimeService.
createProcessInstanceQuery().processInstanceId(
processInstanceId);
ProcessDefinitionEntity procDefEntity = getProcessDefinitionEager(
processInstanceId);
taskDefs = procDefEntity.getTaskDefinitions();
}
taskDefObjects = (Collection<TaskDefinition>) (taskDefs != null ? taskDefs.
values() : new ArrayList<TaskDefinition>());
return taskDefObjects;
}
public ProcessDefinitionEntity getProcessDefinitionEager(
String processInstanceId) {
ProcessInstanceQuery procInstQuery = runtimeService.
createProcessInstanceQuery().processInstanceId(
processInstanceId);
ProcessInstance procInst = procInstQuery.singleResult();
String procDefId = procInst.getProcessDefinitionId();
return (ProcessDefinitionEntity) repositoryService.getProcessDefinition(
procDefId);
}
public List<Task> getTasksByKey(String taskKey, String processInstanceId) {
List<Task> tasks = taskService.createTaskQuery().processInstanceId(
processInstanceId).taskDefinitionKey(taskKey).list();
return tasks;
}
public class SaveTaskActivityInstanceCmd implements Command<ExecutionEntity>,
Serializable {
private TaskEntity newTask;
private ExecutionEntity procInstEntity;
public SaveTaskActivityInstanceCmd(TaskEntity newTaskInit,
ExecutionEntity procInstEntityInit) {
this.newTask = newTaskInit;
this.procInstEntity = procInstEntityInit;
}
public ExecutionEntity execute(CommandContext commandContext) {
ActivityImpl actImpl = new ActivityImpl(newTask.
getTaskDefinitionKey(),
procInstEntity.getProcessDefinition());
actImpl.setActivityBehavior(new UserTaskActivityBehavior(
new CdiExpressionManager(), newTask.getTaskDefinition()));
ExecutionEntity execEntity = new ExecutionEntity();
execEntity.setActivity(actImpl);
execEntity.setActivityInstanceId(newTask.getTaskDefinitionKey()
+ ":" + newTask.getId());
execEntity.setEventName(newTask.getEventName());
execEntity.setProcessDefinitionId(newTask.getProcessDefinitionId());
execEntity.setActive(true);
execEntity.setProcessInstance(procInstEntity);
commandContext.getExecutionManager().insert(execEntity);
return execEntity;
}
}
.
我欣赏任何提示或建议: - )
解决方案
我不会在该级别上弄乱这个级别,正如您已经注意到的那样,您正在绕过Camundas服务。当面对类似的问题时,我们使用以下内容:
- 取消旧进程版本的流程实例
- 启动扩展进程的新实例,并以编程方式转发到所需的状态...
另一个选项:在新流程版本中建模一个入口点(消息启动事件)。然后,而不是以编程方式将实例转发到所需状态,而是通过事件启动新实例并传递旧实例的所有进程变量...
其他提示
从Camunda 7.3开始,您可以使用进程实例修改以在进程中启动任何活动并取消任何活动活动实例。
示例:
runtimeService.createProcessInstanceModification(processInstanceId)
.startBeforeActivity("someActivityId")
.cancelActivityInstance("someActivityInstanceId")
.execute();
.
请参阅 http:// docs。Camunda.org/7.3/Guides/User-Guide/#process-engine-process-instance-modification 用于文档。
不隶属于 StackOverflow