AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / server / 问题 / 1159948
Accepted
Lasse Michael Mølgaard
Lasse Michael Mølgaard
Asked: 2024-05-26 11:27:05 +0800 CST2024-05-26 11:27:05 +0800 CST 2024-05-26 11:27:05 +0800 CST

如何让 ingress-nginx 使用向用户提供的证书?

  • 772

我通过 Ansible 在我的 K3S Kubernetes 集群中进行安装,并且我想使用在集群上运行Home Assistant时生成的 Let's Encrypt 证书作为我的入口。Cert-Manager

我在安装 Home Assistant 时使用以下剧本:

---
- name: Create HomeAssistant namespace
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: v1
      kind: Namespace
      metadata:
        name: home-system

- name: Generate certificate for HomeAssistant
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: cert-manager.io/v1
      kind: Certificate
      metadata:
        name: homeassistant-tls
        namespace: home-system
      spec:
        secretName: homeassistant-tls-secret
        commonName: "homeassistant.example.com" 
        dnsNames:
        - "homeassistant.example.com"
        issuerRef:
          name: letsencrypt-staging
          kind: ClusterIssuer

- name: Get TLS certificate secret
  kubernetes.core.k8s_info:
    api_version: v1
    name: homeassistant-tls-secret
    kind: Secret
    namespace: home-system
  register: tls_secret
  
- name: Create ConfigMap for HomeAssistant
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: v1
      kind: ConfigMap
      metadata:
        name: home-config
        namespace: home-system

      data:
        fullchain.pem: "{{ tls_secret.resources[0].data['tls.crt'] | b64decode }}"

        privkey.pem: "{{ tls_secret.resources[0].data['tls.key'] | b64decode }}"

        groups.yaml: |-

        scripts.yaml: |-

        scenes.yaml: |-

        known_devices.yaml: |- 
    
        automations.yaml: |-

        configuration.yaml: |-

          # Loads Default configuration - DO NOT REMOVE
          default_config:

          # Load frontend themes from the themes folder
          frontend:
            themes: !include_dir_merge_named themes
          
          # Text-to-speech
          tts:
            - platform: google_translate
          
          group: !include groups.yaml
          automation: !include automations.yaml
          script: !include scripts.yaml
          scene: !include scenes.yaml

          http:
            use_x_forwarded_for: true

            trusted_proxies:
            - 127.0.0.1
            - 10.42.0.0/16
            - 10.43.0.0/16
            - 192.168.0.0/24

            ssl_certificate: /ssl/fullchain.pem
            ssl_key: /ssl/privkey.pem

- name: Define storage space for HomeAssistant
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: v1
      kind: PersistentVolumeClaim
      metadata:
        name: homeassist-pvc
        namespace: home-system
      spec:
        accessModes:
          - ReadWriteOnce
        storageClassName: small-longhorn # Store data on only 2 nodes instead of 3
        resources:
          requests:
            storage: 5Gi

- name: Add HomeAssistant StatefulSet
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: apps/v1
      kind: StatefulSet
      metadata:
        name: homeassistant
        namespace: home-system
        labels:
          app: homeassistant
      spec:
        serviceName: home-svc
        replicas: 1
        selector:
          matchLabels:
            app: homeassistant
        template:
          metadata:
            labels:
              app: homeassistant
          spec:
            volumes:
            - name: home-config-vol
              persistentVolumeClaim:
                claimName: homeassist-pvc

            - name: home-config-file
              configMap:
                name: home-config

            containers:
            - name: homeassistant
              image: lscr.io/linuxserver/homeassistant:latest
              ports:
                - containerPort: 8123
              volumeMounts:
              - name: home-config-vol
                mountPath: /config

              - name: home-config-file
                mountPath: /config/configuration.yaml
                subPath: configuration.yaml
                
              - name: home-config-file
                mountPath: /config/automations.yaml
                subPath: automations.yaml

              - name: home-config-file
                mountPath: /config/scripts.yaml
                subPath: scripts.yaml

              - name: home-config-file
                mountPath: /config/scenes.yaml
                subPath: scenes.yaml

              - name: home-config-file
                mountPath: /config/groups.yaml
                subPath: groups.yaml

              - name: home-config-file
                mountPath: /ssl/fullchain.pem
                subPath: fullchain.pem

              - name: home-config-file
                mountPath: /ssl/privkey.pem
                subPath: privkey.pem

              env:
                - name: PUID
                  value: "1000"
                - name: PGID
                  value: "1000"
                - name: TZ
                  value: "Etc/UTC"

            securityContext:
              capabilities:
                add:
                - NET_ADMIN
                - NET_RAW
                - SYS_ADMIN

              priviledged: true

