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;
    }
相关推荐
黄林晴17 小时前
Android 17 Beta 2,隐私这把锁又拧紧了
android
Kapaseker17 小时前
研究表明,开发者对Kotlin集合的了解不到 20%
android·kotlin
bqliang17 小时前
Compose 媒体查询 (Media Query API) 🖱️👇🕹️
android·android jetpack
程序员陆业聪1 天前
Android 平台 AI Agent 技术架构深度解析
android·人工智能
BD_Marathon1 天前
工厂方法模式
android·java·工厂方法模式
王码码20351 天前
Flutter for OpenHarmony:socket_io_client 实时通信的事实标准(Node.js 后端的最佳拍档) 深度解析与鸿蒙适配指南
android·flutter·ui·华为·node.js·harmonyos
勇气要爆发1 天前
吴恩达《LangChain LLM 应用开发精读笔记》2-Models, Prompts and Parsers 模型、提示和解析器
android·笔记·langchain
Railshiqian1 天前
给android源码下的模拟器添加两个后排屏的修改
android·开发语言·javascript
恋猫de小郭1 天前
Flutter 的真正价值是什么?深度解析再结合鸿蒙,告诉你 Flutter 的真正优势
android·前端·flutter
Ehtan_Zheng1 天前
我们如何在不减少功能的前提下,将安卓应用体积缩减 60%
android