Spring集成使用工作流框架Activiti!事務配置,注解配置以及流程資源自動化部署


Spring使用Activiti提供了一些非常不錯的集成特性,只在Activiti與Spring集成時使用

ProcessEngineFactoryBean

  • 可以把流程引擎(ProcessEngine)作為一個普通的Spring bean進行配置
  • org.activiti.spring.ProcessEngineFactoryBean是集成的切入點,這個bean需要一個流程引擎配置來創建流程引擎
  • Spring集成的配置和流程引擎bean,使用的processEngineConfiguration beanorg.activiti.spring.SpringProcessEngineConfiguration類:
<bean id="processEngineConfiguration" class="org.activiti.spring.SpringProcessEngineConfiguration">
    ...
</bean>

<bean id="processEngine" class="org.activiti.spring.ProcessEngineFactoryBean">
  <property name="processEngineConfiguration" ref="processEngineConfiguration" />
</bean>

事務

  • 使用這個例子的Spring配置文件SpringTransactionIntegrationTest-context.xml:
    • dataSource: 數據源
    • transactionManager: 事務管理器
    • processEngine: 流程引擎
    • Activiti引擎服務
  • 當把數據源(DataSource)傳遞給SpringProcessEngineConfiguration(使用"dataSource"屬性)之后,Activiti內部使用了一個org.springframework.jdbc.datasource.TransactionAwareDataSourceProxy代理來封裝傳遞進來的數據源(DataSource).這樣做是為了確保從數據源(DataSource)獲取的SQL連接能夠與Spring的事物結合在一起發揮得更出色.這意味不需要在Spring配置中代理數據源(dataSource). 但是仍然允許傳遞一個TransactionAwareDataSourceProxySpringProcessEngineConfiguration
  • 為了確保在Spring配置中聲明的一個TransactionAwareDataSourceProxy,不能把使用它的應用交給Spring事物控制的資源(例如DataSourceTransactionManagerJPATransactionManager需要非代理的數據源)
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans   http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
                           http://www.springframework.org/schema/tx      http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">

  <bean id="dataSource" class="org.springframework.jdbc.datasource.SimpleDriverDataSource">
    <property name="driverClass" value="org.h2.Driver" />
    <property name="url" value="jdbc:h2:mem:activiti;DB_CLOSE_DELAY=1000" />
    <property name="username" value="sa" />
    <property name="password" value="" />
  </bean>

  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource" />
  </bean>

  <bean id="processEngineConfiguration" class="org.activiti.spring.SpringProcessEngineConfiguration">
    <property name="dataSource" ref="dataSource" />
    <property name="transactionManager" ref="transactionManager" />
    <property name="databaseSchemaUpdate" value="true" />
    <property name="jobExecutorActivate" value="false" />
  </bean>

  <bean id="processEngine" class="org.activiti.spring.ProcessEngineFactoryBean">
    <property name="processEngineConfiguration" ref="processEngineConfiguration" />
  </bean>

  <bean id="repositoryService" factory-bean="processEngine" factory-method="getRepositoryService" />
  <bean id="runtimeService" factory-bean="processEngine" factory-method="getRuntimeService" />
  <bean id="taskService" factory-bean="processEngine" factory-method="getTaskService" />
  <bean id="historyService" factory-bean="processEngine" factory-method="getHistoryService" />
  <bean id="managementService" factory-bean="processEngine" factory-method="getManagementService" />

...
  • Spring配置文件的其余部分包含beans和在特有的例子中的配置:
<beans>
  ...
  <tx:annotation-driven transaction-manager="transactionManager"/>

  <bean id="userBean" class="org.activiti.spring.test.UserBean">
    <property name="runtimeService" ref="runtimeService" />
  </bean>

  <bean id="printer" class="org.activiti.spring.test.Printer" />

</beans>
  • 首先使用任意的一種Spring創建應用上下文的方式創建其Spring應用上下文.可以使用類路徑下面的XML資源來配置我們的Spring應用上下文:
