OKHTTP 默认构建包含 android 4.4 的TLS 1.2 以及设备时间不对兼容

kotlin

Kotlin 复制代码
import android.annotation.SuppressLint
import android.os.Build
import java.net.Proxy
import java.security.KeyStore
import java.security.SecureRandom
import java.security.cert.CertPathValidatorException
import java.security.cert.CertificateException
import java.security.cert.CertificateExpiredException
import java.security.cert.CertificateNotYetValidException
import java.security.cert.X509Certificate
import java.util.concurrent.TimeUnit
import javax.net.ssl.SSLContext
import javax.net.ssl.SSLSocketFactory
import javax.net.ssl.TrustManager
import javax.net.ssl.TrustManagerFactory
import javax.net.ssl.X509TrustManager
import okhttp3.OkHttpClient
import okhttp3.Request

/**
 * <pre>
 * desc   : 用于统一创建okhttp
</pre> *
 */
object DefOkHttp {
    /**
     * 获取默认的OKhttp实例
     * @return
     * 默认的全局okhttp类
     */
    @JvmStatic
    val default: OkHttpClient = newOkHttpClient(OkHttpClient.Builder())

    @JvmStatic
    val defaultHeaders = hashMapOf<String, String>()

    init {
        defaultHeaders["x-brand"] = checkHeaderValue(Build.MODEL, "unknown")
        defaultHeaders["x-platform"] = "android" + Build.VERSION.SDK_INT
    }

    /**
     * 由于okhttp header不容许携带中文,故
     */
    private fun checkHeaderValue(value: String?, defaultVal: String): String {
        if (value == null) {
            return defaultVal
        } else {
            for (c in value) {
                if (c.code <= 31 && c != '\t' || c.code >= 127) {
                    return defaultVal
                }
            }
            return value
        }
    }

    @Synchronized
    @JvmStatic
    fun addDefaultHeader(key: String, value: String) {
        defaultHeaders[key] = value
    }

    /**
     * 创建一个新的okHttp实例
     * @param builder
     * okhttp构建者
     * @return
     * okhttp实例
     */
    @Synchronized
    @JvmStatic
    fun newOkHttpClient(builder: OkHttpClient.Builder): OkHttpClient {
        return patchDefaultBuilder(builder).build()
    }

    /**
     * 创建一个默认的builder类
     * @return
     * 默认的okhttp builder类
     */
    @Synchronized
    @JvmStatic
    fun newDefaultBuilder(): OkHttpClient.Builder {
        return patchDefaultBuilder(OkHttpClient.Builder())
    }

    /**
     * 增加默认配置
     * @param builder:外部构建者
     */
    @JvmStatic
    private fun patchDefaultBuilder(builder: OkHttpClient.Builder): OkHttpClient.Builder {
        patchSSL(builder)
        builder.proxy(Proxy.NO_PROXY)

        // 增加固定header
        builder.addInterceptor { chain ->
            val originalRequest: Request = chain.request()

            // 使用newBuilder复制原始请求
            val newRequestBuilder: Request.Builder = originalRequest.newBuilder()

            // 添加固定的header头部
            for ((key, value) in defaultHeaders) {
                newRequestBuilder.addHeader(key, value)
            }

            // 构建新的请求
            val newRequest: Request = newRequestBuilder.build()

            // 继续处理请求
            chain.proceed(newRequest)
        }

        return builder
    }

