Secure Credentials for GitOps Deployments Using the External Secrets Operator and AWS Secrets Manager

Secure Credentials for GitOps Deployments Using the External Secrets Operator and AWS Secrets Manager

7 min read

The security and storage of secrets is one of the most controversial subjects when it comes to GitOps deployments. Some teams want to go “by the book” and use Git as the storage medium (in an encrypted form of course) while others accept the fact that secrets must be handled in a different way (outside of GitOps). There is no right  or wrong answer here and depending on the organization requirements, either solution might be a great fit. 

In the previous article of the series we explained how to use the Argo CD Vault plugin to fetch secrets from the AWS Secret Manager. In the present article you will learn about an alternative method using the External Secrets Operator (while still using the AWS Secret Manager for the original storage). Like last time we will use Argo CD/Codefresh for the actual deployment of the application.

Solution overview

The assumption is that we already have a Kubernetes cluster (Amazon EKS in our case) and we want to deploy applications to it. The applications need several secure credentials such as database passwords or certificates that are stored in the AWS secret manager.

We want to deploy the applications on the cluster using a GitOps Runtime (the Codefresh runtime in our case). Therefore we need a way to make the secrets available inside the cluster by fetching them from the AWS secret manager.

Here is an overview of the system architecture:

The External Secret Operator (ESO) is the perfect fit for this challenge. It works by retrieving secrets from the AWS Secret Manager and making them available as standard Kubernetes Secrets to be used by our applications. Note that ESO supports several other sources apart from AWS. 

The steps we are going to take are

  1. Install ESO in our Kubernetes cluster
  2. Create secrets in AWS secret manager
  3. Instructing ESO to fetch the secrets
  4. Creating an Argo CD application manifest for our deployment
  5. Deploying everything with the Codefresh runtime.

Note that we assume that your cluster already has a Codefresh runtime installed and it is available in the Codefresh platform.

Prerequisites

Before moving to the main setup we need to create a new  Git Source for your Codefresh  Runtime. When creating the Git source use the path of gitops/argocd

This is to make sure that the “infrastructure” manifests for the operator will be in a different folder than our application that will consume the secrets.

Installing the External Secrets Operator

The next step is to install the External Secrets Operator. To do that, we are going to add an Application to our Git Source. Create a file in your Git Source called external-secrets-operator.yaml and use the following application manifest to install it.

apiVersion: argoproj.io/v1alpha1
kind: Application
metadata:
  name: external-secrets-operator
  finalizers:
    - resources-finalizer.argocd.argoproj.io/foreground
spec:
  project: default
  source:
    repoURL: https://charts.external-secrets.io
    targetRevision: 0.9.11 # make sure to change this to the version you need
    chart: external-secrets
  destination:
    name: in-cluster
    namespace: external-secrets # you can use any namespace
  syncPolicy:
    automated:
      prune: false
      selfHeal: false
      allowEmpty: false
    syncOptions:
      - PrunePropagationPolicy=foreground
      - Replace=false
      - PruneLast=false
      - Validate=true
      - CreateNamespace=true
      - ApplyOutOfSyncOnly=false
      - ServerSideApply=true
      - RespectIgnoreDifferences=false

Once done, save, commit, and push to your Git repository.You should now see the operator when viewing the Codefresh application dashboard.

Notice that we treat the operator like any other GitOps application.

Using the External Secrets Operator

Installing the operator is an one-off process. To actually use the External Secret Operator in a GitOps application we need to do the following:

  1. Define Secrets in AWS secret manager
  2. Instruct ESO to bring them to our deployment cluster
  3. Deploy the application that is using the secrets.

Let’s see these steps in order.

Connect ESO with Amazon Secret Manager

ESO supports several sources of secrets, so we must set it up to work specifically with AWS for our use case. First, we need to create an IAM Role for Service Accounts (IRSA) that is going to be used to access the secrets. You will need to create a role based on the EKS Documentation. Below are the minimum permissions needed to access the secrets that start with testing/. You can edit the Resource section that suits your needs.

{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Effect": "Allow",
            "Action": [
                "secretsmanager:GetResourcePolicy",
                "secretsmanager:GetSecretValue",
                "secretsmanager:DescribeSecret",
                "secretsmanager:ListSecretVersionIds"
            ],
            "Resource": [
                "arn:aws:secretsmanager:us-east-1:<ACCOUNT_ID>>:secret:testing/*"
            ]
        }
    ]
}

Once the IRSA is created, create a secret in AWS Secrets Manager (region us-east-1 in this example). Use the “Other type of Secret” when creating the secret.

Add the key-value pairs that you want. When naming, use the prefix of testing/ for this example. For the rest of the options, use the defaults.

The secrets are now in AWS and we need to bring them into our deployment cluster.

Adding Secrets to Git Source

Now that we have everything set up on AWS, time to create a Service Account, Secret Store, and External Secret. First, create a Directory in your Git Source Repo that’s outside of the application path for the Git Source. 

In this example, our Git Source path is gitops/argocd (for infra) while gitops/test-applications is used for the actual applications.

├── gitops
│   ├── argocd
│   │   └── external-secrets-operator.yaml
│   ├── test-applications

Inside the test-applications directory, create a file called secret-store.yaml. Here we will create a Service Account and a Secret Store Config. The Secret Store will allow us to access the AWS Secrets Manager and use the Service Account to make the API Calls to AWS.