ClassPathXmlApplicationContext applicationContext =
    new ClassPathXmlApplicationContext("org/activiti/examples/spring/SpringTransactionIntegrationTest-context.xml");

或者

@ContextConfiguration("classpath:org/activiti/spring/test/transaction/SpringTransactionIntegrationTest-context.xml")
  • 然后我們就可以得到Activiti的服務beans並且調用該服務上面的方法,ProcessEngineFactoryBean將會對該服務添加一些額外的攔截器,在Activiti服務上面的方法使用的是Propagation.REQUIRED事物語義. 可以使用repositoryService去部署一個流程:
RepositoryService repositoryService = (RepositoryService) applicationContext.getBean("repositoryService");
String deploymentId = repositoryService
  .createDeployment()
  .addClasspathResource("org/activiti/spring/test/hello.bpmn20.xml")
  .deploy()
  .getId();   
  • 其他相同的服務也是同樣可以這么使用.在這個例子中,Spring的事務將會圍繞在userBean.hello()上,並且調用Activiti服務的方法也會加入到這個事務中
UserBean userBean = (UserBean) applicationContext.getBean("userBean");
userBean.hello();
  • 在上面Spring bean的配置中把repositoryService注入到userBean中
public class UserBean {

  /** 由Spring注入 */
  private RuntimeService runtimeService;

  @Transactional
  public void hello() {
        //這里,你可以在你們的領域模型中做一些事物處理。
        //當在調用Activiti RuntimeService的startProcessInstanceByKey方法時,
        //它將會結合到同一個事物中。
    runtimeService.startProcessInstanceByKey("helloProcess");
  }

  public void setRuntimeService(RuntimeService runtimeService) {
    this.runtimeService = runtimeService;
  }
}

表達式

  • 當使用ProcessEngineFactoryBean時候,默認情況下,在BPMN流程中的所有表達式都將會"看見"所有的Spring beans. 可以限制在表達式中暴露出的beans或者甚至可以在配置中使用一個Map不暴露任何beans
  • 想要不暴露任何beans,只需要在SpringProcessEngineConfiguration中傳遞一個空的list作為'beans'的屬性. 當不設置'beans'的屬性時,在應用上下文中Spring beans都是可以使用的
  • 下面的例子暴露了一個單例bean(printer),可以把"printer"當作關鍵字使用:
<bean id="processEngineConfiguration" class="org.activiti.spring.SpringProcessEngineConfiguration">
  ...
  <property name="beans">
    <map>
      <entry key="printer" value-ref="printer" />
    </map>
  </property>
</bean>

  <bean id="printer" class="org.activiti.examples.spring.Printer" />
  • 現在暴露出來的beans就可以在表達式中使用:例如,在SpringTransactionIntegrationTest中的hello.bpmn20.xml展示的是如何使用UEL方法表達式去調用Spring bean的方法
<definitions id="definitions" ...>

  <process id="helloProcess">

    <startEvent id="start" />
    <sequenceFlow id="flow1" sourceRef="start" targetRef="print" />

    <serviceTask id="print" activiti:expression="#{printer.printMessage()}" />
    <sequenceFlow id="flow2" sourceRef="print" targetRef="end" />

    <endEvent id="end" />

  </process>

</definitions>
  • 這里的Printer類似這樣:
public class Printer {

  public void printMessage() {
    System.out.println("hello world");
  }
}
  • Spring bean的配置類似這樣:
<beans ...>
  ...

  <bean id="printer" class="org.activiti.examples.spring.Printer" />

</beans>

資源自動部署

  • Spring的集成有專門針對對資源部署的特性
  • 在流程引擎的配置中,可以指定一組資源,當流程引擎被創建的時候,所有在這里的資源都將會被自動掃描與部署
  • 在這里有過濾以防止資源重新部署,只有當這個資源真正發生改變的時候,它才會向Activiti使用的數據庫創建新的部署.
  • 這對於很多用例來說,當Spring容器經常重啟的情況下,使用是非常不錯的選擇
