Vault as the Secrets Backend - Systems Integration

    • Enabling Vault KV Secrets Engine - Version 2 to store arbitrary secrets
    • Enabling Vault PKI Engine if you are choosing to store and manage either Consul Server TLS credentials or

    Setup per Consul datacenter

    • Installing the Vault Injector within the Consul datacenter installation
    • Configuring a Kubernetes Auth Method in Vault to authenticate and authorize operations from the Consul datacenter
    • Enable Vault as the Secrets Backend in the Consul datacenter

    A one time setup on a Vault deployment is necessary to enable both the Vault KV Secrets Engine and the Vault PKI Engine. These docs assume that you have already setup a Vault cluster for use with Consul on Kubernetes.

    Please read if instructions on setting up a Vault cluster are needed.

    The following secrets can be stored in Vault KV secrets engine, which is meant to handle arbitrary secrets:

    In order to store any of these secrets, we must enable the Vault KV secrets engine - Version 2.

    Vault PKI Engine

    The Vault PKI Engine must be enabled in order to leverage Vault for issuing Consul Server TLS certificates. More details for configuring the PKI Engine is found in Bootstrapping the PKI Engine under the Server TLS section.

    1. $ vault secrets enable pki
    1. $ vault secrets enable pki

    After configuring Vault, Consul datacenters on Kubernetes must be deployed with the Vault Agent injector and configured to leverage the Vault Kubernetes Auth Method to read secrets from a Vault cluster.

    Before installing the Vault Injector and configuring the Vault Kubernetes Auth Method, some environment variables need to be set to better ensure consistent mapping between Vault and Consul on Kubernetes.

    DATACENTER

    • Recommended value: value of global.datacenter in your Consul Helm values file.

        1. $ export DATACENTER=dc1

      VAULT_AUTH_METHOD_NAME

      • Recommended value: a concatenation of a kubernetes- prefix (to denote the auth method type) with DATACENTER environment variable.

        1. $ export VAULT_AUTH_METHOD_NAME=kubernetes-${DATACENTER}
        1. $ export VAULT_AUTH_METHOD_NAME=kubernetes-${DATACENTER}

      VAULT_SERVER_HOST

        • If Vault is installed in a Kubernetes cluster, get the external IP or DNS name of the Vault server load balancer.

          • On GKE or AKS, it’ll be an IP:

            1. $ export VAULT_SERVER_HOST=$(kubectl get svc vault-dc1 -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
            1. $ export VAULT_SERVER_HOST=$(kubectl get svc vault-dc1 -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
          • On EKS, it’ll be a hostname:

            1. $ export VAULT_SERVER_HOST=$(kubectl get svc vault-dc1 -o jsonpath='{.status.loadBalancer.ingress[0].hostname}')
        • If Vault is not running on Kubernetes, utilize the api_addr as defined in the Vault High Availability Parameters configuration:

          1. $ export VAULT_SERVER_HOST=<external IP for vault cluster>
          1. $ export VAULT_SERVER_HOST=<external IP for vault cluster>

      VAULT_ADDR

      • Recommended value: Connecting to port 8200 of the Vault server

        1. $ export VAULT_ADDR=http://${VAULT_SERVER_HOST}:8200
        1. $ export VAULT_ADDR=http://${VAULT_SERVER_HOST}:8200

        Note: If your vault installation is current exposed using SSL, this address will need to use https instead of http. You will also need to setup the VAULT_CACERT environment variable.

      VAULT_TOKEN

      • Recommended value: Your allocated Vault token. If running Vault in dev mode, this can be set to to root.

        1. $ export VAULT_ADDR=<vault token>
        1. $ export VAULT_ADDR=<vault token>

      Install Vault Injector in your Consul k8s cluster

      A minimal valid installation of Vault Kubernetes must include the Agent Injector which is utilized for accessing secrets from Vault. Vault servers could be deployed external to Vault on Kubernetes with the value in the Vault Helm Configuration.

      1. # vault-injector.yaml
      2. server:
      3. enabled: false
      4. injector:
      5. enabled: true
      6. externalVaultAddr: ${VAULT_ADDR}
      7. authPath: auth/${VAULT_AUTH_METHOD_NAME}
      8. EOF
      1. $ cat <<EOF >> vault-injector.yaml
      2. # vault-injector.yaml
      3. server:
      4. enabled: false
      5. injector:
      6. enabled: true
      7. authPath: auth/${VAULT_AUTH_METHOD_NAME}
      8. EOF

      Issue the Helm install command to install the Vault agent injector using the HashiCorp Vault Helm chart.

      1. $ helm install vault-${DATACENTER} -f vault-injector.yaml hashicorp/vault --wait

      Enable the Auth Method

      1. $ vault auth enable -path=kubernetes-${DATACENTER} kubernetes
      1. $ vault auth enable -path=kubernetes-${DATACENTER} kubernetes

      Configure Auth Method with JWT token of service account

      After enabling the Kubernetes auth method, in Vault, ensure that you have configured the Kubernetes Auth method properly as described in Kubernetes Auth Method Configuration.

      First, while targeting your Consul cluster, get the externally reachable address of the Consul Kubernetes cluster.

      1. $ export KUBE_API_URL=$(kubectl config view -o jsonpath="{.clusters[?(@.name == \"$(kubectl config current-context)\")].cluster.server}")

      Next, you will configure the Vault Kubernetes Auth Method for the datacenter. You will need to provide it with:

      • - this a JWT token from the Consul datacenter cluster that the Vault Kubernetes Auth Method will use to query the Consul datacenter Kubernetes API when services in the Consul datacenter request data from Vault.
      • kubernetes_host - this is the URL of the Consul datacenter’s Kubernetes API that Vault will query to authenticate the service account of an incoming request from a Consul data center kubernetes service.
      • - this is the CA certification that is currently being used by the Consul datacenter Kubernetes cluster.
      1. $ vault write auth/kubernetes/config \
      2. token_reviewer_jwt="$(cat /var/run/secrets/kubernetes.io/serviceaccount/token)" \
      3. kubernetes_host="https://${KUBE_API_URL}:443" \
      4. kubernetes_ca_cert=@/var/run/secrets/kubernetes.io/serviceaccount/ca.crt
      1. $ vault write auth/kubernetes/config \
      2. token_reviewer_jwt="$(cat /var/run/secrets/kubernetes.io/serviceaccount/token)" \
      3. kubernetes_host="https://${KUBE_API_URL}:443" \
      4. kubernetes_ca_cert=@/var/run/secrets/kubernetes.io/serviceaccount/ca.crt

      Enable Vault as the Secrets Backend in the Consul datacenter

      Finally, you will configure the Consul on Kubernetes helm chart for the datacenter to expect to receive the following values (if you have configured them) to be retrieved from Vault:

      1. global:
      2. secretsBackend:
      3. vault:
      4. enabled: true

      values.yaml

      1. global:
      2. secretsBackend:
      3. vault:
      4. enabled: true

      As a next step, please proceed to Vault integration with Consul on Kubernetes’ .

      The Vault integration with Consul on Kubernetes makes use of the Vault Agent Injectors. Kubernetes annotations are added to the deployments of the Consul components which cause the Vault Agent Injector to be added as an init-container that will then attach Vault secrets to Consul’s pods at startup. Additionally the Vault Agent sidecar is added to the Consul component pods which is responsible for synchronizing and reissuing secrets at runtime. As a result of these additional sidecar containers the typical location for logging is expanded in the Consul components.

      As a general rule the best way to troubleshoot startup issues for your Consul installation when using the Vault integration is to establish if the vault-agent-init container has completed or not via kubectl logs -f <your-consul-component> -c vault-agent-int and checking to see if the secrets have completed rendering.

      • If the secrets are not properly rendered the underlying problem will be logged in vault-agent-init init-container and generally is related to the Vault Kube Auth Role not having the correct policies for the specific secret e.g. global.secretsBackend.vault.consulServerRole not having the correct policies for TLS.