- name: Define HomeAssistant service ports
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: v1
      kind: Service
      metadata:
        name: home-svc
        namespace: home-system
        annotations: 
          metallb.universe.tf/address-pool: default-pool
      spec:
        type: LoadBalancer
        externalTrafficPolicy: Local
        selector:
          app: homeassistant
        ports:
          - name: "8123"
            port: 8123
            targetPort: 8123

- name: Add HomeAssistant ingress
  kubernetes.core.k8s:
    state: present
    definition:
      apiVersion: networking.k8s.io/v1
      kind: Ingress
      metadata:
        name: homeassistant
        namespace: home-system

        annotations:
          kubernetes.io/ingress.class: nginx
          cert-manager.io/cluster-issuer: letsencrypt-staging
          nginx.ingress.kubernetes.io/backend-protocol: HTTPS
          nginx.ingress.kubernetes.io/ssl-redirect: "true"
          nginx.ingress.kubernetes.io/force-ssl-redirect: "true"

      spec:
        ingressClassName: nginx
        tls:
        - hosts:
          - homeassistant.example.com
          secretName: homeassistant-tls

        rules:
        - host: 'homeassistant.example.com'
          http:
            paths:
            -  path: /
               pathType: Prefix
               backend:
                service:
                  name: home-svc
                  port:
                    number: 8123

我已将实际名称替换FQDN为属于 的名称example.com。 :-)

几个关键点:

家庭助理支持 TLS 加密。

因此,我的第一份工作是创建一个 Let's Encrypt 证书,我将其命名为homeassistant-tls。它有相关的秘密homeassistant-tls-secret。

接下来是从密钥中提取证书和密钥homeassistant-tls-secret,并使用它来创建文件fullchain.pem并privkey.pem在 ConfigMap 中home-config。

这些文件包含 base64 解码的证书和密钥。

然后,证书和密钥文件存储在/ssl容器内的文件夹中,这会返回到 ConfigMap 中定义的最后一个重要文件:configuration.yaml。

为了启用 TLS 加密,我们需要将以下行添加到httpconfiguration.yaml 部分:

  ssl_certificate: /ssl/fullchain.pem
  ssl_key: /ssl/privkey.pem

我尝试调用文件tls.crt和tls.key,但显然家庭助理不喜欢它。 :-)

我唯一确定的是文件必须是 PEM 编码的。

运行剧本后,我得到以下输出:

root@central:~# kubectl get svc -n home-system
NAME       TYPE           CLUSTER-IP     EXTERNAL-IP     PORT(S)          AGE
home-svc   LoadBalancer   10.43.23.186   192.168.0.203   8123:30630/TCP   81m

root@central:~# kubectl get ingress -n home-system
NAME            CLASS   HOSTS                       ADDRESS         PORTS     AGE
homeassistant   nginx   homeassistant.example.com   192.168.0.200   80, 443   85m

当我输入地址:https://192.168.0.203:8123/或https://homeassistant.example.com在浏览器中时,该网站正在运行。

但是,当我检查相关证书时,我得到以下信息:

  • https://192.168.0.203:803(LoadBalancer) 指向 Let's Encrypt 证书,据我所知,这是因为我在文件/ssl夹中定义了证书。
  • https://homeassistant.example.com(Ingress) 指向通用 Kubernetes 集群证书。

那么我该如何Ingress使用正确的证书呢? :-)

更改为secretName肯定不起作用。 :-)homeassist-tls-secret

kubernetes
  • 2 2 个回答
  • 42 Views

