一个 WCF 服务——两个客户端;一个客户端不工作

Posted

技术标签:

【中文标题】一个 WCF 服务——两个客户端;一个客户端不工作【英文标题】:One WCF service – two clients; One client does not work 【发布时间】:2012-09-07 08:53:29 【问题描述】:

我有一个 WCF 服务和两个控制台应用程序客户端。

服务:服务代码是使用 WCSF Blue 工具从 wsdl 联系人创建的。

客户端 1:此客户端正在使用通过浏览 svc 文件获得的 wsdl。这个浏览的 wsdl 文件与合约 wsdl 文件略有不同。

客户端 2:此客户端是使用原始 wsdl 合约创建的。

Cleint1 工作正常。客户端 2 不工作。有哪些潜在问题?

两个客户端的 App.Config 文件看起来很相似——只是名称发生了变化。我认为,问题出在客户端生成的 C# 代码中——很可能在 Action 中——ReplyAction。这里需要纠正什么?

一个明显的区别在于 Action 和 ReplyAction

客户端 1:

Action="urn:lijo:demos:multiplyservice:calculation:v1/ICalculationService/GetMultiplied", ReplyAction="urn:lijo:demos:multiplyservice:calculation:v1/ICalculationService/GetMultipliedRe" + “回应”

客户端 2:

Action="urn:lijo:demos:multiplyservice:calculation:v1:getMultipliedIn", ReplyAction="*"

跟踪消息

由于 EndpointDispatcher 的 ContractFilter 不匹配,接收方无法处理带有 Action 'urn:lijo:demos:multiplyservice:calculation:v1:getMultipliedIn' 的消息。这可能是因为合约不匹配(发送方和接收方之间的操作不匹配)或发送方和接收方之间的绑定/安全不匹配。检查发送方和接收方是否具有相同的合同和相同的绑定(包括安全要求,例如消息、传输、无)。

编辑

这可以通过如下更改 Action 和 ReplyAction 来纠正(从 Service 复制)。

  [System.ServiceModel.OperationContractAttribute(Action = "urn:lijo:demos:multiplyservice:calculation:v1/ICalculationService/getMultiplied", ReplyAction = "urn:lijo:demos:multiplyservice:calculation:v1/ICalculationService/getMultipliedRe" +
        "sponse")]

注意:确保服务中的大小写正确(即 getMultiplied 不是 GetMultiplied)很重要

从服务复制不是一个好的选择,尽管它有效。正确的 Action 和 ReplyAction 是什么?

另外,您能否指出如何在生成的客户端代理中修改 wsdl 以使 ReplyAction 正确?这是将其标记为已回答的重要部分。

WCF: Actions, Asterisk and Metadata

用于元数据发布的 WsdlExporter 忽略带有星号操作的操作(包括 Action 和 ReplyAction)。

来自MSDN -ReplyAction Property

在服务中指定星号指示 WCF 不要向消息添加回复操作,如果您直接针对消息进行编程,这很有用。

参考文献

    WCF metadata missing operations

RestaurantData.xsd

 <?xml version="1.0" encoding="utf-8" ?>
 <xs:schema id="RestaurantData" targetNamespace="urn:lijo:demos:multiplyservice:data:v1"
    elementFormDefault="qualified" xmlns="urn:lijo:demos:multiplyservice:data:v1"
    xmlns:mstns="urn:lijo:demos:multiplyservice:data:v1" xmlns:xs="http://www.w3.org/2001/XMLSchema">

  <xs:complexType name="multipliedResult">
  <xs:sequence>
  <xs:element name="resultNumber" type="xs:int" />
   </xs:sequence>
   </xs:complexType>

  </xs:schema>

原始合同 wsdl

 <definitions xmlns:import0="urn:lijo:demos:multiplyservice:messages:v1" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:import1="urn:lijo:demos:multiplyservice:data:v1" xmlns:tns="urn:lijo:demos:multiplyservice:calculation:v1" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/" name="CalculationService" targetNamespace="urn:lijo:demos:multiplyservice:calculation:v1" xmlns="http://schemas.xmlsoap.org/wsdl/">

 <wsdl:documentation xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" />
 <types>
 <xsd:schema>
  <xsd:import schemaLocation="C:\toolbox\LijosServiceApp\NewService\RestaurantMessages.xsd" namespace="urn:lijo:demos:multiplyservice:messages:v1" />
  <xsd:import schemaLocation="C:\toolbox\LijosServiceApp\NewService\RestaurantData.xsd" namespace="urn:lijo:demos:multiplyservice:data:v1" />
