Android HTTPS SSL双向验证 使用HTTPs

本文涉及的产品
Digicert DV 证书 单域名,20个 3个月
简介:

HTTPS和HTTP的区别

1
2
3
4
5
6
7
1、https协议需要到ca申请证书,一般免费证书很少,需要交费。
 
2、http是超文本传输协议,信息是明文传输,https 则是具有安全性的ssl加密传输协议。
 
3、http和https使用的是完全不同的连接方式,用的端口也不一样,前者是80,后者是443。
 
4、http的连接很简单,是无状态的;HTTPS协议是由SSL+HTTP协议构建的可进行加密传输、身份认证的网络协议,比http协议安全。

二、SSL功能

1
2
3
4
5
6
7
8
1)客户对服务器的身份认证:
SSL服务器允许客户的浏览器使用标准的公钥加密技术和一些可靠的认证中心(CA)的证书,来确认服务器的合法性。
 
2)服务器对客户的身份认证:
也可通过公钥技术和证书进行认证,也可通过用户名,password来认证。
 
3)建立服务器与客户之间安全的数据通道:
SSL要求客户与服务器之间的所有发送的数据都被发送端加密、接收端解密,同时还检查数据的完整性。

SSL协议位于TCP/IP协议与各种应用层协议之间,为数据通讯提供安全支持。SSL协议可分为两层:

SSL记录协议(SSL Record Protocol):它建立在可靠的传输协议(如TCP)之上,为高层协议提供数据封装、压缩、加密等基本功能的支持。

SSL握手协议(SSL Handshake Protocol):它建立在SSL记录协议之上,用于在实际的数据传输开始前,通讯双方进行身份认证、协商加密算法、交换加密密钥等。

三、生成密钥库和证书

可参考以下密钥生成脚本,根据实际情况做必要的修改,其中需要注意的是:服务端的密钥库参数“CN”必须与服务端的IP地址相同,否则会报错,客户端的任意。

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
1、生成服务器证书库
 
keytool -validity 365 -genkey -v -alias server -keyalg RSA -keystore D:\ssl\server.keystore -dname "CN=127.0.0.1,OU=rongyiwang,O=rongyiwang,L=Shanghai,ST=Shanghai,c=cn" -storepass 123456 -keypass 123456
 
2、生成客户端证书库
 
keytool -validity 365 -genkeypair -v -alias client -keyalg RSA -storetype PKCS12 -keystore D:\ssl\client.p12 -dname "CN=client,OU=rongyiwang,O=rongyiwang,L=Shanghai,ST=Shanghai,c=cn" -storepass 123456 -keypass 123456
 
3、从客户端证书库中导出客户端证书
 
keytool -export -v -alias client -keystore D:\ssl\client.p12 -storetype PKCS12 -storepass 123456 -rfc -file D:\ssl\client.cer
 
4、从服务器证书库中导出服务器证书
 
keytool -export -v -alias server -keystore D:\ssl\server.keystore -storepass 123456 -rfc -file D:\ssl\server.cer
 
5、生成客户端信任证书库(由服务端证书生成的证书库)
 
keytool -import -v -alias server -file D:\ssl\server.cer -keystore D:\ssl\client.truststore -storepass 123456
 
6、将客户端证书导入到服务器证书库(使得服务器信任客户端证书)
 
keytool -import -v -alias client -file D:\ssl\client.cer -keystore D:\ssl\server.keystore -storepass 123456
 
7、查看证书库中的全部证书
 
keytool -list -keystore D:\ssl\server.keystore -storepass 123456

通过上面的步骤生成的证书,客户端需要用到的是client.p12(客户端证书,用于请求的时候给服务器来验证身份之用)和client.truststore(客户端证书库,用于验证服务器端身份,防止钓鱼)这两个文件.

其中安卓端的证书类型必须要求是BKS类型,具体生成可以参考这个create a bks bouncycastle,这里涉及到这个JARbcprov-ext-jdk15on-152.jar文件.

