关于 Charles 出现 unknown 的一点探索

用 Charles 截取一些 Android 和 iOS应用的请求数据,发现好多 https 的请求无线显示内容了,只是显示个红色的 unknown

这是因为 Android 从 6.0 之后加强了系统安全性,ssl 证书的验证由系统级别精确到了应用级别,所以即使安装了 Charles 的根证书依然无法看到 https 的内容

但 iOS 还没有像 Android 这么做,但也有一些应用无法看到 https 了,一番搜索后了解到原来有些应用采用了 SSL Pinning 的技术,简单来说就是在建立 ssl 连接的时候对证书做了验证,如果发现证书和请求的域名不匹配的时候就拒绝连接了,所以看到的 unkown 其实是说明这次请求根本没有发送成功,这就牵涉到了 ssl 三次握手的一些知识,可以理解为在建立 ssl 连接的时候服务器和客户端之间会进行几次身份验证,采用的是 RSA 加密, 其中私钥在服务器容器中配置好了,证书信息(包含公钥)会在客户端请求建立 https 连接的时候拿到,至于具体的握手建立连接的过程这里就不详谈了,有兴趣的可以去看其他同学的文章

既然了解了问题的原因,我们可以在本地进行一个简单的测试来验证一下,用 java写一个http客户端和线上的 https 服务建立连接,然后验证服务器公钥,就拿https://www.baidu.com来测试吧,首先我们可以先看一下百度的 https 证书,通过 chrome 访问https://www.baidu.com,点击地址栏左上角的小锁,然后点击证书,就能看到证书信息了

image

接下来我们把证书拖到一个文件夹里,这样就获取到一个 .cer 格式的证书文件,这个证书包含了很多信息,我们主要是要取到证书里的公钥信息,通过 openssl 工具来获取,命令如下

openssl x509 -inform der -in /xxx/xxx.cer -pubkey -noout > public_key.pem

这样就把证书里的公钥导出来了,文件打开后就是下面这个样子:

-----BEGIN PUBLIC KEY-----
MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtMa/2lMgD+pA87hSF2Y7
NgGNErSZDdObbBhTsRkIsPpzRz4NOnlieGEuVDxJfFbawL5hVdVCcGoQvvW9jWSW
IQCTYwmHtxm6DiA+SchT7QKPRgHroQeTc7vt8bPJ4vvd8Dkqg630QZi8huq6dKim
49DlxY6zC7LSrJF0Dv+AECM2YmUItIf1VwwlxwDY9ahduDNBpypf2/pwniG7rkIW
Zgdp/hwmKoEPq3Pj1lIgpG2obNRmSKRv8mgKxWWhTr8EekBDHNN1+3WsGdZKNQVu
z9Vl0UTKawxYBMSFTx++LDLR8cYo+/kmNrVt+suWoqDQvPhR3wdEvY9vZ8DUr9nN
wwIDAQAB
-----END PUBLIC KEY-----

这个就是百度证书的公钥了,然后我们接下来写一个 http 客户端

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.NoSuchProviderException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;

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

import com.bedrock.util.encrypt.Base64;

/**
 * HttpKit
 */
public class HttpsTest {

    private static final SSLSocketFactory sslSocketFactory = initSSLSocketFactory();

    private static SSLSocketFactory initSSLSocketFactory() {
        try {
            TrustManager[] tm = { new HttpsTest().new OptionTrustManager() };
            SSLContext sslContext = SSLContext.getInstance("TLS", "SunJSSE");
            sslContext.init(null, tm, new java.security.SecureRandom());
            return sslContext.getSocketFactory();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * https 证书管理
     */
    private class OptionTrustManager implements X509TrustManager {
        @Override
        public X509Certificate[] getAcceptedIssuers() {
            return null;
        }

        @Override
        public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
        }

        @Override
        public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
            for (X509Certificate x509Certificate : chain) {
                String principal = x509Certificate.getSubjectX500Principal().getName();
                String pubkey = new String(Base64.encode(x509Certificate.getPublicKey().getEncoded(), Base64.DEFAULT));
                if (principal.contains("baidu.com")) {
                    if (!pubkey.contains("MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtMa/2lMgD+p")) {
                        throw new CertificateException("error public key");
                    }
                }
                System.out.println("domain:" + principal + "    public key:\n" + pubkey);
            }
        }
    }

