Let_s_Encrypt 免费Https证书

警告
本文最后更新于 2020-09-03,文中内容可能已过时。

参考文章:Let’s Encrypt,免费好用的 HTTPS 证书

Let’s Encrypt

Let’s Encrypt 是免费、自动化、开放的证书签发服务, 它得到了 Mozilla、Cisco、Akamai、Electronic Frontier Foundation 和 Chrome 等众多公司和机构的支持,发展十分迅猛


  • 一个 HTTP 服务,以 Nginx 为例

  • python

  • 两个目录:

    /site 网站目录

    /site_site_cert 保存证书的目录

证书的颁发有两种方式:

通过网址在线生成,需要验证文件

通过本地 IIS,生成后会自动绑定本地 IIS 站点上的HTTPS 域名。

我使用的是第一种方法:在线生成,原理是:先在你的服务器上传一个密钥,然后Let’s Encrypt会对网站进行访问,下载密钥进行验证。


首先创建一个目录,我是在根目录下创建文件夹site_cert

1
mkdir /site_cert

这个文件夹用来存放各种临时文件和最后的证书文件。进入这个目录,创建一个RSA私钥,用于Let’s Encrypt识别你的身份

1
openssl genrsa 4096 > account.key

在这之前,还需要创建域名私钥(一定不要使用上面的账户私钥)

1
openssl genrsa 4096 > domain.key

我参考的文章提到了两种私钥RSA和ECC,我现在也还不懂什么原理,把两种区别放出来吧

RSA 私钥:兼容性好

ECC 私钥:部分老旧操作系统、浏览器不支持。优点是证书体积小

两个用于身份身份验证的私钥文件创建好,就可以生成CSR(Certificate Signing Request,证书签名请求)文件了,申请时可以把域名带 www 和不带 www 的两种情况都加进去,一张证书最多可以包含100个域名。

1
openssl req -new -sha256 -key domain.key -subj "/" -reqexts SAN -config <(cat /etc/ssl/openssl.cnf <(printf "[SAN]\nsubjectAltName=DNS:yoursite.com,DNS:www.yoursite.com")) > domain.csr
  • 其中DNS的yoursite.comwww.yoursite.com记得要换成自己的域名

我在这里遇到了问题提示找不到/etc/ssl/openssl.cnf文件,在网上找的的解决办法

执行 :

1
openssl version -a

会输出openssl的信息,其中OPENSSLDIR对应的路径就是openssl.cnf文件的地址,自行将上面的cat /etc/ssl/openssl.cnf,改为自己的路径运行。

前面介绍过了Let’s Encrypt验证的原理是在你的服务器上生成一个随机文件,在通过创建CSR时的域名进行访问下载,如果成功表明你对这个域名的拥有权。

创建用于存放网站的目录site以及用于验证文件存放的子目录

1
mkdir -p /site/.well-known/acme-challenge/

然后再Nginx中配置:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
server {
    server_name www.yoursite.com yoursite.com;

    location ^~ /.well-known/acme-challenge/ {
        alias /home/xxx/www/challenges/;
        try_files $uri =404;
    }

    location / {
        rewrite ^/(.*)$ https://yoursite.com/$1 permanent;
    }
}
  • 别忘了改yoursite

这个配置会优先查找/site目录下的网站,建议保留以后证书认证还可以用到,因为颁发的证书一次可以使用90天。

先下载acme-tiny脚本到之前的site_cert目录:

1
wget https://raw.githubusercontent.com/diafygi/acme-tiny/master/acme_tiny.py

指定账户私钥、CSR以及网站上验证文件的目录,执行脚本:

1
python acme_tiny.py --account-key ./account.key --csr ./domain.csr --acme-dir /fakesite/.well-known/acme-challenge/ > ./signed.crt

执行成功的话会在当前目录生成一个signed.crt文件,这个文件就是申请好的证书文件。

我在这里出现了错误提示

1
ValueError: Wrote file to /site/.well-known/acme-challenge/blablabla, but couldn't download http://www.yoursite.com/.well-known/acme-challenge/blablabla

大概的意思就是,在网站目录里写入了一个验证文件,但是Let’s Encrypt的服务器访问不到你的网站,建议先去看一看Nginx配置是否出错,再有可能是自己的域名无法在国外解析,建议暂时使用国外的DNS解析商。推荐的有:

Hurricane Electric Free DNS

ZoneEdit

CloudFlare

这些都是免费的,但是因为我自己的域名后缀为.tk,上面第一个DNS解析商警告因为.tk域名滥用,不给解析。

网站证书到手以后,还要下载 Let’s Encrypt 的中间证书。证书链中大部分都是「站点证书 – 中间证书 – 根证书」这样三级。服务端只需要发送前两个证书即可。我们需要把中间证书和网站证书合在一起:

1
2
3
4
wget -O - https://letsencrypt.org/certs/lets-encrypt-x3-cross-signed.pem > intermediate.pem


cat signed.crt intermediate.pem > chained.pem

最终,在Nginx中添加证书配置,并reload,我的部分配置如下

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
server {
   listen 443 ssl;
   ssl_certificate       /site_cert/chained.pem;
   ssl_certificate_key   /site_cert/domain.key;
   ssl_protocols         TLSv1 TLSv1.1 TLSv1.2;
   ssl_ciphers           HIGH:!aNULL:!MD5;
   server_name           braindance.tk;
   index index.html index.htm;
   root 
        …………………………
}

执行

1
nginx -s reload

​ 至此我们已经成功的获取到了Https证书,但是获取到的Https证书只有90天的时效,到期的话还需要使用相同的方法进行更新,为了避免某次忘记更新导致网站出现问题,我们可以使用linux中的crond服务为我们自动更新证书。

vi/site_cert 文件夹 创建 计划任务脚本 renew_cert.sh

1
vi /site_cert/renew_cert.sh

通过vi输入以下内容

1
2
3
4
5
6
7
#!/bin/bash

cd /fakesite_cert/
python acme_tiny.py --account-key ./account.key --csr ./domain.csr --acme-dir /fakesite/.well-known/acme-challenge/ > ./signed.crt || exit
wget -O - https://letsencrypt.org/certs/lets-encrypt-x3-cross-signed.pem > intermediate.pem
cat signed.crt intermediate.pem > chained.pem
nginx -s reload

更新的大致过程是,运行python脚本再次更新signed.crt申请证书文件,再进行证书合并写入chained.pem文件。

然后给这个文件赋予 可执行 属性

1
chmod +x /fakesite_cert/renew_cert.sh

使用crontab -e指令打开定时任务配置文件,并加入以下内容。

1
0 0 1 * * /home/xxx/shell/renew_cert.sh >/dev/null 2>&1

对于上面指令的具体意思请自行搜索crontab命令