#!/bin/bash # This is an entrypoint for our Docker image that does some minimal bootstrapping before executing. set -e # The default backend URL is the Pulumi Managed Service. # To use one of the alternate supported backends, set the # PULUMI_BACKEND_URL env var according to: # https://www.pulumi.com/docs/intro/concepts/state/#to-a-self-managed-backend. pulumi login $PULUMI_BACKEND_URL # If the PULUMI_CI variable is set, we'll do some extra things to make common tasks easier. if [ ! -z "$PULUMI_CI" ]; then # Capture the PWD before we go and potentially change it. ROOT=$(pwd) # If the root of the Pulumi project isn't the root of the repo, CD into it. if [ ! -z "$PULUMI_ROOT" ]; then cd $PULUMI_ROOT fi # Detect the CI system and configure variables so that we get good Pulumi workflow and GitHub App support. if [ ! -z "$GITHUB_WORKFLOW" ]; then export PULUMI_CI_SYSTEM="GitHub" export PULUMI_CI_BUILD_ID= export PULUMI_CI_BUILD_TYPE= export PULUMI_CI_BUILD_URL= export PULUMI_CI_PULL_REQUEST_SHA="$GITHUB_SHA" # For PR events, we want to take the ref of the target branch, not the current. This ensures, for # instance, that a PR for a topic branch merging into `master` will use the `master` branch as the # target for a preview. Note that for push events, we of course want to use the actual branch. if [ "$PULUMI_CI" = "pr" ]; then # Not all PR events warrant running a preview. Many of them pertain to changes in assignments and # ownership, but we only want to run the preview if the action is "opened", "edited", or "synchronize". PR_ACTION=$(jq -r ".action" < $GITHUB_EVENT_PATH) if [ "$PR_ACTION" != "opened" ] && [ "$PR_ACTION" != "edited" ] && [ "$PR_ACTION" != "synchronize" ]; then echo -e "PR event ($PR_ACTION) contains no changes and does not warrant a Pulumi Preview" echo -e "Skipping Pulumi action altogether..." exit 0 fi BRANCH=$(jq -r ".pull_request.base.ref" < $GITHUB_EVENT_PATH) else BRANCH="$GITHUB_REF" fi BRANCH=$(echo $BRANCH | sed "s/refs\/heads\///g") fi # Respect the branch mappings file for stack selection. Note that this is *not* required, but if the file # is missing, the caller of this script will need to pass `-s ` to specify the stack explicitly. if [ ! -z "$BRANCH" ]; then if [ -z "$PULUMI_STACK_NAME" ]; then if [ -e $ROOT/.pulumi/ci.json ]; then PULUMI_STACK_NAME=$(cat $ROOT/.pulumi/ci.json | jq -r ".\"$BRANCH\"") else # If there's no stack mapping file, we are on master, and there's a single stack, use it. PULUMI_STACK_NAME=$(pulumi stack ls | awk 'FNR == 2 {print $1}' | sed 's/\*//g') fi fi if [ ! -z "$PULUMI_STACK_NAME" ] && [ "$PULUMI_STACK_NAME" != "null" ]; then pulumi stack select $PULUMI_STACK_NAME else echo -e "No stack configured for branch '$BRANCH'" echo -e "" echo -e "To configure this branch, please" echo -e "\t1) Run 'pulumi stack init '" echo -e "\t2) Associated the stack with the branch by adding" echo -e "\t\t{" echo -e "\t\t\t\"$BRANCH\": \"\"" echo -e "\t\t}" echo -e "\tto your .pulumi/ci.json file" echo -e "" echo -e "For now, exiting cleanly without doing anything..." exit 0 fi fi fi # For Google, we need to authenticate with a service principal for certain authentication operations. if [ ! -z "$GOOGLE_CREDENTIALS" ]; then export GOOGLE_APPLICATION_CREDENTIALS="$(mktemp).json" # Check if GOOGLE_CREDENTIALS is base64 encoded if [[ $GOOGLE_CREDENTIALS =~ ^([A-Za-z0-9+/]{4})*([A-Za-z0-9+/]{3}=|[A-Za-z0-9+/]{2}==)?$ ]]; then echo "$GOOGLE_CREDENTIALS"|base64 -d > $GOOGLE_APPLICATION_CREDENTIALS # unset for other gcloud commands using this variable. unset GOOGLE_CREDENTIALS else echo "$GOOGLE_CREDENTIALS" > $GOOGLE_APPLICATION_CREDENTIALS fi gcloud auth activate-service-account --key-file=$GOOGLE_APPLICATION_CREDENTIALS gcloud --quiet auth configure-docker $GOOGLE_DOCKER_HOSTNAME_LIST fi # Next, run npm install. We always call this, as # previous calls (stack select, login, preview) will run the # npm scripts that install plugins; plugin installation directory # is outside of the persisted filesystem (GitHub Actions). # So we need to run this everytime to ensure the plugins are # always available. This shouldn't cause a performance hit, # as the node_modules/lock file will be persisted across runs. # # Similarly, run yarn install as applicable for the same reasons. if [ -e package.json ]; then if [ -f yarn.lock ] || [ ! -z $USE_YARN ]; then yarn install else # Set npm auth token if one is provided. if [ ! -z "$NPM_AUTH_TOKEN" ]; then echo "//registry.npmjs.org/:_authToken=$NPM_AUTH_TOKEN" > ~/.npmrc fi if [ -f package-lock.json ] || [ -f npm-shrinkwrap.json ]; then npm ci else npm install fi fi fi # If the user is running the Python SDK, we will need to install their requirements as well. if [ -e requirements.txt ]; then # Check if should use venv PULUMI_VENV=$(cat Pulumi.yaml | grep "virtualenv:" | cut -d':' -f2) if [ -z $PULUMI_VENV ]; then pip3 install -r requirements.txt else python3 -m venv $PULUMI_VENV $PULUMI_VENV/bin/python -m pip install --upgrade pip setuptools wheel $PULUMI_VENV/bin/python -m pip install -r requirements.txt fi fi # Now just pass along all arguments to the Pulumi CLI, sending the output to a file for # later use. Note that we exit immediately on failure (under set -e), so we `tee` stdout, but # allow errors to be surfaced in the Actions log. PULUMI_COMMAND="pulumi $*" OUTPUT_FILE=$(mktemp) echo "#### :tropical_drink: \`$PULUMI_COMMAND\`" bash -c "$PULUMI_COMMAND" | tee $OUTPUT_FILE EXIT_CODE=${PIPESTATUS[0]} # If the GitHub action stems from a Pull Request event, we may optionally leave a comment if the # COMMENT_ON_PR is set. COMMENTS_URL=$(cat $GITHUB_EVENT_PATH | jq -r .pull_request.comments_url) if [ ! -z $COMMENTS_URL ] && [ ! -z $COMMENT_ON_PR ]; then if [ -z $GITHUB_TOKEN ]; then echo "ERROR: COMMENT_ON_PR was set, but GITHUB_TOKEN is not set." else COMMENT="#### :tropical_drink: \`$PULUMI_COMMAND\` \`\`\` $(cat $OUTPUT_FILE) \`\`\`" PAYLOAD=$(echo '{}' | jq --arg body "$COMMENT" '.body = $body') echo "Commenting on PR $COMMENTS_URL" curl -s -S -H "Authorization: token $GITHUB_TOKEN" -H "Content-Type: application/json" --data "$PAYLOAD" "$COMMENTS_URL" fi fi exit $EXIT_CODE