构建 CA 证书链

我们首先要创建 client/server 使用的证书。创建证书的方法有很多种:不怕麻烦,直接通过openssl创建的,有通过cfssl创建的。这里要介绍的是比较简单的一种:tls-gen。

tls-gen是一个用Python编写的、非常易用的工具。它定义了三种profile。这里我们选择最简单的一种:一个根证书和一组证书、私钥对。

在shell里面执行一下命令:

  1. git clone https://github.com/michaelklishin/tls-gen
  2. cd tls-gen/basic
  3. make CN=www.mytestdomain.io

就这样,我们就为域名 www.mytestdomain.io 创建了一套证书。观察一下当前路径的内容,我们会发现两个新的目录:testca 和 server。前者里面存放了刚刚创建的根证书 (root CA),后者里面存放了我们之后的服务程序要用的的证书和私钥。

  1. testca/
  2. cacert.pemserver/
  3. cert.pem
  4. key.pem

编写服务

接下来开始写代码。Go 对 TLS 的支持还是比较完备的,也比较简单。以下是服务端的代码(server.go):

  1. func HelloServer(w http.ResponseWriter, req *http.Request) {
  2. w.Header().Set("Content-Type", "text/plain")
  3. w.Write([]byte("This is an example server.\n"))
  4. }
  5. func main() {
  6. http.HandleFunc("/hello", HelloServer)
  7. err := http.ListenAndServeTLS(":1443", "server/cert.pem", "server/key.pem", nil)
  8. if err != nil {
  9. log.Fatal("ListenAndServe: ", err)
  10. }
  11. }

可以看到我们创建了一个HTTP服务,这个服务监听1443端口并且只处理一个路径/hello.然后调用了下面这个函数来监听1443端口。注意我们给出了之前创建的服务的证书和私钥,这样就保证了HTTP会用加密的方式来传输。

  1. ListenAndServeTLS(addr, certFile, keyFile string, handler Handler)

运行服务程序:

  1. go run server.go

访问HTTPs服务

假定我们的服务程序是运行在本地的。我们先改一下/etc/hosts来配置域名解析:

  1. # echo 127.0.0.1 www.mytestdomain.io >> /etc/hosts

我们用以下的代码(client.go)来访问服务:

  1. func main() { client := &http.Client{}
  2. resp, err := client.Get("https://www.mytestdomain.io:1443/hello")
  3. if err != nil {
  4. panic("failed to connect: " + err.Error())
  5. }
  6. content, _ := ioutil.ReadAll(resp.Body)
  7. s := strings.TrimSpace(string(content))
  8. fmt.Println(s)
  9. }

运行 go run client.go,只能得到这样的错误:

  1. panic: failed to connect: Get https://www.mytestdomain.io:1443/hello: x509: certificate signed by unknown authorit

这是因为系统不知道如何来处理这个 self signed 证书。
各个 OS 添加根证书的方法是不同的。对于 Linux 系统 (以 Ubuntu 为例) 来说,把证书文件放到相应的目录即可:

  1. # sudo cp testca/cacert.pem /etc/ssl/certs

如果是 macOS,可以用一下的命令:

  1. # sudo security add-trusted-cert -d -r trustRoot -k /Library/Keychains/System.keychain testca/cacert.pem

上面的方法会把我们手工创建的 root CA添加到系统所已知的列表里面。这样一来,所有用该 root CA 创建的证书都可以被认证了。

假如只是一个普通的用户,没有root/sudo权限,不就无法做上面的操作了吗?这种情况下还有另外一个做法:把root CA 放置在代码里面。

在上面的client.go 里面添加这么几行代码:

  1. func main() {
  2. roots := x509.NewCertPool()
  3. ok := roots.AppendCertsFromPEM([]byte(rootPEM))
  4. if !ok { panic("failed to parse root certificate")
  5. }
  6. tr := &http.Transport{
  7. TLSClientConfig: &tls.Config{RootCAs: roots},
  8. }
  9. client := &http.Client{Transport: tr} // ...

其中的rootPEM 就是 testca/cacert.pem 的内容

  1. var rootPEM = `-----BEGIN CERTIFICATE-----MIIDAjCCAeqgAwIBA
  2. gIJAL2faqa73yLvMA0GCSqGSIb3DQEBCwUAMDExIDAeBgNVBAMMF1RMU0dl
  3. blNlbGZTaWduZWR0Um9vdENBMQ0wCwYDVQQHDAQkJCQkMB4XDTE4MDIwNTA
  4. 5Mzc0NVoXDTI4MDIwMzA5Mzc0NVowMTEgMB4GA1UEAwwXVExTR2VuU2VsZl
  5. NpZ25lZHRSb290Q0ExDTALBgNVBAcMBCQkJCQwggEiMA0GCSqGSIb3DQEBA
  6. QUAA4IBDwAwggEKAoIBAQC9eO6Tam4XFDUbK9FAStAg29teYeKtt8WEJvK
  7. GB50xMfXO2pD0StsXhKrspXBYck0FwKIBsTLr97w7dSqa64z3U2V2Borog
  8. FzoEE4JH2sydYGAQqNAqezGx8VZnQVRyZEBifRPebR4WVD5GtXYe+MnSkH
  9. PIgsG0QG0SaiSfMl05dSJHoE9T9Kly9fH6yED88++OYjZZRGKOf2THpQlX
  10. JjF3iwCDLkwz9Z/kjmpK/rR0SEhtanf7bOgGs3OoFmX4DvmFJXoriVUC9jc
  11. j0Z4oX3Ld81XXyd4FJkpKvdKDhYkqcugFgERqdBeRDM+MA38YooKHZh0klL
  12. 2EThNXJxM0r1vAgMBAAGjHTAbMAwGA1UdEwQFMAMBAf8wCwYDVR0PBAQDAgE
  13. GMA0GCSqGSIb3DQEBCwUAA4IBAQBEqp0ON1A/pCKFztfKuzdW+9pauE8dl6Ij
  14. 3++dt6AqW5QYFLOEFQwoMBOkGChGQDxHkakyaA0DfGe5JntMH0yYyZnr4kfs+
  15. AcY6P+2PfgrgVBqadhR6uAGOBaXDW7dlllqIJJ8NRInA/fTDYXMxBJbFrcj2c
  16. GIYVPvAbrosZ5L/YdAdVM76V8uuk8Hmmy5zRQj+gWt/jDkYWFrp0b6k3FBXvM7
  17. +nhqAIdyMjLioAdYwFpPglGj3xHXS5neWjyUDlAYISNe+PKMERSeDrptyDE+ljz
  18. l77hvvfZD9OPhXbDkAeVU/NaDwHG/G5HDVdNbg/FZ6ueevF34Xuzejm3lrdJm-----E
  19. ND CERTIFICATE-----`

也就是说,我们用准备好的 root CA 的内容产生了一个新的 http transport。

运行一下 go run client.go。成功!
This is an example server.

总结

一对 HTTPs client/server 程序中需要一个共同的 root CA。服务器端需要该 root CA,创建的 CA/私钥对。


Go代码打通HTTPs - 图1