Arj*_*Arj 5 c# wcf soap wsdl svcutil.exe
我正在尝试创建一个WCF应用程序,它将侦听来自供应商系统的请求.供应商为我提供了WSDL,因此我需要创建一个服务并将其"端点"暴露给他们.
我已经使用SvcUtil.exe生成C#类,但它输出了相当奇怪的类型.
这是给我的WSDL的片段:
<?xml version="1.0" encoding="utf-8"?>
<wsdl:definitions xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:tm="http://microsoft.com/wsdl/mime/textMatching/" xmlns:soapenc="http://schemas.xmlsoap.org/soap/encoding/" xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/" xmlns:s="http://www.w3.org/2001/XMLSchema" xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/" xmlns:http="http://schemas.xmlsoap.org/wsdl/http/" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/">
<wsdl:types>
<s:schema elementFormDefault="qualified">
<s:element name="Submit">
<s:complexType>
<s:sequence>
<s:element minOccurs="0" maxOccurs="1" name="Incident">
<s:complexType>
<s:sequence>
<s:element minOccurs="0" maxOccurs="1" form="unqualified" name="TransactionId" type="s:string" />
<s:element minOccurs="0" maxOccurs="1" form="unqualified" name="TransactionType" type="s:string" />
<s:element minOccurs="0" maxOccurs="1" form="unqualified" name="TransactionSubType" type="s:string" />
<s:element minOccurs="0" maxOccurs="unbounded" form="unqualified" name="ConfigurationItem">
<s:complexType>
<s:sequence>
<s:element minOccurs="0" maxOccurs="1" form="unqualified" name="AssetTag" type="s:string" />
<s:element minOccurs="0" maxOccurs="1" form="unqualified" name="Name" type="s:string" />
....
Run Code Online (Sandbox Code Playgroud)
我运行的命令很简单
svcutil.exe file_name.wsdl
Run Code Online (Sandbox Code Playgroud)
我希望这会创建一个这样的结构:
class Submit { ... }
class Incident { ... }
class ConfigurationItem { ... }
Run Code Online (Sandbox Code Playgroud)
因此,当它被序列化时,我得到类似的东西:
<Submit>
<Incident>
<TransactionId>12345</TransactionId>
<TransactionType>12345</TransactionType>
<TransactionSubType>12345</TransactionSubType>
<ConfigurationItem>
<AssetTag>xyz</AssetTag>
<Name>name</Name>
</ConfigurationItem>
</Incident>
</Submit>
Run Code Online (Sandbox Code Playgroud)
但是,SvcUtil.exe的输出提供了以下内容:
class SubmitIncident { ... }
class SubmitIncidentConfigurationItem { ... }
Run Code Online (Sandbox Code Playgroud)
似乎将Submit和Incident对象组合成一个,并且它还在每个嵌套元素上添加了"SubmitIncident".所以当序列化我得到这个:
<SubmitIncident>
<TransactionId>...</TransactionId>
<TransactionType>...</TransactionType>
<TransactionSubType>...</TransactionSubType>
<SubmitIncidentConfigurationItem>
<AssetTag>...</AssetTag>
<Name>...</Name>
</SubmitIncidentConfigurationItem>
</SubmitIncident>
Run Code Online (Sandbox Code Playgroud)
这会导致供应商出现问题(我的服务与他们的WSDL不匹配,所以他们无法与我交谈),以及我在应用程序中进行的后续处理.任何人都可以帮助我理解为什么会发生这种情况,如何阻止它,并让SvcUtil正确输出.
TL;博士;
不要使用 WCF 测试客户端来构建 WCF 应用程序来测试它们。我怎么强调都不为过。它迫使您删除会导致应用程序在正常使用中中断的内容。即使从早期开发阶段就使用 SoapUI 等行业标准应用程序。
事实证明,类定义和定义并不是真正的问题(它确实会在稍后引起另一个问题,但一旦您了解发生了什么,这也很容易解决)。这个问题的根源在于我在一次性开发阶段所做的事情。
我做的第一件事就是使用svcutil.exe供应商提供的 WSDL 生成服务器类。一旦我有足够的时间开始调试/测试,我自然会转向 WCF 测试客户端和 Visual Studio 的内置开发服务器。在使用客户端时,它不断出现各种晦涩的错误,例如不支持https、合同和绑定问题等。其中许多是配置不一致,因此通过一些谷歌搜索修复了,但其中的错误是这个:
The operation is not supported in the WCF Test Client because it uses type SubmitIncident
SubmitIncident当 的任何属性属于类型时object(即使嵌套在其他属性的较低位置),就会发生这种情况。基本上我只是一次处理一个错误,剔除那些不起作用的部分,直到最终它起作用为止。我主要是添加和删除属性和类的属性,例如 DataMember、SoapAction、命名空间等。
最终,我能够使用 WCF 测试客户端成功调用应用程序中的端点。然而,当其他应用程序与其通信时,它按照最初的问题失败了。我为使其在 WCF 测试客户端中工作而所做的更改实际上使该服务对原始 WSDL 架构无效!
有用的链接