Android开发Android开发经验谈Android技术知识

Android CA证书库BKS制作、读取和使用

2018-09-14  本文已影响10人  Android技术分享

前言

通常情况下,安卓使用Https与服务器通信时,项目中默认是不用内置CA根证的。
因为手机上默认内置了所有主流的CA根证,我们可以在设置 --> 安全 -->授信的证书列表查看。


image.png

如果我们在设置中,手动停用了服务器对应的根证,再与服务器通信,就会报SSL异常。如果通过浏览器访问服务端的网页,也会弹出类似的提示:未授信的网站。
实际上,用户一般不清楚手机中授信的列表是什么意思,也不会随便禁用根证的。

但是也有很多软件还是选择将CA证书存放在项目本地,一起打包到apk中,原因:
1.怕用户手动禁用掉手机授信列表中的CA根证
2.怕中间人攻击(与CA厂商沟通,这种事情一般不会发生的,如果真被黑客拦截,基本上也破解不了,否则,证书也就没安全性可言了)
3.使用了非官方CA机构颁发的证书(如自己制作的证书)

那么,安卓APP中如何内置服务器对应的根证呢?
在安卓中,通常使用.bks格式的证书库来放置证书,好处是:我们可以将1~多个如.crt格式的CA根证存入bks库中,当与服务器https通信时,会自动匹配、使用bks库中合适的证书。(举例,直白一点:如果bks中内置了a、b两个根证,服务器默认使用a进行通信,当a快过期时,服务器SSL配置切换成未过期的根证b,这时候app不用做任何调整,也不用升级的)

本期,我们就来了解下,bks证书如何制作、读取和使用的。

Keytool 环境

Java的 Keytool工具可以用来制作bks。
环境的配置以mac系统为例(windows类似):
1.安装java环境, 配置环境变量(我下的是jdk1.8.0_20);
2.下载bcprov-ext-jdk15on-151.jar;
3.将jar文件拷贝到\Java\jdk1.8.0_20\jre\lib\ext;
打开终端, 输入keytool, 点击回车键, 如果显示下图效果, 则环境配置正常。


image.png

创建/导入根证书到bks库

要导入的根证类型一般是.cer或者.crt。如果要导入多个根证,执行多次下面的命令即可。

命令:
keytool -importcert -v -trustcacerts -alias myalas1 -file VeriSign.cer -keystore mytrustcerts.bks -storetype BKS -provider org.bouncycastle.jce.provider.BouncyCastleProvider -storepass '123456'

注意:
-alias别名:请修改成自己的别名
-file VeriSign.cer:指要导入的根证文件,请修改成自己的根证文件名称
-keystore:本地bks证书库文件,修改成自己的证书库文件名称,如果文件不存在,会自动创建
-storepass:证书库密码

image.png
image.png

查看bks证书库列表

命令:
keytool -list -rfc -keystore mytrustcerts.bks -storetype BKS -provider org.bouncycastle.jce.provider.BouncyCastleProvider -storepass '123456'
image.png

从bks证书库中导出证书

命令:
keytool -export -alias myalas1 -file VeriSign.cer -keystore mytrustcerts.bks -storetype BKS -provider org.bouncycastle.jce.provider.BouncyCastleProvider -storepass '123456'
image.png
image.png

查看单个证书信息(CA机构、md5、sha、有效期等)

命令:
keytool -printcert -file VeriSign_temp.cer
image.png

代码中的应用:

OKHttp为例:
1.将*.bks文件放在项目的本地,可以是raw下,也可以是assets下
2.编写代码:

public class MyOkhttpClient {
    private static MyOkhttpClient singleton;
    public static OkHttpClient getInstance() {
        if (singleton == null) {
            synchronized (CBOkhttpClient.class) {
                if (singleton == null) {
                    OkHttpClient.Builder builder = new OkHttpClient().newBuilder();
                    builder.connectTimeout(50000, TimeUnit.MILLISECONDS);
                    builder.writeTimeout(50000, TimeUnit.MILLISECONDS);
                    builder.readTimeout(50000, TimeUnit.MILLISECONDS);
                    try {
                        SSLContext sslContext = SSLContext.getInstance("TLS");
                        TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
                        KeyStore trustStore = KeyStore.getInstance("BKS");
                        InputStream ksinstream = CBFramework.getApplication().getResources().openRawResource(R.raw.cbframework_trustcerts);
                        trustStore.load(ksinstream, "".toCharArray());
                        ksinstream.close();
                        trustManagerFactory.init(trustStore);
                        sslContext.init(null, trustManagerFactory.getTrustManagers(), new SecureRandom());
                        builder.sslSocketFactory(sslContext.getSocketFactory(), new X509TrustManager() {
                            @Override
                            public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
                            }

                            @Override
                            public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
                            }

                            @Override
                            public X509Certificate[] getAcceptedIssuers() {
                                return new X509Certificate[0];
                            }
                        });
                    } catch (Exception e) {
                        CBLogger.t(e);
                    }
                    singleton = builder.build();
                }
            }
        }
        return singleton;
    }
}
上一篇 下一篇

猜你喜欢

热点阅读