Java:指定传出连接的本地地址

时间:2013-05-13 10:49:16

标签: java proxy urlconnection virtual-ip-address

我们有许多托管多个IP地址的服务器。额外的是所谓的虚拟IP(“VIP”)。

我们希望启用一个Java应用程序来建立一个传出连接,该连接将源地址列为指定的虚拟IP而不是主机IP地址。

这是我们尝试过的。 InetAddress#getByAddress中列出的IP地址是虚拟IP地址。属性url是用户指定的要连接的目标网址。

public void attemptConnection() {
  try {
    Proxy proxy = new Proxy(Proxy.Type.HTTP,
        new InetSocketAddress(InetAddress.getByAddress(new byte[]{(byte)10,(byte)252,(byte)47,(byte)33}), 0));
    final URL _url = new URL(url);
    final URLConnection conn = _url.openConnection(proxy);
    BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
    String inputLine;
    while ((inputLine = in.readLine()) != null) {
      System.out.println(inputLine);
      in.close();
    }
  } catch (Exception e) {
    e.printStackTrace();
  }
}

上面的代码是在this answer

上建模的

我们正在使用Python SimpleHTTPServer进行测试,只需运行python -m SimpleHTTPServer然后尝试连接。

如果我们在Java代码中不使用代理,那么连接可以正常工作。

但是,一旦定义了VIP代理,我们就会拒绝连接:

java.net.ConnectException: Connection refused
        at java.net.PlainSocketImpl.socketConnect(Native Method)
        at java.net.AbstractPlainSocketImpl.doConnect(AbstractPlainSocketImpl.java:327)
        at java.net.AbstractPlainSocketImpl.connectToAddress(AbstractPlainSocketImpl.java:193)
        at java.net.AbstractPlainSocketImpl.connect(AbstractPlainSocketImpl.java:180)
        at java.net.Socket.connect(Socket.java:546)
        at java.net.Socket.connect(Socket.java:495)
        at sun.net.NetworkClient.doConnect(NetworkClient.java:178)
        at sun.net.www.http.HttpClient.openServer(HttpClient.java:409)
        at sun.net.www.http.HttpClient$2.run(HttpClient.java:457)
        at java.security.AccessController.doPrivileged(Native Method)
        at sun.net.www.http.HttpClient.privilegedOpenServer(HttpClient.java:454)
        at sun.net.www.http.HttpClient.openServer(HttpClient.java:521)
        at sun.net.www.http.HttpClient.<init>(HttpClient.java:240)
        at sun.net.www.http.HttpClient.New(HttpClient.java:321)
        at sun.net.www.http.HttpClient.New(HttpClient.java:338)
        at sun.net.www.protocol.http.HttpURLConnection.getNewHttpClient(HttpURLConnection.java:935)
        at sun.net.www.protocol.http.HttpURLConnection.plainConnect(HttpURLConnection.java:914)
        at sun.net.www.protocol.http.HttpURLConnection.connect(HttpURLConnection.java:801)
        at sun.net.www.protocol.http.HttpURLConnection.getInputStream(HttpURLConnection.java:1139)
        at com.ocado.dsi.ConnectToUrl.attemptConnection(ConnectToUrl.java:38)
        at com.ocado.dsi.ConnectToUrl.run(ConnectToUrl.java:56)
        at com.ocado.dsi.ConnectToUrl.main(ConnectToUrl.java:64)

任何想法为什么?

更新

直接在套接字上指定本地地址时,该过程正常。

我设置了一个在端口8000上侦听的Python简单套接字服务器,就像上面使用的SimpleHTTPServer一样。使用具有所需本地地址的套接字从Java连接到此工作正常工作:python服务器将传入的连接源地址识别为用户定义的地址。

Socket s = new Socket(InetAddress.getByAddress(new byte[]{(byte)10,(byte)252,(byte)47,(byte)33}), 8000, InetAddress.getByAddress(new byte[]{(byte)10,(byte)97,(byte)5,(byte)147}), 0);
s.close();

我将深入研究Socket#setSocketImplFactory,看看是否可以通过自定义每个创建的套接字来解决问题。

问题是,Socket#setSocketImplFactoryURL.setURLStreamHandlerFactory都允许您覆盖默认实现,但不允许您访问默认实现,因此它们似乎是非常全有或全无覆盖。当你只希望在顶部添加逻辑时,这非常烦人。

0 个答案:

没有答案