apiVersion: external-secrets.io/v1beta1
kind: SecretStore
metadata:
  name: secretstore-sample # the name you want to call the Secret Store
spec:
  provider:
    aws:
      service: SecretsManager # Specifing AWS Secret Manager
      region: us-east-1
      auth:
        jwt:
          serviceAccountRef:
            name: aws-secret-store # use an SA with IRSA to gain access to the Secrets.
---
apiVersion: v1
kind: ServiceAccount
metadata:
  annotations:
    eks.amazonaws.com/role-arn: arn:aws:iam::<ACCOUNT_ID>:role/<ROLE_NAME> # The Role that you created to have access to Secrets Manager 
  name: aws-secret-store

Now create another file called external-secret.yaml in the testing-applications directory. This is how we are going to generate a kubernetes secret. We will define a refresh interval so the secret is up to date in the cluster, how to access the secret via the Secret Store, the name of the secret in AWS Secret Manager, and what to name the k8s secret kind once retrieved.

apiVersion: external-secrets.io/v1beta1
kind: ExternalSecret
metadata:
  name: example-secret
spec:
  refreshInterval: 1h # allows us to update secret. Be aware of API limits for AWS if doing shorter intervals.
 secretStoreRef:
    kind: SecretStore
    name: secretstore-sample # name of the secret store so we can access AWS Secret Manager
  target:
    name: my-secret  # name of the k8s Secret to be created. aka kind: Secert
    creationPolicy: Owner
  dataFrom:
  - extract:
      key: testing/my-secret  # name of the secret from AWS Secret Manager
      conversionStrategy: Default
      decodingStrategy: None
      metadataPolicy: None

We now have integrated our Kubernetes cluster (that runs ESO) with secrets in AWS.

Adding an Application that consumes secrets for AWS

The last piece of the puzzle is to deploy an application that needs those secrets. We are going to add another application under gitops/argocd that will contain our example for fetching the secrets.

Create a file called test-application.yaml with the following values.

apiVersion: argoproj.io/v1alpha1
kind: Application
metadata:
  name: test-application
  finalizers:
    - resources-finalizer.argocd.argoproj.io/foreground
  annotations:
    codefresh.io/app-group: example
spec:
  project: default
  source:
    path: gitops/test-applications
    repoURL: https://github.com/OWNER/REPO.git # replace with your repo URL
    targetRevision: main
  destination:
    name: in-cluster
    namespace: testing
  syncPolicy:
    automated:
      prune: false
      selfHeal: false
      allowEmpty: false
    syncOptions:
      - PrunePropagationPolicy=foreground
      - Replace=false
      - PruneLast=false
      - Validate=true
      - CreateNamespace=true
      - ApplyOutOfSyncOnly=false
      - ServerSideApply=true
      - RespectIgnoreDifferences=false

This application is going to be deployed in the testing namespace. All the items before are name-spaced specific and can be reused in different namespaces. Now save, commit, and push these items to your git repo.

Now you should see the Application in your dashboard. You should see a k8s secret that has all the key-value pairs that you created in AWS Secrets Manager.

You can also run the following command to verify the existence of the secret inside the cluster.

kubectl get secrets -n testing my-secret -o yaml

You should see something similar for your secret.

apiVersion: v1
data:
  domain: EXAMPLEguaW8=
  email: EXAMPLEguaW8=
  token: EXAMPLEguaW8=
immutable: false
kind: Secret
metadata:
  annotations:
    argocd.argoproj.io/tracking-id: test-application:external-secrets.io/ExternalSecret:testing/example-secret
    reconcile.external-secrets.io/data-hash: 77cc2a82549cebdce8da1382d888f919
  creationTimestamp: "2024-02-07T19:36:49Z"
  labels:
    app.kubernetes.io/instance: test-application
    reconcile.external-secrets.io/created-by: 1cef718909ba30c1b96f7026c513b567
  name: my-secret
  namespace: testing
  ownerReferences:
  - apiVersion: external-secrets.io/v1beta1
    blockOwnerDeletion: true
    controller: true
    kind: ExternalSecret
    name: example-secret
    uid: dfb6d035-2e76-419b-b287-b2e98d6cb8af
  resourceVersion: "47288"
  uid: 99ef0789-4d70-486f-b846-9c9ef685612c
type: Opaque

The application should be up and running. The main application manifests are located in Git, but the secrets are dynamically fetched from AWS secret manager.

Conclusion

In this blog post we have seen how to use the AWS secret manager with GitOps applications. We have shown you how to

  1. Install the External secret operator in your cluster
  2. Define secrets that will be stored in AWS
  3. Integrate your Kubernetes cluster with the operator and the stored secrets
  4. Deploy an example application that consumes the secrets.

The big advantage of course is that the External Secret manifests of the operator do not contain any confidential information. You can safely commit them to git without worrying about sensitive information being exposed. 

As an added bonus, your Kubernetes secrets will get automatically generated, and refreshed when the secret changes in AWS Secrets Manager. 

If you would like to see an alternative way of using secrets from AWS, we also have a post on using ArgoCD Vault Plugin with the Codefresh GitOps Runtime.

How useful was this post?

Click on a star to rate it!

Average rating 4.6 / 5. Vote count: 14

No votes so far! Be the first to rate this post.

Ready to Get Started?
  • safer deployments
  • More frequent deployments
  • resilient deployments