<bean id="processEngineConfiguration" class="org.activiti.spring.SpringProcessEngineConfiguration">
  ...
  <property name="deploymentResources" value="classpath*:/org/activiti/spring/test/autodeployment/autodeploy.*.bpmn20.xml" />
</bean>

<bean id="processEngine" class="org.activiti.spring.ProcessEngineFactoryBean">
  <property name="processEngineConfiguration" ref="processEngineConfiguration" />
</bean>

默認情況下,上面的配置會把所有匹配的資源發布到Activiti引擎的一個單獨發布包下.用來檢測防止未修改資源重復發布的機制會作用到整個發布包中
有時候,這可能不是想要的.比如,如果你發布了很多流程資源,但是只修改里其中某一個單獨的流程定義,整個發布包都會被認為變更了,導致整個發布包下的所有流程定義都會被重新發布,結果就是每個流程定義都生成了新版本,雖然其中只有一個流程發生了改變

  • 為了定制發布方式, 可以為SpringProcessEngineConfiguration指定一個額外的參數deploymentMode. 這個參數指定了匹配多個資源時的發布處理方式. 默認下這個參數支持設置三個值:
    • default: 把所有資源放在一個單獨的發布包中,對這個發布包進行重復檢測.這是默認值,如果你沒有指定參數值,就會使用它
    • single-resource: 為每個單獨的資源創建一個發布包,並對這些發布包進行重復檢測.你可以單獨發布每個流程定義,並在修改流程定義后只創建一個新的流程定義版本
    • resource-parent-folder: 把放在同一個上級目錄下的資源發布在一個單獨的發布包中,並對發布包進行重復檢測.當需要多資源時需要創建發布包;但是需要根據共同的文件夾來組合一些資源時,可以使用
  • deploymentMode參數配置為single-resource的情況:
<bean id="processEngineConfiguration" class="org.activiti.spring.SpringProcessEngineConfiguration">
  ...
  <property name="deploymentResources" value="classpath*:/activiti/*.bpmn" />
  <property name="deploymentMode" value="single-resource" />
</bean>
  • 如果想使用上面三個值之外的參數值,你需要自定義處理發布包的行為.可以創建一個SpringProcessEngineConfiguration的子類,重寫getAutoDeploymentStrategy(String deploymentMode)方法. 這個方法中處理了對應deploymentMode的發布策略

單元測試

  • 當集成Spring時,使用標准的Activiti測試工具類是非常容易地對業務流程進行測試:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:org/activiti/spring/test/junit4/springTypicalUsageTest-context.xml")
public class MyBusinessProcessTest {

  @Autowired
  private RuntimeService runtimeService;

  @Autowired
  private TaskService taskService;

  @Autowired
  @Rule
  public ActivitiRule activitiSpringRule;

  @Test
  @Deployment
  public void simpleProcessTest() {
    runtimeService.startProcessInstanceByKey("simpleProcess");
    Task task = taskService.createTaskQuery().singleResult();
    assertEquals("My Task", task.getName());

    taskService.complete(task.getId());
    assertEquals(0, runtimeService.createProcessInstanceQuery().count());

  }
}
  • 對於這種方式,需要在Spring配置中定義一個org.activiti.engine.test.ActivitiRulebean
<bean id="activitiRule" class="org.activiti.engine.test.ActivitiRule">
  <property name="processEngine" ref="processEngine" />
</bean>

基於注解的配置

除了基於XML的配置以外,還可以選擇基於注解的方式來配置Spring環境.這與使用XML的方法非常相似,除了要使用@Bean注解.而且配置是使用java編寫的, 可以直接用於Activiti-Spring的集成

  • @EnableActiviti: 會創建一個Spring環境,並對Activiti流程引擎進行配置
    • 默認的內存H2數據庫,啟用數據庫自動升級
    • 一個簡單的DataSourceTransactionManager
    • 一個默認的SpringJobExecutor
    • 自動掃描processes/ 目錄下的bpmn20.xml文件
  @Configuration
  @EnableActiviti
  public static class SimplestConfiguration {

  }
  • 可以直接通過注入操作Activiti引擎:
 @Autowired
  private ProcessEngine processEngine;

  @Autowired
  private RuntimeService runtimeService;

  @Autowired
  private TaskService taskService;

  @Autowired
  private HistoryService historyService;

  @Autowired
  private RepositoryService repositoryService;

  @Autowired
  private ManagementService managementService;

  @Autowired
  private FormService formService;
  • 默認值都可以自定義:
    • 如果配置了DataSource,就會代替默認創建的數據庫配置
    • 事務管理器,ob執行器和其他組件都與之相同
