解决https需要验证问题

2 篇文章 0 订阅
1 篇文章 0 订阅

这两天调试链接问题有两个方面:

HttpsURLConnection 中connect 抛出的异常时IOException

采用下列文章描述的

HttpClient httpClient = new DefaultHttpClient();

返回SSLException异常

javax.net.ssl.SSLException: hostname in certificate didn't match: <58.83.158.101> != <*.360buy.com>


两种方法都正确,只是获取的异常信息后者更为详细,以下是调用堆栈


Exception in thread "main" javax.net.ssl.SSLException: hostname in certificate didn't match: <eps.dev.surepush.cn> != &lt;*.xxxxxxx(公司名称).com>
at org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:220)
at org.apache.http.conn.ssl.BrowserCompatHostnameVerifier.verify(BrowserCompatHostnameVerifier.java:54)
at org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:149)
at org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:130)
at org.apache.http.conn.ssl.SSLSocketFactory.createSocket(SSLSocketFactory.java:399)
at org.apache.http.impl.conn.DefaultClientConnectionOperator.updateSecureConnection(DefaultClientConnectionOperator.java:203)
at org.apache.http.impl.conn.AbstractPoolEntry.layerProtocol(AbstractPoolEntry.java:277)
at org.apache.http.impl.conn.AbstractPooledConnAdapter.layerProtocol(AbstractPooledConnAdapter.java:138)
at org.apache.http.impl.client.DefaultRequestDirector.establishRoute(DefaultRequestDirector.java:704)
at org.apache.http.impl.client.DefaultRequestDirector.execute(DefaultRequestDirector.java:421)
at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:641)
at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:576)
at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:554)
at com.client.ctl.ZTest.main(ZTest.java:195)


具体SSLSocketFactory实现如下,否则不返回SSLException,最终根据调用堆栈,增加this.getHostnameVerifier().verify(host, sslSocket);

class SSLSocketFactoryEx extends SSLSocketFactory {

        SSLContext sslContext = SSLContext.getInstance("TLS");

        public SSLSocketFactoryEx(KeyStore truststore)
                throws NoSuchAlgorithmException, KeyManagementException,
                KeyStoreException, UnrecoverableKeyException {
            super(truststore);

            //
            KeyManager[] km = resetKm();
            TrustManager[] tm = resetTm();
            
            sslContext.init(km, tm, null);
        }

        @Override
        public Socket createSocket(Socket socket, String host, int port,
                boolean autoClose) throws IOException, UnknownHostException {
            System.out.println("=== SSLSocketFactoryEx.createSocket === host : " + host);
            SSLSocket sslSocket = (SSLSocket)sslContext.getSocketFactory().createSocket(socket, host,
                    port, autoClose);
            // 这里没有验证,增加后就能正常判读
            this.getHostnameVerifier().verify(host, sslSocket);
            
            return sslSocket;
        }

        @Override
        public Socket createSocket() throws IOException {
            System.out.println("=== SSLSocketFactoryEx.createSocket ===");
            return sslContext.getSocketFactory().createSocket();
        }
    }


参考:http://my.oschina.net/sourcecoding/blog/80698

今天在开发京东API接口的时候遇到了一个问题。

问题描述:开发京东用户授权,采用https的访问协议获取新的AccessToken时,总是出现异常:

javax.net.ssl.SSLHandshakeException: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target

之后在网上找了很多例子,最后摘了一个不需要证书的例子:

?

之后在运行报另一个异常,执行HttpResponse response = httpClient.execute(httpPost); 报的错

javax.net.ssl.SSLException: hostname in certificate didn't match: <58.83.158.101> != <*.360buy.com>

