美文网首页
使用cert-manager在kubernetes中管理tls证

使用cert-manager在kubernetes中管理tls证

作者: eva_lilizhang | 来源:发表于2020-03-17 18:50 被阅读0次
    • 什么是cert-manager?
    • cert-manager的工作原理
    • 安装和部署cert-manager
    • 具体的使用
    • 需要注意的事项

    什么是cert-manager?

    cert-manager 是一个云原生证书管理开源项目,用于在 Kubernetes 集群中提供 HTTPS 证书并自动续期,支持 Let’s Encrypt, HashiCorp Vault 这些免费证书的签发。在Kubernetes集群中,我们可以通过 Kubernetes Ingress 和 Let’sEncrypt实现外部服务的自动化HTTPS。

    在Kubernetes集群中使用 HTTPS 协议,需要一个证书管理器、一个证书自动签发服务,主要通过 Ingress 来发布 HTTPS 服务,因此需要Ingress Controller并进行配置,启用 HTTPS 及其路由。

    letsencrypt:基于ACME协议, 生成免费的证书工具。</br>
    ACME: 自动证书管理环境 Automated Certificate Management Environment。

    cert-manager的工作原理

    cert-manager的架构图

    这里有2个概念就是issuer、certificate:
    issuer:证书颁发者,它是k8s的资源,接受执行证书签名请求生成签名证书。有issuerclusterIssuer两个类别,区别就在与issuer要指定namespace,只能在指定的namespace下工作,而clusterIssuer是不指定namespace,可以在整个集群下工作。
    certificate:可以把这个资源理解为执行证书,通过执行证书的配置,会向cert-manager对应的issuer发送一个certificateRequest,如果成功,就会将生成的密钥tls.key和证书tls.crt存到secret里面。

    证书颁布原理

    Let’s Encrypt 利用 ACME 协议来校验域名是否真的属于你,校验成功后就可以自动颁发免费证书,证书有效期只有 90 天,在到期前需要再校验一次来实现续期,幸运的是 cert-manager 可以自动续期,这样就可以使用永久免费的证书了。如何校验你对这个域名属于你呢?主流的两种校验方式是 HTTP-01 和 DNS-01。

    HTTP-01 校验原理

    HTTP-01 的校验原理是给你域名指向的 HTTP 服务增加一个临时 location ,Let’s Encrypt 会发送 http 请求到 http://<YOUR_DOMAIN>/.well-known/acme-challenge/<TOKEN>,YOUR_DOMAIN 就是被校验的域名,TOKEN 是 ACME 协议的客户端负责放置的文件,在这里 ACME 客户端就是 cert-manager,它通过修改 Ingress 规则来增加这个临时校验路径并指向提供 TOKEN 的服务。Let’s Encrypt 会对比 TOKEN 是否符合预期,校验成功后就会颁发证书。此方法仅适用于给使用 Ingress 暴露流量的服务颁发证书,并且不支持泛域名证书。

    DNS-01 校验原理

    DNS-01 的校验原理是利用 DNS 提供商的 API Key 拿到你的 DNS 控制权限, 在 Let’s Encrypt 为 ACME 客户端提供令牌后,ACME 客户端 (cert-manager) 将创建从该令牌和您的帐户密钥派生的 TXT 记录,并将该记录放在 _acme-challenge.<YOUR_DOMAIN>。 然后 Let’s Encrypt 将向 DNS 系统查询该记录,如果找到匹配项,就可以颁发证书。此方法不需要你的服务使用 Ingress,并且支持泛域名证书。

    安装和部署cert-manager

    安装

    安装方式有两种

    1. 使用helm安装:
    echo $(helm version) //我这里使用的helm 是v2.14.3
    // 安装Tiller
    helm init --upgrade --wait
    
    echo "Install the CustomResourceDefinition resources separately"
    kubectl apply --validate=false -f https://raw.githubusercontent.com/jetstack/cert-manager/release-0.13/deploy/manifests/00-crds.yaml
    
    echo "Add the Jetstack Helm repository"
    helm repo add jetstack https://charts.jetstack.io
    helm repo update
    
    echo "Install the cert-manager helm chart"
    helm upgrade --install cert-manager --namespace cert-manager --version v0.13.1 jetstack/cert-manager
    

    安装cert-manager chart,如果helm版本不一样,命令也不一样

    # Helm v3+
    $ helm install \
      cert-manager jetstack/cert-manager \
      --namespace cert-manager \
      --version v0.14.0
    # Helm v2
    $ helm install \
      --name cert-manager \
      --namespace cert-manager \
      --version v0.14.0 \
      jetstack/cert-manager
    

    安装成功可以后,kubectl get pods -n cert-manager

    cert-manager pods

    cert-manager-webhook的启动时间会稍微长一点,大约也是在1min之内,到这里,cert-manager也就安装好了。

    1. 也可以使用regular manifests 进行安装,具体的可以参考 https://cert-manager.io/docs/installation/kubernetes/#installing-with-regular-manifests

    具体的使用

    这里是使用http01校验,给使用ingress暴露流量的服务颁发证书。
    接下来需要创建一个证书颁布者Issuer:kubectl apply -f ./kubernetes/issuer.yml

    issuer-stg.yml

    apiVersion: cert-manager.io/v1alpha2
    kind: ClusterIssuer
    metadata:
      name: letsencrypt-staging
    spec:
      acme:
        server: https://acme-staging-v02.api.letsencrypt.org/directory
        email: evalizhangli@gmail.com
        privateKeySecretRef:
          name: letsencrypt-staging
        solvers:
          - http01:
              ingress:
                class: nginx
    
    

    需要注意的地方就是,如果还在开发测试阶段,尽可能的不要使用letsencrypt-prod,因为staging的证书颁布server对于创建证书没有限制,但是prod的证书颁布server有很严格的次数限制。

    issuer.yml

    apiVersion: cert-manager.io/v1alpha2
    kind: ClusterIssuer
    metadata:
      name: letsencrypt-prod
    spec:
      acme:
        server: https://acme-v02.api.letsencrypt.org/directory
        email: evalizhangli@gmail.com
        privateKeySecretRef:
          name: letsencrypt-prod
        solvers:
          - http01:
              ingress:
                class: nginx
    
    

    创建成功以后可以kubectl get clusterIssuer 看到这个issuer的状态为ready,就代表已经创建好了。

    image.png

    接下来,我们只需要在你需要使用到cert-manager管理证书的ingress的configfile里面改动,然后将你的服务重新部署

    apiVersion: extensions/v1beta1
    kind: Ingress
    metadata:
      name: example
      namespace: ${NAMESPACE}
      annotations:
        kubernetes.io/ingress.class: nginx
        cert-manager.io/cluster-issuer: letsencrypt-prod
    spec:
      tls:
      - hosts:
          - my.example.net
        secretName: tls-secret-my
      rules:
      - host: my.example.net
        http:
          paths:
          - path: /
            backend:
              serviceName: my-service
              servicePort: http
    
    

    使用起来非常的方便以及简单,kubectl get certificate -n ${NAMESPACE}就能看到 tls-secret-my的certificate,等到状态ready为true, kubectl get secret -n ${NAMESPACE} 也能看到tls-secret-my的secret。
    如果certificate的ready为false,可以kubectl describe certificate tls-secret-my -n ${NAMESPACE} 查看一下envents,或者查看一下cert-manager pod 的log。

    需要注意的事项

    1. 在开发和测试阶段,尽可能的使用letsencrypt-staging
    2. letsencrypt-prod有rate limit,如果查看cert-manager的log一直显示waiting CertificateRequest status is not complete,有可能就是申请创建证书达到了次数限制
    • 同一个主域下一周只能申请 50 张证书 (例如 http://www.example.com 的主域是 http://example.com)
    • 每个账户每个域名每小时申请验证失败的次数为 5 次
    • 每周只能创建 5 个重复的证书,即使是通过不同账户创建
    • 每个账户同一个 IPv4 地址每 3 小时最多可创建 10 张证书
    • 每个多域名(SAN)证书最多包含 100 个子域
    • 更新证书没有次数的限制,但是更新证书会受到上述重复证书的限制
    1. 如果要重新install cert-manger,可以使用helm delete --purge cert-manager

    章节外:使用DNS01校验,颁发泛域名证书

    apiVersion: cert-manager.io/v1alpha2
    kind: ClusterIssuer
    metadata:
      name: example-issuer
    spec:
      acme:
        email: evalizhangli@gmail.com
        server: https://acme-staging-v02.api.letsencrypt.org/directory
        privateKeySecretRef:
          name: example-issuer-account-key
        solvers:
          - dns01:
              clouddns:
                project: my-project
                serviceAccountSecretRef:
                  name: prod-clouddns-svc-acct-secret
                  key: dns01-solver-credential.json
            selector:
              dnsNames:
                - 'my.example.net'
                - '*.example.net'
              dnsZones:
                - '*.example.net'
    
    

    这里我是google cloud的dns服务,对应的config可以参考https://cert-manager.io/docs/configuration/acme/dns01/

    相关文章

      网友评论

          本文标题:使用cert-manager在kubernetes中管理tls证

          本文链接:https://www.haomeiwen.com/subject/wtlxyhtx.html