Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

gcp-oidc: manage API token in GCP Secret Manager #215

Merged
merged 18 commits into from
Sep 20, 2023
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@

<groupId>com.uid2</groupId>
<artifactId>uid2-operator</artifactId>
<version>5.8.13-20f23891c9</version>
<version>5.8.21-SNAPSHOT</version>

<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
Expand Down
5 changes: 4 additions & 1 deletion scripts/gcp-oidc/Dockerfile
Original file line number Diff line number Diff line change
@@ -1,8 +1,11 @@
FROM eclipse-temurin@sha256:de8e6219ff5360811a453a9237713679a9d9106ba5150290ef37fb23e246ce7d

LABEL "tee.launch_policy.allow_env_override"="API_TOKEN,DEPLOYMENT_ENVIRONMENT,CORE_BASE_URL,OPTOUT_BASE_URL"
LABEL "tee.launch_policy.allow_env_override"="API_TOKEN_SECRET_NAME,DEPLOYMENT_ENVIRONMENT,CORE_BASE_URL,OPTOUT_BASE_URL"
LABEL "tee.launch_policy.log_redirect"="always"

# Install Packages
RUN apk update && apk add jq

WORKDIR /app
EXPOSE 8080

Expand Down
38 changes: 30 additions & 8 deletions scripts/gcp-oidc/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ Run below from [Google Cloud Console](https://console.cloud.google.com/):

3. Enable the following APIs:
```
$ gcloud services enable compute.googleapis.com confidentialcomputing.googleapis.com
$ gcloud services enable compute.googleapis.com confidentialcomputing.googleapis.com secretmanager.googleapis.com
```

4. Create a service account to run the workload:
Expand All @@ -76,6 +76,12 @@ Run below from [Google Cloud Console](https://console.cloud.google.com/):
--member=serviceAccount:{SERVICE_ACCOUNT_NAME}@{PROJECT_ID}.iam.gserviceaccount.com \
--role=roles/logging.logWriter
```
- `roles/secretmanager.secretAccessor`, grants the ability to access operator API token that is managed in Secret Manager.
```
$ gcloud projects add-iam-policy-binding {PROJECT_ID} \
--member=serviceAccount:{SERVICE_ACCOUNT_NAME}@{PROJECT_ID}.iam.gserviceaccount.com \
--role=roles/secretmanager.secretAccessor
```

6. Add VPC rule to allow public 8080 access (default exposed port of UID2 operator):
```
Expand Down Expand Up @@ -104,32 +110,49 @@ Go to Admin portal [Enclave Id Management page](https://admin-integ.uidapi.com/a
- Protocol: "gcp-oidc"
- Enclave ID: the generated value in Step 1

### (For partner) Create secret of your private operator API token in Secret Manager
Store your private operator API token provided by the UID2 team to Secret Manager and get the secret name which will be used to replace the `{API_TOKEN_SECRET_NAME}` placeholder later during VM instance creation.

For example, following script creates a new secret `uid2_operator_api_token`, and prints secret name something like `projects/111111111111/secrets/uid2_operator_api_token/versions/1` which will be used to replace the `{API_TOKEN_SECRET_NAME}` placeholder later.
```
API_TOKEN="<YOUR_OPERATOR_API_TOKEN>"
lunwang-ttd marked this conversation as resolved.
Show resolved Hide resolved
echo -n $API_TOKEN | gcloud secrets create uid2_operator_api_token \
--replication-policy="automatic" \
--data-file=-

gcloud secrets versions describe latest --secret uid2_operator_api_token --format 'value(name)'
```

### (For partner) Create VM Instance
There are a few placeholders that you need to replace in below command:
- `{INSTANCE_NAME}`: your VM name, can be changed as your need.
- `{ZONE}`: which Google Cloud zone will be deployed on.
- `{SERVICE_ACCOUNT}`: in `{SERVICE_ACCOUNT_NAME}@{PROJECT_ID}.iam.gserviceaccount.com` format, the one you created
in Prerequisites phase.
- `{IMAGE_SHA}`: a valid UID2 operator image digest. You should have received this from UID2 team.
- `{API_TOKEN}`: private operator api token, dedicated for you. You should have received this from UID2 team.
- `{API_TOKEN_SECRET_NAME}`: the secret name of your operator API token created in [the above section](#for-partner-create-secret-of-your-private-operator-api-token-in-secret-manager), the format is
`projects/<project_id>/secrets/<secret_id>/versions/<version>`

```
$ gcloud compute instances create {INSTANCE_NAME} \
--zone {ZONE} \
--machine-type n2d-standard-2 \
--confidential-compute \
--shielded-secure-boot \
--maintenance-policy Terminate \
--scopes cloud-platform \
--image-project confidential-space-images \
--image-family confidential-space \
--service-account {SERVICE_ACCOUNT} \
--metadata ^~^tee-image-reference=ghcr.io/iabtechlab/uid2-operator@sha256:{IMAGE_SHA}~tee-restart-policy=Never~tee-container-log-redirect=true~tee-env-DEPLOYMENT_ENVIRONMENT=integ~tee-env-API_TOKEN={API_TOKEN}
--metadata ^~^tee-image-reference=ghcr.io/iabtechlab/uid2-operator@sha256:{IMAGE_SHA}~tee-restart-policy=Never~tee-container-log-redirect=true~tee-env-DEPLOYMENT_ENVIRONMENT=integ~tee-env-API_TOKEN_SECRET_NAME={API_TOKEN_SECRET_NAME}
```

## Production Deployment

We can deploy new UID2 Operator in GCP Confidential Space Enclave into Production Environment by following the same process as for
Integration.

You will be provided a new `{API_TOKEN}`, and `~tee-env-DEPLOYMENT_ENVIRONMENT=integ~` needs to be changed to
You will be provided a new operator API token which should be stored in Secret Manager, and `~tee-env-DEPLOYMENT_ENVIRONMENT=integ~` needs to be changed to
`~tee-env-DEPLOYMENT_ENVIRONMENT=prod~`.

It is recommended that you also specify the machine type in the gcloud script. Currently, it is recommended to run the
Expand All @@ -139,6 +162,7 @@ An example of the script is given below:

```
$ gcloud compute instances create {INSTANCE_NAME} \
--zone {ZONE} \
--machine-type n2d-standard-16 \
--confidential-compute \
--shielded-secure-boot \
Expand All @@ -147,12 +171,10 @@ $ gcloud compute instances create {INSTANCE_NAME} \
--image-project confidential-space-images \
--image-family confidential-space \
--service-account {SERVICE_ACCOUNT} \
--metadata ^~^tee-image-reference=ghcr.io/iabtechlab/uid2-operator@sha256:{IMAGE_SHA}~tee-restart-policy=Never~tee-container-log-redirect=true~tee-env-DEPLOYMENT_ENVIRONMENT=prod~tee-env-API_TOKEN={API_TOKEN}
--metadata ^~^tee-image-reference=ghcr.io/iabtechlab/uid2-operator@sha256:{IMAGE_SHA}~tee-restart-policy=Never~tee-container-log-redirect=true~tee-env-DEPLOYMENT_ENVIRONMENT=prod~tee-env-API_TOKEN_SECRET_NAME={API_TOKEN_SECRET_NAME}
```

Note that compared to the `gcloud` command used in the prior section, an additional option
`--machine-type n2d-standard-16` is added, which ensures production deployment of UID2 Operator runs on
the recommended machine type that matches the production configuration.
Note that compared to the `gcloud` command used in the prior section, parameter `--machine-type n2d-standard-16` is set to ensure production deployment of UID2 Operator runs on the recommended machine type for production.

## Upgrading

Expand Down
17 changes: 15 additions & 2 deletions scripts/gcp-oidc/entrypoint.sh
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,23 @@
# This script must be compatible with Ash (provided in eclipse-temurin Docker image) and Bash

# -- set API tokens
if [ -z "${API_TOKEN}" ]; then
echo "API_TOKEN cannot be empty"
if [ -z "${API_TOKEN_SECRET_NAME}" ]; then
echo "API_TOKEN_SECRET_NAME cannot be empty"
exit 1
fi

GCP_TOKEN=$(wget "http://metadata.google.internal/computeMetadata/v1/instance/service-accounts/default/token" -q --header "Metadata-Flavor: Google" -O - | jq -e -r ".access_token")
if [ $? -ne 0 -o -z "${GCP_TOKEN}" ]; then
echo "Failed to get GCP token"
exit 1
fi

API_TOKEN=$(wget "https://secretmanager.googleapis.com/v1/${API_TOKEN_SECRET_NAME}:access" -q --header "authorization: Bearer ${GCP_TOKEN}" --header "content-type: application/json" -O - | jq -e -r ".payload.data" | base64 -d)
if [ $? -ne 0 -o -z "${API_TOKEN}" ]; then
echo "Failed to get API token"
exit 1
fi

export core_api_token="${API_TOKEN}"
export optout_api_token="${API_TOKEN}"

Expand Down