17

我有一个 wcf 客户端。根据要求,我需要记录请求中的一些元数据(以及请求中未包含的用户数据)。然后,如果请求成功,我可能必须记录响应元数据,并根据标志,完整的肥皂请求。

我正在尝试以正确的方式执行此操作(使用 IParameterInspector 检查元数据并使用 IClientMessageInspector 获取 Soap),但我无法关联两个接口请求。我不确定这里的线程安全。这是我所在位置的精简版...

     public class SoapRequestInfo
{
    public string UserId { get; set; }
    public Guid Key { get; set; }
    //would contain a lot more info
}

public class OperationProfilerParameterInspector : IParameterInspector, IClientMessageInspector
{
    //before serialization
    public object BeforeCall(string operationName, object[] inputs) //IParameterInspector
    {
        //Add the operation, record some specific inputs to db
        return new SoapRequestInfo
                            {
                                UserId = "1234",
                                Key = new Guid()
                            };
    }

    public void AfterCall(string operationName, object[] outputs, object returnValue, object correlationState) //IParameterInspector
    {
       var info = correlationState as SoapRequestInfo;
        //Do some additional logging - easy enough
    }

    public object BeforeSendRequest(ref Message request, IClientChannel channel) //IClientMessageInspector
    {
        //want to correlate this with IParameterInspector
        return null;
    }


    public void AfterReceiveReply(ref Message reply, object correlationState) //IClientMessageInspector
    {
        //May want to log full soap message depending on after call criteria
    }
}      

我知道我不能使用私有变量来保存 Guid。我不能使用会话,因为可能有多个请求连续出现并且不能保证响应是正确的。那么如何才能唯一标识两个接口之间的correlationState呢?

4

2 回答 2

1

如果您的服务在 ASPNET 兼容模式下运行,您可能可以使用 HttpContext.Items 来保留您的对象,否则您可以使用 TLS(线程本地存储),将数据放入插槽并稍后获取/清除。

于 2015-08-30T10:17:18.503 回答
-1

愿eb你可以做这点不同的事情:

从这篇文章将相关令牌传递给 WCF 服务?

如果您使用带有 WS-Addressing 的消息版本,您应该自动拥有它,因为每条消息都将包含其自动生成的 ID (guid),并且每个响应也将包含请求的 ID。您可以通过 OperationContext 访问这些标头

如果使用消息不符合您的要求,可能您可以尝试在发送请求时将自己的 id 放在标头中,并可能使用相同的 id 更新响应。

于 2013-11-21T08:28:52.793 回答