一个 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 操作(Action 和 ReplyAction)遵循标准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 服务——两个客户端;一个客户端不工作的主要内容,如果未能解决你的问题,请参考以下文章