工作流框架--Activiti6.0(二)
我们大概要完成4步。
(1)设计流程
(2)配置流程引擎
(3)加载流程文件
(4)启动流程
一、新建一个springboot项目 (版本:1.5.10和2.1.2版本同时测试)
1.1 新建springboot项目的版本为1.5.10
--activiti6.0不支持springboot的2.x 版本。 根据官方文档解释:https://www.baeldung.com/spring-activiti。
1.2 在src/main/resources下面新建一个prosesses文件夹
如果不建立,项目会报错。
二、设计流程
2.1 使用Eclipse-Activiti插件生成一张流程图。
起名为:first.bpmn(建立流程图)
2.2 可以在该图片上右键-->打开方式-->xml编辑器来查看文件内容(上面的项目是2.1.2版本 下面的项目是1.5.10版本,内容相同)
2.3 生成28张表
在properties中添加一些属性,各个Activiti的属性配置含义如下:
# 是否更新数据库表 spring.activiti.databaseSchemaUpdate=true # 是否激活异步执行器 spring.activiti.asyncExecutorActivate=false # 流程历史记录登录 spring.activiti.historyLevel=audit # 是否检查更新流程定义 spring.activiti.checkProcessDefinitions=false # 流程定义所在前缀 spring.activiti.processDefinitionLocationPrefix=classpath*:/procDef/ # 流程定义后缀 spring.activiti.processDefinitionLocationSuffixes=**.bpmn # 部署流程定义时是否生成图片 spring.activiti.createDiagramOnDeploy=false # 字体 下面内容为转成unicode的'宋体' spring.activiti.activityFontName=\u5b8b\u4f53 spring.activiti.labelFontName=\u5b8b\u4f53
(1)application.properties
server.port: 8081 spring.datasource.url=jdbc:mysql://localhost:3306/activiti?useSSL=true&characterEncoding=utf8 spring.datasource.driverClassName=com.mysql.jdbc.Driver spring.datasource.username=root spring.datasource.password=root activiti.databaseSchemaUpdate=true activiti.checkProcessDefinitions=true
(2)新建一个测试类,创建28张表(在这行代码运行的时候,流程定义的部署已经执行..不过执行了一个默认的数据)
package com.example.demo; import org.activiti.engine.ProcessEngines; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; @RunWith(SpringRunner.class) @SpringBootTest public class ActivitiSpringboot1ApplicationTests { @Test public void contextLoads() { ProcessEngines.getDefaultProcessEngine(); } }
注意:在boot2.0 的时候,系统可能会报一个异常。原因:mysql和本机时差问题
Caused by: com.mysql.cj.exceptions.InvalidConnectionAttributeException: The server time zone value 'Öйú±ê׼ʱ¼ä' is unrecognized or represents more than one time zone. You must configure either the server or JDBC driver (via the serverTimezone configuration property) to use a more specifc time zone value if you want to utilize time zone support.
解决办法:
打开mysql cmd 的命令行,输入密码进入后,直接输入一行命令:
set global time_zone='+8:00'
(3)运行测试类
发现boot1.5.10 版本 和boot 2.1.2 均可以生成这28张表。
但是boot2.1.2 此时又报了异常,不能创建bean
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'org.activiti.spring.boot.SecurityAutoConfiguration': Initialization of bean failed; nested exception is java.lang.ArrayStoreException: sun.reflect.annotation.TypeNotPresentExceptionProxy
解决办法:
springboot Test 测试类中如何排除一个bean类
2.4 流程定义的部署
(1)配置类代码
package com.example; import java.io.IOException; import javax.sql.DataSource; import org.activiti.engine.HistoryService; import org.activiti.engine.IdentityService; import org.activiti.engine.ManagementService; import org.activiti.engine.ProcessEngine; import org.activiti.engine.RepositoryService; import org.activiti.engine.RuntimeService; import org.activiti.engine.TaskService; import org.activiti.spring.SpringProcessEngineConfiguration; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.io.Resource; import org.springframework.core.io.ResourceLoader; import org.springframework.core.io.support.PathMatchingResourcePatternResolver; import org.springframework.jdbc.datasource.DataSourceTransactionManager; import org.springframework.transaction.PlatformTransactionManager; @Configuration public class ActivitiConfiguration { @Autowired private DataSource dataSource; @Autowired private PlatformTransactionManager platformTransactionManager; @Bean public ProcessEngine processEngine(DataSourceTransactionManager transactionManager, DataSource dataSource) throws IOException { SpringProcessEngineConfiguration configuration = new SpringProcessEngineConfiguration(); //自动部署已有的流程文件 Resource[] resources = new PathMatchingResourcePatternResolver().getResources(ResourceLoader.CLASSPATH_URL_PREFIX + "processes/*.bpmn"); configuration.setTransactionManager(transactionManager); configuration.setDataSource(dataSource); configuration.setDatabaseSchemaUpdate("true"); configuration.setDeploymentResources(resources); configuration.setDbIdentityUsed(false); return configuration.buildProcessEngine(); } @Bean public RepositoryService repositoryService(ProcessEngine processEngine) { return processEngine.getRepositoryService(); } @Bean public RuntimeService runtimeService(ProcessEngine processEngine) { return processEngine.getRuntimeService(); } @Bean public TaskService taskService(ProcessEngine processEngine) { return processEngine.getTaskService(); } @Bean public HistoryService historyService(ProcessEngine processEngine) { return processEngine.getHistoryService(); } @Bean public ManagementService managementService(ProcessEngine processEngine) { return processEngine.getManagementService(); } @Bean public IdentityService identityService(ProcessEngine processEngine) { return processEngine.getIdentityService(); } }
(2)部署类代码
package com.example.demo; import org.activiti.engine.RepositoryService; import org.activiti.engine.repository.Deployment; import org.junit.Test; import org.junit.runner.RunWith; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; @RunWith(SpringRunner.class) @SpringBootTest public class ActivitiSpringboot1ApplicationTests { private static final Logger log = LoggerFactory.getLogger(ActivitiSpringboot1ApplicationTests.class); @Autowired private RepositoryService repositoryService; /** * 流程定义的部署 */ @Test public void contextLoads() { Deployment deployment = repositoryService // 与流程定义和部署对象相关的Service .createDeployment()// 创建一个部署对象 .name("请假流程") // 添加部署名称 .addClasspathResource("processes/first.bpmn") // 从classpath下面加载文件,一次只能加载一个文件。 .deploy(); System.out.println("部署ID:" + deployment.getId());// 1 System.out.println("部署名称:" + deployment.getName()); } }
(2)查看表:
act_re_deployment 部署表
act_ge_bytearray 资源表
2.5 流程定义启动
(1)启动代码
/** 启动流程实例 */ @Test public void startProcessInstance() { // 流程定义的key String processDefinitionKey = "myProcess"; ProcessInstance pi = runtimeService// 与正在执行的流程实例和执行对象相关的Service .startProcessInstanceByKey(processDefinitionKey);// 使用流程定义的key启动流程实例,key对应bpmn文件中id的属性值,使用key值启动,默认是按照最新版本的流程定义启动 System.out.println("流程实例ID:" + pi.getId());// 流程实例ID 101 System.out.println("流程定义ID:" + pi.getProcessDefinitionId());// 流程定义ID helloworld:1:4 }
(2)数据库表
详细信息可以查看:act_ru_task表
2.6 查询当前任务
在这儿可以根据人来进行查询,即.taskAssignee(指定个人任务),在设计流程时我们没有指定,所以我们通过指定任务ID来进行查询。
任务ID可以通过表:act_ru_task表
/**查询当前人的个人任务*/ @Test public void findMyPersonalTask(){ // String assignee = "worker vocation"; List<Task> list = taskService//与正在执行的任务管理相关的Service .createTaskQuery()//创建任务查询对象 // .taskAssignee(assignee)//指定个人任务查询,指定办理人 .taskId("2505") .list(); if(list!=null && list.size()>0){ for(Task task:list){ System.out.println("任务ID:"+task.getId()); System.out.println("任务名称:"+task.getName()); System.out.println("任务的创建时间:"+task.getCreateTime()); System.out.println("任务的办理人:"+task.getAssignee()); System.out.println("流程实例ID:"+task.getProcessInstanceId()); System.out.println("执行对象ID:"+task.getExecutionId()); System.out.println("流程定义ID:"+task.getProcessDefinitionId()); System.out.println("########################################################"); } } }
结果:
2.7 结束当前任务
(1)完成代码
/**完成我的任务*/ @Test public void completeMyPersonalTask(){ //任务ID String taskId = "2505"; taskService//与正在执行的任务管理相关的Service .complete(taskId); System.out.println("完成任务:任务ID:"+taskId); }
详情:
2.8 结束流程
现在来说,流程才刚刚完成员工的申请,也就是worker vocation。下面是领导审批,也就是leader approval。
其实过程一样,先来查看数据库表:act_hi_actinst
可以查看现在的所有节点,在最后一行数据中,我们就可以发现,该任务正在进行,还没有结束时间。
我们也可以继续查看正在进行的任务表:act_ru_task
将查看任务的代码中ID改成:5002可以查看任务。
将完成额任务的代码中的ID 改成:5002 可以完成任务。
结论:以上完成了 spingboot1.5.10版本和Activiti6.0.0的结合
spingboot2.1.2版本和Activiti6.0.0的结合
均成功~~
配置不同点:
(1)2.1.2版本 添加 @EnableAutoConfiguration(exclude=SecurityAutoConfiguration.class)
(2)数据库时间报错,需要调节Mysql的时间
打开mysql的命令行,添加:
set global time_zone='+8:00'