以下只给出Android端的请求,具体服务器端的配置可以参考这篇博客-->Java Tomcat SSL 服务端/客户端双向认证(一)


四、Android端SSL认证请求

        一般客户端验证SSL有两种方式,一种是通过SSLSocketFactory方式创建,需要设置域名及端口号(适应于HttpClient请求方式),一种是通过SSLContext方式创建(适用于HttpsURLConnection请求方式).
1、下面给出SslSocketFactory方式进行SSL认证的客户端代码

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
private  static  final  String KEY_STORE_TYPE_BKS =  "bks" ; //证书类型 固定值
     private  static  final  String KEY_STORE_TYPE_P12 =  "PKCS12" ; //证书类型 固定值
 
     private  static  final  String KEY_STORE_CLIENT_PATH =  "client.p12" ; //客户端要给服务器端认证的证书
     private  static  final  String KEY_STORE_TRUST_PATH =  "client.truststore" ; //客户端验证服务器端的证书库
     private  static  final  String KEY_STORE_PASSWORD =  "123456" ; // 客户端证书密码
     private  static  final  String KEY_STORE_TRUST_PASSWORD =  "123456" ; //客户端证书库密码
 
     /**
      * 获取SslSocketFactory
      *
      * @param context 上下文
      * @return SSLSocketFactory
      */
     public  static  SSLSocketFactory getSslSocketFactory(Context context) {
         try  {
             // 服务器端需要验证的客户端证书
             KeyStore keyStore = KeyStore.getInstance(KEY_STORE_TYPE_P12);
             // 客户端信任的服务器端证书
             KeyStore trustStore = KeyStore.getInstance(KEY_STORE_TYPE_BKS);
 
             InputStream ksIn = context.getResources().getAssets().open(KEY_STORE_CLIENT_PATH);
             InputStream tsIn = context.getResources().getAssets().open(KEY_STORE_TRUST_PATH);
             try  {
                 keyStore.load(ksIn, KEY_STORE_PASSWORD.toCharArray());
                 trustStore.load(tsIn, KEY_STORE_TRUST_PASSWORD.toCharArray());
             catch  (Exception e) {
                 e.printStackTrace();
             finally  {
                 try  {
                     ksIn.close();
                 catch  (Exception ignore) {
                 }
                 try  {
                     tsIn.close();
                 catch  (Exception ignore) {
                 }
             }
             return  new  SSLSocketFactory(keyStore, KEY_STORE_PASSWORD, trustStore);
         catch  (KeyManagementException | UnrecoverableKeyException | KeyStoreException | FileNotFoundException | NoSuchAlgorithmException | ClientProtocolException e) {
             e.printStackTrace();
         catch  (IOException e) {
             e.printStackTrace();
         }
         return  null ;
     }
     
     /**
      * 获取SSL认证需要的HttpClient
      *
      * @param context 上下文
      * @param port    端口号
      * @return HttpClient
      */
     public  static  HttpClient getSslSocketFactoryHttp(Context context,  int  port) {
         HttpClient httpsClient =  new  DefaultHttpClient();
         SSLSocketFactory sslSocketFactory = getSslSocketFactory(context);
         if  (sslSocketFactory !=  null ) {
             Scheme sch =  new  Scheme( "https" , sslSocketFactory, port);
             httpsClient.getConnectionManager().getSchemeRegistry().register(sch);
         }
         return  httpsClient;
     }

2、下面给出SSLContext方式进行SSL认证的客户端代码

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
66
67
68
69
70
71
72
73
74
75
76
77
78
private  static  final  String KEY_STORE_TYPE_BKS =  "bks" ; //证书类型 固定值
     private  static  final  String KEY_STORE_TYPE_P12 =  "PKCS12" ; //证书类型 固定值
 
     private  static  final  String KEY_STORE_CLIENT_PATH =  "client.p12" ; //客户端要给服务器端认证的证书
     private  static  final  String KEY_STORE_TRUST_PATH =  "client.truststore" ; //客户端验证服务器端的证书库
     private  static  final  String KEY_STORE_PASSWORD =  "123456" ; // 客户端证书密码
     private  static  final  String KEY_STORE_TRUST_PASSWORD =  "123456" ; //客户端证书库密码
     
     /**     * 获取SSLContext     *     * @param context 上下文     * @return SSLContext     */
     private  static  SSLContext getSSLContext(Context context) {
         try  {
             // 服务器端需要验证的客户端证书
             KeyStore keyStore = KeyStore.getInstance(KEY_STORE_TYPE_P12);
             // 客户端信任的服务器端证书
             KeyStore trustStore = KeyStore.getInstance(KEY_STORE_TYPE_BKS);
 
             InputStream ksIn = context.getResources().getAssets().open(KEY_STORE_CLIENT_PATH);
             InputStream tsIn = context.getResources().getAssets().open(KEY_STORE_TRUST_PATH);
             try  {
                 keyStore.load(ksIn, KEY_STORE_PASSWORD.toCharArray());
                 trustStore.load(tsIn, KEY_STORE_TRUST_PASSWORD.toCharArray());
             catch  (Exception e) {
                 e.printStackTrace();
             finally  {
                 try  {
                     ksIn.close();
                 catch  (Exception ignore) {
                 }
                 try  {
                     tsIn.close();
                 catch  (Exception ignore) {
                 }
             }
             SSLContext sslContext = SSLContext.getInstance( "TLS" );
             TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
             trustManagerFactory.init(trustStore);
             KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance( "X509" );
             keyManagerFactory.init(keyStore, KEY_STORE_PASSWORD.toCharArray());
             sslContext.init(keyManagerFactory.getKeyManagers(), trustManagerFactory.getTrustManagers(),  null );
             return  sslContext;
         catch  (Exception e) {
             Log.e( "tag" , e.getMessage(), e);
         }
         return  null ;
     }
     
     /**     * 获取SSL认证需要的HttpClient     *     * @param context 上下文     * @return OkHttpClient     */
     public  static  OkHttpClient getSSLContextHttp(Context context) {
         OkHttpClient client =  new  OkHttpClient();
         SSLContext sslContext = getSSLContext(context);
         if  (sslContext !=  null ) {
             client.setSslSocketFactory(sslContext.getSocketFactory());
         }
         return  client;
     }
     
     /**     * 获取HttpsURLConnection     *     * @param context 上下文     * @param url     连接url     * @param method  请求方式     * @return HttpsURLConnection     */
     public  static  HttpsURLConnection getHttpsURLConnection(Context context, String url, String method) {
         URL u;
         HttpsURLConnection connection =  null ;
         try  {
             SSLContext sslContext = getSSLContext(context);
             if  (sslContext !=  null ) {
                 u =  new  URL(url);
                 connection = (HttpsURLConnection) u.openConnection();
                 connection.setRequestMethod(method); //"POST" "GET"
                 connection.setDoOutput( true );
                 connection.setDoInput( true );
                 connection.setUseCaches( false );
                 connection.setRequestProperty( "Content-Type" "binary/octet-stream" );
                 connection.setSSLSocketFactory(sslContext.getSocketFactory());
                 connection.setConnectTimeout( 30000 );
             }
         catch  (Exception e) {
             e.printStackTrace();
         }
         return  connection;
     }

参考:http://blog.csdn.net/u012874222/article/details/50339259


//使用https传输:

参考:http://blog.csdn.net/haoaoo/article/details/54614875

http://blog.csdn.net/dlx2chengyt/article/details/53081329 









本文转自yunlielai51CTO博客,原文链接:http://blog.51cto.com/4925054/1949641,如需转载请自行联系原作者


相关文章
|
3天前
|
网络协议 算法 数据建模
IP 地址,包括 IPv6 怎么申请 SSL证书来实现 https
很多企业单位已经开始在使用 IPv6 资源,跟 IPv4 一样,IPv6也是需要SSL证书的。在目前的SSL证书品牌,KeepTrust 是可以支持 IPv6 地址的。跟普通IP地址一样,给IPv6签发SSL证书也是需要验证申请者对 IP 地址的管理权限的。如果是 OV 版,还需要验证组织信息的真实性。
|
13天前
|
应用服务中间件 Linux 网络安全
nginx安装部署ssl证书,同时支持http与https方式访问
为了使HTTP服务支持HTTPS访问,需生成并安装SSL证书,并确保Nginx支持SSL模块。首先,在`/usr/local/nginx`目录下生成RSA密钥、证书申请文件及自签名证书。接着,确认Nginx已安装SSL模块,若未安装则重新编译Nginx加入该模块。最后,编辑`nginx.conf`配置文件,启用并配置HTTPS服务器部分,指定证书路径和监听端口(如20000),保存后重启Nginx完成部署。
203 7
|
2月前
|
安全 网络协议 应用服务中间件
内网ip申请SSL证书实现https访问
内网IP地址虽不能直接申请公网SSL证书,但可通过IP SSL证书保障数据安全。流程包括:确定固定内网IP,选择支持内网IP的CA,注册申请证书,生成CSR,验证IP所有权,下载部署证书至Web服务器,测试HTTPS访问,确保配置正确及证书有效。此方法适用于内网环境,提升数据传输安全性。
内网ip申请SSL证书实现https访问
|
2月前
|
安全 应用服务中间件 网络安全
49.3k star,本地 SSL 证书生成神器,轻松解决 HTTPS 配置痛点
mkcert是一款由Filippo Valsorda开发的免费开源工具,专为生成受信任的本地SSL/TLS证书而设计。它通过简单的命令自动生成并安装本地信任的证书,使本地环境中的HTTPS配置变得轻松无比。mkcert支持多个操作系统,已获得49.2K的GitHub Star,成为开发者首选的本地SSL工具。
168 10
|
2月前
|
存储 网络安全
Curl error (60): SSL peer certificate or SSH remote key was not OK for https://update.cs2c.com.cn/NS/V10/V10SP2/os/adv/lic/base/x86_64/repodata/repomd.xml [SSL: no alternative certificate subject name matches target host name 'update.cs2c.com.cn']
【10月更文挑战第30天】在尝试从麒麟软件仓库(ks10-adv-os)下载元数据时,遇到 SSL 证书验证问题。错误提示为:`Curl error (60): SSL peer certificate or SSH remote key was not OK`。可能原因包括证书不被信任、证书与域名不匹配或网络问题。解决方法包括检查网络连接、导入 SSL 证书、禁用 SSL 证书验证(不推荐)、联系仓库管理员、检查系统时间和尝试其他镜像。
418 1
|
2月前
|
安全 数据建模 应用服务中间件
如何给IP地址添加SSL证书(https)
为IP地址配置SSL证书实现HTTPS访问,需拥有固定公网IP,选择支持IP证书的CA,完成账户注册、证书申请、所有权验证及证书安装。验证过程涉及在服务器上放置特定文件,确保可访问。安装后需测试连接,注意兼容性和安全性,定期维护证书。
|
3月前
|
安全 应用服务中间件 Shell
nginx配置https的ssl证书和域名
nginx配置https的ssl证书和域名
|
3月前
|
Web App开发 网络安全
requests库的使用—SSL证书验证
requests库的使用—SSL证书验证
100 0
|
5月前
|
算法 Java 测试技术
java 访问ingress https报错javax.net.ssl.SSLHandshakeException: Received fatal alert: protocol_version
java 访问ingress https报错javax.net.ssl.SSLHandshakeException: Received fatal alert: protocol_version
|
5月前
|
安全 网络安全 Windows
【Azure App Service】遇见az命令访问HTTPS App Service 时遇见SSL证书问题,暂时跳过证书检查的办法
【Azure App Service】遇见az命令访问HTTPS App Service 时遇见SSL证书问题,暂时跳过证书检查的办法
【Azure App Service】遇见az命令访问HTTPS App Service 时遇见SSL证书问题,暂时跳过证书检查的办法