在与WCF服务交互过程中传递用户信息等附加信息
2008-03-28 16:18
519 查看
在Web应用中,我们可以使用Session或Cookies在浏览器与Server间维护会话状态及传递附加信息。假设有这样的需求,用户正常登陆系统后,Server产生一个UserIdentity发送到Client端,在此后的Client每次调用Service时都需要传递此UserIdentity传递到Server端以进行用户身份认证及权限审核。在WCF应用中有没有什么方法可以实现同样功能呢?
首先我们看看MSDN中对WCF的Session的说明:
它们由调用应用程序显式启动和终止。
会话期间传递的消息按照接收消息的顺序进行处理。
会话将一组消息相互关联,从而形成对话。该关联的含义是抽象的。例如,一个基于会话的通道可能会根据共享网络连接来关联消息,而另一个基于会话的通道可能会根据消息正文中的共享标记来关联消息。可以从会话派生的功能取决于关联的性质。
不存在与 WCF 会话相关联的常规数据存储区。
最后一句告诉我们,WCF中的Session是无法像Web应用一样存储附加信息的。
经过研究,我们可以通过扩展MessageHeader实现一个附加的数据存储区在Client端每次请求Service时发送到Server端。具体实现如下(以前述需求为例)。
首先我们定义一个类作为交互过程中的UserIdentity,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
namespace BNCommonPlus.BNCommonModel
...{
[Serializable]
public class UserIdentity
...{
public string UserName = string.Empty;
public string UserRuningSystemID = string.Empty;
public string UserSysGUID = string.Empty;
}
}
这里使用[Serializable]作为序列化方式。在这个类中,UserName对应用户名,UserRuningSystemID对应用户已登陆的系统ID,UserSysGUID对应用户当前的有效登陆。用户登录后Server产生这个类的一个实例回传Client,Client将其保存在缓存里,这里我定义了一个类来作为缓存,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.Data;
using BNCommonPlus.BNCommonModel;
namespace BNIIClientLayerPlus
...{
public class UserPermissionInfo
...{
private UserPermissionInfo()
...{ }
private static UserPermissionInfo _Instance = null;
public static UserPermissionInfo GetInstance()
...{
if (_Instance == null)
...{
_Instance = new UserPermissionInfo();
}
return _Instance;
}
private UserIdentity _userIdentity = null;
public UserIdentity _UserIdentity
...{
get
...{
return _userIdentity;
}
}
public void SetUserIdentity(UserIdentity ui)
...{
_userIdentity = ui;
}
}
}
这是一个单件类,Client正常登陆得到Server端回传的UserIdentity实例后可以通过如下代码将其存入缓存:
UserPermissionInfo.GetInstance().SetUserIdentity(ServerReturnedUserIdentity);
其中ServerReturnedUserIdentity就是Server产生并回传的UserIdentity
下面我们扩展MessageHeader将我们自己定义的UserIdentity加入进去,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceProcess;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
namespace BNCommon.ClientHelper
...{
public class BNClientMessageInspector : IClientMessageInspector
...{
IClientMessageInspector 成员#region IClientMessageInspector 成员
public void AfterReceiveReply(ref Message reply, object correlationState)
...{
}
public object BeforeSendRequest(ref Message request, IClientChannel channel)
...{
MessageHeader mh = MessageHeader.CreateHeader("UserIdentity", "UINS", BNIIClientLayerPlus.UserPermissionInfo.GetInstance()._UserIdentity);
request.Headers.Add(mh);
return null;
}
#endregion
}
}
这个类实现了IClientMessageInspector接口,实现该接口可以在Client每次向Server请求前及请求返回后控制Client的行为对发送和接收的数据进行处理。
现在我们需要实现BehaviorExtensionElement, IEndpointBehavior将刚刚建立的行为加入Client行为集合,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceProcess;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
namespace BNCommon.ClientHelper
...{
public class BNClientEndpointBehavior : BehaviorExtensionElement, IEndpointBehavior
...{
IEndpointBehavior 成员#region IEndpointBehavior 成员
public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
...{}
public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
...{
clientRuntime.MessageInspectors.Add(new BNClientMessageInspector());
}
public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
...{
}
public void Validate(ServiceEndpoint endpoint)
...{
return;
}
#endregion
public override Type BehaviorType
...{
get ...{ return typeof(BNClientEndpointBehavior); }
}
protected override object CreateBehavior()
...{
return new BNClientEndpointBehavior();
}
}
}
现在我们定义了Client在用户正常登陆后,每次请求都会发送缓存里面的UserIdentity实例。然后需要修改Client端的代理类,这里要分两种情况:如果客户端使用的是”SvcUtil.exe”工具生成的代理类,则需要修改对应Client的构造方法,加入如下代码:
Endpoint.Behaviors.Add(new BNCommonPlus.ClientHelper.BNClientEndpointBehavior());
如果使用ChanelFactory则需要在Create对应得ChanelFactory后加入如下代码:
_ChannelFactory.Endpoint.Behaviors.Add(new BNClientEndpointBehavior());
当然也可以通过配置文件实现上述目的。
搞定Client端后,再来看看Server端如何取Client每次发送过来的附加信息.
首先在Server端要实现IDispatchMessageInspector接口,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
using BNCommonPlus.BNCommonModel;
namespace BNCommonPlus.ServiceHelper
...{
public class DispatchMessageInspector : IDispatchMessageInspector
...{
IDispatchMessageInspector 成员#region IDispatchMessageInspector 成员
public object AfterReceiveRequest(ref Message request, IClientChannel channel, InstanceContext instanceContext)
...{
//得到身份信息
UserIdentity ui = request.Headers.GetHeader<UserIdentity>("UserIdentity", "UINS");
……..//这里写验证用户身份,检测权限的代码。
return null;
}
public void BeforeSendReply(ref Message reply, object correlationState)
...{
}
#endregion
}
}
和客户端一样将上面定义的行为加入行为集合,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
using BNCommonPlus.BNCommonModel;
namespace BNCommonPlus.ServiceHelper
...{
public class BNServiceOperationBehavior : BehaviorExtensionElement, IEndpointBehavior
...{
IEndpointBehavior 成员#region IEndpointBehavior 成员
public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
...{ }
public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
...{ }
public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
...{
endpointDispatcher.DispatchRuntime.MessageInspectors.Add(new DispatchMessageInspector());
}
public void Validate(ServiceEndpoint endpoint)
...{
}
public override Type BehaviorType
...{
get ...{ return typeof(BNServiceOperationBehavior); }
}
protected override object CreateBehavior()
...{
return new BNServiceOperationBehavior();
}
#endregion
}
}
最后将扩展出来的行为加入到ServiceHost上:
[align=left]for (int j = 0; j < _ServiceHost.Description.Endpoints.Count; j++)[/align]
[align=left]{[/align]
[align=left] _ServiceHost.Description.Endpoints[j].Behaviors.Add(new BNServiceOperationBehavior());[/align]
}
首先我们看看MSDN中对WCF的Session的说明:
它们由调用应用程序显式启动和终止。
会话期间传递的消息按照接收消息的顺序进行处理。
会话将一组消息相互关联,从而形成对话。该关联的含义是抽象的。例如,一个基于会话的通道可能会根据共享网络连接来关联消息,而另一个基于会话的通道可能会根据消息正文中的共享标记来关联消息。可以从会话派生的功能取决于关联的性质。
不存在与 WCF 会话相关联的常规数据存储区。
最后一句告诉我们,WCF中的Session是无法像Web应用一样存储附加信息的。
经过研究,我们可以通过扩展MessageHeader实现一个附加的数据存储区在Client端每次请求Service时发送到Server端。具体实现如下(以前述需求为例)。
首先我们定义一个类作为交互过程中的UserIdentity,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
namespace BNCommonPlus.BNCommonModel
...{
[Serializable]
public class UserIdentity
...{
public string UserName = string.Empty;
public string UserRuningSystemID = string.Empty;
public string UserSysGUID = string.Empty;
}
}
这里使用[Serializable]作为序列化方式。在这个类中,UserName对应用户名,UserRuningSystemID对应用户已登陆的系统ID,UserSysGUID对应用户当前的有效登陆。用户登录后Server产生这个类的一个实例回传Client,Client将其保存在缓存里,这里我定义了一个类来作为缓存,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.Data;
using BNCommonPlus.BNCommonModel;
namespace BNIIClientLayerPlus
...{
public class UserPermissionInfo
...{
private UserPermissionInfo()
...{ }
private static UserPermissionInfo _Instance = null;
public static UserPermissionInfo GetInstance()
...{
if (_Instance == null)
...{
_Instance = new UserPermissionInfo();
}
return _Instance;
}
private UserIdentity _userIdentity = null;
public UserIdentity _UserIdentity
...{
get
...{
return _userIdentity;
}
}
public void SetUserIdentity(UserIdentity ui)
...{
_userIdentity = ui;
}
}
}
这是一个单件类,Client正常登陆得到Server端回传的UserIdentity实例后可以通过如下代码将其存入缓存:
UserPermissionInfo.GetInstance().SetUserIdentity(ServerReturnedUserIdentity);
其中ServerReturnedUserIdentity就是Server产生并回传的UserIdentity
下面我们扩展MessageHeader将我们自己定义的UserIdentity加入进去,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceProcess;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
namespace BNCommon.ClientHelper
...{
public class BNClientMessageInspector : IClientMessageInspector
...{
IClientMessageInspector 成员#region IClientMessageInspector 成员
public void AfterReceiveReply(ref Message reply, object correlationState)
...{
}
public object BeforeSendRequest(ref Message request, IClientChannel channel)
...{
MessageHeader mh = MessageHeader.CreateHeader("UserIdentity", "UINS", BNIIClientLayerPlus.UserPermissionInfo.GetInstance()._UserIdentity);
request.Headers.Add(mh);
return null;
}
#endregion
}
}
这个类实现了IClientMessageInspector接口,实现该接口可以在Client每次向Server请求前及请求返回后控制Client的行为对发送和接收的数据进行处理。
现在我们需要实现BehaviorExtensionElement, IEndpointBehavior将刚刚建立的行为加入Client行为集合,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceProcess;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
namespace BNCommon.ClientHelper
...{
public class BNClientEndpointBehavior : BehaviorExtensionElement, IEndpointBehavior
...{
IEndpointBehavior 成员#region IEndpointBehavior 成员
public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
...{}
public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
...{
clientRuntime.MessageInspectors.Add(new BNClientMessageInspector());
}
public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
...{
}
public void Validate(ServiceEndpoint endpoint)
...{
return;
}
#endregion
public override Type BehaviorType
...{
get ...{ return typeof(BNClientEndpointBehavior); }
}
protected override object CreateBehavior()
...{
return new BNClientEndpointBehavior();
}
}
}
现在我们定义了Client在用户正常登陆后,每次请求都会发送缓存里面的UserIdentity实例。然后需要修改Client端的代理类,这里要分两种情况:如果客户端使用的是”SvcUtil.exe”工具生成的代理类,则需要修改对应Client的构造方法,加入如下代码:
Endpoint.Behaviors.Add(new BNCommonPlus.ClientHelper.BNClientEndpointBehavior());
如果使用ChanelFactory则需要在Create对应得ChanelFactory后加入如下代码:
_ChannelFactory.Endpoint.Behaviors.Add(new BNClientEndpointBehavior());
当然也可以通过配置文件实现上述目的。
搞定Client端后,再来看看Server端如何取Client每次发送过来的附加信息.
首先在Server端要实现IDispatchMessageInspector接口,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
using BNCommonPlus.BNCommonModel;
namespace BNCommonPlus.ServiceHelper
...{
public class DispatchMessageInspector : IDispatchMessageInspector
...{
IDispatchMessageInspector 成员#region IDispatchMessageInspector 成员
public object AfterReceiveRequest(ref Message request, IClientChannel channel, InstanceContext instanceContext)
...{
//得到身份信息
UserIdentity ui = request.Headers.GetHeader<UserIdentity>("UserIdentity", "UINS");
……..//这里写验证用户身份,检测权限的代码。
return null;
}
public void BeforeSendReply(ref Message reply, object correlationState)
...{
}
#endregion
}
}
和客户端一样将上面定义的行为加入行为集合,代码如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.ServiceModel;
using System.ServiceModel.Dispatcher;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Configuration;
using BNCommonPlus.BNCommonModel;
namespace BNCommonPlus.ServiceHelper
...{
public class BNServiceOperationBehavior : BehaviorExtensionElement, IEndpointBehavior
...{
IEndpointBehavior 成员#region IEndpointBehavior 成员
public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
...{ }
public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
...{ }
public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
...{
endpointDispatcher.DispatchRuntime.MessageInspectors.Add(new DispatchMessageInspector());
}
public void Validate(ServiceEndpoint endpoint)
...{
}
public override Type BehaviorType
...{
get ...{ return typeof(BNServiceOperationBehavior); }
}
protected override object CreateBehavior()
...{
return new BNServiceOperationBehavior();
}
#endregion
}
}
最后将扩展出来的行为加入到ServiceHost上:
[align=left]for (int j = 0; j < _ServiceHost.Description.Endpoints.Count; j++)[/align]
[align=left]{[/align]
[align=left] _ServiceHost.Description.Endpoints[j].Behaviors.Add(new BNServiceOperationBehavior());[/align]
}
相关文章推荐
- IFC标准是为了满足建筑行业的信息交互与共享而产生的统一数据标准,是建 筑行业事实上的数据交换与共享标准。本文概要介绍了IFC标准的产生及发展 历程,IFC的整体框架结构,简要说明了IFC标准的实现方法和过程,描述了 当前的应用以及我们应该更加积极地利用IFC标准为建筑软件行业服务。
- 如何在 Internet 信息服务中为所有与您的网站交互的用户启用 SSL
- 热插拔时间从内核空间到用户空间的传递过程
- jQuery调用WCF服务传递JSON对象
- android Button用户交互——监听机制调用过程
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- WCF热门问题编程示例(5):WCF服务如何获取客户端在线用户数量?
- 获取共享服务中用户配置的其它额外信息
- Spring Cloud Spring Boot mybatis分布式微服务云架构(二十九)LDAP来管理用户信息(1)
- WCF后续之旅(6): 通过WCF Extension实现Context信息的传递
- V4L2用户空间和kernel层driver的交互过程
- Python通过suds调用WCF服务 传递Python List 到 WCF Service