`
dingchao.lonton
  • 浏览: 49767 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

osworkflow dynamic split 动态会签的实现

 
阅读更多

目前 osworkflow的版本是2.8,很久没有更新了,而且这个版本的动态会签无法实现,虽然她提供了 split和join这样的标签,但是split中result的个数是固定的,这往往在我们现实的系统中用的很少,比如 我要给你一个组提供会签功能,而这个组的人数事先是不知道的,所以必须提供动态会签,也就是动态的split。

附件里面是我通过改写osworkflow使其能够动态会签的例子

 

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE workflow PUBLIC "-//OpenSymphony Group//DTD OSWorkflow 2.8//EN" "workflow_2_8.dtd" >
<workflow>
  <initial-actions>
    <action id="1" name="Start Workflow">
      <results>
        <unconditional-result old-status="Finished" status="Queued" step="1"/>
      </results>
    </action>
  </initial-actions>
  <steps>
    <step id="1" name="First Part">
      <actions>
        <action id="2" name="Finish First Part">
          <results>
            <unconditional-result old-status="Finished" status="Queued" split="1" owner="test1,test2,test3"/>
          </results>
        </action>
      </actions>
    </step>
    <step id="2" name="Second Part">
      <actions>
        <action id="3" name="Back to First Part">
          <results>
            <unconditional-result old-status="Rejected" status="Queued" join="2"/>
          </results>
        </action>
        <action id="4" name="Finish">
          <results>
            <unconditional-result old-status="Finished" status="Queued" join="1"/>
          </results>
        </action>
      </actions>
    </step>

    <step id="4" name="End" />
  </steps>

  <splits>
    <split id="1" dynamic="TRUE" obtain-groups-callback="com.siemens.dingchao.test.DynamicSplitGroup"

>
      <unconditional-result old-status="Finished" status="Queued" step="2"/>
    </split>
  </splits>

  <joins>
    <join id="1">
      <conditions type="AND">
        <condition type="beanshell">
          <arg name="script">"Finished".equals(jn.getStep(2,"test1").getStatus())</arg>
        </condition>
        <condition type="beanshell">
          <arg name="script">"Finished".equals(jn.getStep(2,"test2").getStatus())</arg>
        </condition>
         <condition type="beanshell">
          <arg name="script">"Finished".equals(jn.getStep(2,"test3").getStatus())</arg>
        </condition>
      </conditions>
      <unconditional-result old-status="Finished" status="Underway" step="4"/>
    </join>
    <join id="2">
      <conditions type="OR">
        <condition type="beanshell">
          <arg name="script">"Rejected".equals(jn.getStep(2,"test1").getStatus())</arg>
        </condition>
        <condition type="beanshell">
          <arg name="script">"Rejected".equals(jn.getStep(2,"test2").getStatus())</arg>
        </condition>
        <condition type="beanshell">
          <arg name="script">"Rejected".equals(jn.getStep(2,"test3").getStatus())</arg>
        </condition>
      </conditions>
      <unconditional-result old-status="Finished" status="Underway" step="1"/>
    </join>
  </joins>
</workflow>

 

上面的红颜色部分是我为split标签添加的 属性,dynamic 属性 : 可以为 TRUE,true,FALSE,false,指定是否是动态的,如果是动态的 则必须指定obtain-group-callback 属性,该callback必须实现com.opensymphony.workflow.util.DynamicSplitGroupCallback接口(该接口为新添加)

 

 

 

test case :

 

/*
 * Copyright (c) 2002-2003 by OpenSymphony
 * All rights reserved.
 */
package com.siemens.dingchao.test;

import junit.framework.TestCase;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.opensymphony.workflow.Workflow;
import com.opensymphony.workflow.basic.BasicWorkflow;


/**
 * @author hani Date: Feb 17, 2005 Time: 4:24:20 PM
 */
public class JoinTestCase extends TestCase {
    //~ Instance fields ////////////////////////////////////////////////////////

    private Workflow workflow;

    //~ Methods ////////////////////////////////////////////////////////////////

    public void testWithReject() throws Exception {
    	//说明:
        checkRoute(new int[] {2, 3, 2, 4 ,4 ,4});

    	//checkRoute(new int[] {4,4});
    }

    protected void setUp() throws Exception {
       /* workflow = new BasicWorkflow("testuser");*/
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        workflow = (Workflow)context.getBean("workflow");
    
    }

    private void checkRoute(int[] actions) throws Exception {
        long workflowId = workflow.initialize("join", 1, null);

        for (int i = 0; i < actions.length; i++) {
            workflow.doAction(workflowId, actions[i], null);
        }
    }
}
 

 

test case 的红颜色部分标明了 步骤的 route 均是 xml里面的action id ,你可以把这个数字和xml对照起来看。

 

 

调试结果     以下图例都是 表 os_currentstep:

 

流程实例初始化


 

运行action  2 此时动态split为3个任务,因为 owner为 test1,test2,test3



 运行action : 3  reject(驳回) 回到step1

运行action 2 ,相当于再次提交任务,此时任务再次分裂成为3个任务,owner分别为test1 ,test2,test3


test1 执行action 4 : test1 用户提交任务


test2执行action 4 : test2 用户提交任务


test3 执行action 4 : test3 用户提交任务


此时3个用户都提交了任务,任务移动到下一步 , 即 结束节点,至此 这个工作流实例完成

 

在看看 os_historystep 表的数据 :

 


我们可以发现历史信息业完整的记录了这个操作轨迹。

 

希望能给你们带来帮助

  • 大小: 17.7 KB
  • 大小: 14.6 KB
  • 大小: 17.8 KB
  • 大小: 16.3 KB
  • 大小: 14.7 KB
  • 大小: 12.9 KB
  • 大小: 29.1 KB
  • 大小: 15.8 KB
分享到:
评论
7 楼 zhaoyangjay 2015-06-25  
你是不是改过JAR包中的源代码了,AbstractWorkflow是不是被你改过了?
6 楼 dingchao.lonton 2013-10-23  
TheMatrix 写道
有个问题,例子中都是用同一个用户去执行doAction三遍,现实中应该是不同的三个人去执行doAction,比如test1、test2、test3三人,怎么保证任何一个人先执行doAction,能处理owner=自己的那一条记录呢(OS_CURRENTSTEP)?
因为我发现,无论谁先执行doAction,都只是把这三条记录的第一条记录给执行掉了,那条记录的owner!=自己。

是有问题,需要修改一下源代码了。
5 楼 TheMatrix 2013-10-21  
有个问题,例子中都是用同一个用户去执行doAction三遍,现实中应该是不同的三个人去执行doAction,比如test1、test2、test3三人,怎么保证任何一个人先执行doAction,能处理owner=自己的那一条记录呢(OS_CURRENTSTEP)?
因为我发现,无论谁先执行doAction,都只是把这三条记录的第一条记录给执行掉了,那条记录的owner!=自己。
4 楼 opera4130 2012-10-18  
这个很赞,谢谢分享
3 楼 dingchao.lonton 2012-09-02  
public class ApproveJoinCondition implements Condition {

	public boolean passesCondition(Map transientVars, Map args, PropertySet ps)throws WorkflowException {
		// TODO Auto-generated method stub
		JoinNodes jn = (JoinNodes)transientVars.get("jn");
		
		Collection steps  = (Collection)jn.getSteps();
		Iterator it = steps.iterator();
		Step step = null;
		boolean result = true;
		while(it.hasNext()){
			step = (Step)it.next();
			result = result&&"Finished".equals(step.getStatus());
		}
		return result;
	}

}


处理 Approval的condition;

public class RejectJoinCondition implements Condition {

	public boolean passesCondition(Map transientVars, Map args, PropertySet ps)
			throws WorkflowException {
		// TODO Auto-generated method stub
		JoinNodes jn = (JoinNodes)transientVars.get("jn");
		
		Collection steps  = (Collection)jn.getSteps();
		Iterator it = steps.iterator();
		Step step = null;
		boolean result = false;
		while(it.hasNext()){
			step = (Step)it.next();
			result = result || "Rejected".equals(step.getStatus());
		}
		return result;
	}

}

处理 Reject的 condition

另外需要修改xml


<joins>
    <join id="1">
      <conditions type="AND">
        <condition type="class">
          <arg name="class.name">com.siemens.dingchao.test.ApproveJoinCondition</arg>
        </condition>
      </conditions>
      
      <unconditional-result old-status="Finished" status="Underway" step="4"/>
    </join>
    <join id="2">
      <conditions type="AND">
        <condition type="class">
          <arg name="class.name">com.siemens.dingchao.test.RejectJoinCondition</arg>
        </condition>
      </conditions>
      <unconditional-result old-status="Finished" status="Underway" step="1"/>
    </join>
  </joins>


另外 源代码还需要修改一个地方 就是JoinNodes 需要加一个 getSteps 方法 你可以反编译了再加,完后替换这个class即可

以上2个condition对全部通过,和一票否决的情况可以通用,但如果 比如说 2/3的人通过 ,否决这样的情况需要自己写自己的判断逻辑了
2 楼 dingchao.lonton 2012-09-01  
kjj 写道
 <join id="1">  
      <conditions type="AND">  
        <condition type="beanshell">  
          <arg name="script">"Finished".equals(jn.getStep(2,"test1").getStatus())</arg>  
        </condition>  
        <condition type="beanshell">  
          <arg name="script">"Finished".equals(jn.getStep(2,"test2").getStatus())</arg>  
        </condition>  
         <condition type="beanshell">  
          <arg name="script">"Finished".equals(jn.getStep(2,"test3").getStatus())</arg>  
        </condition>  
      </conditions>  
      <unconditional-result old-status="Finished" status="Underway" step="4"/>  
    </join>  


难道这个join只能join 固定的三个步骤吗!!!
如果会签数不确定能否可以,对付。。。。。。。

当然可以对付,我是用beanshell写的,你可以写自己的condition啊?

import java.util.Map;

import com.opensymphony.module.propertyset.PropertySet;
import com.opensymphony.workflow.Condition;
import com.opensymphony.workflow.JoinNodes;
import com.opensymphony.workflow.WorkflowException;

public enum JoinCondition implements Condition {
	;

	@Override
	public boolean passesCondition(Map transientVars, Map args, PropertySet ps)
			throws WorkflowException {
		// TODO Auto-generated method stub
		JoinNodes jn = (JoinNodes)transientVars.get("jn");
		
		//bla bla 你自己的判断逻辑
		return false;
	}

}

1 楼 kjj 2012-09-01  
 <join id="1">  
      <conditions type="AND">  
        <condition type="beanshell">  
          <arg name="script">"Finished".equals(jn.getStep(2,"test1").getStatus())</arg>  
        </condition>  
        <condition type="beanshell">  
          <arg name="script">"Finished".equals(jn.getStep(2,"test2").getStatus())</arg>  
        </condition>  
         <condition type="beanshell">  
          <arg name="script">"Finished".equals(jn.getStep(2,"test3").getStatus())</arg>  
        </condition>  
      </conditions>  
      <unconditional-result old-status="Finished" status="Underway" step="4"/>  
    </join>  


难道这个join只能join 固定的三个步骤吗!!!
如果会签数不确定能否可以,对付。。。。。。。

相关推荐

    osworkflow+spring+jdbc实现

    标题 "osworkflow+spring+jdbc实现" 涉及到的是一个集成开源工作流引擎OSWorkflow(OpenSymphony Workflow)与Spring框架,并利用JDBC作为持久化存储的解决方案。这个集成使得开发者能够在Java应用中方便地引入工作...

    OSWorkflow

    OSWorkflow

    OSWorkFlow

    OSWorkFlow是一个基于Java的工作流引擎,用于在企业级应用中实现复杂的业务流程自动化。它提供了灵活的流程定义和执行模型,使得开发者可以方便地设计、实施和管理各种工作流。下面将详细介绍OSWorkFlow的核心概念、...

    目前osworkflow最新最全的资料合集

    **osworkflow** 是一个强大的开源工作流引擎,主要用于在Java应用程序中实现...通过深入学习这些文档,你可以掌握如何利用osworkflow来实现动态、可扩展的工作流解决方案,从而提升你的Java应用程序的业务处理能力。

    OSWorkflow中文手册 OSWorkflow中文手册

    OSWorkflow提供了这样的平台,支持动态流程修改,使得在运行时可以更新流程模型。 2. **OSWorkflow核心概念** - **流程定义**:定义了工作流的各个步骤和转换规则,包括活动(Activities)、决策(Decisions)和...

    基于osworkflow框架的OA系统demo

    本DEMO是基于osworkflow框架构建的一个会议室使用申请的实例,旨在展示如何利用osworkflow来实现一个具体的办公自动化功能。 **1. osworkflow简介** OSWorkflow提供了强大的工作流定义和执行能力,包括流程设计、...

    workflow_osworkflow_工作流_

    其中,OSWorkflow是一款开源的工作流引擎,特别适用于Java应用环境,提供了丰富的API和灵活的配置,使得开发者能够轻松地实现复杂的工作流程。 **OSWorkflow简介** OSWorkflow,全称OpenSymphony Workflow,是由...

    osworkflow指导文档 + 源码

    osWorkflow 支持动态流程更改、事件驱动的流程控制以及多种持久化机制,如 JNDI、XML 文件和数据库存储。 ### 二、核心概念 1. **流程定义(Workflow Definition)**:osWorkflow 中的流程定义是通过 XML 文件来...

    osworkflow中文入门指南,osworkflow+hibernate+spring集成配置,osworkflow-2.8.0.jar

    **利用osworkflow实现业务流程.doc** 可能包含了一些实际案例,演示如何使用osworkflow设计和实现具体的业务流程,比如审批流程、订单处理流程等。 **Osworkflow2.8.0+hibernate3.1.3+spring1.2.8集成环境配置安装...

    osworkflow

    osWorkflow 提供了一种灵活的方式来定义、修改和控制应用程序中的工作流程,使得开发者可以轻松地实现业务流程自动化。 **工作流管理系统** 工作流管理系统(WfMS)是用于自动化业务过程的软件工具,它允许用户...

    osworkflow_bundle2

    OSWorkflow 是一个开源的工作流引擎,它允许开发者在应用程序中实现复杂的业务流程。"osworkflow_bundle2" 是一个专门用于学习 OSWorkflow 的示例集合,它包含了一系列的资源和实例,帮助用户理解并掌握如何使用这个...

    OSWorkflow源码及电子书

    OSWorkflow,全称为OpenSymphony Workflow,是一款轻量级、高度灵活的开源工作流引擎,专为满足各类组织和企业的工作流程管理需求而设计。它提供了丰富的API和配置选项,使得开发者可以轻松地将工作流集成到自己的...

    osworkflow2.8

    OSWorkflow 是一个开源的工作流引擎,它允许开发者在应用程序中实现复杂的业务流程管理。这个"osworkflow2.8"版本是OSWorkflow的一个特定发行版,包含了一整套用于设计、执行和管理工作流程的工具和资源。下面我们将...

    OsWorkflow工作流实例

    这个工作流实例是作者基于OsWorkflow实现的,已经过测试,确保其功能正常。 首先,我们需要理解OsWorkflow的基本结构。工作流定义通常是一个XML文件,其中包含了流程的所有步骤和它们之间的转移规则。例如,一个...

    OSWorkFlow 请假实例

    OSWorkFlow是一个强大的工作流引擎,它在IT领域中被广泛应用来实现业务流程自动化。工作流,简单来说,就是一系列相互关联的任务,按照特定的顺序执行,以完成一个业务过程。OSWorkFlow作为开源的工作流解决方案,为...

    工作流 osworkflow 例子和原理

    OSWorkflow的实现基于XML配置文件,这使得工作流定义易于理解和修改。通过XML,你可以定义每个步骤的属性,如参与者、后继步骤和相关条件。此外,OSWorkflow提供了API和事件监听器机制,允许开发者在流程执行过程中...

Global site tag (gtag.js) - Google Analytics