This SPIFFE demo repository aims to showcase the possibilities of different SPIFFE use-cases and requirements end-users might have in real production environments to integrate with their applications.
This repo contains the following:
- Golang application
- Terraform
- Kubernetes deployment
- Dockerfiles
The architecture of the SPIFFE-demo applications is the following when deployed:
A simple Golang tool to showcase SPIFFE possibilities. It has 3 subcommands:
- customer
- backend
- httpbackend
The customer is the entry point for customers through an Ingress. It serves a simple webserver that is exposed over an Ingress and shows a page with buttons that allows an end-user to take actions. The following actions can be taken:
- Connect to a SPIFFE server backend. This connects to another application that runs with the backend subcommand. The connection is SPIFFE authenticated and authorized. This showcases the potential when SPIFFE is integrated in the application layer.
- Connect to a non-SPIFFE server backend. connects to another application that runs with the httpbackend subcommand. In the Kubernetes deployment we have put an Envoy in front that will authenticate and authorize the SPIFFE connection. This showcases the potential when SPIFFE can't be integrated in the application layer
- Talk to AWS Services. This writes and reads from an AWS S3 bucket with a SPIFFE JWT identity. In the container we abstract everything away from the application (for the application it is as it would run natively in AWS). This is done through the spiffe-aws-assume-role binary. That binary gets called through the AWS Profile
credential_process
. - Talk to Google Cloud Service. This writes and reads from an GCS bucket with a SPIFFE JWT identity. In the container we abstract everything away from the application (for the application it is as it would run natively in Google Cloud). This is done through the spiffe-gcp-proxy proxy. That proxy gets called when making a call to the internal metadata API of Google Cloud.
- Talk to a PostgreSQL database with its SVID. It writes a randomly generated user to a database every time you click the button. With the retrieval function it will retrieve all previous generated users from the database. No username or password authentication is required. It uses the SPIFFE-helper to let PostgreSQL consume the SVID that it got issued. The SPIFFE-helper is responsible for writing it to an in-memory filesystem that is accessible by the PostgreSQL container and than reloads the PostgreSQL config to make sure that PostgreSQL is aware of the latest certificates. As PostgreSQL doesn't understand SPIFFE IDs, it does verification based on the CN on the X.509. By configuring SPIRE in such a way, it will create those extra entries for the application SVID and that way it can authenticate and authorize itself to PostgreSQL
- A SPIFFE retriever endpoint
HOSTNAME/spifferetriever
to show the SVID details.
The setup of the OIDC federation between our SPIRE install with AWS and Google Cloud happens through Terraform. It also creates the necessary GCS, S3 buckets and IAM roles and policies so our customer application can authenticate to AWS and Google Cloud.
The different manifests to deploy the different components to Kubernetes.
We need to package our Golang tool so it can be deployed in our Kubernetes cluster. For the customer application we also require an init-container that sets the AWS config correctly. All of these images have been published to Docker hub.
The initcontainer is responsible for creating an AWS Profile config and making that available to the customer
application.
Adds 2 extra things to the default PostgreSQL image:
init-user-db.sh
: Creates the necessary database, user and tables in the PostgreSQL during initializationset-pg-hba.sh
: Overwrites the defaultpg_hba.conf
during initialization to force authentication and authorization with certificate instead of the normal username/password
Creates a Docker image for the latest version of the SPIFFE-helper. The SPIFFE-helper will than be used as a sidecar container next to the PostgreSQL container to make sure PostgreSQL always has the latest SVID loaded.
Creates a Docker image for the latests version of the spiffe-gcp-proxy. The SPIFFE GCP Proxy will than be used as a sidecar container next to our customer application to intercept calls to Google Cloud and add the necessary SPIFFE authentication credentials to it.
The Golang application gets built with golang through a built container. Afterwards ca-certificates
and the spiffe-aws-assume-role
binary get added to it as well.
To be able to run this demo, there are a few prerequisites
- Ingress controller that is publicly exposed. This is required to make the OIDC endpoint available as well as the demo application.
- cert-manager installed and configured with a ClusterIssuer
letsencrypt-prod
to be able to request certificates for your OIDC endpoint and the demo application. - DNS hostnames configured for OIDC and the demo application. These need to your Ingress controller.
The AWS and Google Cloud bits are optional. When you deploy the spiffe-demo
application on your Kubernetes cluster, it will deploy everything but off-course if you haven't done the cloud provider setup those specific bits will not work.
Some values are going to be specific to your environment. We are going to prepare these now:
export DEMO_HOSTNAME=demo.yourdomain.com
export DEMO_ROGUE_HOSTNAME=demo-rogue.yourdomain.com
export OIDC_HOSTNAME=oidc.yourdomain.com
export AWS_BUCKET_NAME=myrandombucketname
export GCP_BUCKET_NAME=mygcpbucketname
export GCP_PROJECT_NAME=my-gcp-project-name
export GCP_PROJECT_NUMBER=11111111
sed -i '' "s/OIDC_HOSTNAME/$OIDC_HOSTNAME/g" deploy/spire/values.yaml
sed -i '' "s/OIDC_HOSTNAME/$OIDC_HOSTNAME/g; s/AWS_BUCKET_NAME/$AWS_BUCKET_NAME/g" deploy/terraform/aws/variables.tf
sed -i '' "s/OIDC_HOSTNAME/$OIDC_HOSTNAME/g; s/GCP_BUCKET_NAME/$GCP_BUCKET_NAME/g; s/GCP_PROJECT_NAME/$GCP_PROJECT_NAME/g" deploy/terraform/google/variables.tf
sed -i '' "s/AWS_BUCKET_NAME/$AWS_BUCKET_NAME/g; s/GCP_BUCKET_NAME/$GCP_BUCKET_NAME/g; s/GCP_PROJECT_NAME/$GCP_PROJECT_NAME/g; s/GCP_PROJECT_NUMBER/$GCP_PROJECT_NUMBER/g; s/DEMO_HOSTNAME/$DEMO_HOSTNAME/g; s/DEMO_ROGUE_HOSTNAME/$DEMO_ROGUE_HOSTNAME/g" deploy/chart/spiffe-demo/values.yaml
Install SPIRE with Helm. Make sure that the OIDC endpoint is publicly available as AWS will need to be able to hit that endpoint. You can take a look at the values.yaml
that is used.
helm upgrade --install -n spire spire-crds spire-crds --repo https://spiffe.github.io/helm-charts-hardened/ --create-namespace
# Change ./deploy/spire/values.yaml to match your environment before running the next command
helm upgrade --install -n spire spire spire -f ./deploy/spire/values.yaml --repo https://spiffe.github.io/helm-charts-hardened/
Make sure you have access to an AWS account through the CLI, Terraform will use the same method to create the necessary resources. Take a look at deploy/terraform/aws/variables.tf
to verify the expected environment specific values.
cd deploy/terraform/aws
# Change variables.tf to match your environment before running the next command
terraform init
terraform apply
Make sure you have access to an Google Cloud account. Take a look at deploy/terraform/google/variables.tf
to verify the expected environment specific values.
cd deploy/terraform/google
# Change variables.tf to match your environment before running the next command
terraform init
terraform apply
Take a look deploy/chart/spiffe-demo/values.yaml
to verify it matches your environment and after that do a Helm install.
helm upgrade --install -n spiffe-demo2 spife-demo ./deploy/chart/spiffe-demo --create-namespace
Contributions of new use-cases or improvements are more than welcome!