Azure Static Web Apps に Next.js ハイブリッド サイトをデプロイする (プレビュー)
このチュートリアルでは、React サーバー コンポーネント、サーバー側レンダリング (SSR)、API ルートなどの Next.js の機能のサポートを使って、Next.js の Web サイトを Azure Static Web Apps にデプロイする方法について説明します。
Note
Next.js ハイブリッドのサポートはプレビュー段階です。
前提条件
リソース | 説明 |
---|---|
Azure アカウント | Azure アカウントとアクティブなサブスクリプションをお持ちでない場合は、無料で作成できます。 |
GitHub アカウント | GitHub アカウントをお持ちでない場合は、無料でアカウントを作成できます。 |
Node.js | 最新バージョンの Node.js をインストールします。 |
Next.js CLI | 最新バージョンの Next.js CLI をインストールします。 詳しくは、Next.js の概要ガイドに関するページをご覧ください。 |
プレビューでサポートされていない機能
Static Web Apps の次の機能は、ハイブリッド レンダリングを使用する Next.js ではサポートされていません。
- Azure サービスの選択: Azure Functions、Azure App Service、Azure Container Apps、または Azure API Management を使用するリンクされた API。
- SWA CLI の機能: SWA CLI のローカル エミュレーションとデプロイ。
- 部分的な機能のサポート:
staticwebapp.config.json
ファイル内の次のプロパティはサポートされていません。- ナビゲーション フォールバックはサポートされていません。
- Next.js アプリケーション内のルートへのルートの書き換えは、
next.config.js
内で構成する必要があります。 staticwebapp.config.json
ファイル内の構成は、next.config.js
内の構成より優先されます。- 機能の完全な互換性のためには、
next.config.js
を使って Next.js サイトの構成を処理する必要があります。
- ビルドのスキップ: Next.js アプリケーションの場合、
skip_api_build=true
が設定されていると、Static Web Apps は既定では開発依存関係の削除も、sharp パッケージの追加も行いません。 これらの最適化が必要な場合は、skip_app_build=true
を渡す前に、カスタム ビルド ステップにそれらを追加します。 - 増分静的再生成 (ISR): イメージ キャッシュはサポートされていません。
Note
ハイブリッド Next.js アプリケーションの最大アプリ サイズは 250 MB です。 最適化されたアプリ サイズには、Next.js によるスタンドアロン機能を使用します。 これでも十分でない場合、アプリ サイズの要件が 250 MB を超える場合は、静的 HTML エクスポート済み Next.js の使用を検討してください。
リポジトリを作成する
この記事では、簡単に作業を開始できるように、GitHub テンプレート リポジトリを使用します。 テンプレートには、Azure Static Web Apps にデプロイするスターター アプリが含まれます。
次の場所に移動して、新しいリポジトリを作成します。
https://github.com/staticwebdev/nextjs-hybrid-starter/generate
リポジトリの名前を my-first-static-web-app に設定します
[Create repository from template](テンプレートからリポジトリを作成する) を選択します。
静的 Web アプリを作成する
リポジトリが作成されたので、Azure portal から静的 Web アプリを作成できます。
- Azure ポータルにアクセスします。
- [リソースの作成] を選択します。
- Static Web Apps を検索します。
- Static Web Apps を選択します。
- [作成] を選択します
[基本] セクションで、新しいアプリを構成し、それを GitHub リポジトリにリンクすることから始めます。
設定 | 値 |
---|---|
サブスクリプション | Azure サブスクリプションを選択します。 |
リソース グループ | [新規作成] リンクを選択し、テキスト ボックスに「static-web-apps-test」と入力します。 |
名前 | テキスト ボックスに、「my-first-static-web-app」と入力します。 |
プランの種類 | [無料] を選択します。 |
ソース | 必要な場合は、[GitHub] を選んで GitHub にサインインします。 |
[GitHub アカウントでサインイン] を選び、GitHub で認証します。
GitHub にサインインした後、リポジトリ情報を入力します。
設定 | 値 |
---|---|
Organization | 自分の組織を選択します。 |
リポジトリ | [my-first-web-static-app] を選択します。 |
[Branch]\(ブランチ) | [main](メイン) を選択します。 |
Note
リポジトリが表示されない場合、次の手順を実行します。
- GitHub で Azure Static Web Apps を承認する必要がある場合があります。 GitHub リポジトリを参照し、[設定] > [アプリケーション] > [認可された OAuth アプリ] の順に移動して、[Azure Static Web Apps]、[許可] の順に選択します。
- Azure DevOps 組織で Azure Static Web Apps を承認する必要がある場合があります。 アクセス許可を付与する組織の所有者である必要があります。 OAuth を使用してサード パーティ アプリケーションのアクセスを要求します。 詳細については、OAuth 2.0 を使用した REST API へのアクセスの承認に関する記事を参照してください。
[ビルドの詳細] セクションで、使用するフロントエンド フレームワークに固有の構成の詳細を追加します。
[ビルドのプリセット] ドロップダウンから [Next.js] を選びます。
[App location](アプリの場所) ボックスは既定値のままにします。
[Api location](API の場所) ボックスは空のままにします。
[出力場所] ボックスは空のままにします。
[Review + create](レビュー + 作成) を選択します。
Web サイトを表示する
静的アプリのデプロイには 2 つの側面があります。 1 つ目は、アプリを構成する基になる Azure リソースを作成することです。 2 つ目は、アプリケーションをビルドして発行するワークフローです。
新しい静的サイトに移動する前にまず、デプロイ ビルドの実行が完了している必要があります。
Static Web Apps の "概要" ウィンドウには、Web アプリとの対話に役立つ一連のリンクが表示されます。
"Select here to check the status of your GitHub Actions runs (こちらを選択して、GitHub Actions の実行の状態を確認してください)" というバナーを選択すると、リポジトリに対して実行されている GitHub アクションが表示されます。 デプロイ ジョブが完了したことを確認したら、生成された URL を使用して Web サイトに移動できます。
GitHub Actions ワークフローが完了したら、 [URL] リンクを選択して、新しいタブで Web サイトを開くことができます。
変更を行うためにローカル環境に Next.js プロジェクトを設定する
新しいリポジトリをお使いのコンピューターに複製します。 <GITHUB_ACCOUNT_NAME> は、必ず自分のアカウント名に置き換えてください。
git clone http://github.com/<GITHUB_ACCOUNT_NAME>/my-first-static-web-app
Visual Studio Code または任意のコード エディターでプロジェクトを開きます。
サーバー側のレンダリングを設定する
マネージド バックエンドは、すべてのプランのすべてのハイブリッド Next.js デプロイで自動的に利用できるようになります。 しかし、カスタム バックエンドをサイトに割り当てることで、パフォーマンスを微調整したり、バックエンドをより細かく制御したりできます。 マネージド バックエンドをリンクされたバックエンドに切り替える場合、サイトでダウンタイムは発生しません。
独自のバックエンドを使用する
独自のバックエンドを使用すると、パフォーマンスを向上させ、Next.js のサーバー側レンダリングをより詳細に制御できます。 サイトのカスタム バックエンドを設定するには、次の手順に従います。
次の手順では、カスタム バックエンドを Standard プラン以上の静的 Web アプリに関連付ける方法を示します。
Note
リンクされたバックエンドは、Standard プラン以上を使用しているサイトでのみ利用できます。
Azure portal で、静的 Web アプリに移動します。
[設定] を選択し、サイド メニューから [API] を選びます。
[リンクされたバックエンドの構成] を選択します。
新しい App Service プランを作成するか、既存の App Service プランを選択します。
選択した App Service プランでは、少なくとも S1 SKU を使用する必要があります。
[リンク] をクリックします。
サーバー コンポーネントを使用してサーバー レンダリング済みデータを追加する
アプリ ルーターを使って Next.js プロジェクトにサーバー レンダリング済みデータを追加するには、Next.js コンポーネントを編集して、データをレンダリングするためのサーバー側操作をコンポーネントに追加します。 既定では、Next.js コンポーネントはサーバー レンダリング可能なサーバー コンポーネントです。
app/page.tsx
ファイルを開き、サーバー側で計算される変数の値を設定する操作を追加します。 たとえば、データのフェッチやその他のサーバー操作などです。export default function Home() { const timeOnServer = new Date().toLocaleTimeString('en-US'); return( ... ); }
next/cache
からunstable_noStore
をインポートし、Home
コンポーネント内でそれを呼び出して、ルートが動的にレンダリングされるようにします。import { unstable_noStore as noStore } from 'next/cache'; export default function Home() { noStore(); const timeOnServer = new Date().toLocaleTimeString('en-US'); return( ... ); }
Note
この例では、サーバーの現在時刻のサーバー レンダリングを見ていただくために、このコンポーネントを強制的に動的レンダリングします。 Next.js のアプリ ルーター モデルでは、Next.js アプリのパフォーマンスを最適にするため、個々のデータ要求をキャッシュすることをお勧めします。 詳しくは、Next.js でのデータのフェッチとキャッシュに関するページをご覧ください。
サーバー側のデータをレンダリングするように、app/pages.tsx の
Home
コンポーネントを更新します。import { unstable_noStore as noStore } from 'next/cache'; export default function Home() { noStore(); const timeOnServer = new Date().toLocaleTimeString('en-US'); return( <main className="flex min-h-screen flex-col items-center justify-between p-24"> <div> This is a Next.js application hosted on Azure Static Web Apps with hybrid rendering. The time on the server is <strong>{timeOnServer}</strong>. </div> </main> ); }
API ルートの追加
Next.js には、サーバー コンポーネントに加えて、Next.js アプリケーションへの API ルートの作成に使用できるルート ハンドラーが用意されています。 これらの API は、クライアント コンポーネントでフェッチできます。
まず API ルートを追加します。
app/api/currentTime/route.tsx
に新しいファイルを作成します。 このファイルは、新しい API エンドポイントのためのルート ハンドラーを保持します。API からデータを返すハンドラー関数を追加します。
import { NextResponse } from 'next/server'; export const dynamic = 'force-dynamic'; export async function GET() { const currentTime = new Date().toLocaleTimeString('en-US'); return NextResponse.json({ message: `Hello from the API! The current time is ${currentTime}.` }); }
app/components/CurrentTimeFromAPI.tsx
に新しいファイルを作成します。 このコンポーネントは、ブラウザーから API をフェッチするクライアント コンポーネントのコンテナーを作成します。API をフェッチするクライアント コンポーネントをこのファイルに追加します。
'use client'; import { useEffect, useState } from 'react'; export function CurrentTimeFromAPI(){ const [apiResponse, setApiResponse] = useState(''); const [loading, setLoading] = useState(true); useEffect(() => { fetch('/api/currentTime') .then((res) => res.json()) .then((data) => { setApiResponse(data.message); setLoading(false); }); }, []); return ( <div className='pt-4'> The message from the API is: <strong>{apiResponse}</strong> </div> ) }
このクライアント コンポーネントは、読み込みが完了した後でコンポーネントをレンダリングするために、useEffect
React フックを使って API をフェッチします。 'use client'
ディレクティブは、この要素をクライアント コンポーネントとして識別します。 詳しくは、「クライアント コンポーネント」をご覧ください。
CurrentTimeFromAPI
クライアント コンポーネントをインポートしてレンダリングするように、app/page.tsx を編集します。import { unstable_noStore as noStore } from 'next/cache'; import { CurrentTimeFromAPI } from './components/CurrentTimeFromAPI'; export default function Home() { noStore(); const timeOnServer = new Date().toLocaleTimeString('en-US'); return( <main className="flex min-h-screen flex-col items-center justify-between p-24"> <div> This is a Next.js application hosted on Azure Static Web Apps with hybrid rendering. The time on the server is <strong>{timeOnServer}</strong>. </div> <CurrentTimeFromAPI /> </main> ); }
API ルートからの結果がページに表示されます。
Next.js のランタイム バージョンを構成する
Next.js の特定のバージョンでは、Node.js の特定のバージョンが必要です。 Node の特定のバージョンを構成するには、package.json
ファイルの engines
プロパティを設定してバージョンを指定できます。
{
...
"engines": {
"node": "18.17.1"
}
}
Next.js 用の環境変数を設定する
Next.js は、ビルド時と要求時に環境変数を使って、サーバー側レンダリングで静的ページ生成と動的ページ生成の両方をサポートします。 そのため、ビルド タスクとデプロイ タスク内、および Azure Static Web Apps リソースの [環境変数] 内の両方で、環境変数を設定します。
...
- name: Build And Deploy
id: builddeploy
uses: Azure/static-web-apps-deploy@v1
with:
azure_static_web_apps_api_token: ${{ secrets.AZURE_STATIC_WEB_APPS_API_TOKEN }}
repo_token: ${{ secrets.GITHUB_TOKEN }} # Used for GitHub integrations (i.e. PR comments)
action: "upload"
app_location: "/"
api_location: ""
output_location: ""
env:
DB_HOST: ${{ secrets.DB_HOST }}
DB_USER: ${{ secrets.DB_USER }}
DB_DATABASE: ${{ secrets.DB_DATABASE }}
DB_PASSWORD: ${{ secrets.DB_PASSWORD }}
DB_PORT: ${{ secrets.DB_PORT }}
...
スタンドアロン機能を有効にする
アプリケーションのサイズが 250 MB を超える場合は、Next.js の出力ファイル トレース機能を使うと、アプリのサイズを最適化し、パフォーマンスを向上させることができます。
出力ファイル トレースでは、必要なパッケージ依存関係を含むアプリケーション全体の圧縮バージョンが作成されます。 このパッケージは、.next/standalone という名前のフォルダーに組み込まれます。 このパッケージを使うと、node_modules 依存関係がなくても、アプリをそれだけでデプロイできます。
next.config.js
機能を有効にするには、次のプロパティを standalone
に追加します。
module.exports ={
output:"standalone",
}
次に、静的ファイルをスタンドアロン出力にコピーするために、package.json
ファイルで build
コマンドを構成します。
{
...
"scripts": {
...
"build": "next build && cp -r .next/static .next/standalone/.next/ && cp -r public .next/standalone/"
...
}
...
}
デプロイ用にルーティングとミドルウェアを構成する
カスタム リダイレクト、書き換え、ミドルウェアを使って、ルートの Next.js プロジェクト ハンドルを構成できます。 これらのハンドラーは、認証、パーソナル化、ルーティング、国際化に一般的に使用されます。 カスタム処理は Next.js サイトの既定のルーティングに影響を与えるため、構成は Static Web Apps でのホスティングと互換性がある必要があります。
Static Web Apps は、ビルド時にサイトにページを追加して、Next.js サイトが正常にデプロイされていることを確認します。 このページは public/.swa/health.html
という名前です。Static Web Apps は、/.swa/health.html
に移動して応答の成功を確認することにより、サイトが正常に起動され、デプロイされたことを確認します。 リダイレクトと書き換えを含むミドルウェアとカスタム ルーティングは、/.swa/health.html
パスのアクセスに影響を与える可能性があり、Static Web Apps のデプロイの検証が妨げられるおそれがあります。 Static Web Apps へのデプロイが成功するようにミドルウェアとルーティングを構成するには、次の手順に従います。
ミドルウェアの構成では、
middleware.ts
(または.js
) ファイルで、.swa
で始まるルートを除外します。export const config = { matcher: [ /* * Match all request paths except for the ones starting with: * - .swa (Azure Static Web Apps) */ '/((?!.swa).*)', ], }
.swa
で始まるルートを除外するように、next.config.js
でリダイレクトを構成します。module.exports = { async redirects() { return [ { source: '/((?!.swa).*)<YOUR MATCHING RULE>', destination: '<YOUR REDIRECT RULE>', permanent: false, }, ] }, };
.swa
で始まるルートを除外するように、next.config.js
で書き換え規則を構成します。module.exports = { async rewrites() { return { beforeFiles: [ { source: '/((?!.swa).*)<YOUR MATCHING RULE>', destination: '<YOUR REWRITE RULE>', } ] } }, };
これらのコード スニペットでは、.swa
で始まるパスが除外されて、カスタム ルーティングまたはミドルウェアによるこれらの要求の処理が停止されます。 これらの規則により、デプロイの検証中にパスが想定どおりに解決されることが保証されます。
Next.js のログ記録を有効にする
Next.js サーバー API のトラブルシューティングのベスト プラクティスに従って、ログ記録を API に追加し、これらのエラーをキャッチします。 Azure でのログ記録では、Application Insights が使用されます。 この SDK を事前に読み込むには、カスタム スタートアップ スクリプトを作成する必要があります。 詳細については、以下を参照してください。
リソースをクリーンアップする
このアプリケーションを引き続き使用しない場合は、次の手順を使用して Azure Static Web Apps インスタンスを削除することができます。
- Azure portal を開きます。
- 上部の検索バーから my-first-web-static-app を検索します。
- アプリの名前を選択します。
- [削除] を選択します。
- [はい] を選択して削除アクションを確定します (このアクションが完了するまでにしばらく時間がかかる場合があります)。