【问题标题】:Certificate Pinning fails even with CA-signed cert即使使用 CA 签名的证书,证书固定也会失败
【发布时间】:2018-01-03 07:38:21
【问题描述】:

我正在尝试通过 Retrofit 在 Android 上使用证书固定。我正在尝试评估有效的 Verisign 签名证书。

我收到以下错误:

HTTP FAILED: javax.net.ssl.SSLPeerUnverifiedException: 未能找到签署证书的可信证书。

为什么证书 pinner 不能针对设备的 CA 根证书进行评估?它是否无权访问设备信任?或者设备信任可能不包含整个证书链。但是为什么我的 SSL 通信没有失败呢?

// Pin Certificate
CertificatePinner certificatePinner = new CertificatePinner.Builder()
        .add("www.mydomain.com", "sha256/somerandompublickeystring")
        .build();

// To handle self-signed cert
OkHttpClient.Builder clientBuilder = new OkHttpClient.Builder();

OkHttpClient client = clientBuilder.connectTimeout(120, TimeUnit.SECONDS)
        .writeTimeout(120, TimeUnit.SECONDS)
        .readTimeout(120, TimeUnit.SECONDS)
        .certificatePinner(certificatePinner)
        .build();

【问题讨论】:

    标签: android retrofit okhttp certificate-pinning


    【解决方案1】:

    找到了答案。我可以获得如下所示的 Root 信任,并在 sslSocketFactory 调用中使用它。这对我有用。

    OkHttpClient client = clientBuilder.connectTimeout(120, TimeUnit.SECONDS)
            .writeTimeout(120, TimeUnit.SECONDS)
            .readTimeout(120, TimeUnit.SECONDS)
            .sslSocketFactory(getSystemDefaultSSLSocketFactory(app))
            .certificatePinner(certificatePinner)
            .build();
    
    private static SSLSocketFactory getSystemDefaultSSLSocketFactory(Application app) {
        SSLContext sslContext = null;
        try
        {
            TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(
                    TrustManagerFactory.getDefaultAlgorithm());
            trustManagerFactory.init((KeyStore) null);
            TrustManager[] trustManagers = trustManagerFactory.getTrustManagers();
            if (trustManagers.length != 1 || !(trustManagers[0] instanceof X509TrustManager)) {
                throw new IllegalStateException("Unexpected default trust managers:" + Arrays.toString(trustManagers));
            }
            sslContext = SSLContext.getInstance("TLS");
            sslContext.init(null, trustManagers, null);
    
        }
        catch(Exception ex)
        {
            Log.e("TAG",ex.getMessage());
        }
        return sslContext.getSocketFactory();
    

    }

    【讨论】:

    • 考虑为 OkHttp 提供证书固定器和信任管理器。否则 OkHttp 需要通过反射查找信任管理器,这会比较慢。
    • 经过数小时的尝试解决这个问题后,这对我有用!谢谢
    • 取自 OkHttp 规范,您可以获得更多上下文:square.github.io/okhttp/4.x/okhttp/okhttp3/-ok-http-client/…
    猜你喜欢
    • 2013-08-30
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-09-13
    • 2020-07-30
    相关资源
    最近更新 更多