一种消息处理器的的设计与实现

时间:2011-07-20

  XML和简单对象访问协议(Simple Object Access Protocol,SOAP)带来了信息交换技术革命性的进步,极大地促进了分布式应用的发展。XML(Extensible Markup Language)即可扩展标记语言,它与HTML一样,都是SGML(STandard Generalized Markup Language,标准通用标记语言)。Xml是Internet环境中跨平台的,依赖于内容的技术,是当前处理结构化文档信息的有力工具。扩展标记语言XML是一种简单的数据存储语言,使用一系列简单的标记描述数据,而这些标记可以用方便的方式建立,虽然XML占用的空间比二进制数据要占用更多的空间,但XML极其简单易于掌握和使用。目前,在基于SOAP信息交换应用的集成化开发环境和工具中关于SOAP的技术对开发人员基本上是透明的,即屏蔽和隐藏了SOAP实现和处理的细节。深入了解SOAP消息的处理过程和SOAP处理器的工作机理,可以对分布式应用进行更加灵活地控制。

  1  SOAP概述

  简单对象访问协议(SOAP)是一种轻量的、简单的、基于 XML 的协议,它被设计成在 WEB 上交换结构化的和固化的信息。 SOAP 可以和现存的许多因特网协议和格式结合使用,包括超文本传输协议( HTTP),简单邮件传输协议(SMTP),多用途网际邮件扩充协议(MIME)。它还支持从消息系统到远程过程调用(RPC)等大量的应用程序。SOAP 消息基本上是从发送端到接收端的单向传输,但它们常常结合起来执行类似于请求 / 应答的模式。所有的 SOAP 消息都使用 XML 编码。一条 SOAP 消息就是一个包含有一个必需的 SOAP 的封装包,一个可选的 SOAP 标头和一个必需的 SOAP 体块的 XML 文档。

  2  SOAP消息处理模型

  2.1 SOAP的应用架构和消息处理模型框架

  SOAP成功地解决了异构网络环境中的软件组件和应用程序之间进行可靠信息交换的难题。以Web Service为例,其SOAP RPC的应用架构如图1所示。一般开发环境都隐藏了SOAP消息处理的过程,这部分功能封装于由开发环境自动生成的客户端和服务器端的SOAP代理模块中,其中都使用了编译好的高层类库,用户据此难以探究其内部的处理细节。

  实际上,客户端和服务器端的SOAP代理模块内部做了大量的处理工作。SOAP RPC的处理过程如图2所示。客户端模块按照Web Service使用接口要求的格式和规范,通过XML处理器将客户端的RPC调用编码成SOAP请求消息。SOAP请求消息中按规定封装了请求服务所必需的信息,借助传输协议传送到服务器端;服务器的传输协议监听器得到传输的内容,由服务器端的代理模块按照WSDL文档描述的Web Service接口的信息采用XML处理器对SOAP请求消息进行解析,提取出适当的信息,生成提供相应的Web服务的对象,调用请求的方法,得到服务程序的处理结果,再按WSDL文档的要求将服务处理结果生成SOAP响应消息,借助传输协议传回客户端;客户端的代理按照WSDL文档描述的信息解析SOAP响应消息,提取出适当的信息返回客户程序。在客户端和服务器端对SOAP消息进行串行化和反串行化期间,涉及到客户程序的数据类型与XML的类型系统XSD之间,以及XML的类型系统XSD与服务程序的数据类型之间的相互映射,还有对象参数的序列化和反序列化工作。

  当SOAP与HTTP绑定时,服务器端的消息监听可以采用ASP或ISAPI处理方式。

  SOAP消息的生成和解析工作与WSDL文档密切相关,只有遵照Web Service对应的WSDL文档中描述的信息格式,SOAP消息才能得到正确的、自动化的处理。

  2.2 WSDL简介

  WSDL是Web Service的描述语言,用于描述Web Service的服务,接口绑定等,它以一种XML模式来描述Web Service的接口。服务的使用方根据这些描述信息来理解如何使用服务。由于描述方法和格式具有统一的标准和规范,因此便于由机器来自动处理描述信息。实际上,很多开发环境就是由软件工具根据服务模块自动生成相应的WSDL文档。

  在具体实现SOAP处理器的过程中需要编程访问WSDL文档。为使篇幅简洁,本文将采用MS SOAP TK 3.0中提供的WSDL文档模型对象接口(详情可参见MSTK3.0开发文档)来编程分析WSDL文档,即通过编程对象及其方法遍历文档中的元素,得到的接口描述信息用以处理SOAP消息。

  3 实现SOAP处理器的功能

  由图2所示的SOAP消息处理器的功能和过程架构,可以实现自行开发的SOAP处理器。

  3.1 具体的实现模型

  由于SOAP消息和WSDL文档都是基于XML格式的,所以代理的工作主要是分析和处理XML文档。可以根据SOAP消息封装格式、编码规则、RPC表示以及WSDL文档结构的协议规范,采用任一种XML解析器工具来处理SOAP和WSDL这类特殊的XML文档(如图2所示)。本文将采用MSTK3.0中的低层API来实现SOAP代理,处理过程反映了技术机理和细节。将准备实现的客户端和服务器端的SOAP代理类命名为mySoapClient和mySoapServer。其内部实现过程模型分别如图3和图4所示。

  3.2 具体实现

  下面采用VB6.0来实现mySoapClient类和mySoapServer类。

  3.2.1 客户端SOAP代理

  新建ActiveX DLL工程MyClientSoap,添加并引用Microsoft Soap Type Library v3.0。将以下代码加入mySoapClient类模块中。

  Dim Port As IWSDLPort

  Public Sub Initialize(ByVal WSDLFileName As String,

  Optional ByVal ServiceName As String=″″,

  Optional ByVal PortName As String=″″,

  Optional ByVal WSMLFileName As String=″″)

  Dim Fetched As Long

  Dim WSDLReader As New WSDLReader30

  ′load WSDL file

  WSDLReader.Load WSDLFileName,WSMLFileName

  ′get the service

  Dim ServiceEnumerator As IEnumWSDLService,Service As IWSDLService

  WSDLReader.GetSoapServices ServiceEnumerator

  If ServiceName=″″Then

  ServiceEnumerator.Next 1,Service,Fetched

  Else

  ServiceEnumerator.Find ServiceName,Service

  End If

  ′get the port

  Dim PortEnumerator As IEnumWSDLPorts

  Service.GetSoapPorts PortEnumerator

  If PortName=″″Then

  PortEnumerator.Next 1,Port,Fetched

  Else

  PortEnumerator.Find PortName,Port

  End If

  End Sub

  Public Function Invoke(ByVal OperationName As String,

  ParamArray Parameter() As Variant) As Variant

  ′find the operation

  Dim OperationEnumerator As IEnumWSDLOperations,

  Operation As IWSDLOperation

  Port.GetSoapOperations OperationEnumerator

  OperationEnumerator.Find OperationName,Operation

  ′prepare request

  Dim MapperEnumerator As IEnumSoapMappers,

  Mapper As ISoapMapper

  Operation.GetOperationParts MapperEnumerator

  Dim Fetched As Long

  MapperEnumerator.Next 1,Mapper,Fetched

  Do While Fetched=1

  If (Mapper.IsInput=smInput) Or (Mapper.IsInput=smInOut) Then

  Mapper.ComValue=Parameter(Mapper.ParameterOrder)

  End If

  MapperEnumerator.Next 1,Mapper,Fetched

  Loop

  ′construct the request message and send it.

  ′(that is,invoking the operation and getting result)

  Dim Serializer As SoapSerializer30

  Dim Connector As SoapConnector30

  Set Connector=New HttpConnector30

  Connector.ConnectWSDL Port

  Connector.BeginMessageWSDL Operation

  Set Serializer=New SoapSerializer30

  Serializer.Init Connector.InputStream

  Serializer.StartEnvelope

  Serializer.StartBody

  Operation.Save Serializer,True′writes the XML

  Serializer.EndBody

  Serializer.EndEnvelope

  ′load response

  Dim SoapReader As New SoapReader30

  SoapReader.Load Connector.OutputStream

  Operation.Load SoapReader,False

  ′return the outgoing parameters and result

  MapperEnumerator.Reset

  MapperEnumerator.Next 1,Mapper,Fetched

  Do While Fetched=1

  If Mapper.IsInput=smOutput Then

  If Mapper.VariantType=vbObject Then

  Set Invoke=Mapper.ComValue

  Else

  Invoke=Mapper.ComValue

  End If

  ElseIf Mapper.IsInput=smInOut Then

  Parameter(Mapper.ParameterOrder)=Mapper.ComValue

  End If

  MapperEnumerator.Next 1,Mapper,Fetched

  Loop

  End Function

  将mySoapClient类实例化就成为客户端SOAP代理对象。mySoapClient类向外提供了二个可调用的接口:(1)Initialize方法利用WSDL文档中描述的Web Service接口消息来初始化客户端代理对象,用户还可指定要远程调用的Web Service的服务名称ServiceName和端口名称PortName。(2)Invoke方法向客户程序提供了使用Web Service的接口,用户只需提供Web Service中的方法名称和此方法的参数,就可以调用Web Service。

  3.2.2 服务器端SOAP代理

  新建ActiveX DLL工程MyServerSoap,并添加引用Microsoft Soap Type Library v3.0。

  将以下代码加入mySoapServer类模块中。

  Dim WSDLReader As WSDLReader30

  Public Sub Initialize(ByVal WSDLFileName As String,ByVal WSMLFileName As String)

  Set WSDLReader=New WSDLReader30

  WSDLReader.SetProperty ″LoadOnServer″,True

  WSDLReader.Load WSDLFileName,WSMLFileName

  End Sub

  Public Sub ProcessRequest(ByVal Request As IStream,ByVal Response As IStream)

  Dim WSDLPort As IWSDLPort

  Dim WSDLOperation As IWSDLOperation

  Dim Serializer As New SoapSerializer30

  Dim SoapReader As New SoapReader30

  SoapReader.Load Request

  WSDLReader.ParseRequest SoapReader,WSDLPort,

  WSDLOperation

  WSDLOperation.Load SoapReader,True

  Serializer.Init Response

  Serializer.StartEnvelope

  WSDLOperation.ExecuteOperation SoapReader,Serializer

  Serializer.StartBody

  WSDLOperation.Save Serializer,False

  Serializer.EndBody

  Serializer.EndEnvelope

  End Sub

  编译后生成MyServerSoap.dll。将mySoapServer类实例化就成为服务器端SOAP代理对象。Initialize方法利用WSDL文档来初始化服务器端代理对象。ProcessRequest方法分析接收到的SOAP请求消息,调用相应的Web Service方法,将结果编码成SOAP响应消息回传。

  以上详细剖析了SOAP消息的处理过程,设计实现了SOAP处理器。本文中实现代理类的代码还可进一步细化,甚至完全可以不引用MSTK3.0中的低层API类库,而采用XML解析器来编写SOAP处理器。由于篇幅所限,还有一些细节问题未能进一步展开探讨,如复杂数据类型的编码、SOAP头部的处理、错误的捕获和附件的处理等。本文已实现了SOAP处理器重要的功能框架,用户可以在此基础上,增加调用附加功能的接口,这样可以实现功能更加丰富的SOAP处理器。


  
上一篇:一种代理服务器的设计
下一篇:一个设备无关的移动电子商务开发平台的实现

免责声明: 凡注明来源本网的所有作品,均为本网合法拥有版权或有权使用的作品,欢迎转载,注明出处。非本网作品均来自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

相关技术资料