随着Web Servcie技术日益成熟和流行,许多企业的很多部门相应地都创建了Web Service服务。如何在不改变这些Web Service正常运行的情况下,将这些Web Service集成起来创造出新的业务模型、业务流程就成为一个比较突出的业务需求。
要解决这个问题,一定要有一种新的流程语言能够将Web Service给串起来,这种新的业务流程语言就是BPEL。
BPEL需要在BPEL的运行环境(BPEL引擎)下运行。一个BPEL流程一般会创建一个BPEL流程实例,BPEL就是定义这个BPEL流程如何和合作伙伴(外部的Web Service)进行交互的。
BPEL将通过合作伙伴连接来实现服务的调用。BPEL的合作伙伴连接只定义所要调用的接口,一个抽象的 WSDL接口,这个接口在BPEL运行时再绑定到真正的服务提供者上面。这样使接口和实现之间达到了一种松散耦合的效果,如果说Web Service实现了接口的可重用性,那么BPEL实现的是流程的可重用性。因为BPEL实现了抽象的WSDL接口的集成,所以它也属于SOA的解决方案 之一。
本章主要内容:
l 结合Java程序的实例来说明BPEL的基本思想;
l 通过房屋贷款的实例详细说明BPEL的创建过程。
l 通过实例阐明BPEL的各种过程组件的概念和用法;
l 通过实例阐明BPEL的各种活动的概念和用法。
许多开发人员觉得BPEL很神秘,不知道到底是什么意思。主要是因为它是根据很抽象的基于WSDL的Web Service再定义一些抽象执行的流程。
其实BPEL一点都不神秘,是一个很简单的东西。
首先谈一下BPEL和WSDL的区别,WSDL只是定义接口参数,不会定义如何实现接口,而BPEL不仅有自 己的接口定义(也是一个WSDL文件,包括输入参数、方法操作名、返回参数),BPEL会定义如何调用其他服务的接口来实现自己的接口。简单地说, BPEL通过流程编程将各种接口组合在一起,其目的在于提供一个“集成了各种接口”的接口。
首先BPEL会有一个起点和终点。
l 它的起点就是“receive”,也就是接收它自己接口的输入参数。
l 它的终点就是“reply”,也就是得到它自己接口的返回参数。
整个BPEL就是定义如何通过它的接口输入参数,调用其他外部服务的接口,得到其接口的返回参数。
它最重要的是两个定义:
l 一个是赋值命令(Assign/Copy),通过赋值命令将某一变量值赋给所要调用的接口的输入参数。
l 另一个是调用命令(Invoke),通过Invoke命令来调用外部服务。
另外一个比较重要的就是条件命令(Case Condition),根据变量的不同来定义各种条件,然后根据各种条件来调用不同的服务。这些变量可以来自于输入参数,也可以来自于调用外部服务的结果。作为BPEL运行环境的一个重要功能就是能够保存并查询到这些变量。
这里用一个简单的实例来进一步说明BPEL的基本思想和核心本质,并用Java实例来模拟BPEL的创建过程。
首先假定已经有了两个Web Service,一个为“加法服务”,专门处理两个数相加,名为AddService;另外一个为“减法服务”,专门处理两个数相减,名为SubtractService。
现在有一个新的需求,需要将上面两个Web Service集成起来,也就是需要创建一个新的服务,称为“运算服务”,名为CaculatorService,它有一个运算类型的参数,当运算类型为“加法”时,调用加法服务,当运算类型为“减法”时,调用减法服务。
看到这里,读者可能会想,直接写一个Java程序,调用者两个Web Service不就可以了吗?笔者的回答是:直接用Java编程当然可以,事实上现在大家就是这么做的。但是Java本质上是一种具体的程序语言,只能运 行于JVM的Java运行环境,不是一种抽象性的通用的标准语言,而BPEL是一种标准化的执行语言,如何能够真正运行BPEL不是它所关心的事情,它还 需要各个厂商去开发自己的BPEL运行环境,尽管各个厂商所开发的BPEL的运行环境可能会不一样,但是同一个BPEL所开发的程序,应该可以运行于这些 不同的BPEL运行环境,得到的结果应该是一样的。就像Web Service的WSDL文件一样,它只管如何定义服务、服务接口、服务操作、服务参数等,如何具体实现Web Service不是WSDL所要关心的事情。
下面继续前面的实例,两个Service都会有自己的WSDL定义。下面用实际的Java语言来类似地描述一下,以便于理解。
下面的AddService .java相当于AddService.wsdl(用Java的Interface可能更贴切一点,用Java的class可以说明得更详细一点)。
Public class AddService{
Public double add(double addParameter1, double addParameter2) {
Double addResposne;
addResposne = addParameter1 +addParameter2;
// WSDL 不会定义具体如何实现,
//此处只是说明如何实现操作
Return addResponse;
}
}
下面的SubstractService .java相当SubstractService.wsdl。
Public class SubstractService{
Public double substract(double substractParameter1, double substractParameter2) {
Double substractResposne = substractParameter1 - substractParameter2;
Return substractResposne;
}
}
下面用Java模拟BPEL的创建过程。
首先需要创建BPEL的接口,下面用CaculatorBPELInterface.java来说明,它将有3个参数,其中paramter1和paramter2是需要运算的两个数,第3个参数processType表示运算的类型。
Public Interface CaculatorBPELInterface{
Public double caculatorProcess(double parameter1, double parameter2, String processType) ;
}
下面将创建一个Java实现类,说明如何对应于BPEL的创建过程(许多语句(如一些变量定义和赋值定义)对Java来说是不必要的;为了模拟BPEL的创建过程,让Java开发人员更好地把握BPEL的创建过程,相应地加入了这些程序语句)。
Public class CaculatorBPELImple implements CaculatorBPELInterface {
Public double caculatorProcess (double parameter1, double parameter2, String processType) {
//步骤1:定义所要调用的外部类(相当于定义BPEL里面partnerLink)
AddService addServer = new AddService();
SubstractService substractSevice = new SubstractService();
/*步骤2:定义输入和输出变量(相当于定义BPEL里面变量variable)
定义的变量如下:
●BPEL接口的输入和输出变量
●所要调用的外部类的接口方法的输入和输出变量 */
//定义BPEL接口的输入变量
Double caculatorProcessParameter1Request;
Double caculatorProcessParameter2Request;
Double caculatorProcessTypeRequest;
//定义BPEL接口的输出变量
Double caculatorProcessResponse;
//定义加法服务的输入变量
Double addParameter1Request;
Double addParameter2Request;
//定义加法服务的输出变量
Double addResponse;
//定义减法服务的输入变量
Double substractParameter1Request;
Double substractParameter2Request;
//定义减法服务的输出变量
Double substractParameter1Response;
//***将请求参数赋值给BPEL接口的输入变量 (相当于BPEL的receive)
caculatorProcessParameter1Request = parameter1;
caculatorProcessParameter2Request= parameter2;
Double caculatorProcessTypeRequest= processType;
//步骤3:定义条件,并调用外部接口
If (caculatorProcessTypeRequest.equals.(“add”’)
// (相当于BPEL的switch/condition/case)
{
//**下面将调用加法服务
//将接口请求变量传给加法服务的请求变量 (相当于BPEL的assign/copy)
addParameter1Request = caculatorProcessParameter1Request;
addParameter2Request = caculatorProcessParameter2Request;
//调用addService的接口 (相当于BPEL的Invoke)
addResponse = addServer.add(addParameter1Request, addParameter2Request);
//将addResponse赋值给BPEL接口的输出变量 (相当于BPEL的assign/copy)
caculatorProcessResponse = addResponse;
} else //(相当于BPEL的 condition/otherwise)
{
//将接口请求变量传给减法服务的请求变量 (相当于BPEL的assign/copy)
substractParameter1Request = caculatorProcessParameter1Request;
substractParameter2Request = caculatorProcessParameter2Request;
//调用substractService的接口 (相当于BPEL的invoke)
substractResponse =
substractServer.substract(substractParameter1Request, substractParameter2Request);
//将substractResponse赋值给BPEL接口的输出变量(相当于BPEL的assign/copy)
caculatorProcessResponse = substractResponse;
}
Return caculatorProcessResponse; //相当于BPEL的reply
}
}
上面已经用Java模拟了BPEL的创建过程。下面将基于同样的实例,用描述性的WSDL和BPEL来实现其创建过程,使读者对BPEL能有一个整体性的掌握。
假设已经有了前面的加法服务和减法服务的WSDL文件,加法服务为AddService.wsdl,减法服务为Substract.wsdl,它们的主要内容如下所示:
AddService.wsdl
|――getRequest (请求消息)
|――addParameter1(double)
|――addParameter2(double)
|――getResponse(返回消息)
|――addResponse(double)
|――addServcie(portType接口)
|――add(operation接口操作)
|――AddService(service 服务名称)
substractService.wsdl
|――getRequest(请求消息)
|――subtractParameter1(double)
|――subtractParameter2(double)
|――getResponse(返回消息)
|――substractResponse(double)
|――substractService(portType接口)
|――substract(operation接口操作)
|――SubstractService(服务名称)
下面需要为BPEL创建一个服务接口,caculator.wsdl如下:
caculatorService.wsdl
|――getRequest(请求消息)
|――parameter1(double)
|――parameter2(double)
|――processType (String)
|――getResponse(返回消息)
|――caculatorProcessResponse(double)
|――caculatorService(portType接口)
|――caculatorPorcess(operation接口操作)
|――CaculatorService(service 服务名称)
下面介绍创建BPEL的基本过程,即caculatorServiceProcess.bpel(下面只是说明BPEL的创建过程,没有完全按照BPEL的语法,具体的BPEL语法和BPEL编程实例后面会详细介绍)。
所定义的变量包括:
l BPEL接口的输入和输出变量。
l 所要调用的外部服务partnerLink的接口操作的输入和输出变量。
<bpel:variables>
<!—BPEL流程的请求变量和返回变量 -->
variable name="request-bpel" messageType采用caculatorService/getRequest
variable name="response-bpel" messageType采用caculatorService/getResponse
<!—加法服务的请求变量和返回变量 -->
variable name="request-add" messageType采用addService/getRequest
variable name="response-add" messageType采用addService/getResponse
<!—减法服务请求变量和返回变量 -->
variable name="request-substract" messageType采用substractService/getRequest
variable name="response-substract" messageType采用substractService/getResponse
</bpel:variables>
BPEL定义变量的方式与Java是不一样的,主要因为BPEL所调用的WSDL是XML语言上面的所定义的变量并不是对应一个具体的值,它实际上对应的是一个数据结构。
如caculatorService/getRequest 对应的是caculatorService.wsdl下面的
getRequest
|――parameter1(double)
|――parameter2 (double)
|――processType (String)
也就是说getRequest下面的所有参数都包含进去了。
<bpel:receive name="request" partnerLink=" CaculatorService "
portType=" caculatorService " operation=" caculatorPorcess "
variable=" request-bpel " >
Receive是整个BPEL的起点,所定义的变量request-bpel从服务请求中得到赋值,整个后面的业务过程将以这个请求变量作为触发点。
程序将根据不同的条件调用不同的服务,所以先要设立各种条件:
<bpel:switch>
<bpel:case
condition="getVariableData('request-bpel','payload','getRequest/processType'>= 'add' >
… 调用加法服务
</bpel:case>
上面的程序表示在request-bpel的变量中取出路径为getRequest/ processType所对应的变量值,如果满足这个值为“add”的条件时,可以在里面加入程序,完成相应的任务。
<bpel:otherwise>
…… 调用减法服务
<bpel:otherwise>
如果上面的所有条件都不满足时,可以在里面加入程序,完成默认的任务。
为了调用外部服务,先要给外部服务赋值:
<bpel:assign>
<bpel:copy>
<bpel:from variable="request-bpel" part="payload" query="/getRequest/parameter1" />
<bpel:to variable="request-add" part="payload" query="/getRequest /addParameter1" />
</bpel:copy>
</bpel:assign>
上面表示将BPEL过程所收到的初始变量赋给加法服务的请求变量。
<bpel:invoke name="addService" partnerLink="AddService"
portType="addService" operation="add"
inputVariable=" request-add"
outputVariable="response-add" />
上面将会调用加法服务AddSerivce,其中request-add为输入变量。
在完成了外部服务的调用之后,就可以将外部服务的输出结果赋给BPEL流程的返回变量。文法同步骤4。
调用BPEL的reply命令,将BPEL流程的返回变量返回给服务请求者。
<bpel:reply name="response" partnerLink="CaculatorService"
portType=" caculatorService" operation="caculatorPorcess"
variable=" response-bpel" />
这里比较一下步骤2的<bpel:receive>和步骤6的<bpel:reply>,可以 看到它们的partnerLink的名字、portType的名字、operation的名字都是一样的。它们是调用的同一个服务下面的同一个接口操作, 只是两个命令的变量不一样。<bpel:receive>是收到请求消息的变量值,<bpel:reply>是将响应变量的结果 返回给服务请求者。
事实上,BPEL运行环境执行<bpel:receive>后,就在等待<bpel:reply>的返回结果。
上面通过实例介绍了BPEL的基本创建过程,有了这些基本的BPEL整体创建思路后,就可以为更好地理解一些具体的语法打下了基础。
<!-- page -->因为BPEL本质上只是一个程序语言,要让开发人员知道程序语言的用法,详细介绍程序语言用法,最后是结合实 际的典型的实例。所以这里先提出一个房屋贷款实例的实例来,然后在后面的章节再详细介绍用法。本节的BPEL房屋贷款实例需要结合7.9节的 ServiceMix来运行,7.9节将提供外部服务让BPEL来集成,同时ServiceMix也将提供BPEL的运行环境。
IT技术本质上都是为业务服务的,只有真正理解了业务,才能真正理解IT。下面将详细介绍房屋贷款业务的来龙去脉。
目前我国的房价正在以飞快的速度上涨,这与“炒房者”一人拥有多套房子不无关系,其实大部分“炒房者”大都是 用银行贷款的钱在“炒房”,这与我国目前的银行贷款政策不无关系。目前我国的银行房屋贷款政策对购买一套房和多套房的首付和贷款利率是一样的,这样银行政 策是将“炒房者”和“真正的购房者”置于同等的地位,为“炒房者”提供了机会,使他们能够“以小搏大”,以少量的首付贷款得到房子,然后倒手得到巨大的差 价。
银行应该根据“购房者”目前的拥有的房屋数量提供不同的首付比例和贷款利率。房屋数量为0的客户应该得到最小的首付比例和最低的贷款利率;然后根据客户的房屋数量依次相应提高首付比例和贷款利率,这样应该能够起到平抑房价的作用。下面的实例假定银行政策已经这么执行了。
目前有5个独立的Web Service,它们相互之间互不知道,也不能互相调用。
l HouseLoanAgency:专门负责根据客户姓名查询客户的已有的房屋数量。
l Bank0:专门负责处理已有房屋数量为0的客户的房屋贷款需求,有专门针对房屋数量为0的贷款首付和贷款利率。
l Bank1:专门负责处理已有房屋数量为1的客户的房屋贷款需求,有专门针对房屋数量为1的贷款首付和贷款利率。
l Bank2:专门负责处理已有房屋数量为2的客户的房屋贷款需求,有专门针对房屋数量为2的贷款首付和贷款利率。
l Bank3:专门负责处理已有房屋数量大于2的客户的房屋贷款需求,有专门针对房屋数量大于2的贷款首付和贷款利率。
在完全不影响上面的5个Web Service正常运行的情况下,包括不修改上面5个Web Service的任何代码的情况下,将上面的5个Web Service集成起来。这个新的系统只需要输入用户姓名,这个系统会首先自动查出客户的房屋数量,然后自动转到相应的银行进行房屋贷款服务,客户最后可以得到对应于他的目前已有房屋数量的贷款首付和贷款利率。
图6-1显示了用BPEL实现的业务流程,具体过程如下。
(1)BPEL将首先以客户姓名name作为输入变量,调用HouseLoanAgency的Web Service,得到客户的目前拥有的房屋数量。
(2)如果客户房屋数量为0,BPEL将调用Bank0的服务;如果客户房屋数量为1,BPEL将调用Bank1的服务;如果客户房屋数量为2,BPEL将调用Bank2的服务;如果客户房屋数量大于2,BPEL将调用Bank3的服务。
(3)BPEL将从Bank返回的首付比率和贷款利率返回给服务请求者。
图6-1 通过BPEL实现房屋贷款业务
定义BPEL流程的接口WSDL主要有两个目的:
(1)让客户端知道如何使用相应的组织数据来调用这个BPEL服务。例如下面的接口portType 名称为“HouseLoanBroker”;其下的输入input为“tns:getLoanQuoteRequest”,其中“tns”为命名空间 “urn:sample:soa:houseloanbroker”;getLoanQuoteRequest下面含有元素“name”;总的结构如下:
HouseLoanBroker(portType)
|――tns:getLoanQuoteRequest (input)
|――name (String)
这样客户端可以创建响应的SOAP请求消息:
<getLoanQuoteRequest xmlns='urn:sample:soa:houseloanbroker'>
<name>Zhang San</name>
</getLoanQuoteRequest>
(2)BPEL引擎可以根据请求消息创建相应的BPEL实例,执行BPEL流程。例如:
BPEL在收到上面的服务请求后,根据命名空间“urn:sample:soa:houseloanbroker”和请求消息“getLoanQuoteRequest”,就知道创建和执行该BPEL流程houseloanbroker。
例程6-1显示了BPEL流程的接口WSDL文件。
例程6-1 houseloanbroker.wsdl
<?xml version="1.0" encoding="UTF-8"?>
<definitions targetNamespace="urn:sample:soa:houseloanbroker"
xmlns:tns="urn:sample:soa:houseloanbroker"
xmlns:typens="urn:sample:soa:houseloanbroker:types"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:plnk="http://schemas.xmlsoap.org/ws/2003/05/partner-link/"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:bpws="http://schemas.xmlsoap.org/ws/2003/03/business-process/">
<import namespace="urn:sample:soa:houseloanagency" location="houseloanagency.wsdl" />
<import namespace="urn:sample:soa:bank" location="bank.wsdl" />
<!-- type defs -->
<types>
<xsd:schema
targetNamespace="urn:sample:soa:houseloanbroker:types"
xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<xsd:complexType name="getLoanQuoteRequest">
<xsd:sequence>
<xsd:element name="name" type="xsd:string" />
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name="getLoanQuoteResponse">
<xsd:sequence>
<xsd:element name="rate" type="xsd:double" />
<xsd:element name="firstpaidratio" type="xsd:double" />
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name="unknownNAMEFault">
<xsd:sequence>
<xsd:element name="name" type="xsd:string" />
</xsd:sequence>
</xsd:complexType>
</xsd:schema>
</types>
<message name="getLoanQuoteRequest">
<part name="payload" type="typens:getLoanQuoteRequest" />
</message>
<message name="getLoanQuoteResponse">
<part name="payload" type="typens:getLoanQuoteResponse" />
</message>
<message name="unknownNAMEFault">
<part name="payload" type="typens:unknownNAMEFault" />
</message>
<portType name="HouseLoanBroker">
<operation name="getLoanQuote">
<input message="tns:getLoanQuoteRequest" />
<output message="tns:getLoanQuoteResponse" />
<fault name="UnknownNAME" message="tns:unknownNAMEFault" />
</operation>
</portType>
<plnk:partnerLinkType name="HouseLoanBrokerPL">
<plnk:role name="HouseLoanBrokerService" portType="tns:HouseLoanBroker" />
</plnk:partnerLinkType>
<binding name="HouseLoanBroker" type="tns:HouseLoanBroker">
<operation name="request"></operation>
</binding>
<service name="HouseLoanBrokerService">
<port name="houseloanbroker" binding="tns:HouseLoanBroker" />
</service>
</definitions>
下面是BPEL需要调用的外部服务houseloanagency的WSDL文件,这个外部服务接收顾客姓 名,输出顾客已有的房屋数量。通过这个WSDL,BPEL可以知道所要调用的外部服务的接口、接口操作、输入消息和输出消息、异常,这都是BPEL流程在 调用外部服务时需要知道的,对于houseloanagency.wsdl,这些参数如下。
l 接口:HouseLoanAgency;
l 接口操作:getHouseNumber;
l 输入消息为下面的树状结构:
getHouseNumberRequest
|――name (String)
l 输出消息为下面的树状结构:
getHouseNumberResponse
|――housenumber (int)
l 异常:unknownNAMEFault。
此外,BPEL需要引入该WSDL,并定义该服务为合作伙伴,所以还需要在此加入合作伙伴的定义。
<plnk:partnerLinkType name="HouseLoanAgencyPL">
<plnk:role name="HouseLoanAgencyService" portType="tns:HouseLoanAgency" />
</plnk:partnerLinkType>
上面portType="tns:HouseLoanAgency"指该合作伙伴所对应的WSDL的接口。BPEL的一个合作伙伴,对应于WSDL的一个接口。如果WSDL有多个接口,则需要定义多个合作伙伴。
例程6-2显示了houseloanagency.wsdl的内容。
例程6-2 houseloanagency.wsdl
<?xml version="1.0" encoding="UTF-8"?>
<definitions targetNamespace="urn:sample:soa:houseloanagency"
xmlns:tns="urn:sample:soa:houseloanagency"
xmlns:typens="urn:sample:soa:houseloanagency:types"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:plnk="http://schemas.xmlsoap.org/ws/2003/05/partner-link/"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:bpws="http://schemas.xmlsoap.org/ws/2003/03/business-process/">
<types>
<xsd:schema
targetNamespace="urn:sample:soa:houseloanagency:types"
xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<xsd:complexType name="getHouseNumberRequest">
<xsd:sequence>
<xsd:element name="name" type="xsd:string" />
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name="getHouseNumberResponse">
<xsd:sequence>
<xsd:element name="housenumber" type="xsd:int" />
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name="unknownNAMEFault">
<xsd:sequence>
<xsd:element name="name" type="xsd:string" />
</xsd:sequence>
</xsd:complexType>
</xsd:schema>
</types>
<message name="getHouseNumberRequest">
<part name="payload" type="typens:getHouseNumberRequest" />
</message>
<message name="getHouseNumberResponse">
<part name="payload" type="typens:getHouseNumberResponse" />
</message>
<message name="unknownNAMEFault">
<part name="payload" type="typens:unknownSSNFault" />
</message>
<portType name="HouseLoanAgency">
<operation name="getHouseNumber">
<input message="tns:getHouseNumberRequest" />
<output message="tns:getHouseNumberResponse" />
<fault name="UnknownNAME" message="tns:unknownNAMEFault"/>
</operation>
</portType>
<plnk:partnerLinkType name="HouseLoanAgencyPL">
<plnk:role name="HouseLoanAgencyService" portType="tns:HouseLoanAgency" />
</plnk:partnerLinkType>
</definitions>
下面是所要调用的4个Bank的服务,它们的服务接口一样,只是输出结果的首付比例和贷款比例不同,所以采用同一个WSDL文件即可,但是在BPEL中需要定义4个合作伙伴,这4个合作伙伴所指向的服务地址是不同的,Bank的WSDL内容如下。
l 接口:Bank;
l 接口操作:getLoanQuote;
l 输入消息为如下树状结构;
getLoanQuoteRequest
|――housenumber(int)
l 输出消息为如下的树状结构:
getHouseNumberResponse
|――rate (double)
|――firstpaidratio (double)
此外,BPEL需要引入该WSDL,并定义该服务为合作伙伴,所以还需要在此加入合作伙伴定义:
<plnk:partnerLinkType name="BankPL">
<plnk:role name="BankService" portType="tns:Bank" />
</plnk:partnerLinkType>
例程6-3显示了bank.wsdl的内容。
例程6-3 bank.wsdl
<?xml version="1.0" encoding="UTF-8"?>
<definitions targetNamespace="urn:sample:soa:bank"
xmlns:tns="urn:sample:soa:bank"
xmlns:typens="urn:sample:soa:bank:types"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:plnk="http://schemas.xmlsoap.org/ws/2003/05/partner-link/"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:bpws="http://schemas.xmlsoap.org/ws/2003/03/business-process/">
<types>
<xsd:schema
targetNamespace="urn:sample:soa:bank:types"
xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<xsd:complexType name="getLoanQuoteRequest">
<xsd:sequence>
<xsd:element name="housenumber" type="xsd:int" />
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name="getLoanQuoteResponse">
<xsd:sequence>
<xsd:element name="rate" type="xsd:double" />
<xsd:element name="firstpaidratio" type="xsd:double" />
</xsd:sequence>
</xsd:complexType>
</xsd:schema>
</types>
<message name="getLoanQuoteRequest">
<part name="payload" type="typens:getLoanQuoteRequest" />
</message>
<message name="getLoanQuoteResponse">
<part name="payload" type="typens:getLoanQuoteResponse" />
</message>
<portType name="Bank">
<operation name="getLoanQuote">
<input message="tns:getLoanQuoteRequest" />
<output message="tns:getLoanQuoteResponse" />
</operation>
</portType>
<plnk:partnerLinkType name="BankPL">
<plnk:role name="BankService" portType="tns:Bank" />
</plnk:partnerLinkType>
</definitions>
因为本例需要调用5个外部服务Web Servcie,每个外部服务有一个接口(portType),所以总共需要定义5个合作伙伴,它们的分别为HouseLoanBroker 、Bank1、Bank2、Bank3、Bank4。
这里需要说明的是,尽管4个Bank的名称不一样,但是它们的partnerRole名称和partnerLinkType名称是一样的。因为它们都用同样的WSDL接口。
下面是合作伙伴的具体内容:
<bpel:process name="houseloanbrokerProcess"
targetNamespace="urn:sample:soa:houseloanbroker"
xmlns:tns="urn:sample:soa:houseloanbroker"
xmlns:ca="urn:sample:soa:houseloanagency"
xmlns:bk="urn:sample:soa:bank"
xmlns:svc="urn:sample:soa:service" suppressJoinFailure="yes"
xmlns:bpel="http://schemas.xmlsoap.org/ws/2003/03/business-process/"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:sm="http://servicemix.apache.org/schemas/bpe/1.0"
xsi:schemaLocation="http://schemas.xmlsoap.org/ws/2003/03/business-process/
http://schemas.xmlsoap.org/ws/2003/03/business-process/">
<bpel:import importType="http://schemas.xmlsoap.org/wsdl/" location="houseloanbroker.wsdl"
namespace="urn:sample:soa:houseloanbroker"/>
<bpel:partnerLinks>
<bpel:partnerLink myRole="HouseLoanBrokerService" name="HouseLoanBroker"
partnerLinkType="tns:HouseLoanBrokerPL"/>
<bpel:partnerLink partnerRole="HouseLoanAgencyService" name="HouseLoanAgency"
partnerLinkType="ca:HouseLoanAgencyPL"/>
<bpel:partnerLink partnerRole="BankService" name=
"Bank0" partnerLinkType="bk:BankPL"/>
<bpel:partnerLink partnerRole="BankService" name=
"Bank1" partnerLinkType="bk:BankPL"/>
<bpel:partnerLink partnerRole="BankService" name=
"Bank2" partnerLinkType="bk:BankPL"/>
<bpel:partnerLink partnerRole="BankService" name=
"Bank3" partnerLinkType="bk:BankPL"/>
</bpel:partnerLinks>
对应于上面定义的每一个合作伙伴都应该定义一个输入和输出变量,以便调用<invoke>、 <receive>、<reply>时用到。因为Bank的接口一样,所以对4个Bank服务只定义一个输入变量;因为有可能要 对Bank服务的结果进行聚集和比较,所以对每个Bank合作伙伴都定义一个输出变量。
本例中所定义的变量如下。
l Request:对应于BPEL接口的输入变量;
l Response:对应于BPEL接口的输出变量;
l ca-housenumber-request:合作伙伴HouseLoanAgency的输入变量;
l ca-housenumber-response:合作伙伴HouseLoanAgnecy的输出变量;
l bk-loanquote-request:Bank的输入变量(对应于所有Bank);
l bk-loanquote-response-0:Bank0的输出变量;
l bk-loanquote-response-1:Bank1的输出变量;
l bk-loanquote-response-1:Bank2的输出变量;
l bk-loanquote-response-1:Bank3的输出变量。
本例所定义的变量程序如下:
<bpel:variables>
<bpel:variable name="request" messageType="tns:getLoanQuoteRequest" />
<bpel:variable name="response" messageType="tns:getLoanQuoteResponse" />
<bpel:variable name="ca-housenumber-request" messageType="ca:getHouseNumberRequest" />
<bpel:variable name="ca-housenumber-response" messageType="ca:getHouseNumberResponse" />
<bpel:variable name="bk-loanquote-request" messageType="bk:getLoanQuoteRequest" />
<bpel:variable name="bk-loanquote-response-0" messageType="bk:getLoanQuoteResponse" />
<bpel:variable name="bk-loanquote-response-1" messageType="bk:getLoanQuoteResponse" />
<bpel:variable name="bk-loanquote-response-2" messageType="bk:getLoanQuoteResponse" />
<bpel:variable name="bk-loanquote-response-3" messageType="bk:getLoanQuoteResponse" />
<bpel:variable name="unknownNAME" messageType="tns:unknownNAMEFault" />
</bpel:variables>
本例中将会定义一个异常处理,它会截获HouseLoanAgency所抛出的“unknownNAME”。它表示HouseLoanAgency房屋数量查询机构没有该顾客姓名的记录,该顾客姓名不在数据库中。
<bpel:faultHandlers>
<bpel:catch faultName="ca:UnkownNAME">
<bpel:sequence>
<bpel:assign>
<bpel:copy>
<bpel:from variable="request" part="payload"
query="/tns:getLoanQuoteRequest/tns:name" />
<bpel:to variable="unknownNAME" part="payload"
query="/tns:unknownNAMEFault/tns:name" />
</bpel:copy>
</bpel:assign>
<bpel:reply name="response" partnerLink="HouseLoanBrokerResponse"
portType="tns:HouseLoanBroker" operation="getLoanQuote"
variable="unknownNAME" faultName="tns:unknownNAME" >
</bpel:reply>
</bpel:sequence>
</bpel:catch>
</bpel:faultHandlers>
下面介绍BPEL的流程。
<receive>是整个BPEL的起点,它将接收来自客户端的请求消息,程序如下:
<bpel:sequence>
<bpel:receive name="request" partnerLink="HouseLoanBroker"
portType="tns:HouseLoanBroker" operation="getLoanQuote" variable="request"
createInstance="yes">
</bpel:receive>
下面将BPEL接口收到的输入变量“request”中的“name”参数通过<bpel:assign>赋值给ca-housenumber-request变量的“name”参数(这两个变量都是上面刚刚定义的):
<bpel:flow>
<bpel:sequence>
<bpel:assign>
<bpel:copy>
<bpel:from variable="request" part="payload"
query="/tns:getLoanQuoteRequest/tns:name" />
<bpel:to variable="ca-housenumber-request" part="payload"
query="/ca:getHouseNumberRequest/ca:name" />
</bpel:copy>
</bpel:assign>
下面通过合作伙伴的名称HouseLoanAgency调用其接口的接口操作“getHouseNumber”,并且以上面刚刚得到赋值的ca-housenumber-request作为输入变量inputVariable:
<bpel:invoke name="service" partnerLink="HouseLoanAgency"
portType="ca:HouseLoanAgency" operation="getHouseNumber"
inputVariable="ca-housenumber-request"
outputVariable="ca-housenumber-response" />
</bpel:sequence>
</bpel:flow>
下面将<invoke>得到的房屋数量输出变量赋值给上面定义的Bank服务的输入变量:
<bpel:assign>
<bpel:copy>
<bpel:from variable="ca-housenumber-response" part="payload"
query="/ca:getHouseNumberResponse/ca:housenumber" />
<bpel:to variable="bk-loanquote-request" part="payload"
query="/bk:getLoanQuoteRequest/bk:housenumber"/>
</bpel:copy>
</bpel:assign>
下面是BPEL的分支,即房屋数量为0时的一个分支:
<bpel:switch>
<bpel:case condition="getVariableData('bk-loanquote-request', 'payload',
'/bk:getLoanQuoteRequest/bk:housenumber') = 0 ">
<bpel:sequence>
<bpel:flow>
下面表示房屋数量为0时,将会调用Bank0的服务:
<bpel:invoke name="bank0" partnerLink="Bank0"
portType="bk:Bank" operation="getLoanQuote"
inputVariable="bk-loanquote-request"
outputVariable="bk-loanquote-response-0"
sm:endpoint="urn:logicblaze:soa:bank:Bank0:bank" />
</bpel:flow>
将Bank0输出变量中的rate参数赋值给BPEL接口输出变量response的rate参数:
<bpel:assign>
<bpel:copy>
<bpel:from expression="getVariableData
('bk-loanquote-response-0',
'payload', '/bk:getLoanQuoteResponse/
bk:rate')" />
<bpel:to variable="response" part="payload"
query="/tns:getLoanQuoteResponse/
tns:rate" />
</bpel:copy>
将Bank0输出变量中的firstpaidratio赋值给BPEL输出变量的firstpaidratio:
<bpel:copy>
<bpel:from expression="getVariableData
('bk-loanquote-response-0',
'payload', '/bk:getLoanQuoteResponse/
bk:firstpaidratio')" />
<bpel:to variable="response" part="payload"
query="/tns:getLoanQuoteResponse/
tns:firstpaidratio" />
</bpel:copy>
</bpel:assign>
</bpel:sequence>
</bpel:case>
下面省略房屋数量为1的BPEL分支情形代码,因为业务一样、程序一样,将会调用Bank1的服务,将Bank输出变量中的rate和firstpaidratio参数赋值给BPEL接口输出变量response的rate和firstpaidratio参数:
<bpel:case condition="getVariableData('bk-loanquote-request', 'payload',
'/bk:getLoanQuoteRequest/
bk:housenumber') = 1 " >
<bpel:sequence>
……
</bpel:sequence>
</bpel:case>
下面省略房屋数量为2的BPEL分支情形,过程同上。
<bpel:case condition="getVariableData('bk-loanquote-request', 'payload',
'/bk:getLoanQuoteRequest/
bk:housenumber') = 2 ">
<bpel:sequence>
……
</bpel:sequence>
</bpel:case>
下面是用<otherwise>来实现BPEL条件的最后一个分支,房屋数量大于2时的一个分支。
<bpel:otherwise>
<bpel:sequence>
<bpel:flow>
在默认情形下将会调用Bank3的服务:
<bpel:invoke name="bank3" partnerLink="Bank3"
portType="bk:Bank" operation="getLoanQuote"
inputVariable="bk-loanquote-request"
outputVariable="bk-loanquote-response-3"
sm:endpoint="urn:logicblaze:soa:bank:Bank3:bank" />
</bpel:flow>
将Bank3输出变量中的rate参数赋值给BPEL接口输出变量response的rate参数:
<bpel:assign>
<bpel:copy>
<bpel:from expression="getVariableData
('bk-loanquote-response-3',
'payload', '/bk:getLoanQuoteResponse/
bk:rate')" />
<bpel:to variable="response" part="payload"
query="/tns:getLoanQuoteResponse/
tns:rate" />
</bpel:copy>
将Bank3输出变量中的firstpaidratio赋值给BPEL输出变量response的firstpaidratio:
<bpel:copy>
<bpel:from expression="getVariableData
('bk-loanquote-response-3',
'payload', '/bk:getLoanQuoteResponse/
bk:firstpaidratio')" />
<bpel:to variable="response" part="payload"
query="/tns:getLoanQuoteResponse/
tns:firstpaidratio" />
</bpel:copy>
</bpel:assign>
</bpel:sequence>
</bpel:otherwise>
</bpel:switch>
<reply>是BPEL的结束,将变量结果返回非服务请求者,程序如下:
<bpel:reply name="response" partnerLink="HouseLoanBroker"
portType="tns:HouseLoanBroker" operation="getLoanQuote"
variable="response" />
</bpel:sequence>
</bpel:process>
BPEL流程的本质就是通过BPEL的活动(Activity)将BPEL的过程组件“串起来”。所谓“串起来”就是通过BPEL的活动让BPEL的过程组件产生一种动态的交互。比如说将一个变量赋值给另外一个变量,就是一种变量之间的动态的交互。
BPEL流程中经常用到的一些组件如下:
l 合作伙伴连接(Partner Links);
l 变量Varaibles;
l 相关集Correlation Sets;
l 错误处理Fault handles;
l 补偿处理(Compensation Handlers)等。
合作伙伴连接(Partner Links)是指在BPEL中的服务提供者,它主要分为两种,一种是BPEL流程所要调用的外部服务;另一种是指BPEL自己所要提供的服务。
定义合作伙伴连接(Partner Links)实际上包括定义两件事情:
l 在BPEL中定义<partnerLink>
l 在对应的WSDL中定义<partnerLinkType>
BPEL首先需要定义合作伙伴的<partnerLink>,然后需要在所要引入的WSDL中定义所对应的WSDL接口PortType。
如上面的实例在BPEL文件中定义Bank0这么一个partnerLink:
<bpel:partnerLink partnerRole="BankService" name="Bank0"
partnerLinkType="bk:BankPL"/>
如果是外部服务,需要采用partnerRole来定义合作伙伴类型,如上例中:
<bpel:partnerLink partnerRole="BankService" name="Bank0"
partnerLinkType="bk:BankPL"/>
如果是BPEL自己的接口,则通过myRole来定义了一个合作伙伴,它表示这个服务接口是BPEL自己提供服务的,如前面实例:
<bpel:partnerLink myRole="HouseLoanBrokerService" name="HouseLoanBroker"
partnerLinkType="tns:HouseLoanBrokerPL"/>
为了使合作伙伴能够对应相应的WSDL的接口,需要在对应的bank.wsdl里面通过<partnerLinkType>定义它所对应的WSDL的portType:
<plnk:partnerLinkType name="BankPL">
<plnk:role name="BankService" portType="tns:Bank" />
</plnk:partnerLinkType>
上面就将BPEL的partnerLinkTyp(BankPL)和WSDL的portType(Bank)对应上了。
BPEL是通过合作伙伴的名字来调用<invoke>外部服务的,BPEL引擎也是通过合作伙伴的名字来和真正的服务提供者进行绑定的,如前面的实例将通过“Bank0”来调用外部的Bank0服务。
<bpel:invoke name="bank0" partnerLink="Bank0"
portType="bk:Bank" operation="getLoanQuote"
inputVariable="bk-loanquote-request"
outputVariable="bk-loanquote-response-0"
sm:endpoint="urn:sample:soa:bank:Bank0:bank" />
BPEL里面的变量Varaibles类似于Java的变量的概念,是一个保存数据的地方。但是BPEL和Java语言 不太一样,Java语言的参数可以直接调用,比方说,Java程序的其他函数接口可以直接调用上面的double parameter1。但是作为BPEL语言是不行的。因为BPEL需要集成WSDL定义的Web Service。同一个参数名字在不同的WSDL中参数是不一样的,因为WSDL是一种XML结构,其本质是一种树状的数据结构,直接赋值是赋不进去的, 所以需要先定义变量。一般来说对每一个合作伙伴,不管有多少输入参数和输出参数,定义两个变量即可,一个请求(request)变量,一个响应变量。
<bpel:variables>
<bpel:variable name="request" messageType="tns:getLoanQuoteRequest" />
</bpel:variables>
上面房屋贷款中的变量实例,可以通过messageType将WSDL文件中的变量类型取过来,如上面的变量“request”实际上对应houseloanbroker.wsdl所定义的变量,结构如下:
getLoanQuoteRequest
|――name(String)
这样,通过messageType可以直接定义变量的数据结构。
相关集(Correlation Sets)用一组特定的数据,来关联和标定一个BPEL过程实例。因为本质上BPEL是一个分布式的消息系统,不像分布式的对象系统,可以通过 Instance ID来标识实例。BPEL是通过交换消息里面的一些属性值来标识的。而相关集可以不需要Instance ID就可以识别BPEL过程实例,对许多需要保持会话状态的BPEL业务需求来说,是非常有用的。
下面用一个日常生活中的实例进一步说明这个问题。
假设一个小学一个班级有30个学生。小学校长想知道每个学生的数学、语文和英语的成绩。小学校长准备了30个空白信封,每个信封里面有一张白纸让相应的老师填入成绩。
小学校长直接将30个空白信封先交给了数学老师,数学老师在信封里的白纸上填好数学成绩交回给校长。
小学校长接着将这30个信封交给语文老师,希望在原有数学成绩的基础上再填上语文成绩。语文老师没法完成此工作,因为信封上没有标志,语文老师无法知道哪个学生对应哪个信封。
小学校长首先将30个空白信封上标上学生的姓名和学生证号码,交给数学老师,数学老师就可以根据信封上的标志 分别填上每个学生相应的数学成绩,然后交给语文老师;语文老师就可以根据信封上的标志分别填上每个学生相应的语文成绩,然后交给英语老师;英语老师就可以 根据信封上的标志分别填上每个学生相应的英语成绩。这样每个学生的各科成绩就都填进去了。
上面实例中的信封相当于BPEL的一个实例,信封上的学生的姓名和学生证号码相当于一个相关集。
在BPEL中设置相关集比较复杂,下面以房屋贷款的实例来说明在BPEL中设置相关集的过程思路,下面以顾客姓名“name”数据建立相关集。
首先需要建立“相关属性”如下:
<definitions name="properties"
targetNamespace="http://example.com/houseloanCorrelation.wsdl"
xmlns:tns="http://example.com/ houseloanCorrelation.wsdl"
xmlns:bpws="http://schemas.xmlsoap.org/ws/2003/03/business-process/"
xmlns="http://schemas.xmlsoap.org/wsdl/">
<!-- define correlation properties -->
<bpws:property name="name" type="xsd:string"/>
</definitions>
接下来需要在WSDL中定义“相关属性”。因为同一个属性名,比如“name”,它可能是请求变量的参数,也可能是响应变 量的参数,还有路径的关系,因为变量在SOAP消息中是树状的数据结构。所以这里需要用<bpws:propertyAlias>标签从请求 消息中得到“name”值。
<?xml version="1.0" encoding="UTF-8"?>
<definitions targetNamespace=" http://example.com/messageCorrelation.wsdl "
xmlns:tns=" http://example.com/messageCorrelation.wsdl "
xmlns:cor="http://example.com/houseloanCorrelation.wsdl"
xmlns="http://schemas.xmlsoap.org/wsdl/"
……
xmlns:bpws="http://schemas.xmlsoap.org/ws/2003/03/business-process/">
<types>
<xsd:schema>
<xsd:complexType name="getLoanQuoteRequest">
<xsd:sequence>
<xsd:element name="name" type="xsd:string" />
</xsd:sequence>
</xsd:complexType>
……
</xsd:schema>
</types>
<message name="getLoanQuoteRequest">
<part name="payload" type="typens:getLoanQuoteRequest" />
</message>
<message name="getLoanQuoteResponse">
<part name="payload" type="typens:getLoanQuoteResponse" />
</message>
……
<bpws:propertyAlias propertyName="cor:name"
messageType="tns: getLoanQuoteRequest " part=" payload "
query="/ getLoanQuoteRequest /name"/>
……
</definitions>
接下来在BPEL中定义相关集:
<correlationSets xmlns:cor="http://example.com/houseloanCorrelation.wsdl">
<correlationSet name="HouseLoanCor" properties="cor:name "/>
</correlationSets>
进而在BPEL的< receive>活动中调用相关集:
<bpel:receive name="request" partnerLink="HouseLoanBroker"
portType="tns:HouseLoanBroker" operation="getLoanQuote" variable="request">
<correlations>
<correlation set=" HouseLoanCor " initiate="yes">
</correlations>
</bpel:receive>
这样就将房屋贷款的流程实例和相关集关联起来了。
BPEL程序在调用合作伙伴的过程中,或者合作伙伴的服务有可能会抛出异常,或者在BPEL流程内部调用中也 会抛出异常(如将字符串String类型的变量赋值给整数类型int的变量)。BPEL流程有一套机制可以捕获异常,并可以定义捕获异常后的下一步行动。 如本例中定义了一个“unknownNAME”(系统中无此顾客姓名的异常),BPEL流程在捕获此异常后,通过<reply>返回给服务请 求者。
需要说明的是,这个“unknownNAME”必须事先在BPEL的接口WSDL中定义,本例中的定义如下:
<xsd:complexType name="unknownNAMEFault">
<xsd:sequence>
<xsd:element name="name" type="xsd:string" />
</xsd:sequence>
</xsd:complexType>
而且合作伙伴返回的异常必须符合上面的格式,才能捕获这个异常,否则是不会捕获这类异常的。合作伙伴返回的异常实例如下:
<InvalidNAME xmlns="urn:sample:soa:houseloanagency">
<name>Zhang San1</name>
</InvalidNAME>
下面是BPEL异常处理的写法:
<bpel:faultHandlers>
<bpel:catch faultName="ca:UnkownNAME">
<bpel:sequence>
<bpel:assign>
<bpel:copy>
<bpel:from variable="request" part="payload"
query="/tns:getLoanQuoteRequest/tns:name" />
<bpel:to variable="unknownNAME" part="payload"
query="/tns:unknownNAMEFault/tns:name" />
</bpel:copy>
</bpel:assign>
<bpel:reply name="response" partnerLink="HouseLoanBrokerResponse"
portType="tns:HouseLoanBroker" operation="getLoanQuote"
variable="unknownNAME" faultName="tns:unknownNAM