是否可以使 WcfTestClient 为自定义传输通道工作?

Posted

技术标签:

【中文标题】是否可以使 WcfTestClient 为自定义传输通道工作?【英文标题】:Is it possible to make the WcfTestClient work for custom transport channels? 【发布时间】:2010-10-03 20:07:45 【问题描述】:

目标

我希望能够在我正在设计的托管框架内通过 WCF 托管和连接到 vanilla sockets 服务器。我希望能够使用 WCF 来编码当今必须由套接字程序员手动管理的传输和协议通信。这将允许我与只公开传统套接字和专有协议的 Linux 服务器守护进程实现最终的互操作性。我现在只对通常使用 WcfTestClient 验证传输通道层感兴趣。我的理解是 WcfTestClient 不支持复杂的服务方法。

有人认为可以让 WcfTestClient 为自定义传输通道工作吗? 能够通用地使用此客户端来测试任意数量的自定义传输通道真是太好了。

概述

我正在努力了解 Windows SDK 中包含的 WCF Udp 示例,该示例通常位于 C:\Program Files\Microsoft SDKs\Windows\v6.1\Samples\WCFSamples\TechnologySamples\Extensibility\Transport\Udp\ CS,假设 WCFSamples.zip 文件是从 Samples 目录中完全解压出来的。

这些是我到目前为止所采取的步骤:

(成功):在 Visual Studio 2008 中成功运行解决方案的服务和客户端。

(成功):使用通常位于 C:\Program Files\Microsoft Visual Studio 9.0\Common7\IDE 的 WcfTestClient 连接到 MEX 端点

(失败):使用 WcfTestClient 尝试从 ICalculatorContract 或 IDatagramContract 服务合同执行方法。

例如,当我执行 Hello() 方法时,我收到以下错误:

友情提示:

调用服务失败。可能的 原因:服务离线或 无法访问;客户端 配置不匹配 代理;现有代理无效。 有关更多信息,请参阅堆栈跟踪 细节。您可以尝试通过以下方式恢复 启动一个新的代理,恢复到 默认配置,或者刷新 服务。

错误详情:

