WCF发现功能根本无法正常工作。

16

我正在尝试将临时发现添加到简单的WCF服务客户端设置中(目前通过控制台应用程序进行自托管)。在Windows 7上使用VS2010进行调试,并按照在线教程中找到的任何内容进行操作,但是发现客户端仍然找不到。不用说,如果我打开正确服务端点的客户端,则可以从该客户端访问服务。

服务代码:

using (var selfHost = new ServiceHost(typeof(Renderer)))
{
    try
    {
        selfHost.Open();
        ...
        selfHost.Close();

服务应用程序配置:

<?xml version="1.0"?>
<configuration>
  <system.serviceModel>
    <services>
      <service name="TestApp.Renderer">
        <host>
          <baseAddresses>
            <add baseAddress="http://localhost:9000" />
          </baseAddresses>
        </host>
        <endpoint address="ws" binding="wsHttpBinding" contract="TestApp.IRenderer"/>
        <endpoint kind="udpDiscoveryEndpoint"/>
      </service>
    </services>
    <behaviors>
      <serviceBehaviors>
        <behavior>
          <serviceDiscovery/>
          <serviceMetadata httpGetEnabled="True"/>
        </behavior>
      </serviceBehaviors>
    </behaviors>
  </system.serviceModel>
</configuration>

客户端发现代码:

DiscoveryClient discoveryClient = new DiscoveryClient(new UdpDiscoveryEndpoint());
var criteria = new FindCriteria(typeof(IRenderer)) { Duration = TimeSpan.FromSeconds(5) };
var endpoints = discoveryClient.Find(criteria).Endpoints;

我的'endpoints'集合总是空的。我已经尝试了从调试器、命令行和管理员命令行中运行服务和客户端,但都没有用(当然,这些都是在本地机器上进行的,未来还需要在整个子网上运行)

希望能得到帮助 :-)


我也尝试在serviceDiscovery行为中添加公告端点,但这也没有帮助。 - kbo
客户端是否有任何app.config信息? - Mark W
你尝试过添加作用域吗? - Mark W
是的,客户端的app.config是由svcutil.exe自动生成的。同时,在服务端尝试了Security=none(从客户端app.config中删除了角色)。 - kbo
没有,我没有尝试添加作用域,主要是因为我没有看到任何好的例子 :-) - 最简单的方法是什么? - kbo
2个回答

37

这是一个超级简单的示例程序。它不使用配置文件,全部都是C#代码,但你可能可以将其移植到配置文件中。

在主机程序和客户端程序之间共享此接口(暂时复制到每个程序中)。

[ServiceContract]
public interface IWcfPingTest
{
  [OperationContract]
  string Ping();
}
将此代码放入主程序中。
public class WcfPingTest : IWcfPingTest
{
  public const string magicString = "djeut73bch58sb4"; // this is random, just to see if you get the right result
  public string Ping() {return magicString;}
}
public void WcfTestHost_Open()
{
  string hostname = System.Environment.MachineName;
  var baseAddress = new UriBuilder("http", hostname, 7400, "WcfPing");
  var h = new ServiceHost(typeof(WcfPingTest), baseAddress.Uri);

  // enable processing of discovery messages.  use UdpDiscoveryEndpoint to enable listening. use EndpointDiscoveryBehavior for fine control.
  h.Description.Behaviors.Add(new ServiceDiscoveryBehavior());
  h.AddServiceEndpoint(new UdpDiscoveryEndpoint());

  // enable wsdl, so you can use the service from WcfStorm, or other tools.
  var smb = new ServiceMetadataBehavior();
  smb.HttpGetEnabled = true;
  smb.MetadataExporter.PolicyVersion = PolicyVersion.Policy15;
  h.Description.Behaviors.Add(smb);

  // create endpoint
  var binding = new BasicHttpBinding(BasicHttpSecurityMode.None);
  h.AddServiceEndpoint(typeof(IWcfPingTest) , binding,   "");
  h.Open();
  Console.WriteLine("host open");
}

将此代码放在客户端程序中

private IWcfPingTest channel;
public Uri WcfTestClient_DiscoverChannel()
{
  var dc = new DiscoveryClient(new UdpDiscoveryEndpoint());
  FindCriteria fc = new FindCriteria(typeof(IWcfPingTest));
  fc.Duration = TimeSpan.FromSeconds(5);
  FindResponse fr = dc.Find(fc);
  foreach(EndpointDiscoveryMetadata edm in fr.Endpoints) 
  {
    Console.WriteLine("uri found = " + edm.Address.Uri.ToString());
  }
  // here is the really nasty part
  // i am just returning the first channel, but it may not work.
  // you have to do some logic to decide which uri to use from the discovered uris
  // for example, you may discover "127.0.0.1", but that one is obviously useless.
  // also, catch exceptions when no endpoints are found and try again.
  return fr.Endpoints[0].Address.Uri;  
}
public void WcfTestClient_SetupChannel()
{
  var binding = new BasicHttpBinding(BasicHttpSecurityMode.None);
  var factory = new ChannelFactory<IWcfPingTest>(binding);
  var uri = WcfTestClient_DiscoverChannel();
  Console.WriteLine("creating channel to " + uri.ToString());
  EndpointAddress ea = new EndpointAddress(uri);
  channel = factory.CreateChannel(ea);
  Console.WriteLine("channel created");
  //Console.WriteLine("pinging host");
  //string result = channel.Ping();
  //Console.WriteLine("ping result = " + result);
}
public void WcfTestClient_Ping()
{
  Console.WriteLine("pinging host");
  string result = channel.Ping();
  Console.WriteLine("ping result = " + result);
}

在主机上,只需调用WcfTestHost_Open()函数,然后永久睡眠或执行其他操作。

在客户端上运行这些函数。由于主机开启需要一些时间,所以这里有几次延迟。

System.Threading.Thread.Sleep(8000);
this.server.WcfTestClient_SetupChannel();
System.Threading.Thread.Sleep(2000);
this.server.WcfTestClient_Ping();

主机输出应该如下所示

host open

客户端输出应该像这样

uri found = http://wilkesvmdev:7400/WcfPing
creating channel to http://wilkesvmdev:7400/WcfPing
channel created
pinging host
ping result = djeut73bch58sb4

这是我为了解释发现示例能想到的最简单的东西。这些东西很快就会变得相当复杂。


3

该死!是防火墙的问题...由于某种原因,所有UDP通信都被阻塞了 - 禁用防火墙解决了问题。现在我只需要弄清楚正确的防火墙配置...


你最终解决了如何配置防火墙的问题吗?我也遇到了同样的问题。 - odyth
1
是的,Discovery依赖于UDP广播来定位服务。 - Chris Marisic

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接