@Configuration
  @EnableActiviti
  public static class Config {

    @Bean
    public DataSource dataSource() {
        BasicDataSource basicDataSource = new BasicDataSource();
        basicDataSource.setUsername("sa");
        basicDataSource.setUrl("jdbc:h2:mem:anotherDatabase");
        basicDataSource.setDefaultAutoCommit(false);
        basicDataSource.setDriverClassName(org.h2.Driver.class.getName());
        basicDataSource.setPassword("");
        return basicDataSource;
    }

  }

其他數據庫會代替默認的

  • 注意AbstractActivitiConfigurer用法,它暴露了流程引擎的配置,可以用來對它的細節進行詳細的配置:
@Configuration
@EnableActiviti
@EnableTransactionManagement(proxyTargetClass = true)
class JPAConfiguration {

    @Bean
    public OpenJpaVendorAdapter openJpaVendorAdapter() {
        OpenJpaVendorAdapter openJpaVendorAdapter = new OpenJpaVendorAdapter();
        openJpaVendorAdapter.setDatabasePlatform(H2Dictionary.class.getName());
        return openJpaVendorAdapter;
    }

    @Bean
    public DataSource dataSource() {
        BasicDataSource basicDataSource = new BasicDataSource();
        basicDataSource.setUsername("sa");
        basicDataSource.setUrl("jdbc:h2:mem:activiti");
        basicDataSource.setDefaultAutoCommit(false);
        basicDataSource.setDriverClassName(org.h2.Driver.class.getName());
        basicDataSource.setPassword("");
        return basicDataSource;
    }

    @Bean
    public LocalContainerEntityManagerFactoryBean entityManagerFactoryBean(
        OpenJpaVendorAdapter openJpaVendorAdapter, DataSource ds) {
        LocalContainerEntityManagerFactoryBean emf = new LocalContainerEntityManagerFactoryBean();
        emf.setPersistenceXmlLocation("classpath:/org/activiti/spring/test/jpa/custom-persistence.xml");
        emf.setJpaVendorAdapter(openJpaVendorAdapter);
        emf.setDataSource(ds);
        return emf;
    }

    @Bean
    public PlatformTransactionManager jpaTransactionManager(
        EntityManagerFactory entityManagerFactory) {
        return new JpaTransactionManager(entityManagerFactory);
    }

    @Bean
    public AbstractActivitiConfigurer abstractActivitiConfigurer(
        final EntityManagerFactory emf,
        final PlatformTransactionManager transactionManager) {

        return new AbstractActivitiConfigurer() {

            @Override
            public void postProcessSpringProcessEngineConfiguration(SpringProcessEngineConfiguration engine) {
                engine.setTransactionManager(transactionManager);
                engine.setJpaEntityManagerFactory(emf);
                engine.setJpaHandleTransaction(false);
                engine.setJobExecutorActivate(false);
                engine.setJpaCloseEntityManager(false);
                engine.setDatabaseSchemaUpdate(ProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE);
            }
        };
    }

    // A random bean
    @Bean
    public LoanRequestBean loanRequestBean() {
        return new LoanRequestBean();
    }
}

JPA和Hibernate

  • 在Activiti引擎的serviceTasklistener中使用Hibernate 4.2.x JPA時,需要添加Spring ORM依賴,Hibernate 4.1.x及以下版本是不需要的:
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-orm</artifactId>
  <version>${org.springframework.version}</version>
</dependency>


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM