概要
GitHub Actions を使用して Cloud Functions をデプロイできるようにする。
一度設定すれば、GitHub に push したソースを自動で Cloud Functions へ反映できる。
前提
なお、ここで扱う Cloud Funtctions は第一世代。
第二世代を扱う場合は実態が Cloud Run なので設定等は微妙に異なるはず。
https://cloud.google.com/functions/docs/concepts/version-comparison?hl=ja
設定方法
よく記事で見かける Secret にサービスアカウントキーを設定する方法ではなく、Workload Identity 連携を利用して設定を行う。
サービスアカウントキーを発行する必要がないので、セキュリティ上のリスクを抑えることができる。
手順
GCP 側の設定
ここではコンソール上で設定を行う。
gcloud コマンドを使う設定方法は、こちらの記事にわかりやすくまとまっている。
サービスアカウントを作成する
「IAM と管理」の 「サービスアカウント」のページを開き「サービスアカウントを作成」を選択。
サービスアカウント名とIDを設定し、必要なロールを付与する。
- Cloud Functions 開発者(roles/cloudfunctions.developer)
- サービスアカウントユーザー(roles/iam.serviceAccountUser)
Workload Identity プールを作成する
「IAM と管理」の「Workload Identity 連携」のページを開き、「使ってみる」を選択。
必要事項を入力する
- プール名:任意の名前
- プールID:任意のID
- デフォルトはプール名と同じ値
- 説明:空欄でもOK
プールにプロバイダを追加する
- プロバイダの選択:OpenID Connect(OIDC)
- プロバイダ名:任意の名前
- プロバイダID:任意のID
- 発行元 (URL):https://token.actions.githubusercontent.com
- JWK ファイル:空欄
- 指定しなければ発行元、つまり GitHub Actions から取得される
- オーディエンス:デフォルトのオーディエンス
プロバイダの属性を構成する
Google トークンの属性と GitHub Actions から送られるトークンの属性をマッピングする
- google.subject:assertion.sub
- 必須項目
- attribute.actor:assertion.actor
- 任意項目
- これ以外にも色々ある
公式ブログによると
属性マッピングは、GitHub Actions JWT のクレームを、リクエストについて行うことのできるアサーションにマップします(リポジトリや GitHub Action を呼び出すプリンシパルの GitHub ユーザー名など)。これらを使用して、--attribute-condition フラグで認証をさらに制限できます。たとえば、属性の repository 値をマップして、認証を特定のリポジトリに制限するためにこの値を後で使用することができます。
つまり、Google と GitHub Actions の属性情報をマッピングすることで、トークン発行を制限するための属性条件が設定できるということ。
任意の項目には設定したい属性条件に必要なものを設定する。
どんな属性が含まれるかはこちらを参照。
今回は自分しか触らないリポジトリで GitHub Actions を設定するため、「自分のアカウントIDで実行された場合だけトークンが発行される」という条件を追加した。
attribute.actor == <GitHubのアカウントID>
actor とよく似た属性に actor_id があるが、全くの別物なので注意
actor_id には数値が入るようだが、アカウントの何にあたるのかはわからなかった
サービスアカウントへのアクセス権を付与する
作成したプールの詳細画面で「アクセスを許可」を選択。
先ほど作成したサービスアカウントを指定し、プリンシパルを選択する。
これによって指定したサービスアカウントが持つ権限を一時的に借りられる。
属性名に指定した値が入っている外部ID(GCP外部からの接続)にだけ権限を与える。
この辺りは筆者も理解しきれていない部分もあるが、複数のリポジトリの GitHub Actions で使い回すことを想定して自分の GitHub アカウントが入っている外部IDに同様の権限を付与する設定にした。
登録後、構成ファイルをダウンロードできる。
このファイル自体は不要だが、audience に記載されたプロバイダのIDはこの後の工程で必要になる。
中身はこんな感じ
{
"type": "external_account",
"audience": "//iam.googleapis.com/projects/123456789/locations/global/workloadIdentityPools/github-actions-pool/providers/github-actions-provider",
"subject_token_type": "urn:ietf:params:oauth:token-type:jwt",
"token_url": "https://sts.googleapis.com/v1/token",
"service_account_impersonation_url": "https://iamcredentials.googleapis.com/v1/projects/-/serviceAccounts/github-actions@xxxxxxxxxxx.iam.gserviceaccount.com:generateAccessToken",
"credential_source": {
"url": "https://token.actions.githubusercontent.com",
"headers": {},
"format": {
"type": "json",
"subject_token_field_name": "access_token"
}
}
}
GitHub Actions 側の設定
リポジトリシークレットの設定(任意)
念の為に先ほど作成したプロバイダIDと、GCPプロジェクトのIDをシークレットとして設定する。
リポジトリから「Settings」→「Secrets and variables」→「Actions」へ遷移し「New repository secret」を選択。
- WORKLOAD_IDENTITY_PROVIDER_ID(プロバイダID)
- GCP_PROJECT_ID(プロジェクトID)
の2つを登録する。
プロバイダIDは先ほどの JSON の audience の project 以降のパスにあたる
projects/123456789/locations/global/workloadIdentityPools/github-actions-pool/providers/github-actions-provider
ymlファイルの作成
リポジトリのルートディレクトリに .github/workflows ディレクトリを作成し、その中に任意の名前の yml ファイルを作成する。
name: Deploy Cloud Functions
on:
# mainへのPush時、または手動でワークフローを実行する
workflow_dispatch:
push:
branches: [main]
jobs:
job_id:
runs-on: "ubuntu-latest"
permissions:
contents: "read"
id-token: "write"
steps:
- uses: "actions/checkout@v4"
- id: "auth"
uses: "google-github-actions/auth@v2"
with:
workload_identity_provider: ${{ secrets.WORKLOAD_IDENTITY_PROVIDER_ID}} # 作成したプロバイダのID
service_account: "github-actions@${{ secrets.GCP_PROJECT_ID }}.iam.gserviceaccount.com" # プロバイダで指定したサービスアカウント
- id: "deploy"
uses: "google-github-actions/deploy-cloud-functions@v2"
with: # Cloud Functions の設定(下記は一例)
name: "function_name" # 関数名
entry_point: "main" # エントリポイント
runtime: "python312" # 関数に使うランタイム
env_vars: ENV1=hogehoge, ENV2=fugafuga # 環境変数
source_dir: "cloud_functions" # ルートディレクトリに main.py が無い時はこれでディレクトリを指定する
timeout: 120 # 関数のタイムアウト時間
max_instances: 10 # 最大インスタンス数
region: "asia-northeast1" # リージョン
内容は deploy-cloud-functions のREADMEに従う。
- on
- ワークフローを実行する条件を指定する
-
id: "auth"
- Workload Identity連携を利用してGitHub ActionsからGoogle Cloudへ認証を行う
- パラメータの意味はこちらを参照
-
id: "deploy"
で Cloud Functions をデプロイする- ここに記載した設定で Cloud Functions をデプロイする
- パラメータの意味はこちらを参照
ワークフローの実行
作成した yml ファイルを push する。
GitHub のリポジトリにアクセスし、Actions タブに選択するとワークフローが表示されるはず。
トリガーの条件を満たしていれば自動で実行されている。
手動で動かす場合は、ワークフロー名を選択して「Run workflow」から実行できる。
ジョブが成功すれば Cloud Functions がデプロイされているはず。
失敗した場合は Workload Identity か yml を見直す。
終わりに
Workload Identity を利用して GitHub Actions から Cloud Functions をデプロイできるようにしました。もちろん、GitHub Actions や Cloud Functions 以外にも応用が可能です。
サービスアカウントキーを利用するよりも手間はかかりますが、よりセキュアな設定が求められる場合はぜひ利用したい。