メリット
Cloud Functionsもサーバーレスだが、Cloud Runにするメリットはいくつかある
- コンテナイメージで実行されるため、どの言語でもデプロイが可能
- Cloud Runの最大タイムアウト時間は60分。一方でCloud Functionsは最大9分
- Cloud Runは複数の同時リクエストの送信が可能。Cloud Functionsは1インスタンス(1リクエスト)につき1送信のみ
※Cloud Run と Cloud Functions の違い
料金
またCloud Runも最初の200万リクエストまでは無料枠で動かせる。
トリガー
起動トリガーとなるサービスは若干異なる
Trigger | Cloud Run | Cloud Functions |
---|---|---|
HTTP | ✓ | ✓ |
Pub/Sub | ✓ | ✓ |
Scheduling | ✓ | ✓ |
Cloud Scheduler | ✓ | ✓ |
gRPC | ✓ | |
Cloud Tasks | ✓ | |
WebSockets | ✓ | |
Cloud Storage | ✓ | |
Firestore | ✓ | |
Firebase | ✓ |
Cloud Run採用の理由
コンテナのイメージそのままデプロイできるのが大きいメリットで、Cloud Functionsのフォーマット(ランタイム、ファイル名、関数名)に合わせる必要がないのと、デバッグが容易であったため
ちなみにCloud Functionsでもデバッグできるフレームワークはある(functions-framework-nodejs)が、これも結局フレームワークにはめた形で実装する必要がある
実装
create-next-app
でreactのセットアップ
npx create-next-app nextjs-for-cloudrun
cd nextjs-for-cloudrun
npm run dev
上図のトップページは pages/index.js
に書かれていて、ここを少し編集。Reactについては割愛します
import Head from 'next/head'
import styles from '../styles/Home.module.css'
export default function Home() {
return (
<div className={styles.container}>
<Head>
<title>Cloud Run デプロイ</title>
<meta name="description" content="Generated by create next app" />
<link rel="icon" href="/favicon.ico" />
</Head>
<h1 className={styles.title}>
こんにちはCloud Run
</h1>
</div>
)
}
コンテナ化
rootディレクトリにDockerfileを作成する。内容は 公式のドキュメント のコードそのまま
# Install dependencies only when needed
FROM node:alpine AS deps
# Check <https://github.com/nodejs/docker-node/tree/b4117f9333da4138b03a546ec926ef50a31506c3#nodealpine> to understand why libc6-compat might be needed.
RUN apk add --no-cache libc6-compat
WORKDIR /app
COPY package.json yarn.lock ./
RUN yarn install --frozen-lockfile
# Rebuild the source code only when needed
FROM node:alpine AS builder
WORKDIR /app
COPY . .
COPY --from=deps /app/node_modules ./node_modules
RUN yarn build && yarn install --production --ignore-scripts --prefer-offline
# Production image, copy all the files and run next
FROM node:alpine AS runner
WORKDIR /app
ENV NODE_ENV production
RUN addgroup -g 1001 -S nodejs
RUN adduser -S nextjs -u 1001
# You only need to copy next.config.js if you are NOT using the default configuration
# COPY --from=builder /app/next.config.js ./
COPY --from=builder /app/public ./public
COPY --from=builder --chown=nextjs:nodejs /app/.next ./.next
COPY --from=builder /app/node_modules ./node_modules
COPY --from=builder /app/package.json ./package.json
USER nextjs
EXPOSE 3000
ENV PORT 3000
# Next.js collects completely anonymous telemetry data about general usage.
# Learn more here: <https://nextjs.org/telemetry>
# Uncomment the following line in case you want to disable telemetry.
# ENV NEXT_TELEMETRY_DISABLED 1
CMD ["node_modules/.bin/next", "start"]
そのままだとlockファイルがpackage-lock.jsonなのでコンテナビルドするとエラーになる。公式のコードに合わせる場合は以下のコマンドでyarn.lockファイルを生成する
yarn import
Dockerでも動くか確認
docker build -t nextjs-on-cloudrun .
docker run -p 3000:3000 nextjs-on-cloudrun
問題なければイメージをgcpにあげる。
gcloud builds submit --tag gcr.io/{GCPのプロジェクトID}/nextjs-on-cloudrun --project {GCPのプロジェクトID}
Cloud Runにデプロイ
gcloud run deploy nextjs-on-cloudrun \\
--image gcr.io/{GCPのプロジェクトID}/nextjs-on-cloudrun \\
--project {GCPのプロジェクトID} \\
--platform managed \\
--region asia-northeast1 \\
--memory 128Mi \\
--cpu 1 \\
--port 3000 \\
--allow-unauthenticated
成功するとURLがログに出力される
Deploying container to Cloud Run service [nextjs-on-cloudrun] in project [{GCPのプロジェクトID}] region [asia-northeast1]
✓ Deploying new service... Done.
✓ Creating Revision...
✓ Routing traffic...
✓ Setting IAM Policy...
Done.
Service [nextjs-on-cloudrun] revision [nextjs-on-cloudrun-00001-gen] has been deployed and is serving 100 percent of traffic.
Service URL: {サービスURL}
このサービスURLに入れば、ローカルで見た画面と同じものが見える
クリーンアップ
Cloud Runサービスから削除
gcloud run services delete nextjs-on-cloudrun \\
--region asia-northeast1 \\
--platform managed
Containerレジストリから削除
gcloud container images delete gcr.io/{GCPのプロジェクトID}/nextjs-on-cloudrun \\
--force-delete-tags
# イメージの検索(+ハッシュ値)
gcloud container images list-tags gcr.io/{GCPのプロジェクトID}/nextjs-on-cloudrun --format="get(digest)"
gcloud container images delete gcr.io/{GCPのプロジェクトID}/nextjs-on-cloudrun@sha256:{ハッシュ値}
参照
- Cloud Run: サーバーレス コンテナの話
- Google Cloud Run vs Cloud Functions
- NextJS Deployment
- gcloud container images delete
kunimitsu higashi