我们有许多托管多个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#setSocketImplFactory
和URL.setURLStreamHandlerFactory
都允许您覆盖默认实现,但不允许您访问默认实现,因此它们似乎是非常全有或全无覆盖。当你只希望在顶部添加逻辑时,这非常烦人。