Workload identity is the best practice for authenticating as a service account when running on GKE.
See https://cloud.google.com/kubernetes-engine/docs/how-to/workload-identity for how this works
First enable workload identity on the cluster and all node-pools in the cluster:
enable-workload-identity.sh K8S_PROJECT ZONE CLUSTER
Next ensure the kubernetes service account exists and that it has an
iam.gke.io/gcp-service-account
annotation. This associates it with the desired
GCP service account.
apiVersion: v1
kind: ServiceAccount
metadata:
annotations:
iam.gke.io/gcp-service-account: [email protected]
name: SOMETHING
namespace: SOMEWHERE
Once this service account exists in the cluster, then an owner of
[email protected]
-- typically a PROJECT
owner --
runs bind-service-accounts.sh
:
bind-service-accounts.sh \
K8S_PROJECT ZONE CLUSTER SOMEWHERE SOMETHING \
[email protected]
This script assumes the same person can access both K8S_PROJECT
and
PROJECT
. If that is not true then the PROJECT
owner can just run this
command directly:
# Note: K8S_PROJECT is the project owning the GKE cluster
# whereas PROJECT owns the service account (may be the same)
gcloud iam service-accounts add-iam-policy-binding \
--project=PROJECT \
--role=roles/iam.workloadIdentityUser \
--member=serviceAccount:K8S_PROJECT.svc.id.goog[SOMEWHERE/SOMETHING] \
[email protected]
This is what tells GCP that the SOMEWHERE/SOMETHING
service account in
K8S_PROJECT
is authorized to act as
[email protected]
.
These are all described in the how-to GKE doc link at the top.
At this point any pod that:
- Runs in a
K8S_PROJECT
GKE cluster - Inside the
SOMEWHERE
namespace - Using the
SOMETHING
service
will authenticate as [email protected]
to GCP. The
bind-service-accounts.sh
script will verify this (see the how-to doc above for
the manual command).
Whenever you want a pod to authenticate this way, just configure the
serviceAccountName
on the PodSpec
. See an example
pod
and
prowjob:
Here are minimal deployment and prow job that use an image and args to print the
authenticated user to STDOUT
:
apiVersion: apps/v1
kind: Deployment
metadata:
name: foo
namespace: SOMEWHERE # from above
labels:
app: foo
spec:
replicas: 1
selector:
matchLabels:
app: foo
template:
metadata:
labels:
app: foo
spec:
serviceAccountName: SOMETHING # from above
containers:
- name: whatever
image: google/cloud-sdk:slim
args:
- gcloud
- auth
- list
periodics:
- name: foo
interval: 10m
decorate: true
spec:
serviceAccountName: SOMETHING # from above, note: namespace is chosen by prow
containers:
- image: google/cloud-sdk:slim
args:
- gcloud
- auth
- list