2 个回答

  • Voted
  1. Best Answer
    Sai Chandini Routhu
    2024-05-26T15:21:42+08:002024-05-26T15:21:42+08:00

    您面临的问题是入口控制器并不总是获取包含 homeassistant.example.com 的 Let's Encrypt 证书的游戏名称。

    要解决您的问题:

    使用注释来引用 TLS 密钥和集群颁发者,而不是为您的入口指定指定密钥名称;

    annotations:
    
    
                  certmanager.K8s.Io/provider: letsencrypt-staging
                  nginx.Ingress.Kubernetes.Io/ssl-secret:homeassistant-tls-secret
    

    更新入口后出现注释,建议重新启动 nginx-ingress 控制器 pod 以选择更改

    通过使用注释,您可以为入口控制器提供更多上下文。它可以利用集群颁发者来验证证书并使用引用的秘密 homeassistant-tls-secret 为 homeassistant.example.Com 上的安全访问者提供服务

    对于 k8s 和 k3s,与通过注释使用 Lets Encrypt 证书的 ingress-nginx 相关的标准和配置步骤通常是相同的。

    欲了解更多信息,请参阅Ivan Khramov 的Medium博客

    • 1
  2. Lasse Michael Mølgaard
    2024-05-26T17:26:58+08:002024-05-26T17:26:58+08:00

    对我对 Sai Chandini Routhu 的回答的评论进行快速重新格式化和解释:

    Ivan Khramov 的 Medium 博客几乎说到了正在发生的事情,尽管他的实现与我的集群不兼容,但考虑到他的博客文章已经有6 年了,我将其归因于 Kubernetes 同时进行了一些重命名。

    正如我的评论中所述,没有名为 的注释nginx.ingress.kubernetes.io/ssl-secret。

    最接近的匹配是nginx.ingress.kubernetes.io/proxy-ssl-secret: SecretName,但即使这样在实验时也不起作用。

    我发现有效的是,如果我secretName在入口配置中将其重命名为homeassistant-tls,homeassistant-ingress-tls那么我将得到我想要的结果。

    作为参考,请参阅下面的配置,该配置按预期工作:

    apiVersion: networking.k8s.io/v1
    kind: Ingress
    metadata:
      name: homeassistant
      namespace: home-system
    
      annotations:
        kubernetes.io/ingress.class: nginx
        kubernetes.io/tls-acme: "true"
        cert-manager.io/cluster-issuer: letsencrypt-staging
        nginx.ingress.kubernetes.io/ssl-redirect: "true"
        nginx.ingress.kubernetes.io/backend-protocol: HTTPS
        nginx.ingress.kubernetes.io/force-ssl-redirect: "true"
    
    spec:
      ingressClassName: nginx
      tls:
      - hosts:
        - homeassistant.example.com
        secretName: homeassistant-ingress-tls
      rules:
      - host: 'homeassistant.example.com'
        http:
          paths:
          -  path: /
              pathType: Prefix
              backend:
              service:
                name: home-svc
                port:
                  number: 8123
    

    它起作用的原因是因为入口正在调用cert-manager创建一个新证书,该证书本质上是容器中安装的证书的副本。

    (我还没有检查两个证书上的哈希码是否相同)。

    之后我留下了两张证书,如下所示:

    root@central:~# kubectl get certs -n home-system
    NAME                        READY   SECRET                      AGE
    homeassistant-tls           True    homeassistant-tls-secret    10h
    homeassistant-ingress-tls   True    homeassistant-ingress-tls   42m
    

    (我想我应该在我的定义中进行一些重命名homeassistant-tls)。 :-)

    这使得我的浏览器始终看到 Let's Encrypt 证书,无论我是否通过 IP 地址调用该网站,从而访问我的网站Load Balancer或通过 DNS 访问Ingress.

    我现在很开心。 :-)

    • 0

相关问题

  • Nvidia Config 后 Containerd 无法启动

  • 在使用 kubeadm 引导集群之前,如何修改 CoreDNS 配置映射?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    新安装后 postgres 的默认超级用户用户名/密码是什么?

    • 5 个回答
  • Marko Smith

    SFTP 使用什么端口?

    • 6 个回答
  • Marko Smith

    命令行列出 Windows Active Directory 组中的用户?

    • 9 个回答
  • Marko Smith

    什么是 Pem 文件,它与其他 OpenSSL 生成的密钥文件格式有何不同?

    • 3 个回答
  • Marko Smith

    如何确定bash变量是否为空?

    • 15 个回答
  • Martin Hope
    Tom Feiner 如何按大小对 du -h 输出进行排序 2009-02-26 05:42:42 +0800 CST
  • Martin Hope
    Noah Goodrich 什么是 Pem 文件,它与其他 OpenSSL 生成的密钥文件格式有何不同? 2009-05-19 18:24:42 +0800 CST
  • Martin Hope
    Brent 如何确定bash变量是否为空? 2009-05-13 09:54:48 +0800 CST
  • Martin Hope
    cletus 您如何找到在 Windows 中打开文件的进程? 2009-05-01 16:47:16 +0800 CST

热门标签

linux nginx windows networking ubuntu domain-name-system amazon-web-services active-directory apache-2.4 ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve