WEB

NextJSをGoogle Cloud Runにデプロイ

メリット

Cloud Functionsもサーバーレスだが、Cloud Runにするメリットはいくつかある

  • コンテナイメージで実行されるため、どの言語でもデプロイが可能
  • Cloud Runの最大タイムアウト時間は60分。一方でCloud Functionsは最大9分
  • Cloud Runは複数の同時リクエストの送信が可能。Cloud Functionsは1インスタンス(1リクエスト)につき1送信のみ

Cloud Run と Cloud Functions の違い

料金

またCloud Runも最初の200万リクエストまでは無料枠で動かせる。

Cloud Run pricing

トリガー

起動トリガーとなるサービスは若干異なる

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

http://localhost:3000/

上図のトップページは 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:{ハッシュ値}

参照

 

kunimitsu higashi