在我的 WCF 项目中,我需要在响应中使用自定义标头,因此我实现了 IDispatchMessageInspector。老实说,一切都很好,但我对一件小事感到不安。
问题是,即使我只是将 .svc 作为页面打开,或者将服务加载到 WCF 测试客户端中,BeforeSendReply 和 AfterReceiveRequest 都会触发。
那么,第一个问题:这种行为正常吗?是否有一些方法可以以声明方式处理(可能是一些 web.config 技巧)?
目前我使用下一个代码:
public void BeforeSendReply(ref Message reply, object correlationState)
{
if (reply.Properties.Any(x => x.Key == "httpResponse"))
return;
MessageHeader header = MessageHeader.CreateHeader("Success", "NS", !reply.IsFault);
reply.Headers.Add(header);
}
所以现在我使用它来处理所有不是服务调用的调用:
if (reply.Properties.Any(x => x.Key == "httpResponse"))
return;
但我很确定还有其他更好的方法来处理这个问题。所以我的主要问题是:请建议我一种更好的方法来处理所描述的情况。
提前致谢!
更新 1
我的 system.serviceModel 部分
<system.serviceModel>
<services>
<service behaviorConfiguration="someBehavior" name="serviceName">
<endpoint address="" binding="basicHttpBinding" contract="my contract" />
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior name="someBehavior">
<serviceMetadata httpGetEnabled="true" httpGetUrl=""/>
<serviceDebug includeExceptionDetailInFaults="false"/>
<exceptionInspector/>
</behavior>
</serviceBehaviors>
</behaviors>
<extensions>
<behaviorExtensions>
<add name="exceptionInspector" type="class which implements BehaviorExtensionElement" />
</behaviorExtensions>
</extensions>
</system.serviceModel>
更新 2(接受的解决方案)
我花了一些时间调查问题的根源,最后我找到了可以接受的解决方案。
所以我发现了什么:
首先Message
是一个抽象类。所以BeforeSendReply
每次都会收到不同类型的具体消息。
最常用的是:
1) System.ServiceModel.Description.ServiceMetadataExtension.HttpGetImpl.MetadataOnHelpPageMessage
- 表示客户端将 svc 作为页面打开。结果 = 众所周知的 html 格式页面,其中包含有关 svc 服务的常见信息。对于这种类型reply.Version.Envelope
是EnvelopeVersion.None
.
2) 获取元数据请求。这是一个有点棘手的部分,它取决于我们是否使用 MEX。System.ServiceModel.Dispatcher.OperationFormatter.OperationFormatterMessage
因此,如果我们使用 MEX,那么请求将执行到 .svc/mex 端点,其消息类型将reply.Version.Envelope
等于.svc/mex EnvelopeVersion.Soap12
。
如果我们不使用 MEX,那么客户端会执行很少的请求来获取 wsdl 数据。消息类型将为XMLSchemaMessage
.
3) 执行web方法请求。这仅对我类型的请求有用。它是System.ServiceModel.Dispatcher.OperationFormatter.OperationFormatterMessage
并且reply.Version.Envelope
等于EnvelopeVersion.Soap11
。
我使用的是 basicHttpBinding,所以 SOAP 版本是 1.1。所以我的最终代码应该只检查回复是否有 SOAP 信封并检查它的版本。如果信封存在并且版本为 1.1,那么我们可以非常确定我们有 Web 方法调用,并且可能会添加自定义标头:
public void BeforeSendReply(ref Message reply, object correlationState)
{
if(reply.Version.Envelope == EnvelopeVersion.Soap11)
{
MessageHeader header = MessageHeader.CreateHeader("Success", "NS", !reply.IsFault);
reply.Headers.Add(header);
}
}