    @Suppress("CAST_NEVER_SUCCEEDS")
    private fun patchSSL(builder: OkHttpClient.Builder): OkHttpClient.Builder {
        try {
            val tmfAlgorithm = TrustManagerFactory.getDefaultAlgorithm()
            val tmf = TrustManagerFactory.getInstance(tmfAlgorithm)
            val socketFactory: SSLSocketFactory
            tmf.init(null as? KeyStore)
            val sslContext = SSLContext.getInstance("TLS")
            val trustManagers = tmf.trustManagers
            val x509TrustManager = trustManagers[0] as X509TrustManager
            sslContext.init(null, getWrappedTrustManagers(trustManagers), SecureRandom())
            socketFactory = if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT_WATCH) {
                val socketFactoryNoTls = sslContext.socketFactory
                TLS12SocketFactory(socketFactoryNoTls)
            } else {
                sslContext.socketFactory
            }
            builder.connectTimeout(8, TimeUnit.SECONDS)
                .readTimeout(8, TimeUnit.SECONDS)
                .writeTimeout(8, TimeUnit.SECONDS)
                .sslSocketFactory(socketFactory, x509TrustManager)
                .followRedirects(true)
                .followSslRedirects(true)
                .retryOnConnectionFailure(true)
                .build()
        } catch (_: Exception) {
            builder.connectTimeout(8, TimeUnit.SECONDS)
                .readTimeout(8, TimeUnit.SECONDS)
                .writeTimeout(8, TimeUnit.SECONDS)
                .followRedirects(true)
                .followSslRedirects(true)
                .retryOnConnectionFailure(true)
        }

        return builder
    }

    @SuppressLint("CustomX509TrustManager")
    private fun getWrappedTrustManagers(trustManagers: Array<TrustManager>): Array<TrustManager> {
        val originalTrustManager = trustManagers[0] as X509TrustManager
        return arrayOf(
            object : X509TrustManager {
                override fun getAcceptedIssuers(): Array<X509Certificate> {
                    return originalTrustManager.acceptedIssuers
                }

                @Throws(CertificateException::class)
                override fun checkClientTrusted(certs: Array<X509Certificate>, authType: String) {
                    try {
                        originalTrustManager.checkClientTrusted(certs, authType)
                    } catch (e: CertificateException) {
                        if (e.cause is CertificateNotYetValidException ||
                            e.cause?.cause is CertificateNotYetValidException ||
                            e.cause is CertPathValidatorException ||
                            e.cause ?.cause is CertPathValidatorException ||
                            e.cause is CertificateExpiredException ||
                            e.cause ?.cause is CertificateExpiredException
                        ) {
                            Log.d("checkClientTrusted post CertificateNotYetValidEvent----")
                        } else {
                            throw e
                        }
                    }
                }

                @Throws(CertificateException::class)
                override fun checkServerTrusted(certs: Array<X509Certificate>, authType: String) {
                    try {
                        originalTrustManager.checkServerTrusted(certs, authType)
                    } catch (e: CertificateException) {
                        if (e.cause is CertificateNotYetValidException ||
                            e.cause?.cause is CertificateNotYetValidException ||
                            e.cause is CertPathValidatorException ||
                            e.cause ?.cause is CertPathValidatorException ||
                            e.cause is CertificateExpiredException ||
                            e.cause ?.cause is CertificateExpiredException
                        ) {
                            Log.d("checkServerTrusted post CertificateNotYetValidEvent----")
                        } else {
                            throw e
                        }
                    }
                }
            })
    }
}

使用:

Kotlin 复制代码
DefOkHttp.newDefaultBuilder()

旧版JAVA

java 复制代码
import android.os.Build;
import android.util.Log;


import java.security.KeyStore;
import java.security.SecureRandom;
import java.security.cert.CertPathValidatorException;
import java.security.cert.CertificateException;
import java.security.cert.CertificateExpiredException;
import java.security.cert.CertificateNotYetValidException;
import java.security.cert.X509Certificate;

import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSocketFactory;
import javax.net.ssl.TrustManager;
import javax.net.ssl.TrustManagerFactory;
import javax.net.ssl.X509TrustManager;

public class CertImpl {

    public long getTimeInMs() {
        return System.currentTimeMillis();
    }

