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链接时会出现的一些问题;

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

  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 target

上面提到SunJSSE,JSSE(JavaSecure 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
以及javaSecurity Guide:http://java.sun.com/j2se/1.5.0/docs/guide/security/

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

那遇到这种情况,怎么处理呢?有以下两种方案:
1、按照以上信任管理器的规则,将服务端的公钥导入到jssecacerts,或者是在系统属性中设置要加载的trustStore文件的路径;证书导入可以用如下命令:keytool -import -file src_cer_file –keystore dest_cer_store;至于证书可以通过浏览器导出获得;
2、实现自己的证书信任管理器类,比如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都必须做相同的操作;第二种方式灵活性更高,但是要小心实现,否则可能会留下安全隐患;

时间: 2024-08-14 04:28:59

HTTPS 信任证书的相关文章

HTTPS的证书未经权威机构认证的情况下,访问HTTPS站点的两种方法

注意一下文章中提到的jsse在jdk1.4以后已经集成了,不必纠结. 摘 要 JSSE是一个SSL和TLS的纯Java实现,通过JSSE可以很容易地编程实现对HTTPS站点的访问.但是,如果该站点的证书未经权威机构的验证,JSSE将拒绝信任该证书从而不能访问HTTPS站点.本文在简要介绍JSSE的基础上提出了两种解决该问题的方法. 引言 过去的十几年,网络上已经积累了大量的Web应用.如今,无论是整合原有的Web应用系统,还是进行新的Web开发,都要求通过编程来访问某些Web页面.传统的方法是使

https数字证书交换过程介绍

文章转自:https://www.2cto.com/kf/201804/739010.html,感谢原作者的辛苦整理,讲解的很清楚,谢谢. [https数字证书交换过程介绍] 注意:该问的背景用到了非对称加密,请先了解非对称加密与对称加密的区别. 1. 鲍勃有两把钥匙,一把是公钥,另一把是私钥. 2. 鲍勃把公钥送给他的朋友们----帕蒂.道格.苏珊----每人一把. 3. 苏珊要给鲍勃写一封保密的信.她写完后用鲍勃的公钥加密,就可以达到保密的效果. 4.鲍勃收信后,用私钥解密,就看到了信件内容

keytool+tomcat配置HTTPS双向证书认证

系统需求: 1.  Windows系统或Linux系统 2.  安装并配置JDK 1.6.0_13 3.  安装并配置Tomcat 6.0 一.服务器证书 创建证书存储目录"D:\home","运行"控制台,进入%JAVA_HOME%/bin目录,使用keytool为Tomcat生成证书,假定目标机器的域名是"localhost",keystore文件存放在"D:\home\tomcat.keystore",口令为"

vcenter 6.5上传文件至存储卷:提示不信任证书,导致无法上传

故障现象:chrome浏览器浏览器上传镜至vcenter 6.5 存储卷提示:由于不确定的原因,操作失败.通常,当浏览器 不信任证书时会发生此问题. 解决方法:(1)登录https://vcenter.com //也就是vcenter管理页面 (2)下载好CA证书解压CA证书,即可获得linux/win/mac三个版本的CA签名证书 (3)将证书导入浏览器,我使用的是windows系统,和chrome浏览器.①谷歌浏览器设置 ②高级-->隐私设置和安全性-->管理证书 ③证书-->受信任

jdk信任证书

在使用httpclient请求时,使用proxy设置后,然后就报错了. javax.net.ssl.SSLHandshakeException: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requ

抓包工具fiddler下载配置(一):下载/安装&amp;信任证书

简介 Fiddler一个http协议调试代理工具,它能够记录并检查所有你的电脑和互联网之间的http通讯,设置断点,查看所有的“进出”Fiddler的数据(指cookie,html,js,css等文件). 下载安装 一.下载 1,官网下载,地址:https://www.telerik.com/download/fiddler 进入官网,填写如下信息,后下载自己需要版本 2,不想到官网下载可以到小编百度云盘下载(windows64位) 链接:https://pan.baidu.com/s/1-UQ

nginx搭建https单向证书

一)默认情况下ssl模块并未被安装,如果使用该模块则需要在编译nginx的时指定--with-http_ssl_module参数. wget http://nginx.org/download/nginx-1.3.16.tar.gz tar -xf nginx-1.3.16.tar.gz -C /usr/local/ cd /usr/local/nginx-1.3.16/ ./configure --prefix=/usr/local/nginx --user=nginx --group=ngi

IIS上架设https网站证书处理备忘

1. 免费SSL证书申请 https://www.startssl.com 教程:http://hxs.fd.fj.cn/?action=show&id=13 2. 证书转换 申请到的证书有两个关键文件(适用于openssl文本格式), xxx.key & xxx.crt,要在IIS上安装需要转换成pfx格式,转换命令: openssl pkcs12 -export -out server.pfx -inkey xxx.key -in xxx.crt 3. 证书安装 在IIS中,打开“服务

HTTPS安全证书访问连接知识讲解

HTTPS安全证书访问连接知识讲解 01:网络安全涉及的问题: ①. 网络安全问题-数据机密性问题 传输的数据可能会被第三方随时都能看到 ②. 网络安全问题-数据完整性问题 传输的数据不能随意让任何人进行修改 ③. 网络安全问题-身份验证问题 第一次通讯时,需要确认通讯双方的身份正确 02:网络安全涉及的问题解决: ①. 网络安全问题-数据机密性问题解决 a) 利用普通加密算法解决机密性 利用相应算法,对传输数据(明文数据)进行加密(密文数据):再利用对应算法,将加密数据解密变为 真实数据 优点