**
对 WSE 使用总结的补充
随着项目的进行,对 WSE 的了解也多了一点,前几天突然想到了一个在 Server 端动态加入 Filter 的办法。个人认为这个办法是可行的,赶紧拿出来和大家分享。
在 WSE 总结一文中我所说的无对整个站点的服务请求进行动态的加载自定义 Filter 其实是不正确的。例如,我们可以继承 HttpMoudle 来对所有进入的 http 请求进行处理, 微软网站有相关的例子 。只是就实际情况出发,在我所说的案例当中,我们的自定义 Filter 其实是与 Soap 中的一个 ServiceID 相关联的。也就是说,不同的 serviceID 对应不同的 Filter 。如果我们自己编写相应的 Http 处理程序而不使用 WSE 提供的功能,无疑是一种费时费力的事情。但是,这些 Filter 却只能在 global.asax 中添加,那么我们应该如何处理?
有一种变通的方法可以很好的解决上面的问题。那就是将自定义的 Filter 与 global.asax 结合起来使用。为了方便说明我假设一共有 4 个自定义的 Filter ,它们是 ,
,
名字
|
用途
---|---
InputHeaderFilter
|
验证 Soap Header 是否符合 Schema
InputAAFilter
|
加密
OutputHeaderFilter
|
加入 oap Header 信息
OutputAAFilter
|
解密
OutputAllFilter
|
处理输出的 Soap 信息(转换用途)
假设 ServiceID=”001” 的用户请求只需要经过 InputAAFilter ,而服务器再返回时只需要经过 OutputAAFilter 。但是, ServiceID = “002” 的用户请求可能需要经过上面的 4 个 Filter 。那么我们就无法再 global.asax 或者 web.config 文件中静态的加入 Filter 。不同的 ServiceID 对应的 Filter 以及 Filter 顺序可能是不同的,因此我们需要动态的加入这些 Filter 并执行。
InputFilter的处理
首先,我们可以如此假定,我们可以在 Web Method 中得到 ServiceID 并且创建一个新的 pipeline 对象,并把所有的 Filter 按一定顺序加入(数据库有相应字段表示顺序),然后利用 pipeline 的 IprocessInputMessage() 方法完成设计中 Input Filter 应执行的操作,代码如下:
//Input Filter
Pipeline replyPip = new Pipeline();
// 加入一个 filter
// 为了便于说明,没有使用从数据库读取的方法添加,其原理同样
// 需要注意添加 Filter 的顺序不能颠倒!
replyPip.InputFilters.Add (new AAInboundFilter());
// 执行加入的所有 InputFilter
replyPip.ProcessInputMessage(requestContext.Envelope);
这样,我们便完成了服务器端的 InputFilter 的功能,上述代码将在 WebMethod 中被调用。
OutputFilter 的处理
对于 OutPutFilter 情况要复杂一些,因为如果我们使用类似上面的技术处理 OutputFilter ( responseContext.Envelope )由于 responseContext.Envelope 是 readonly ,我们无法在程序中人为的创建并修改这个属性,因此我将使用一个替身转换的方法来实现。首先,创建一个新的 SoapEnvelope 对象,然后利用和 InputFilter 类似的技术加入 OutFilter 并处理,得到一个和真实的需要的 Response Sope 信息一致的 SoapEnvelope 。然后将这个 SoapEnvelope 保存到 ResponseContext 的中,然后在 OutputAllFIlter 中取出,赋给服务器返回的 Soap 信息即可。整个代码如下:
WebMethod 中:
//Add Output Filter
replyPip.OutputFilters.Add (new AAOutboundFilter());
SoapEnvelope env = new SoapEnvelope();
// 处理 Output SoapEnvelope
// 此处的 envelope 并不是真正的返回信息,只是替身
// 用户可以控制哪些 Output filter 可以使用
replyPip.ProcessOutputMessage(env);
// 将修改后的模拟的 envelope 信息保存在 ResponseContext 对象中
HttpSoapContext.ResponseContext.Add("env",env);
在 global.asax 的 application_start() 加入下列代码 :
WebServicesConfiguration.FilterConfiguration.OutputFilters.Add
(new OutputAllFIlter());
这样这个站点的所有响应的 Soap 信息都会经过这个 OutFilter
最后, OutputAllFIlter.cs 中代码如下:
using System;
using System.Xml;
using Microsoft.Web.Services;
using System.Xml.Schema;
using System.Collections;
namespace rottenapple
{
public class OutputAllFIlter: SoapOutputFilter
{
public OutputAllFIlter ()
{
//
// TODO: Add constructor logic here
//
}
public override void ProcessMessage(SoapEnvelope envelope)
{
// 如果 env 这个 object 则执行,否则不执行转换
// 在调用 pipeline/ProcessOutFilter 的时候也会执行这个 Filter ,但那时
// 候不做任何处理
if (envelope.Context.Contains("env") )
{
SoapEnvelope env = (SoapEnvelope)envelope.Context["env"];
XmlNode header = envelope.Header;
if (header == null)
header = envelope.CreateHeader();
header.InnerXml = env.Header.InnerXml ;
}
}
}
}
以上代码在 VS2003,WSE1.0sp1 下测试通过。
**