网络环境:
Https客户端<=============>代理服务器<==============>Https服务器
192.168.17.11<-----外网--- --->192.168.17.22
10.100.21.10<----内网----->10.100.21.11ps: Http Client 没有默认网关,但是可以ping 10.100.21.11
描述:
操作系统:Ubuntu 12.04 on 3 hosts
Https 客户端:使用 java(openjdk-6) 实现。有一个网络接口。
代理服务器:Apache2.2。有两个网络接口。
Https Server:Tomcat6。有一个网络接口。
我使用两种方法通过代理实现httpsurlconnection
:(
为了方便我不写关于检查serverTrusted和hostnameVerifier问题的ssl句柄功能。如果需要我会更新。)
1.代理类
InetSocketAddress proxyInet = new InetSocketAddress("10.100.21.11",80);
Proxy proxy = new Proxy(Proxy.Type.HTTP, proxyInet);
URL httpsUrl = new URL("https://192.168.17.22:8443/test");
HttpsURLConnection httpsCon = (HttpsURLConnection) httpsUrl.openConnection(proxy);
httpsCon.setDoOutput(true);
httpsCon.setDoInput(true);
httpsCon.setRequestMethod("POST");
OutputStream out = httpsCon.getOutputStream();
OutputStreamWriter owriter = new OutputStreamWriter(out);
owriter.write("<request>test</request>");
owriter.flush();
owriter.close();
...
这种方法可行,我观察到数据包流也符合我的预期。
HttpClient ---> 代理服务器 ---> HttpServer
但是当我使用 set Property 方法时:
2.setProperty
System.setProperty("http.proxySet", "true");
System.setProperty("http.proxyHost",10.100.21.11);
System.setProperty("http.proxyPort","80");
URL httpsUrl = new URL("https://192.168.17.22:8443/test");
HttpsURLConnection httpsCon = (HttpsURLConnection)httpsUrl.openConnection();
httpsCon.setDoOutput(true);
httpsCon.setDoInput(true);
httpsCon.setRequestMethod("POST");
OutputStream out = httpsCon.getOutputStream();
OutputStreamWriter owriter = new OutputStreamWriter(out);
owriter.write("<request>test</request>");
owriter.flush();
owriter.close();
...
我有一个NoRouteToHostException: Network is unreachable
.
这让我很困惑。我没有看到 HttpClient 和 ProxyServer 之间的任何数据包。
但是 HttpClient 可以 ping 到 ProxyServer(10.100.12.10 ping 10.100.21.11)
所以我删除了代理设置(不使用代理):
也得到了NoRouteToHostException: Network is unreachable
.
我认为这是合理的。因为没有通往外联网的路线。
我想似乎是setProperty方法,httpsUrlConnection的内部函数将检查此 url 是否可以访问。
但这很奇怪。第一种方法可以成功。
有什么想法吗?或者第一种和第二种方法有什么不同?
++++++++++++++++++++++++++++++++++++++++++++++++++++++ +++++++++++++++++
更新
System.setProperty("https.proxyHost",10.100.21.11);
System.setProperty("https.proxyPort","80");
它可以工作并且数据包流是我期望的正确的。
但是设置 https.proxyPort=443 对我来说不可行
System.setProperty("https.proxyPort","443");
它将抛出一个异常,如下所示:
java.net.SocketException: Unexpected end of file from server
at sun.net.www.http.HttpClient.parseHTTPHeader(HttpClient.java:770)
....
所以我认为 Apache Proxy 也必须修改为正确的配置。