美文网首页Android零散知识
几行代码轻松搞定Retrofit 2.0 https的支持

几行代码轻松搞定Retrofit 2.0 https的支持

作者: 丶叫我官人 | 来源:发表于2017-08-11 13:04 被阅读192次

网上或许零零散散有解决Retrofit https的问题,现在做个总结吧
首先初始化证书,你需要一个httpsUtils,代码如下

    import java.io.IOException;
    import java.io.InputStream;
    import java.security.KeyManagementException;
    import java.security.KeyStore;
    import java.security.KeyStoreException;
    import java.security.NoSuchAlgorithmException;
    import java.security.UnrecoverableKeyException;
    import java.security.cert.CertificateException;
    import java.security.cert.CertificateFactory;
    import java.security.cert.X509Certificate;
    
    import javax.net.ssl.HostnameVerifier;
    import javax.net.ssl.KeyManager;
    import javax.net.ssl.KeyManagerFactory;
    import javax.net.ssl.SSLContext;
    import javax.net.ssl.SSLSession;
    import javax.net.ssl.SSLSocketFactory;
    import javax.net.ssl.TrustManager;
    import javax.net.ssl.TrustManagerFactory;
    import javax.net.ssl.X509TrustManager;
    
    /**
     * describe https 绑定证书工具类
     *
     * @author emilia.Zhang
     * @time 2017/8/11 9:43.
     */
    public class HttpUtils {
        public static class SSLParams
        {
            public SSLSocketFactory sSLSocketFactory;
            public X509TrustManager trustManager;
        }
    
        public static SSLParams getSslSocketFactory(InputStream[] certificates, InputStream bksFile, String password)
        {
            SSLParams sslParams = new SSLParams();
            try
            {
                TrustManager[] trustManagers = prepareTrustManager(certificates);
                KeyManager[] keyManagers = prepareKeyManager(bksFile, password);
                SSLContext sslContext = SSLContext.getInstance("TLS");
                X509TrustManager trustManager = null;
                if (trustManagers != null)
                {
                    trustManager = new MyTrustManager(chooseTrustManager(trustManagers));
                } else
                {
                    trustManager = new UnSafeTrustManager();
                }
                sslContext.init(keyManagers, new TrustManager[]{trustManager},null);
                sslParams.sSLSocketFactory = sslContext.getSocketFactory();
                sslParams.trustManager = trustManager;
                return sslParams;
            } catch (NoSuchAlgorithmException e)
            {
                throw new AssertionError(e);
            } catch (KeyManagementException e)
            {
                throw new AssertionError(e);
            } catch (KeyStoreException e)
            {
                throw new AssertionError(e);
            }
        }
    
        private class UnSafeHostnameVerifier implements HostnameVerifier
        {
            @Override
            public boolean verify(String hostname, SSLSession session)
            {
                return true;
            }
        }
    
        private static class UnSafeTrustManager implements 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 java.security.cert.X509Certificate[]{};
            }
        }
    
        private static TrustManager[] prepareTrustManager(InputStream... certificates)
        {
            if (certificates == null || certificates.length <= 0) return null;
            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)
    
                    {
                    }
                }
                TrustManagerFactory trustManagerFactory = null;
    
                trustManagerFactory = TrustManagerFactory.
                        getInstance(TrustManagerFactory.getDefaultAlgorithm());
                trustManagerFactory.init(keyStore);
    
                TrustManager[] trustManagers = trustManagerFactory.getTrustManagers();
    
                return trustManagers;
            } catch (NoSuchAlgorithmException e)
            {
                e.printStackTrace();
            } catch (CertificateException e)
            {
                e.printStackTrace();
            } catch (KeyStoreException e)
            {
                e.printStackTrace();
            } catch (Exception e)
            {
                e.printStackTrace();
            }
            return null;
    
        }
    
        private static KeyManager[] prepareKeyManager(InputStream bksFile, String password)
        {
            try
            {
                if (bksFile == null || password == null) return null;
    
                KeyStore clientKeyStore = KeyStore.getInstance("BKS");
                clientKeyStore.load(bksFile, password.toCharArray());
                KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
                keyManagerFactory.init(clientKeyStore, password.toCharArray());
                return keyManagerFactory.getKeyManagers();
    
            } catch (KeyStoreException e)
            {
                e.printStackTrace();
            } catch (NoSuchAlgorithmException e)
            {
                e.printStackTrace();
            } catch (UnrecoverableKeyException e)
            {
                e.printStackTrace();
            } catch (CertificateException e)
            {
                e.printStackTrace();
            } catch (IOException e)
            {
                e.printStackTrace();
            } catch (Exception e)
            {
                e.printStackTrace();
            }
            return null;
        }
    
        private static X509TrustManager chooseTrustManager(TrustManager[] trustManagers)
        {
            for (TrustManager trustManager : trustManagers)
            {
                if (trustManager instanceof X509TrustManager)
                {
                    return (X509TrustManager) trustManager;
                }
            }
            return null;
        }
    
    
        private static class MyTrustManager implements X509TrustManager
        {
            private X509TrustManager defaultTrustManager;
            private X509TrustManager localTrustManager;
    
            public MyTrustManager(X509TrustManager localTrustManager) throws NoSuchAlgorithmException, KeyStoreException
            {
                TrustManagerFactory var4 = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
                var4.init((KeyStore) null);
                defaultTrustManager = chooseTrustManager(var4.getTrustManagers());
                this.localTrustManager = localTrustManager;
            }
    
    
            @Override
            public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException
            {
    
            }
    
            @Override
            public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException
            {
                try
                {
                    defaultTrustManager.checkServerTrusted(chain, authType);
                } catch (CertificateException ce)
                {
                    localTrustManager.checkServerTrusted(chain, authType);
                }
            }
    
    
            @Override
            public X509Certificate[] getAcceptedIssuers()
            {
                return new X509Certificate[0];
            }
        }
    
    }

