mbp*_*mbp 10 .net performance wcf
我使用安全模式"TransportWithMessageCredential"将WCF用于soap端点.
WCF客户端/服务器使用SCT(安全上下文令牌)来维护安全连接,并且它在一般情况下按预期工作.
但是,在一段时间不活动后,SCT将过期,下一个方法调用将导致MessageSecurityException:
从另一方收到了无担保或不正确安全的故障.请参阅内部FaultException以获取故障代码和详细信息
内部异常:
邮件无法处理.这很可能是因为" http://tempuri.org/IMyService/MyMethod "操作不正确,或者因为邮件包含无效或过期的安全上下文令牌,或者因为绑定之间存在不匹配.如果服务由于不活动而中止了通道,则安全上下文令牌将无效.要防止服务中止空闲会话,请过早增加服务端点绑定的接收超时.
在后续调用中,当我看到CommunicationState出现故障时,我会更新连接.但在进行方法调用之前,我无法找到预先检查SCT是否已过期的方法.
您可以使用委托来解决您的问题.这将允许安全地调用操作,如果失败,则捕获异常,构造新的服务实例并再次执行操作.
using System;
using System.ServiceModel;
using System.ServiceModel.Security;
public static class Service
{
    private static IService _service;
    public static void Invoke(Action<IService> action)
    {
        try
        {
            action(_service);
        }
        catch (MessageSecurityException)
        {
            if (_service.State != CommunicationState.Faulted)
            {
                throw;
            }
            _service.Abort();
            _service = CreateFreshInstance();
            action(_service);
        }           
    }
}
然后,您可以调用您的帮助程序类Service.Invoke(s => s.Method());来调用IService.Method().
在第一个答案的基础上,我想到了此解决方案,该解决方案通常包装由svcutil.exe创建的自动生成的客户端代理:
public class ProxyWrapper<T> where T : ICommunicationObject
{
    private T _service;
    public ProxyWrapper()
    {
        _service = CreateNewInstance();
    }
    public void Invoke(Action<T> action)
    {
        try
        {
            action(_service);
        }
        catch (MessageSecurityException)
        {
            if (_service.State != CommunicationState.Faulted)
            {
                throw;
            }
            _service.Abort();
            _service = CreateNewInstance();
            action(_service);
        }
    }
    public TResult Invoke<TResult>(Func<T, TResult> func)
    {
        try
        {
            return func(_service);
        }
        catch (MessageSecurityException)
        {
            if (_service.State != CommunicationState.Faulted)
            {
                throw;
            }
            _service.Abort();
            _service = CreateNewInstance();
            return func(_service);
        }
    }
    private T CreateNewInstance()
    {
        Type type = typeof(T);
        return (T)type.GetConstructor(Type.EmptyTypes).Invoke(null);
    }
}
要使用此功能,您需要做的是:
ProxyWrapper<ServiceClient> client = new ProxyWrapper<ServiceClient>();
client.Invoke(s => s.SomeAction());
int val = client.Invoke<int>(s => s.ReturnsAnInteger());
注意:由于我仅使用默认的构造函数作为客户端代理,因此就可以了。
看一下这篇文章,您可以下载一个代理包装器,该包装器在会话过期时重试。
http://www.dasblonde.net/2008/04/24/MyProxyWrapperAndTheEVILSUOFile.aspx
| 归档时间: | 
 | 
| 查看次数: | 6773 次 | 
| 最近记录: |