HTTPS 信任证书

  1. 使用HttpsURLConnection访问HTTPS连接时通常须要引入证书,不然会产生异常。
  2. 可是也可使用信任全部证书的方式来达到访问的目的。
  3. 经上网查询资料发现一个很好用的类来实现信任全部证书的功能。特此记录。
  4. 代码来自[这里](http://javaweb.org/?p=1237)

类代码

  1. import java.security.cert.CertificateException;
  2. import java.security.cert.X509Certificate;
  3. import javax.net.ssl.HostnameVerifier;
  4. import javax.net.ssl.HttpsURLConnection;
  5. import javax.net.ssl.SSLContext;
  6. import javax.net.ssl.SSLSession;
  7. import javax.net.ssl.TrustManager;
  8. import javax.net.ssl.X509TrustManager;
  9. public class SslUtils {
  10. private static void trustAllHttpsCertificates() throws Exception {
  11. TrustManager[] trustAllCerts = new TrustManager[1];
  12. TrustManager tm = new miTM();
  13. trustAllCerts[0] = tm;
  14. SSLContext sc = SSLContext.getInstance("SSL");
  15. sc.init(null, trustAllCerts, null);
  16. HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
  17. }
  18. static class miTM implements TrustManager, X509TrustManager {
  19. public X509Certificate[] getAcceptedIssuers() {
  20. return null;
  21. }
  22. public boolean isServerTrusted(X509Certificate[] certs) {
  23. return true;
  24. }
  25. public boolean isClientTrusted(X509Certificate[] certs) {
  26. return true;
  27. }
  28. public void checkServerTrusted(X509Certificate[] certs, String authType)
  29. throws CertificateException {
  30. return;
  31. }
  32. public void checkClientTrusted(X509Certificate[] certs, String authType)
  33. throws CertificateException {
  34. return;
  35. }
  36. }
  37. /**
  38. * 忽略HTTPS请求的SSL证书,必须在openConnection以前调用
  39. *
  40. * @throws Exception
  41. */
  42. public static void ignoreSsl() throws Exception {
  43. HostnameVerifier hv = new HostnameVerifier() {
  44. public boolean verify(String urlHostName, SSLSession session) {
  45. System.out.println("Warning: URL Host: " + urlHostName + " vs. " + session.getPeerHost());
  46. return true;
  47. }
  48. };
  49. trustAllHttpsCertificates();
  50. HttpsURLConnection.setDefaultHostnameVerifier(hv);
  51. }
  52. }

调用方式

  1. openConnection以前调用SslUtils.ignoreSsl();便可忽略全部HTTPS连接的证书。

 


 

 

 

 

 

 

在web应用交互过程当中,有不少场景须要保证通讯数据的安全;在前面也有好多篇文章介绍了在Web Service调用过程当中用WS-Security来保证接口交互过程的安全性,值得注意的是,该种方式基于的传输协议仍然是Http,采用这种方式可扩展性和数据交互效率比较高;另一种实现方式就是用Https,他是在协议层对Http的再次封装,加入了SSL/TLS,采用该协议进行通讯的数据所有都会被加密,因为目前Web开发编程中对此都有了必定程度的封装,因此采用Https对外提供服务,除了证书之外,对编程能力的要求并不高,相对于前者门槛较低,可是因为对双方通讯的全部数据都进行加密,并且交互过程当中还有屡次握手等,因此效率较低;如下就介绍下在Java中访问Https连接时会出现的一些问题;html

在Java中要访问Https连接时,会用到一个关键类HttpsURLConnection;参见以下实现代码:java

  1.         // 建立URL对象
  2.         URL myURL = new URL("https://www.sun.com");
  3.         // 建立HttpsURLConnection对象,并设置其SSLSocketFactory对象
  4.         HttpsURLConnection httpsConn = (HttpsURLConnection) myURL
  5.                 .openConnection();
  6.         // 取得该链接的输入流,以读取响应内容
  7.         InputStreamReader insr = new InputStreamReader(httpsConn
  8.                 .getInputStream());
  9.         // 读取服务器的响应内容并显示
  10.         int respInt = insr.read();
  11.         while (respInt != -1) {
  12.             System.out.print((char) respInt);
  13.             respInt = insr.read();
  14.         }

在取得connection的时候和正常浏览器访问同样,仍然会验证服务端的证书是否被信任(权威机构发行或者被权威机构签名);若是服务端证书不被信任,则默认的实现就会有问题,通常来讲,用SunJSSE会抛以下异常信息:
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 targetweb

上面提到SunJSSE,JSSE(Java Secure Socket Extension)是实现Internet安全通讯的一系列包的集合。它是一个SSL和TLS的纯Java实现,能够透明地提供数据加密、服务器认证、信息完整性等功能,可使咱们像使用普通的套接字同样使用JSSE创建的安全套接字。JSSE是一个开放的标准,不仅是Sun公司才能实现一个SunJSSE,事实上其余公司有本身实现的JSSE,而后经过JCA就能够在JVM中使用。
关于JSSE的详细信息参考官网Reference:http://java.sun.com/j2se/1.5.0/docs/guide/security/jsse/JSSERefGuide.html
以及java Security Guide:http://java.sun.com/j2se/1.5.0/docs/guide/security/编程

在深刻了解JSSE以前,须要了解一个有关Java安全的概念:客户端的TrustStore文件。客户端的TrustStore文件中保存着被客户端所信任的服务器的证书信息。客户端在进行SSL链接时,JSSE将根据这个文件中的证书决定是否信任服务器端的证书。在SunJSSE中,有一个信任管理器类负责决定是否信任远端的证书,这个类有以下的处理规则:
一、若系统属性javax.net.sll.trustStore指定了TrustStore文件,那么信任管理器就去jre安装路径下的lib/security/目录中寻找并使用这个文件来检查证书。
二、若该系统属性没有指定TrustStore文件,它就会去jre安装路径下寻找默认的TrustStore文件,这个文件的相对路径为:lib/security/jssecacerts
三、若jssecacerts不存在,可是cacerts存在(它随J2SDK一块儿发行,含有数量有限的可信任的基本证书),那么这个默认的TrustStore文件就是lib/security/cacerts浏览器

那遇到这种状况,怎么处理呢?有如下两种方案:
一、按照以上信任管理器的规则,将服务端的公钥导入到jssecacerts,或者是在系统属性中设置要加载的trustStore文件的路径;证书导入能够用以下命令:keytool -import -file src_cer_file –keystore dest_cer_store;至于证书能够经过浏览器导出得到;
二、实现本身的证书信任管理器类,好比MyX509TrustManager,该类必须实现X509TrustManager接口中的三个method;而后在HttpsURLConnection中加载自定义的类,能够参见以下两个代码片断,其一为自定义证书信任管理器,其二为connect时的代码:安全

  1. package test;
  2. import java.io.FileInputStream;
  3. import java.security.KeyStore;
  4. import java.security.cert.CertificateException;
  5. import java.security.cert.X509Certificate;
  6. import javax.net.ssl.TrustManager;
  7. import javax.net.ssl.TrustManagerFactory;
  8. import javax.net.ssl.X509TrustManager;
  9. public class MyX509TrustManager implements X509TrustManager {
  10.     /*
  11.      * The default X509TrustManager returned by SunX509.  We'll delegate
  12.      * decisions to it, and fall back to the logic in this class if the
  13.      * default X509TrustManager doesn't trust it.
  14.      */
  15.     X509TrustManager sunJSSEX509TrustManager;
  16.     MyX509TrustManager() throws Exception {
  17.         // create a "default" JSSE X509TrustManager.
  18.         KeyStore ks = KeyStore.getInstance("JKS");
  19.         ks.load(new FileInputStream("trustedCerts"),
  20.             "passphrase".toCharArray());
  21.         TrustManagerFactory tmf =
  22.         TrustManagerFactory.getInstance("SunX509", "SunJSSE");
  23.         tmf.init(ks);
  24.         TrustManager tms [] = tmf.getTrustManagers();
  25.         /*
  26.          * Iterate over the returned trustmanagers, look
  27.          * for an instance of X509TrustManager.  If found,
  28.          * use that as our "default" trust manager.
  29.          */
  30.         for (int i = 0; i < tms.length; i++) {
  31.             if (tms[i] instanceof X509TrustManager) {
  32.                 sunJSSEX509TrustManager = (X509TrustManager) tms[i];
  33.                 return;
  34.             }
  35.         }
  36.         /*
  37.          * Find some other way to initialize, or else we have to fail the
  38.          * constructor.
  39.          */
  40.         throw new Exception("Couldn't initialize");
  41.     }
  42.     /*
  43.      * Delegate to the default trust manager.
  44.      */
  45.     public void checkClientTrusted(X509Certificate[] chain, String authType)
  46.                 throws CertificateException {
  47.         try {
  48.             sunJSSEX509TrustManager.checkClientTrusted(chain, authType);
  49.         } catch (CertificateException excep) {
  50.             // do any special handling here, or rethrow exception.
  51.         }
  52.     }
  53.     /*
  54.      * Delegate to the default trust manager.
  55.      */
  56.     public void checkServerTrusted(X509Certificate[] chain, String authType)
  57.                 throws CertificateException {
  58.         try {
  59.             sunJSSEX509TrustManager.checkServerTrusted(chain, authType);
  60.         } catch (CertificateException excep) {
  61.             /*
  62.              * Possibly pop up a dialog box asking whether to trust the
  63.              * cert chain.
  64.              */
  65.         }
  66.     }
  67.     /*
  68.      * Merely pass this through.
  69.      */
  70.     public X509Certificate[] getAcceptedIssuers() {
  71.         return sunJSSEX509TrustManager.getAcceptedIssuers();
  72.     }
  73. }
  1.         // 建立SSLContext对象,并使用咱们指定的信任管理器初始化
  2.         TrustManager[] tm = { new MyX509TrustManager() };
  3.         SSLContext sslContext = SSLContext.getInstance("SSL", "SunJSSE");
  4.         sslContext.init(null, tm, new java.security.SecureRandom());
  5.         // 从上述SSLContext对象中获得SSLSocketFactory对象
  6.         SSLSocketFactory ssf = sslContext.getSocketFactory();
  7.         // 建立URL对象
  8.         URL myURL = new URL("https://ebanks.gdb.com.cn/sperbank/perbankLogin.jsp");
  9.         // 建立HttpsURLConnection对象,并设置其SSLSocketFactory对象
  10.         HttpsURLConnection httpsConn = (HttpsURLConnection) myURL.openConnection();
  11.         httpsConn.setSSLSocketFactory(ssf);
  12.         // 取得该链接的输入流,以读取响应内容
  13.         InputStreamReader insr = new InputStreamReader(httpsConn.getInputStream());
  14.         // 读取服务器的响应内容并显示
  15.         int respInt = insr.read();
  16.         while (respInt != -1) {
  17.             System.out.print((char) respInt);
  18.             respInt = insr.read();
  19.         }

对于以上两种实现方式,各有各的优势,第一种方式不会破坏JSSE的安全性,可是要手工导入证书,若是服务器不少,那每台服务器的JRE都必须作相同的操做;第二种方式灵活性更高,可是要当心实现,不然可能会留下安全隐患;服务器

相关文章
相关标签/搜索