|
在這篇文章中,主要介紹osworkflow的核心概念以及重要的部分,讓大家對osworkflow有個比較全面的認識。 在osworkflow中最最核心的東西就是工作流定義的xml文件。盡管它并不是一定要定義成xml文件。但是xml格式是一種標(biāo)準(zhǔn)的通用的格式。 這個xml文件為某一個給定的工作流進行描述steps、states,transitions,和functionality。下面闡述一下此xml的一般規(guī)則: 1、 一個工作流由多個steps組成 2、 對于每個step,可以包括多個actions。一個action可以被設(shè)置成自動運行或者需要通過人工交互才可以運行。 3、 每個action都要包括至少一個unconditional result和0或多個conditional results。 4、 如果設(shè)定了多個concitioanl results,所有當(dāng)中的第一個將被執(zhí)行,如果沒有設(shè)定conditional results或者沒有conditions滿足,那么執(zhí)行unconditional result 5、 一個result過后可能依舊停留在當(dāng)前的step中,一個新的step,一個split,一個join。在所有的情形中,工作流的state跟著變化(例子工作流中的states分別為:Underway,Queued,和finished) 6、 如果一個result引起一個split,這個result會指定split的屬性,以指向一個split元素。 7、 一個split可以有一個或者多個unconditional results,但是沒有conditional results。Unconditional results。Unconditional results需要指定steps。 8、 一個propertyset是一個持久層數(shù)據(jù)的map,在全局應(yīng)用中都是可用的。 9、 還有一種叫做transientVars的map,它只存活于一個工作流調(diào)用過程中的一定的生命周期,它將會對所有functions和conditions,包括所有的registers,user input,以及工作流上下文狀態(tài)等起作用。
工作流概念: 下面開始理解osworkflow的核心概念: 對于step,status,actions部分就不多說明了,其實我覺得理解概念的最快方法應(yīng)該是參照實例,即使我們不能用高高大大的詞匯描繪出來,能自己理解是什么意思就可以了。 Unconditional result 和 conditional results 這里做以簡單介紹,對于每個action,要求至少存在一個Unconditional result,一個result也就是通過一系列指示來告訴osworkflow下一步的任務(wù)是干什么。這種調(diào)用使得產(chǎn)生變遷進而從一個state到另外一個state。這種概念是在UML的狀態(tài)機里有講,希望了解狀態(tài)機相關(guān)概念的可以到UML相關(guān)書籍中查看。 Conditional result是unconditional result的一種擴展。不同的地方在于他需要一些子元素:condition。用and 和 or來標(biāo)志各個condition之間的關(guān)系。 Conditional 和unconditional 的最終result可以產(chǎn)生三種效應(yīng)或者說是結(jié)果: 1、 一個新的step/status 2、 一個split,出現(xiàn)一或多個step/status 3、 一個join,一個新的step/status 普遍的,一個split或者join不能result出另外一個split或join。 一個step/status result可以按下面方式簡單的設(shè)定: <unconditional-result old-status="Finished" step="2" status="Underway" owner="${someOwner}"/>
從一個state split 到多個 states可以按以下方式達到: <unconditional-result split="1"/>...<splits> <split id="1"> <unconditional-result old-status="Finished" step="2" status="Underway" owner="${someOwner}"/> <unconditional-result old-status="Finished" step="2" status="Underway" owner="${someOtherOwner}"/> </split></splits>
Joins是比較復(fù)雜的用例。一個典型的join看起來大致如下: <!-- for step id 6 -><unconditional-result join="1"/>...<!- for step id 8 -><unconditional-result join="1"/>...<joins> <join id="1"> <join id="1"> <conditions type="AND"> <condition type="beanshell"> <arg name="script"> "Finished".equals(jn.getStep(6).getStatus() && "Finished".equals(jn.getStep(8).getStatus()) </arg> </condition> </conditions> </join> <unconditional-result old-status="Finished" status="Underway" owner="test" step="2"/> </join></joins>
上面這段代碼中最需要關(guān)心的就應(yīng)該是jn。當(dāng)join實際發(fā)生的時候,這個特殊的變量jn可以被用來建立表達式。本質(zhì)上,可以很容易理解出這個段xml的意思就是:當(dāng)step6和8都finish時候在此處進行join。 Functions部分: Osworkflow用function來定義商業(yè)邏輯和一些需要定義執(zhí)行的服務(wù)。用functions標(biāo)簽來表示。 兩種functions(pre和post) Pre 是在工作流進行某個變遷之前需要被執(zhí)行的。一個比較好的例子:為了在result中state變更產(chǎn)生,而先建立caller。 Post是在之后執(zhí)行的。如當(dāng)某一個state改變后,發(fā)送一email到某處。 Functions可以在兩個分別的地方被指定:steps和actions。
Trigger Functions
Validators
Registers 一個function:用來返回一個用以被其他普通對象能夠容易訪問得到的對象。尤其是指workflow 的實體類。返回的對象類型不閑典型的例子如:document,metadata,issue,task等。非常便利。 <registers> <register name="doc" class="com.acme.DocumentRegister"/></registers>...<results> <result condition="doc.priority == 1" step="1" status="Underway" owner="${someManager}"/> <unconditional-result step="1" status="Queued"/></results>
Conditions
變量
授權(quán)與限權(quán)
自動執(zhí)行的action 設(shè)置auto=true
Common and global actions: Common和globalactions的主要作用在于在工作流定義文件中能夠避免代碼重復(fù)。 基本思想就是簡單。這兩種actions是在最開始就進行說明的,在initial-actions元素后面。 這兩處還不能完全理解好! Common actions:在最開始定義好,可以在其他地方如此引用 <common-action id="100" /> 例如一個“send mail”的action Global actions:不同之處在于顯式的被某一個step引用。它通常對所有的steps都是可用的。一個例子:“終止工作流”,在任何一步,都有可能終止工作流。
需要注意的是:這兩種actions要具備唯一的ID,而不能和其他action的ID重復(fù)。
接下來主要講解關(guān)于function的四種情況: osworkflow中的function就是可以在變遷之前或者后進行執(zhí)行的內(nèi)容。 1、 基于java的functions 從classloader中加載java 類,通過jndi找會java類,遠程ejbs,本地ejb。 這一類型的function必須實現(xiàn)接口:com.opensymphony.workflow.FunctionProvider,在這個接口中有一個方法execute。這個方法(execute)需要三個參數(shù) 可以自己去查api即可找到這三個參數(shù),兩個map一個propertyset 基于java的functions在以下類型是可用的。 (1)、class 對于一個class function,classloader必須能夠知道你的function的類名。如下: <function type="class"> <arg name="class.name">com.acme.FooFunction</arg> <arg name="message">The message is ${message}</arg></function> (2)、jndi jndi function與 class想類似,當(dāng)然必須已經(jīng)確實存在于jndi樹中,在這里,需要一個jndi.location來進行配置: <function type="jndi"> <arg name="jndi.location">java:/FooFunction</arg> <arg name="message">The message is ${message}</arg></function> (3)、remote-ejb 這部分跳過 <function type="remote-ejb"> <arg name="ejb.location">java:/comp/env/FooEJB</arg> <arg name="message">The message is ${message}</arg></function> (4)、local-ejb 跳過 <function type="local-ejb"> <arg name="ejb.location">java:/comp/env/FooEJB</arg> <arg name="message">The message is ${message}</arg></function> 2、 beanshell function osworkflow支持beanshell,這是一種腳本語言??梢栽?/SPAN>http://www./來查看beanshell的相關(guān)內(nèi)容。 在這種情況下,需要將xml過程定義中的type設(shè)置成beanshell。另外還需要有個一個參數(shù)名字為script的參數(shù),此參數(shù)值是實際需要被執(zhí)行的。 例如: <function type="beanshell"> <arg name="script"> System.out.println("Hello, World!"); </arg></function> 三種變量:entry,context,store entry:實現(xiàn)com.opensymphony.workflow.spi.WorkflowEntry并且表示工作流實例。 Context:com.opensymphony.workflow.WorkflowContext。允許beanshell functions回滾事務(wù)或者確定caller的name Store:com.opensymphony.workflow.WorkflowStore。允許function訪問工作流的持久存儲層。 <function type="beanshell"> <arg name="script"> propertySet.setString("world", "Earth"); </arg></function><function type="beanshell"> <arg name="script"> System.out.println("Hello, "+propertySet.getString("world")); </arg></function>
輸出結(jié)果是“hello,earth”因為任何存儲在propertyset中的變量將可以在整個工作流中被持久使用。 3、 BSF function(perlscript, vbscript, javascript) 除了上面說過的兩種type的function。還有bsf類型的function。 BSF(bean scripting framework)是IBM的一個組織做的一個通用環(huán)境可以使用VBScript, Perlscript, Python, and JavaScript <function type="bsf"> <arg name="source">foo.pl</arg> <arg name="row">0</arg> <arg name="col">0</arg> <arg name="script"> print $bsf->lookupBean("propertySet").getString("foo"); </arg></function> 個人覺得這部分可以先跳過去,知道有這么一回事就可以了。 4、 Utility Function 可以到api的com.opensymphony.workflow.util這部分查看。下面只是列出一些主要的功能。相當(dāng)于java.util里的東西。 主要用于創(chuàng)建動態(tài)的工作流定義,主要是包括一些實用功能,如caller、webworkexecutor、ejbinvoker、jmsmessage、mostrecentowner、schedulejob、unschdulejob、sendmail。 這部分碰到不懂就去查api是個好辦法,這里就不去多寫的。
validators: 與functions類似,osworkflow有下面幾種不同形式的validators:java-based,beanshell,和bsf。Java-based的validators必須實現(xiàn)com.opensymphony.workflow.Validator接口(如果是remote-ejb,則需實現(xiàn)com.opensymphony.workflow.ValidatorRemote接口)。Java-based這種情況是通過拋出個InvalidInputException異常表明一個輸入是不合法的,并且停止工作流action。
在beanshell和bsf中,有一點小小不同,即使異常可以在腳本中拋出,但是不能抵達到jre。所以在beanshell和bsf中用錯誤信息來完成。邏輯如下: u 如果返回值是一個InvalidInputException對象,這個對象立刻拋出到client。 u 如果返回值是一個map,map被用做一個error/errormessage對。 u 如果返回值是一個String [],偶數(shù)字被做為key。奇數(shù)做為value來構(gòu)造一個map u 其他情況,把值轉(zhuǎn)換成string并且作為一個普通的錯誤信息來添加。
Registers: Register是一個在工作流定義文件中用來動態(tài)注冊的運行時(塊) 它也是和validators,functions差不多,都是能夠以java-based,beanshell和bsf不同格式出現(xiàn)。
Conditions: Conditions與osworkflow小小不相似之處在于:在bsf或者beanshell腳本中,有一個額外的對象叫做“jn”。這個jn來源于com.opensymphony.workflow.joinnodes。它的作用就是連接條件(join-conditions)。除此之外,condition必須有一個返回值(true or false)。 Condition必須被conditions包含成為其子元素。Conditions有一個屬性type??梢詾?/SPAN>and或者or。And表示:所有condition元素必須都為true或者false。Or表示只要有一個condition元素為true。如果你需要更多復(fù)雜的condition邏輯??梢钥紤]實現(xiàn)condition或者conditionremote接口,beanshell,bsf。如果只有一個condition子元素的時候,conditions的type屬性值可以省略。 在2.7中??梢郧短?/SPAN>conditions,這樣可以讓你實現(xiàn)更為復(fù)雜的商業(yè)邏輯。 下面是一個寫標(biāo)準(zhǔn)的conditions OSUserGroupCondition、StatusCondition、AllowOwnerOnlyCondition、DenyOwnerCondition
Soap支持,暫時略!因為暫時可能涉及不到!
Gui Designer部分略。 |
|
|