    public static void main(String[] args) {
        String responseStr = get("https://www.baidu.com");
        System.out.println(responseStr.length());
    }

    public static HttpURLConnection getHttpConnection(String url)
            throws IOException, NoSuchAlgorithmException, NoSuchProviderException, KeyManagementException {
        URL _url = new URL(url);
        HttpURLConnection conn = (HttpURLConnection) _url.openConnection();
        if (conn instanceof HttpsURLConnection) {
            ((HttpsURLConnection) conn).setSSLSocketFactory(sslSocketFactory);
        }

        conn.setRequestMethod("GET");
        conn.setDoOutput(true);
        conn.setDoInput(true);

        conn.setConnectTimeout(30000);
        conn.setReadTimeout(30000);

        conn.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");
        conn.setRequestProperty("User-Agent",
                "Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.146 Safari/537.36");

        return conn;
    }

    /**
     * Send GET request
     */
    public static String get(String url) {
        HttpURLConnection conn = null;
        try {
            conn = getHttpConnection(url);
            conn.connect();
            return readResponseString(conn);
        } catch (Exception e) {
            throw new RuntimeException(e);
        } finally {
            if (conn != null) {
                conn.disconnect();
            }
        }
    }

    private static String readResponseString(HttpURLConnection conn) {
        StringBuilder sb = new StringBuilder();
        InputStream inputStream = null;
        try {
            inputStream = conn.getInputStream();
            BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, "UTF-8"));
            String line = null;
            while ((line = reader.readLine()) != null) {
                sb.append(line).append("\n");
            }
            return sb.toString();
        } catch (Exception e) {
            throw new RuntimeException(e);
        } finally {
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

}

主要关注下 * OptionTrustManager 的 checkServerTrusted * 这个方法,我们在自定义的 X509TrustManager 中对证书进行了简单的验证,取了证书的一部分内容做校验(这个只是例子,线上使用的话最好做全部字符的校验)

好了,客户端也有了,接下来我们用 Charles 来验证下,看加上这个验证后 charles 还能不能截取到我们的通信内容了(这里 java 代码相当于一个客户端,baidu 相当于服务器)

首先,我们开启 Charles 的 macOS Proxy来获取系统的 http 通信内容.如下图(开启 macOS Proxy 需要安装根证书,安装方法看下面第二张图)


image.png

image.png

首先开启 macOS Proxy,接下来我们运行下 main 方法,发现现在访问 baidu.com 显示的是 unknown 了,再看 console 输出了异常信息,公钥验证不通过


image.png

这是因为 Charles 是作为中间人来劫持我们的请求的,我们访问 baidu.com 实际是先访问了 Charles 服务,然后 Charles 服务再去跟 baidu.com 进行交互,完了再把请求数据返回给我们,所以我们拿到的公钥信息是 Charles 的根证书的,自然是跟 baidu 的证书不匹配的,所以就不会再发起请求了

然后关闭 macOS Proxy,再次运行 main 方法,可以看到这次没有抛异常,请求到了 baidu.com 的页面内容,而且打印了两个证书信息,其中第一个证书是 baidu 的,第二个是 CA 的证书

有一点需要注意,这里验证的是证书的公钥信息,正式情况下,还需要验证证书的过期时间等信息,验证公钥是为了再证书过期后不至于客户端留的证书和新的证书不一致,只要我们的服务器私钥不变,生成的证书的公钥信息也就不会变

到此,我们已经对 unknown 有了比较清晰的认识,以上只是自己个人浅显的理解,如有纰漏还望指正

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容