原因我猜测可能是在验证HOST的时候出现了问题,但我始终传给程序的是域名,为啥就非要解析成ip呢。。。我郁闷。我就一直在想怎么能报HOST变成域名的形式,我debug了httpPost里的HOST没有问题还是域名的,我就想那肯定是httpClient中有问题。最后发现httpClient与socketFactory有关系,而socketFactory就像是一个配置信息的存储,我在这里找到了一个setHostnameVerifier的方法,貌似是验证hostName的,需要参数hostnameVerifier,我像之前那个X509TrustManager一样new了一个X509HostnameVerifier付给setHostnameVerifier方法,然后重写里面的方法,发现了验证的方法verify(String arg0, SSLSession arg1),跳过直接返回true,果然问题解决了。

以下为正确代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
/**
      * 发送HTTPS  POST请求
      *
      * @param 要访问的HTTPS地址,POST访问的参数Map对象
      * <a href="http://my.oschina.net/u/556800" class="referer" target="_blank">@return</a>  返回响应值
      * */
     public static final String sendHttpsRequestByPost(String url, Map<String, String> params) {
         String responseContent = null ;
         HttpClient httpClient = new DefaultHttpClient();
         //创建TrustManager
         X509TrustManager xtm = new X509TrustManager() {
             public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
             public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
             public X509Certificate[] getAcceptedIssuers() {
                 return null ;
             }
         };
         //这个好像是HOST验证
         X509HostnameVerifier hostnameVerifier = new X509HostnameVerifier() {
             public boolean verify(String arg0, SSLSession arg1) {
                 return true ;
             }
             public void verify(String arg0, SSLSocket arg1) throws IOException {}
             public void verify(String arg0, String[] arg1, String[] arg2) throws SSLException {}
             public void verify(String arg0, X509Certificate arg1) throws SSLException {}
         };
         try {
             //TLS1.0与SSL3.0基本上没有太大的差别,可粗略理解为TLS是SSL的继承者,但它们使用的是相同的SSLContext
             SSLContext ctx = SSLContext.getInstance( "TLS" );
             //使用TrustManager来初始化该上下文,TrustManager只是被SSL的Socket所使用
             ctx.init( null , new TrustManager[] { xtm }, null );
             //创建SSLSocketFactory
             SSLSocketFactory socketFactory = new SSLSocketFactory(ctx);
             socketFactory.setHostnameVerifier(hostnameVerifier);
             //通过SchemeRegistry将SSLSocketFactory注册到我们的HttpClient上
             httpClient.getConnectionManager().getSchemeRegistry().register( new Scheme( "https" , socketFactory, 443 ));
             HttpPost httpPost = new HttpPost(url);
             List<NameValuePair> formParams = new ArrayList<NameValuePair>(); // 构建POST请求的表单参数
             for (Map.Entry<String, String> entry : params.entrySet()) {
                 formParams.add( new BasicNameValuePair(entry.getKey(), entry.getValue()));
             }
             httpPost.setEntity( new UrlEncodedFormEntity(formParams, "UTF-8" ));
             HttpResponse response = httpClient.execute(httpPost);
             HttpEntity entity = response.getEntity(); // 获取响应实体
             if (entity != null ) {
                 responseContent = EntityUtils.toString(entity, "UTF-8" );
             }
         } catch (KeyManagementException e) {
             e.printStackTrace();
         } catch (NoSuchAlgorithmException e) {
             e.printStackTrace();
         } catch (UnsupportedEncodingException e) {
             e.printStackTrace();
         } catch (ClientProtocolException e) {
             e.printStackTrace();
         } catch (ParseException e) {
             e.printStackTrace();
         } catch (IOException e) {
             e.printStackTrace();
         } finally {
             // 关闭连接,释放资源
             httpClient.getConnectionManager().shutdown();
         }
         return responseContent;
     }

为了大家不引错包,我把import也放在这里:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Field;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
 
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLException;
import javax.net.ssl.SSLSession;
import javax.net.ssl.SSLSocket;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
 
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.ParseException;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.conn.scheme.Scheme;
import org.apache.http.conn.ssl.SSLSocketFactory;
import org.apache.http.conn.ssl.X509HostnameVerifier;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值