美文网首页Android进阶
OkHttp加载https的配置

OkHttp加载https的配置

作者: poweru | 来源:发表于2018-01-29 17:30 被阅读98次

    https简介

    现在网络安全越来越受到重视,https作为更加安全的通信协议应用也越来越广泛。https相对与http有以下优势:

    • 认证用户和服务器,确保数据发送到正确的客户机和服务器;(验证证书)
    • 加密数据以防止数据中途被窃取;(加密)
    • 维护数据的完整性,确保数据在传输过程中不被改变。(摘要算法)

    对于一个android开发来说,目前的网络请求框架大部分都是使用okhttp进行网络请求的,okhttputils,或者retrofit, 所以了解okhttp是如何加载https请求的对于我们平时开发有很大的帮助。

    访问自签名的网站

    首先我们将.cer证书文件放到assets文件夹下,其实你可以随便放哪,反正能读取到就行。
    然后在我们的OkHttpClientManager里面添加如下的方法:

    public void setCertificates(InputStream... certificates) {
        try {
            CertificateFactory certificateFactory = CertificateFactory.getInstance("X.509");
            KeyStore keyStore = KeyStore.getInstance(KeyStore.getDefaultType());
            keyStore.load(null);
            int index = 0;
            for (InputStream certificate : certificates) {
                String certificateAlias = Integer.toString(index++);
                keyStore.setCertificateEntry(certificateAlias, certificateFactory.generateCertificate(certificate));
    
                try {
                    if (certificate != null)
                        certificate.close();
                } catch (IOException e) {
                }
            }
    
            SSLContext sslContext = SSLContext.getInstance("TLS");
    
            TrustManagerFactory trustManagerFactory =
                    TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
    
            trustManagerFactory.init(keyStore);
            sslContext.init
                    (
                            null,
                            trustManagerFactory.getTrustManagers(),
                            new SecureRandom()
                    );
            //.setSslSocketFactory(sslContext.getSocketFactory());
    
    
        } catch (Exception e) {
            e.printStackTrace();
        }
    
    }
    

    为了代码可读性,我把异常捕获的部分简化了,可以看到我们提供了一个方法传入InputStream流,InputStream就对应于我们证书的输入流。

    代码内部,我们:

    • 构造CertificateFactory对象,通过它的generateCertificate(is)方法得到Certificate。
    • 然后讲得到的Certificate放入到keyStore中。
    • 接下来利用keyStore去初始化我们的TrustManagerFactory
    • trustManagerFactory.getTrustManagers获得TrustManager[]初始化我们的SSLContext
    • 最后,设置我们mOkHttpClient.setSslSocketFactory即可。

    这样就完成了我们代码的编写,其实挺短的,当客户端进行SSL连接时,就可以根据我们设置的证书去决定是否新人服务端的证书。

    记得在Application中进行初始化:

    public class MyApplication extends Application
    {
     @Override
      public void onCreate()
      {
        super.onCreate();
    
        try
        {
            OkHttpClientManager.getInstance()
                    .setCertificates(getAssets().open("srca.cer"));
        } catch (IOException e)
        {
            e.printStackTrace();
        }
    
      }
    

    然后我们就可以尝试访问自签名的htpps服务器接口了。

    使用字符串替代证书

    下面继续,有些人可能觉得把证书copy到assets下还是觉得不舒服,其实我们还可以将证书中的内容提取出来,写成字符串常量,这样就不需要证书根据着app去打包了。

    图片.png

    使用keytool命令,以rfc样式输出。keytool命令是JDK里面自带的。

    有了这个字符串以后,我们就不需要srca.cer这个文件了,直接编写以下代码:

    public class MyApplication extends Application
    {
    private String CER_12306 = "-----BEGIN CERTIFICATE-----\n" +
            "MIICmjCCAgOgAwIBAgIIbyZr5/jKH6QwDQYJKoZIhvcNAQEFBQAwRzELMAkGA1UEBhMCQ04xKTAn\n" +
            "BgNVBAoTIFNpbm9yYWlsIENlcnRpZmljYXRpb24gQXV0aG9yaXR5MQ0wCwYDVQQDEwRTUkNBMB4X\n" +
            "DTA5MDUyNTA2NTYwMFoXDTI5MDUyMDA2NTYwMFowRzELMAkGA1UEBhMCQ04xKTAnBgNVBAoTIFNp\n" +
            "bm9yYWlsIENlcnRpZmljYXRpb24gQXV0aG9yaXR5MQ0wCwYDVQQDEwRTUkNBMIGfMA0GCSqGSIb3\n" +
            "DQEBAQUAA4GNADCBiQKBgQDMpbNeb34p0GvLkZ6t72/OOba4mX2K/eZRWFfnuk8e5jKDH+9BgCb2\n" +
            "9bSotqPqTbxXWPxIOz8EjyUO3bfR5pQ8ovNTOlks2rS5BdMhoi4sUjCKi5ELiqtyww/XgY5iFqv6\n" +
            "D4Pw9QvOUcdRVSbPWo1DwMmH75It6pk/rARIFHEjWwIDAQABo4GOMIGLMB8GA1UdIwQYMBaAFHle\n" +
            "tne34lKDQ+3HUYhMY4UsAENYMAwGA1UdEwQFMAMBAf8wLgYDVR0fBCcwJTAjoCGgH4YdaHR0cDov\n" +
            "LzE5Mi4xNjguOS4xNDkvY3JsMS5jcmwwCwYDVR0PBAQDAgH+MB0GA1UdDgQWBBR5XrZ3t+JSg0Pt\n" +
            "x1GITGOFLABDWDANBgkqhkiG9w0BAQUFAAOBgQDGrAm2U/of1LbOnG2bnnQtgcVaBXiVJF8LKPaV\n" +
            "23XQ96HU8xfgSZMJS6U00WHAI7zp0q208RSUft9wDq9ee///VOhzR6Tebg9QfyPSohkBrhXQenvQ\n" +
            "og555S+C3eJAAVeNCTeMS3N/M5hzBRJAoffn3qoYdAO1Q8bTguOi+2849A==\n" +
            "-----END CERTIFICATE-----";
    
    @Override
    public void onCreate()
    {
        super.onCreate();
    
        OkHttpClientManager.getInstance()
                .setCertificates(new Buffer()
                        .writeUtf8(CER_12306)
                        .inputStream());
    }
    

    注意Buffer是okio包下的,okhttp依赖okio。

    ok,这样就省去将cer文件一起打包进入apk了。

    参考

    相关文章

      网友评论

        本文标题:OkHttp加载https的配置

        本文链接:https://www.haomeiwen.com/subject/hxwxzxtx.html