大型 WCF Web 服务请求失败(400) HTTP 错误请求

我遇到了这个显然很常见的问题,但一直无法解决。

如果我使用数组参数中相对较少的条目调用 WCF Web 服务(我已经测试了50个) ,那么一切都很好。

然而,如果我调用带有500个条目的 Web 服务,就会得到 Bad Request 错误。

有趣的是,我在服务器上运行了 金线鲨,发现请求甚至没有到达服务器——400错误是在客户端生成的。

唯一的例外是:

系统。服务模式。ProtocolException: 远程服务器返回一个意外响应: (400)错误请求。—— > 制度。网。WebException: 远程服务器返回一个错误: (400) Bad Request。

我的客户端配置文件的 system.serviceModel部分是:

 <system.serviceModel>
<bindings>
<wsHttpBinding>
<binding name="WSHttpBinding_IMyService" closeTimeout="00:01:00"
openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
bypassProxyOnLocal="false" transactionFlow="false" hostNameComparisonMode="StrongWildcard"
maxBufferPoolSize="524288" maxReceivedMessageSize="2147483647"
messageEncoding="Text" textEncoding="utf-8" useDefaultWebProxy="true"
allowCookies="false">
<readerQuotas maxDepth="32" maxStringContentLength="8192" maxArrayLength="2147483647"
maxBytesPerRead="4096" maxNameTableCharCount="16384" />
<reliableSession ordered="true" inactivityTimeout="00:10:00"
enabled="false" />
<security mode="None">
<transport clientCredentialType="Windows" proxyCredentialType="None"
realm="" />
<message clientCredentialType="Windows" negotiateServiceCredential="true"
establishSecurityContext="true" />
</security>
</binding>
</wsHttpBinding>
</bindings>
<client>
<endpoint address="http://serviceserver/MyService.svc"
binding="wsHttpBinding" bindingConfiguration="WSHttpBinding_IMyService"
contract="SmsSendingService.IMyService" name="WSHttpBinding_IMyService" />
</client>
</system.serviceModel>

在服务器端,web.config 文件包含以下 system.serviceModel部分:

<system.serviceModel>
<services>
<service name="MyService.MyService" behaviorConfiguration="MyService.MyServiceBehaviour" >
<endpoint address="" binding="wsHttpBinding" bindingConfiguration="MyService.MyServiceBinding" contract="MyService.IMyService">
</endpoint>
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange"/>
</service>
</services>
<bindings>
<wsHttpBinding>
<binding name="MyService.MyServiceBinding">
<security mode="None"></security>
</binding>
</wsHttpBinding>
</bindings>
<behaviors>
<serviceBehaviors>
<behavior name="MyService.MyServiceBehaviour">
<!-- To avoid disclosing metadata information, set the value below to false and remove the metadata endpoint above before deployment -->
<serviceMetadata httpGetEnabled="true"/>
<!-- To receive exception details in faults for debugging purposes, set the value below to true.  Set to false before deployment to avoid disclosing exception information -->
<serviceDebug includeExceptionDetailInFaults="true"/>
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>

看着 相当大 号码 答案这个问题没有成功

有人能帮我吗?

103637 次浏览

它可能是有用的调试客户端,关闭工具选项调试常规“启用只是我的代码”,单击调试异常“捕捉所有第一次机会异常”为托管 CLR 异常,并查看是否有一个异常在客户端协议异常之前,消息触发线。(我猜可能是某种序列化失败。)

尝试在服务器上设置 maxReceivedMessageSize,例如设置为4MB:

<binding name="MyService.MyServiceBinding" maxReceivedMessageSize="4194304">

默认值(我认为是65535)如此之低的主要原因是为了降低分布式拒绝服务攻击攻击的风险。您需要将其设置为大于服务器上的最大请求大小和客户端上的最大响应大小。如果您处于 Intranet 环境中,DoS 攻击的风险可能很低,所以使用比您预期需要的高得多的值可能是安全的。

顺便提一下关于连接到 WCF 服务的故障排除的一些技巧:

然而,我也遇到了这个问题,但是上面的方法对我来说都不管用,因为我使用了一个自定义绑定(用于 BinaryXML) ,经过长时间的研究,我在这里找到了答案: 将大型 XML 从 Silverlight 发送到 WCF

与使用 customBinding 一样,maxReceivedMessageSize 必须在 web.config 的 bind 元素下的 httpTransport 元素上设置:

<httpsTransport maxReceivedMessageSize="4194304" />

无论如何,在使用。NET 4.0是,如果在配置中没有找到有效的端点,将自动创建和使用默认端点。

默认端点将使用所有默认值,因此如果您认为您有一个有效的服务配置,其中 maxReceivedMessageSize 等值较大,但是配置有问题,您仍然会得到400 Bad Request,因为将创建并使用一个默认端点。

这是无声的,所以很难检测到。如果您打开服务器上的跟踪,但是没有其他指示(据我所知) ,您将看到这种效果的消息(例如“没有为服务找到端点,创建默认端点”或类似的消息)。

我只是想指出

除了 MaxRecivedMessageSize 之外,ReaderQuotas 下还有一些属性,您可能会达到项目数量限制,而不是大小限制。MSDN 链路是 给你

在服务器中。NET 4.0在 web.config 中,您还需要在默认绑定中进行更改。设置以下3个帕尔玛:

<basicHttpBinding>
<!-- http://www.intertech.com/Blog/post/NET-40-WCF-Default-Bindings.aspx
Enable transfer of large strings with maxBufferSize, maxReceivedMessageSize and maxStringContentLength
-->
<binding **maxBufferSize="2147483647" maxReceivedMessageSize="2147483647"**>
<readerQuotas **maxStringContentLength="2147483647"**/>
</binding>
</basicHttpBinding>

我找到了“坏请求400”问题的答案。

这是默认的服务器绑定设置。您需要添加到服务器和客户端的默认设置。

<binding name="" openTimeout="00:10:00" closeTimeout="00:10:00" receiveTimeout="00:10:00" sendTimeout="00:10:00" maxReceivedMessageSize="2147483647" maxBufferPoolSize="2147483647" maxBufferSize="2147483647">
<security mode="None"/>
<readerQuotas maxStringContentLength="2147483647"/>
</binding>

您还可以打开 WCF 日志以获得关于原始错误的更多信息。这帮助我解决了这个问题。

将以下内容添加到 web.config 中,它会将日志保存到 C: log Traces.svclog

<system.diagnostics>
<sources>
<source name="System.ServiceModel"
switchValue="Information, ActivityTracing"
propagateActivity="true">
<listeners>
<add name="traceListener"
type="System.Diagnostics.XmlWriterTraceListener"
initializeData= "c:\log\Traces.svclog" />
</listeners>
</source>
</sources>
</system.diagnostics>

在我的情况下,即使在尝试了所有的解决方案并将所有的限制设置为最大值之后,它仍然不起作用。最后我发现在 IIS/网站上安装了一个微软 IIS 过滤模块 Url Scan 3.1,它有自己的限制,可以根据内容大小拒绝传入的请求并返回“404未找到页面”。

通过将 MaxAllowedContentLength设置为所需的值,可以在 %windir%\System32\inetsrv\urlscan\UrlScan.ini文件中更新它的限制。

例如,以下将允许高达300mb 的请求

MaxAllowedContentLlength = 314572800

希望它能帮到别人!