当操作返回元页面 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 服务?的主要内容,如果未能解决你的问题,请参考以下文章

jquery ajax - 更好地返回 json 或纯 html

元宇宙医疗或将改变医疗格局

使用 lxml 有效地解析元标记?

从 Pepper QiSDK 获取诊断

使用mvc返回给浏览器html元素时需要注意的一个问题

仅当用户单击 [重复] 时如何执行 JS 文件