然后在你application初始化

  public class AppApplication extends Application {
  public static HttpUtils.SSLParams sslParams;
        @Override
        public void onCreate() {
            super.onCreate();
            sslParams = HttpUtils.getSslSocketFactory(new InputStream[]{
                    getResources().openRawResource(R.raw.inpet)}, null, null);
    
        }
    }

接着就是配置你请求,Retrofit 其实就是常规配置,不做讲解,主要是okhttp的配置

// 证书文件放在了res/raw/目录下
HttpUtils.SSLParams sslParams = HttpUtils.getSslSocketFactory(new InputStream[]{
                   context.getResources().openRawResource(R.raw.inpet)}, null, null);
// 接下来给okhttp绑定证书
OkHttpClient.Builder builder = new OkHttpClient.Builder();
builder.sslSocketFactory(sslParams.sSLSocketFactory, sslParams.trustManager).build();

到此就已经成功搞定了Retrofit 2.0 https的请求

可能会有小伙伴本地测试,服务器给你证书HostName不对,返回域名验证不过,报错

Android OKHttp https java.io.IOException: Hostname was not verified

别慌,现在有两种解决办法:
1.把服务器端打一顿,然后让他重新生成服务器的证书,用真实的本地域名信息
2.不理那个菜逼,自己解决,办法如下:

// 只需要在build多加一个选项,verify默认返回的是false,只要强制返回true 搞定
OkHttpClient.Builder builder = new OkHttpClient.Builder();
builder.sslSocketFactory(sslParams.sSLSocketFactory, sslParams.trustManager)
       .hostnameVerifier(new HostnameVerifier() {
                @Override
                public boolean verify(String hostname, SSLSession session) {
                    //强行返回true,忽略HostName验证 即验证成功
                    return true;
                }
            }).build();

到此问题已完美解决!!!

相关文章

网友评论

    本文标题:几行代码轻松搞定Retrofit 2.0 https的支持

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