    public SSLSocketFactory getCertImpl() {
        try {
            SSLSocketFactory socketFactory = null;

            String tmfAlgorithm = TrustManagerFactory.getDefaultAlgorithm();
            TrustManagerFactory tmf = TrustManagerFactory.getInstance(tmfAlgorithm);
            tmf.init((KeyStore) null);
            SSLContext sslContext = SSLContext.getInstance("TLS");
            TrustManager[] trustManagers = tmf.getTrustManagers();
            sslContext.init(null, getWrappedTrustManagers(trustManagers), new SecureRandom());
            if(Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT_WATCH){
                socketFactory = new Tls12SocketFactory(sslContext.getSocketFactory());
            }else{
                socketFactory = sslContext.getSocketFactory();
            }
            return socketFactory;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    private static TrustManager[] getWrappedTrustManagers(TrustManager[] trustManagers) {
        final X509TrustManager originalTrustManager = (X509TrustManager) trustManagers[0];
        return new TrustManager[]{ new X509TrustManager() {
            public X509Certificate[] getAcceptedIssuers() {
                return originalTrustManager.getAcceptedIssuers();
            }

            public void checkClientTrusted(X509Certificate[] certs, String authType) throws CertificateException {
                try {
                    originalTrustManager.checkClientTrusted(certs, authType);
                } catch (CertificateException e) {
                    if (e.getCause() instanceof CertificateNotYetValidException
                            || (e.getCause() != null && e.getCause().getCause() instanceof CertificateNotYetValidException)
                            || (e.getCause() instanceof CertPathValidatorException)
                            || e.getCause() instanceof CertificateExpiredException
                            || (e.getCause() != null && e.getCause().getCause() instanceof CertificateExpiredException)) {
                        Log.i("checkServerTrusted post CertificateNotYetValidEvent----");
                    } else {
                        throw e;
                    }
                }
            }

            public void checkServerTrusted(X509Certificate[] certs, String authType) throws CertificateException {
                try {
                    originalTrustManager.checkServerTrusted(certs, authType);
                } catch (CertificateException e) {
                    if (e.getCause() instanceof CertificateNotYetValidException
                            || (e.getCause() != null && e.getCause().getCause() instanceof CertificateNotYetValidException)
                            || (e.getCause() instanceof CertPathValidatorException)
                            || e.getCause() instanceof CertificateExpiredException
                            || (e.getCause() != null && e.getCause().getCause() instanceof CertificateExpiredException)) {
                        Log.d("CertImpl", "checkServerTrusted post CertificateNotYetValidEvent--");
                    } else {
                        throw e;
                    }
                }
            }
        }};
    }

}

使用:

java 复制代码
    public static OkHttpClient getClient() {
        if (mOkHttpClient == null) {
            okhttp3.OkHttpClient.Builder clientBuilder = getBuilder();
            try {
                SSLSocketFactory factory = new CertImpl().getCertImpl();
                clientBuilder.sslSocketFactory(factory);
            }catch (Exception e){
                e.printStackTrace();
            }
            mOkHttpClient = clientBuilder.build();
        }
        return mOkHttpClient;
    }
相关推荐
落魄Android在线炒饭18 小时前
Android 自定义HAL开发篇之 HIDL篇——从入门到实战(上)
android
plainGeekDev19 小时前
广播接收器 → Flow + Lifecycle
android·java·kotlin
plainGeekDev19 小时前
EventBus → SharedFlow
android·java·kotlin
37手游移动客户端团队2 天前
招聘-高级安卓开发工程师
android·客户端
用户41659673693552 天前
WebView 请求异常排查操作手册
android·前端
Kapaseker2 天前
学不动了,入门 Compose Styles API
android·kotlin
墨狂之逸才3 天前
Android TV WebView 遥控器按键处理:从全透传到白名单
android
plainGeekDev3 天前
MVC 写法 → MVVM
android·java·kotlin
恋猫de小郭3 天前
Flutter Patchwork,不用 Fork 改依赖包源码的第三方工具
android·前端·flutter
三少爷的鞋3 天前
“结构化”这个词,本质上就是——把混乱的东西变成有组织、有规则、有边界的东西
android