指定要与WCF客户端一起使用的传出IP地址

时间:2010-07-14 19:43:35

标签: wcf ip-address wcf-binding wcf-client

如何在调用WCF服务时定义WCF客户端使用的LocalEndPoint(如果客户端计算机有多个IP地址)?

我的机器位于带有两个IP地址的DMZ中,外部IP地址可以通过防火墙通过我们位于外部服务提供商的网络服务器的VPN连接到达。在此计算机上运行基于WCF和Unity的自定义应用程序服务器,该服务器应充当代理或应用程序级网关(ALG)。它应接受来自Web服务器的服务调用,并使用wcf客户端工厂重新生成服务调用,并将它们转发到LAN中的真实应用服务器。

使用wcf客户端工厂在此代理上重新创建服务调用时,wcf客户端应使用此计算机的第二个内部IP地址,因为只允许来自此内部IP地址的消息通过防火墙到达局域网中的应用服务器。不幸的是,我们的wcf客户端代理始终选择使用第一个“外部”IP地址创建传出消息。我正在寻找一种方法来明确设置wcf客户端代理使用的IP地址。

我只能找到一个允许定义LocalEndPoint或ClientBaseAddress的WCF绑定元素:CompositeDuplexBindingElement。据我所知,这个属性是告诉服务器在哪里发送asynch回复消息,所以它与我正在寻找的设置不同。

任何想法我能做些什么才能找到可行的解决方案?

提前感谢任何有用的建议!!

这似乎是一个类似的问题,只是使用TcpClient / Sockets而不是WCF: Specify the outgoing IP address to use with TCPClient / Socket in C#

另一个,这次是关于SoapClient: Binding a new SoapClient to a specific IP address before sending outgoing request

3 个答案:

答案 0 :(得分:5)

我很难自己找到这个答案,所以我会分享我的解决方案以防其他人来到这里。

下面是一个返回wcf soapservice的方法。代码将尝试使用特定的ip(如果可用)。我做了这个检查,让它也可以在测试机上工作。

    public static proposalSoapClient getService()
    {
        string serviceurl = "http://somesite.dk/proposal.asmx";
        var localIpAddress = IPAddress.Parse("123.123.123.123");
        var hasLocalAddress = Dns.GetHostEntry(Dns.GetHostName()).AddressList.Any(ip => ip.AddressFamily == AddressFamily.InterNetwork && ip.Equals(localIpAddress));

        var binding = new System.ServiceModel.BasicHttpBinding("proposalSoap"); //name of binding in web.config
        var endpoint = new EndpointAddress(serviceurl);
        ServicePoint servicePoint = ServicePointManager.FindServicePoint(new Uri(serviceurl));
        servicePoint.BindIPEndPointDelegate = (sp, rm, retryCount) => { return new IPEndPoint(hasLocalAddress ? localIpAddress : IPAddress.Any, 0); };
        return new proposalSoapClient(binding, endpoint);
    }

当我在测试服务器上时,我不得不使用代理来获取服务。 (我在有权访问该服务的机器上使用fiddler作为代理)。下面是相同的代码,但在测试服务器上添加了代理部分。

    public static proposalSoapClient getService()
    {
        string serviceurl = "http://somesite.dk/proposal.asmx";
        var localIpAddress = IPAddress.Parse("123.123.123.123");
        var hasLocalAddress = Dns.GetHostEntry(Dns.GetHostName()).AddressList.Any(ip => ip.AddressFamily == AddressFamily.InterNetwork && ip.Equals(localIpAddress));

        var binding = new System.ServiceModel.BasicHttpBinding("proposalSoap"); //name of binding in web.config
        var endpoint = new EndpointAddress(serviceurl);
        ServicePoint servicePoint = ServicePointManager.FindServicePoint(new Uri(serviceurl));

    #if DEBUG
        Uri proxyUri = new Uri("http://someothersite.dk:8888");
        binding.ProxyAddress = proxyUri;
        binding.BypassProxyOnLocal = false;
        binding.UseDefaultWebProxy = false;
        servicePoint = ServicePointManager.FindServicePoint(serviceurl, new WebProxy(proxyUri, false));
    #endif

        servicePoint.BindIPEndPointDelegate = (sp, rm, retryCount) => { return new IPEndPoint(hasLocalAddress ? localIpAddress : IPAddress.Any, 0); };
        return new proposalSoapClient(binding, endpoint);
    }

答案 1 :(得分:1)

这是一个较旧的主题,但我建议任何人找到这个,因为它引导我找到一个稍微不同的解决方案。我们需要在同一台服务器上的不同进程上运行两个进程。

问题是每个默认选择绑定到NIC的主地址。我们找不到让应用程序绑定到NIC上的辅助地址的方法。

解决方法是向VM添加第二个NIC,将辅助IP作为第二个NIC的主IP移动到第二个NIC。在Windows中,您不能拥有两个默认网关(即使它们指向相同的地址)。

因此,在第二个NIC上,我们仅分配了IP和子网掩码。为了让流量路由出正确的接口,我们添加了一个路由语句,格式为:

route add 4.3.2.1 mask 255.255.255.255 1.2.3.4 IF 0x10002

我们需要进程与之通信的主机是4.3.2.1,本地默认网关是1.2.3.4,第二个NIC在路由表中显示为接口号0x10002(来自路线打印)。

我希望这会有所帮助。因为这个,我还有一些白发。

答案 2 :(得分:0)

听起来路由表在服务器上有点乱,因为你想要做什么。而不是试图在应用程序中纠正它,调整机器上的接口度量和/或路由表,以优先选择内部接口用于发往该网络的流量。 Here是一个旧参考,但应该或多或少有效。

我不认为.NET套接字会允许您明确选择源接口 - 最接近的是DontRoute SocketOption,但即便如此,您可能需要滚动自定义绑定和传输以使该选项可用