</xsd:schema>
</types>
<message name="getMultipliedIn">
<wsdl:documentation xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" />
<part name="parameters" element="import0:getMultiplied" />
</message>
<message name="getMultipliedOut">
<wsdl:documentation xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" />
<part name="parameters" element="import0:getMultipliedResponse" />
</message>
<portType name="CalculationServiceInterface">
<wsdl:documentation xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" />
<operation name="getMultiplied">
  <wsdl:documentation xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" />
  <input message="tns:getMultipliedIn" />
  <output message="tns:getMultipliedOut" />
</operation>
</portType>
<binding name="BasicHttpBinding_CalculationServiceInterface" type="tns:CalculationServiceInterface">
<soap:binding transport="http://schemas.xmlsoap.org/soap/http" />
<operation name="getMultiplied">
  <soap:operation soapAction="urn:lijo:demos:multiplyservice:calculation:v1:getMultipliedIn" style="document" />
  <input>
    <soap:body use="literal" />
  </input>
  <output>
    <soap:body use="literal" />
  </output>
 </operation>
 </binding>
 <service name="CalculationServicePort">
 <port name="CalculationServicePort" binding="tns:BasicHttpBinding_CalculationServiceInterface">
  <soap:address location="http://localhost/CalculationService" />
 </port>
 </service>
 </definitions>

【问题讨论】:

参考:***.com/questions/5487791/… 和 ***.com/questions/1264431/… 当您说第二个客户端不工作时,您遇到了什么错误? @MilanRaval 异常与问题中发布的跟踪消息相同 【参考方案1】:

我想通了。为了其他人的利益,我将在这里解释。

在此之前,请参考400 Bad Request Exception: Simple SOAP WCF service with small data 的答案以获得一些调试想法。

这是由于WCSF Blue 工具中的Format SOAP Action 选项。

我在使用 WCSF Blue 生成代码时使用了“Format Soap Actions”。但是当客户时,我没有使用该工具。这种不匹配是关键问题。

Format Soap Actions 强制应用于每个操作合同的 SOAP 操作(ActionReplyAction)遵循标准WCF 格式

  <namespace>/<service>/<operation>[Response] 

如果我无法控制客户端,我不应该使用 WCSF Blue Tool 中的 Format SOAP Action 选项。

请参考Service works from wcfTestClient but fails in Console Application 获取工作示例。

[我还有一个问题 - 如果我无法控制客户端仍然需要使用 ReplyAction 怎么办?在这种情况下,要在客户端和服务中使用的 xml 格式的 URI 是什么? ]

一般调试思路:

    使用 wcfTestClient 确保服务良好(在 VS 命令提示符下输入 wcfTestClient 启动)

    使用How to turn on WCF tracing?中提到的跟踪

    验证配置值在 web.config/app.config 中,而不是在 output.config 中(在使用工具自动生成的情况下)

    验证您是否引用了正确的 wsdl(是本地文件还是来自正在运行的服务的 url?)

    通过浏览 svc 文件验证是否可以查看 wsdl。元数据已启用

    检查服务中的“地址”是相对路径还是绝对路径

【讨论】:

【参考方案2】:

您说得对,ReplyAction 存在问题。当 ReplyAction 设置为“*”时,WCF 会忽略该操作。将ReplyAction 更正为您的操作合同将起作用。

http://social.msdn.microsoft.com/Forums/en-US/wcf/thread/41f5fe72-3ab3-4741-867e-a93119fe62aa

【讨论】:

在 getMultipliedOut 的位置使用 getMultipliedResponse,因为 ReplyAction 应该是 ("OperationName + "Response") 并且在您的情况下操作名称是 GetMultiplied Action="urn:lijo:demos:multiplyservice:calculation:v1:getMultipliedIn",ReplyAction="urn:lijo:demos:multiplyservice:calculation:v1:getMultipliedInResponse"。还有一篇好博客:shishkin.wordpress.com/2007/03/21/… 其实这行不通。当我从服务中复制确切的操作和回复操作时,它可以工作;但这是不可接受的。有没有办法强制工具生成所需的回复操作?

以上是关于一个 WCF 服务——两个客户端;一个客户端不工作的主要内容,如果未能解决你的问题,请参考以下文章

WCF 客户端证书停止工作

WCF - 无法发送数据

从两个单独的客户端调用服务时 WCF 死锁

“参数不正确。” RSACryptoServiceProvider WCF 客户端/服务器 C#

如何在 jQuery 中使用 WCF 方法

异步 WCF 服务超时