当操作返回元页面 html 时,如何更好地诊断 WCF 服务?
Posted
技术标签:
【中文标题】当操作返回元页面 html 时,如何更好地诊断 WCF 服务?【英文标题】:How can I better diagnose a WCF service when an operation returns the meta page html? 【发布时间】:2011-11-04 17:11:42 【问题描述】:我有一个托管在 IIS7.5 (.NET 4) 中的 WCF 服务应用程序 (wsHttpBinding),我可以在浏览器中导航 .svc 文件,元数据页面成功显示,并且可以从 wsdl 生成客户端。
我添加了我自己的自定义服务行为来记录未处理的异常(下面的“ErrorHandling”),并且我正在使用 MembershipProvider 作为服务凭据。这是我的完整 system.serviceModel 配置:
<system.serviceModel>
<diagnostics wmiProviderEnabled="true">
<messageLogging
logEntireMessage="true"
logMalformedMessages="true"
logMessagesAtServiceLevel="true"
logMessagesAtTransportLevel="true"
maxMessagesToLog="3000"
maxSizeOfMessageToLog="-1" />
</diagnostics>
<extensions>
<behaviorExtensions>
<add name="ErrorHandling" type="MyCompany.MyProduct.Services.ErrorHandlerBehavior, MyCompany.MyProduct.Services" />
</behaviorExtensions>
</extensions>
<services>
<service name="MyCompany.MyProduct.Services.ApplicationService" behaviorConfiguration="MyProductServiceBehavior">
<endpoint bindingConfiguration="MyProductWsHttpBinding" binding="wsHttpBinding" contract="MyCompany.MyProduct.Services.IApplicationService" />
</service>
<service name="MyCompany.MyProduct.Services.AccountService" behaviorConfiguration="MyProductServiceBehavior">
<endpoint bindingConfiguration="MyProductWsHttpBinding" binding="wsHttpBinding" contract="MyCompany.MyProduct.Services.IAccountService" />
</service>
<service name="MyCompany.MyProduct.Services.InvoiceService" behaviorConfiguration="MyProductServiceBehavior">
<endpoint bindingConfiguration="MyProductWsHttpBinding" binding="wsHttpBinding" contract="MyCompany.MyProduct.Services.IInvoiceService" />
</service>
<service name="MyCompany.MyProduct.Services.ReportService" behaviorConfiguration="MyProductServiceBehavior">
<endpoint bindingConfiguration="MyProductWsHttpBinding" binding="wsHttpBinding" contract="MyCompany.MyProduct.Services.IReportService" />
</service>
<service name="MyCompany.MyProduct.Services.PaymentService" behaviorConfiguration="MyProductServiceBehavior">
<endpoint bindingConfiguration="MyProductWsHttpBinding" binding="wsHttpBinding" contract="MyCompany.MyProduct.Services.IPaymentService" />
</service>
</services>
<bindings>
<wsHttpBinding>
<binding name="MyProductWsHttpBinding">
<security mode="Message">
<message clientCredentialType="UserName" establishSecurityContext="true" negotiateServiceCredential="true" />
</security>
</binding>
</wsHttpBinding>
</bindings>
<behaviors>
<serviceBehaviors>
<behavior name="MyProductServiceBehavior">
<serviceMetadata httpGetEnabled="true" />
<serviceDebug includeExceptionDetailInFaults="true" />
<serviceCredentials>
<serviceCertificate findValue="MyProductServices" storeLocation="CurrentUser" storeName="My" x509FindType="FindBySubjectName" />
<userNameAuthentication userNamePasswordValidationMode="MembershipProvider" membershipProviderName="AspNetSqlMembershipProvider" />
</serviceCredentials>
<ErrorHandling />
<serviceSecurityAudit auditLogLocation="Application" suppressAuditFailure="false" serviceAuthorizationAuditLevel="SuccessOrFailure" messageAuthenticationAuditLevel="SuccessOrFailure" />
</behavior>
</serviceBehaviors>
</behaviors>
<serviceHostingEnvironment multipleSiteBindingsEnabled="true" aspNetCompatibilityEnabled="true" />
</system.serviceModel>
您可以看到我还启用了安全审核,但事件日志中除了显示消息日志记录已打开的消息外,什么都没有显示。因此,当客户端调用操作时,它似乎永远不会进入身份验证阶段。
我还启用了跟踪功能,如下所示:
<system.diagnostics>
<sources>
<source name="System.ServiceModel"
switchValue="All"
propagateActivity="true" >
<listeners>
<add name="xml"/>
</listeners>
</source>
<source name="System.ServiceModel.MessageLogging" switchValue="All">
<listeners>
<add name="xml"/>
</listeners>
</source>
<source name="myUserTraceSource"
switchValue="Information, ActivityTracing">
<listeners>
<add name="xml"/>
</listeners>
</source>
</sources>
<sharedListeners>
<add name="xml"
type="System.Diagnostics.XmlWriterTraceListener"
initializeData="C:\Users\Public\Documents\services.svclog" />
</sharedListeners>
</system.diagnostics>
当客户端调用主服务中的主操作时,它会收到带有以下消息的协议异常:
内容类型text/html;响应消息的 charset=UTF-8 与绑定的内容类型不匹配 (application/soap+xml; charset=utf-8)。如果使用自定义编码器,请确保正确实现 IsContentTypeSupported 方法。响应的前 1024 个字节是:'#content FONT-SIZE: 0.7em;垫底:2em; MARGIN-LEFT: 30pxBODYMARGIN-TOP: 0px;左边距:0px;颜色:#000000;字体家族:Verdana;背景颜色:白色P边距顶部:0px;边距底部:12px;颜色:#000000; FONT-FAMILY: VerdanaPREBORDER-RIGHT: #f0f0e0 1px solid;填充右:5px;边框顶部:#f0f0e0 1px 实心;边距顶部:-5px;填充左:5px;字体大小:1.2em;填充底部:5px;左边界:#f0f0e0 1px 实心;填充顶部:5px;边框底部:#f0f0e0 1px 实心; FONT-FAMILY:Courier New;背景颜色:#e5e5cc.heading1MARGIN-TOP:0px;填充左:15px;字重:正常;字体大小:26px;边距底部:0px;填充底部:3px;左边距:-30px;宽度:100%;颜色:#ffffff;填充顶部:10px;字体家族:Tahoma;背景颜色:#003366.introMARGIN-LEFT:-15pxApplicationService 服务。
当您浏览到 .svc 文件时,您看到的元数据页面的 html 是什么。所以看起来好像服务正在返回该页面作为对操作的响应。
查看 services.svclog 也是如此,实际上该日志是格式错误的 xml,因为该响应包含在其中而没有转义标签。跟踪中没有任何有用的信息。
服务器上没有抛出异常,日志什么也没有。这一切都可以在我设置的另一台服务器上的 IIS6 上进行测试,并且错误日志也可以在那里工作。
哪些工具和技术可以帮助我找出根本问题是什么?
更新:以下是来自服务器端跟踪的 svclog,它针对来自新生成的控制台应用程序客户端的单个请求(经过清理以删除个人信息),注意它在最后是如何被截断的: http://pastebin.com/mksL0FFY
【问题讨论】:
尝试查看服务器上的事件日志。它应该在应用程序文件夹中有失败的消息。 事件日志中什么都没有,唯一的条目是消息日志已打开:消息日志已打开。敏感信息可能会以明文形式记录,即使它是在网络上加密的:例如,消息体。进程名称:w3wp 进程ID:6604 您是否已将 svclog 加载到 ServiceTraceViewer 工具中?通过它阅读它比尝试阅读原始 XML 更容易。 是的,我已经在跟踪查看器工具中加载并查看了它。它最初抱怨一些标签没有关闭,但它确实显示了它可以显示的内容。 【参考方案1】:由于事件日志为空,您应该检查 IIS 日志以确保调用成功。我遇到过类似的错误消息,问题通常与权限有关。检查用于应用程序池和网站的帐户是否有足够的权限。我认为如果与会员提供商进行身份验证时出现问题,您会看到一条事件日志消息。 此外,请尝试使用 WcfTestClient 应用程序调用该方法,以便您可以看到每种方式传输的 XML。
【讨论】:
不幸的是,WcfTestClient 不允许您指定凭据,这对我的场景毫无用处。但是,我确实找到了一个名为 soapUI 的应用程序,它允许您在请求中指定用户名和密码。【参考方案2】:检查站点的 IIS 中的 URL 重写规则。如果其中之一正在重定向,则可能导致客户端接收元数据页面作为对操作的响应。我以前见过这种情况,这可能会产生误导。
【讨论】:
以上是关于当操作返回元页面 html 时,如何更好地诊断 WCF 服务?的主要内容,如果未能解决你的问题,请参考以下文章