springboot集成flowable工作流的实战继续研究,以实例展开。
1、pom依赖
<dependency>
<groupId>org.flowable</groupId>
<artifactId>flowable-spring-boot-starter</artifactId>
<version>6.6.0</version>
</dependency>
2、application.yml
spring:
datasource:
url: jdbc:mysql://localhost:3306/flowable?useSSL=false&characterEncoding=UTF-8
driver-class-name: com.mysql.jdbc.Driver
username: root
password: root
3、leave.bpmn20.xml文件,此为学生请假流程文件,只是最简单的审批操作userTask
<?xml version="1.0" encoding="UTF-8"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:flowable="http://flowable.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" typeLanguage="http://www.w3.org/2001/XMLSchema" expressionLanguage="http://www.w3.org/1999/XPath" targetNamespace="http://www.flowable.org/processdef">
<process id="leave" name="leave" isExecutable="true">
<documentation>leave flowable</documentation>
<startEvent id="start" name="开始" flowable:formFieldValidation="true"></startEvent>
<userTask id="student" name="学生请假" flowable:assignee="${studentId}" flowable:formFieldValidation="true">
</userTask>
<userTask id="teacher" name="教师审核" flowable:assignee="${teacherId}" flowable:formFieldValidation="true">
</userTask>
<userTask id="header" name="校长审核" flowable:assignee="${headerId}" flowable:formFieldValidation="true">
</userTask>
<endEvent id="end" name="结束"></endEvent>
<sequenceFlow id="flow1" name="开始" sourceRef="start" targetRef="student"></sequenceFlow>
<sequenceFlow id="flow2" name="同意" sourceRef="student" targetRef="teacher">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='agree'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow3" name="同意" sourceRef="teacher" targetRef="header">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='agree'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow4" name="结束" sourceRef="header" targetRef="end">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='agree'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow5" name="拒绝" sourceRef="teacher" targetRef="student">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='refuse'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow6" name="拒绝" sourceRef="header" targetRef="student">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='refuse'}]]></conditionExpression>
</sequenceFlow>
</process>
<bpmndi:BPMNDiagram id="BPMNDiagram_leave">
...
</bpmndi:BPMNDiagram>
</definitions>
4、启动项目,在数据库中会自动生成79个表
5、请求接口
// 待办的标红
@RestController
public class DemoController {
@Autowired
private RuntimeService runtimeService;
@Autowired
private TaskService taskService;
@Autowired
private RepositoryService repositoryService;
@Resource
private ProcessEngine processEngine;
@RequestMapping(value = "startLeave")
public String startLeave(String userId) {
Map<String, Object> map = new HashMap<>();
map.put("studentId", userId);
map.put("leaveDays",2);
map.put("leaveReason","生病");
ProcessInstance processInstance = runtimeService.startProcessInstanceByKey("leave3", map);
return "请假流程processInstanceId:" + processInstance.getId();
}
@RequestMapping(value = "getTasks")
public String getTasks(String userId) {
StringBuilder sb = new StringBuilder();
List<Task> tasks = taskService.createTaskQuery().taskAssignee(userId).orderByTaskCreateTime().desc().list();
for (Task task : tasks) {
sb.append("任务taskId: " + task.getId() + ";");
}
return sb.toString();
}
@RequestMapping(value = "agree")
public String agree(String user,String userId,String taskId) {
Task task = taskService.createTaskQuery().taskId(taskId).singleResult();
if (task == null) {
System.out.println("该任务对应的流程不存在");
return "该任务对应的流程不存在";
}
Map<String, Object> map = new HashMap<>();
map.put(user, userId);
map.put("command", "agree");
taskService.complete(taskId, map);
return "审核通过";
}
@RequestMapping(value = "refuse")
public String refuse(String user,String userId,String taskId) {
Task task = taskService.createTaskQuery().taskId(taskId).singleResult();
if (task == null) {
System.out.println("该任务对应的流程不存在");
return "该任务对应的流程不存在";
}
Map<String, Object> map = new HashMap<>();
map.put(user, userId);
map.put("command", "refuse");
taskService.complete(taskId, map);
return "审核驳回";
}
@RequestMapping(value = "processDiagram")
public void genProcessDiagram(HttpServletResponse httpServletResponse, String ProcessInstanceId) throws Exception {
ProcessInstance processInstance = runtimeService.createProcessInstanceQuery().processInstanceId(ProcessInstanceId).singleResult();
// 流程走完的不显示图
if (processInstance == null) {
//return;
}
Task task = taskService.createTaskQuery().processInstanceId(ProcessInstanceId).singleResult();
if (task == null) {
}
List<Execution> executions = runtimeService.createExecutionQuery().processInstanceId(ProcessInstanceId).list();
// 得到正在执行的Activity的Id
List<String> activityIds = new ArrayList<>();
List<String> flows = new ArrayList<>();
for (Execution execution : executions) {
List<String> ids = runtimeService.getActiveActivityIds(execution.getId());
activityIds.addAll(ids);
}
// 获取流程图
BpmnModel bpmnModel = repositoryService.getBpmnModel(processInstance.getProcessDefinitionId());
ProcessEngineConfiguration engineConfiguration = processEngine.getProcessEngineConfiguration();
ProcessDiagramGenerator diagramGenerator = engineConfiguration.getProcessDiagramGenerator();
InputStream in = diagramGenerator.generateDiagram(bpmnModel,
"png",
activityIds,
flows,
engineConfiguration.getActivityFontName(),
engineConfiguration.getLabelFontName(),
engineConfiguration.getAnnotationFontName(),
engineConfiguration.getClassLoader(),
1.0,
true);
OutputStream out = null;
byte[] buf = new byte[1024];
int length = 0;
try {
out = httpServletResponse.getOutputStream();
while ((length = in.read(buf)) != -1) {
out.write(buf, 0, length);
}
} finally {
if (in != null) {
in.close();
}
if (out != null) {
out.close();
}
}
}
}
6、测试
或
// 完成的标红
@RequestMapping(value = "processDiagram")
public void genProcessDiagram(HttpServletResponse httpServletResponse, String processInstanceId) {
String processDefinitionId;
if (this.isFinished(processInstanceId)) {
HistoricProcessInstance historicProcessInstance = historyService.createHistoricProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
processDefinitionId = historicProcessInstance.getProcessDefinitionId();
} else {
ProcessInstance processInstance = runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
processDefinitionId = processInstance.getProcessDefinitionId();
}
List<String> highLightedActivityIds = new ArrayList<>();
List<String> highLightedFlows = new ArrayList<>();
List<HistoricActivityInstance> historicActivityInstanceList = historyService.createHistoricActivityInstanceQuery().processInstanceId(processInstanceId).orderByHistoricActivityInstanceStartTime().asc().list();
for(HistoricActivityInstance historicActivityInstance : historicActivityInstanceList){
String activityId = historicActivityInstance.getActivityId();
highLightedActivityIds.add(activityId);
if ("startEvent".equals(historicActivityInstance.getActivityType())) {
System.out.println("startEvent" + historicActivityInstance.getActivityName());
//highLightedActivityIds.add(activityId);
}
if ("userTask".equals(historicActivityInstance.getActivityType())) {
System.out.println("userTask" + historicActivityInstance.getActivityName());
//highLightedActivityIds.add(activityId);
}
if ("sequenceFlow".equals(historicActivityInstance.getActivityType())) {
System.out.println("sequenceFlow" + historicActivityInstance.getActivityName());
highLightedFlows.add(historicActivityInstance.getActivityId());
}
if ("endEvent".equals(historicActivityInstance.getActivityType())) {
System.out.println("endEvent" + historicActivityInstance.getActivityName());
//highLightedActivityIds.add(activityId);
}
}
BpmnModel bpmnModel = repositoryService.getBpmnModel(processDefinitionId);
ProcessEngineConfiguration engineConfiguration = processEngine.getProcessEngineConfiguration();
ProcessDiagramGenerator diagramGenerator = engineConfiguration.getProcessDiagramGenerator();
InputStream in = diagramGenerator.generateDiagram(bpmnModel,
"png",
highLightedActivityIds,
highLightedFlows,
engineConfiguration.getActivityFontName(),
engineConfiguration.getLabelFontName(),
engineConfiguration.getAnnotationFontName(),
engineConfiguration.getClassLoader(),
1.0,
true);
OutputStream out = null;
byte[] buf = new byte[1024];
int legth = 0;
try {
out = httpServletResponse.getOutputStream();
while ((legth = in.read(buf)) != -1) {
out.write(buf, 0, legth);
}
} catch (IOException e) {
} finally {
IOUtils.closeQuietly(out);
IOUtils.closeQuietly(in);
}
}
注意审核操作时的参数指定的是下一个节点的审核人,且必须与配置文件中的保持一致。还需继续研究若最后审核完成后如何查看流程图。