Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ayoul3/asm-webhook
Kubernetes mutating webhook for fetching secrets from SecretsManager at runtime
https://github.com/ayoul3/asm-webhook
aws container docker eks kubernetes secretsmanager
Last synced: 22 days ago
JSON representation
Kubernetes mutating webhook for fetching secrets from SecretsManager at runtime
- Host: GitHub
- URL: https://github.com/ayoul3/asm-webhook
- Owner: ayoul3
- License: apache-2.0
- Created: 2021-02-07T10:54:26.000Z (almost 4 years ago)
- Default Branch: master
- Last Pushed: 2021-03-07T21:36:58.000Z (almost 4 years ago)
- Last Synced: 2024-11-19T01:42:08.747Z (about 2 months ago)
- Topics: aws, container, docker, eks, kubernetes, secretsmanager
- Language: Go
- Homepage:
- Size: 7.58 MB
- Stars: 4
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# ASM Webhook
asm-webhook is a mutating Webhook designed to dynamically fetch secrets from SecretsManager and inject them as env variables in pods.
It is heavily inspired by the great [Banzai Vaults](https://github.com/banzaicloud/bank-vaults/tree/master/charts/vault-secrets-webhook) that supports Vault.
## How does it work?
1. InterceptionA mutating webhook will intercept all pod submissions bearing the annotation `asm-webhook:"true"`.
2. Validation
It will go through all its environment variables and secrets looking for SecretsManager ARNs. If found, the Pod will be mutated.
3. Mutation
An init container `ayoul3/asm-env` is injected which copies its main binary `asm-env` to a volume `/asm` shared with the other containers in the pod.
4. Execution
The command of the container is overwritten so that it first starts a binary `asm-env` that will decrypt SecretsManager secrets. After which it will start the original command with its arguments.
```yaml
Command: sh
Args:
- -c
- trap 'exit' TERM; while :; do sleep 1; echo decrypted $KEY_ID; done
```
becomes:
```yaml
Command: /asm/asm-env
Args:
- sh
- -c
- trap 'exit' TERM; while :; do sleep 1; echo decrypted $KEY_ID; done
```## Install
Most default values work just fine if you want to test on minikube.
On an EKS cluster change at least the roleArn in `./chart/values.yaml` to point to an IAM role **capable of pulling ECR images**:
```yaml
serviceAccountName: webhook
roleArn: "arn:aws:iam::1111111111111:role/webhook"
```Once you're done, execute the `generate.sh` script to provision certificates used by the webhook:
```bash
./generate.sh
```
This command will prepare certificates and store them a `secret` resource in Kube that will be mounted and used by the Webhook.If everything goes alright, you can then deploy the chart using helm:
```bash
helm upgrade --install asm-webhook ./chart
```Check that the webhook is running:
```bash
$ kubectl get mutatingwebhookconfigurationsNAME WEBHOOKS AGE
asm-webhook.default.svc 1 17s$ k get deployments
NAME READY UP-TO-DATE AVAILABLE AGE
asm-webhook 1/1 1 1 17s
...snip...
```Try deploying [pod.yaml](./sample/pod.yaml) file to test your webhook. Make sure to put in a proper secret ARN.
## Test
To test that your installation succeeded, try submitting the sample pod in the sample folder.
**First change the serviceaccount and the secret's ID to match your setup.**
```bash
$ kubectl deploy -f sample/pod.yaml -n default
pod "asm-sample-pod" created$ kubectl logs asm-sample-pod
time="2021-02-14T13:39:45Z" level=debug msg="Preparing env variables"
time="2021-02-14T13:39:45Z" level=debug msg="Decrypting key ID arn:aws:secretsmanager:eu-west-1:111111111111:secret:/key1-mIdVIP#password"
time="2021-02-14T13:39:45Z" level=debug msg="Setting secret value in env var KEY_ID"
time="2021-02-14T13:39:45Z" level=debug msg="Looking for nested key password in secret arn:aws:secretsmanager:eu-west-1:111111111111:secret:/key1-mIdVIP#password"
time="2021-02-14T13:39:45Z" level=debug msg="Found nested key password in secret arn:aws:secretsmanager:eu-west-1:111111111111:secret:/key1-mIdVIP#password"
time="2021-02-14T13:39:45Z" level=debug msg="Found absolute path /bin/sh"
decrypted value password
```## Usage
Add the following label to the pods or deployments you wish to send to the webhook:
```yaml
metadata:
labels:
asm-webhook: "true"
...
env:
- name: KEY_ID
value: arn:aws:secretsmanager:us-east-1:111111111111:secret:key-us-cmo1Hc
```
It's the service account of the pod that will fetch the secret so make sure to give it proper access rights.
Check the [pod.yaml](./sample/pod.yaml) file for a full example## Prerequisites
It is the pod that fetches its own secrets, so obviously it needs to use a service account mapped to an IAM role capable of reading such secrets. You can read more about it [here](https://docs.aws.amazon.com/eks/latest/userguide/create-service-account-iam-policy-and-role.html) and find an actual example [here](https://aws.amazon.com/blogs/containers/aws-secrets-controller-poc/).
The webhook needs a service account capable of getting container manifest from the registry hosting the container image. This capability is only used when the submitted container lacks the `command` argument.
If you are using ECR, grant read-only access to the role mapped to the service account `webhook`.
The default `AmazonEC2ContainerRegistryReadOnly` works just fine.You can put any value if you only use public images, or you always specify the command argument for your containers.
## Secret formats
Secrets stored in SecretsManager can be of two formats:
* Simple strings
* Flat JSON
You can specify which JSON key to fetch by adding it after the character **#**.If the secret `arn:aws:secretsmanager:us-east-1:111111111111:secret:key-us-cmo1Hc` contains `{"user":"test", "password": "secret"}` then you can choose to only fetch the password key as such:
```yaml
DB_PASS: arn:aws:secretsmanager:us-east-1:886477354405:secret:key-us-cmo1Hc#password
```
*Nested keys are not supported at the moment.*## Annotations
The webhook supports the following annotations at the pod level:
| Tables | Decription | Value |
| ----------------------------------- |:-------------:| :----------:|
| asm.webhook.debug | Activate debug logs for the webhook handler of this pod | false/true
| asm.webhook.asm-env.image | image of the init container to inject | `ayoul3/asm-env:latest`
| asm.webhook.asm-env.path | path to binary inside the init container that will fetch secrets | /app/
| asm.webhook.asm-env.bin | name of the binary that will fetch secrets | asm-env
| asm.webhook.asm-env.mountPath | mount path where containers will find this binary | /asm/
| asm.webhook.asm-env.skip-cert-check | skip certificate check when contacting SecretsManager. Useful for bare pods without `ca-certificates` package | false## asm-env
asm-env binary is maintained as a [separate project](https://github.com/ayoul3/asm-env).
## Monitoring
The webhook exposes Prometheus metrics on `/metrics`. A [grafana dashboard](https://grafana.com/grafana/dashboards/13685) is also available courtesy of Xabier Larrakoetxea.
## Credit
[Banzai Vaults](https://github.com/banzaicloud/bank-vaults/tree/master/charts/vault-secrets-webhook)
[Kubewebhook](https://github.com/slok/kubewebhook)## Author
Ayoub Elaassal