CustomBinding 带有合约的 ServiceEndpoint 'IDatagramContract' 缺少一个 传输绑定元素。每一个 绑定必须至少有一个绑定 派生自的元素 传输绑定元素。在 System.ServiceModel.Channels.Binding.EnsureInvariants(字符串 合同名称)在 System.ServiceModel.Description.ServiceEndpoint.EnsureInvariants() 在 System.ServiceModel.Channels.ServiceChannelFactory.BuildChannelFactory(ServiceEndpoint 服务端点)在 System.ServiceModel.ChannelFactory.CreateFactory() 在 System.ServiceModel.ChannelFactory.OnOpening() 在 System.ServiceModel.Channels.CommunicationObject.Open(时间跨度 超时)在 System.ServiceModel.ChannelFactory.EnsureOpened() 在 System.ServiceModel.ChannelFactory1.CreateChannel(EndpointAddress address, Uri via) at System.ServiceModel.ChannelFactory1.CreateChannel() 在 System.ServiceModel.ClientBase1.CreateChannel() at System.ServiceModel.ClientBase1.CreateChannelInternal() 在 System.ServiceModel.ClientBase`1.get_Channel() 在 DatagramContractClient.Hello()

了解客户端错误

在 UdpTransport 项目中定义的 UdpTransportBindingElement 肯定是从 TransportBindingElement 派生的,如下所示,所以我认为 WcfTestClient 配置文件中一定缺少某些东西和/或可能我必须以某种方式提供测试客户端更多信息。我基本上尝试将 Udp 解决方案的客户端项目的 System.ServiceModel 部分复制到 WcfTestClient 的配置文件中,并将传输程序集 dll 复制到与测试客户端相同的文件夹中,但我收到了相同的错误。

我的理解是,MEX 端点应该足以获取在服务上调用简单方法所需的信息。当然,我知道这个故事可能还有更多内容,考虑到我正在尝试让一个旨在测试开箱即用传输通道的客户端与自定义传输通道一起工作。

/// <summary>
/// Udp Binding Element.  
/// Used to configure and construct Udp ChannelFactories and ChannelListeners.
/// </summary>
public class UdpTransportBindingElement 
    : TransportBindingElement // to signal that we're a transport
    , IPolicyExportExtension // for policy export
    , IWsdlExportExtension

ITransportPolicyImport 对自定义传输通道重要吗?

我询问 ITransportPolicyImport 是因为它是由 TransportBindingElement 的标准 WCF 派生实现的接口,但 Udp 示例没有实现此接口,而且我在 Web 搜索或 Safari 上找不到任何有用的东西。它甚至可能不相关。

例如...

public class HttpTransportBindingElement :
    : TransportBindingElement
    , IWsdlExportExtension
    , IPolicyExportExtension
    , ITransportPolicyImport

示例配置...

服务配置如下所示:

  <system.serviceModel>

    <!-- 
         add our udpTransport handler for use by binding declarations 
     -->
    <!-- 
         add our standard binding handler for use by binding declarations 
     -->
    <extensions>
      <bindingElementExtensions>
        <add name="udpTransport" type="Microsoft.ServiceModel.Samples.UdpTransportElement, UdpTransport" />
      </bindingElementExtensions>
      <bindingExtensions>
        <add name="sampleProfileUdpBinding" type="Microsoft.ServiceModel.Samples.SampleProfileUdpBindingCollectionElement, UdpTransport" />
      </bindingExtensions>
    </extensions>

    <services>
      <service name="Microsoft.ServiceModel.Samples.ConfigurableCalculatorService" behaviorConfiguration="udpServiceBehavior">
        <host>
          <baseAddresses>
            <add baseAddress="http://localhost:8000/udpsample"/>
          </baseAddresses>
        </host>
        <endpoint address="soap.udp://localhost:8001/" 
            binding="sampleProfileUdpBinding" 
            bindingConfiguration="CalculatorServer"
            contract="Microsoft.ServiceModel.Samples.ICalculatorContract" />
        <endpoint address="soap.udp://localhost:8002/datagram" 
            binding="customBinding" 
            bindingConfiguration="DatagramServer"
            contract="Microsoft.ServiceModel.Samples.IDatagramContract" />
        <endpoint address="mex"
            binding="mexHttpBinding"
            contract="IMetadataExchange" />
      </service>
    </services>

    <bindings>
        <!-- 
         server bindings 
          -->
        <sampleProfileUdpBinding>
            <binding name="CalculatorServer" clientBaseAddress="soap.udp://localhost:8003/" />
            <binding name="DatagramServer" reliableSessionEnabled="false" />
        </sampleProfileUdpBinding>

        <customBinding>
        <binding name="DatagramServer">
            <binaryMessageEncoding />
            <udpTransport/>
        </binding>
      </customBinding>
    </bindings>

    <behaviors>
      <serviceBehaviors>
        <behavior name="udpServiceBehavior">
          <serviceMetadata httpGetEnabled="True"/>
        </behavior>
      </serviceBehaviors>
    </behaviors>
    <client>
      <endpoint address="" binding="sampleProfileUdpBinding" bindingConfiguration="CalculatorServer"
        contract="Microsoft.ServiceModel.Samples.ICalculatorContract"
        name="CalculatorClient" />
    </client>    
  </system.serviceModel>

解决方案中的客户端配置如下所示:

  <system.serviceModel>
    <bindings>
      <customBinding>
        <binding name="CustomBinding_IDatagramContract">
          <binaryMessageEncoding maxReadPoolSize="64" maxWritePoolSize="16"
              maxSessionSize="2048">
            <readerQuotas maxDepth="32" maxStringContentLength="8192" maxArrayLength="16384"
                maxBytesPerRead="4096" maxNameTableCharCount="16384" />
          </binaryMessageEncoding>
          <UpdTransportElementClientSide maxBufferPoolSize="524288" maxMessageSize="65536"
              multicast="false" />
        </binding>
      </customBinding>
      <sampleProfileUdpBinding>
        <binding name="SampleProfileUdpBinding_ICalculatorContract" closeTimeout="00:01:00"
            openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
            orderedSession="true" reliableSessionEnabled="true" sessionInactivityTimeout="00:10:00" 
            clientBaseAddress="soap.udp://localhost:8003/" />
      </sampleProfileUdpBinding>
    </bindings>
    <client>
      <endpoint address="soap.udp://localhost:8001/" binding="sampleProfileUdpBinding"
          bindingConfiguration="SampleProfileUdpBinding_ICalculatorContract"
          contract="ICalculatorContract" name="SampleProfileUdpBinding_ICalculatorContract" />
      <endpoint address="soap.udp://localhost:8002/datagram" binding="customBinding"
          bindingConfiguration="CustomBinding_IDatagramContract" contract="IDatagramContract"
          name="CustomBinding_IDatagramContract" />
    </client>
    <extensions>
      <bindingElementExtensions>
        <add name="UpdTransportElementClientSide" type="Microsoft.ServiceModel.Samples.UdpTransportElement, UdpTransport" />
      </bindingElementExtensions>
      <!-- This was added manually because svcutil.exe does not add this extension to the file -->
      <bindingExtensions>
        <add name="sampleProfileUdpBinding" type="Microsoft.ServiceModel.Samples.SampleProfileUdpBindingCollectionElement, UdpTransport" />
      </bindingExtensions>
    </extensions>
  </system.serviceModel>

【问题讨论】:

天哪,这是个大问题…… 我提供了所有这些信息,以防其他人也有同样的兴趣。我希望它不是那么大。如果我可以附加文件而不是列表,那将很有帮助。 【参考方案1】:

可以在 WCF 中创建自定义传输。不过,这涉及到很多!

Roman Kiss 构建了 Null 传输,允许您在同一进程中使用和托管服务,而无需通过内置传输之一编组数据的开销。他的 CodePlex 文章可在以下网址获得:

http://www.codeproject.com/KB/WCF/NullTransportForWCF.aspx

这可能是帮助您学习如何构建自己的交通工具的良好起点。祝你好运!

【讨论】:

奇怪,我已经为这个添加了书签,但我认为它不适用于这个案例。我现在一定会通读的,谢谢。【参考方案2】:

我不知道自定义传输通道。我认为根据我的经验,可能无法让 WcfTestClient 为自定义传输通道工作。

【讨论】:

以上是关于是否可以使 WcfTestClient 为自定义传输通道工作?的主要内容,如果未能解决你的问题,请参考以下文章

是否有办法在春季为自定义依赖项找出application.properties文件的属性名?

PHP 检查当前帖子类型是否为自定义帖子类型 - WordPress

Application Insights - ILogger 参数呈现为自定义维度中的对象名称

在 Android 上为自定义形状添加阴影

将 PySpark 命令转换为自定义函数

如何将自定义 json 转换为自适应卡片 json 格式