メインコンテンツまでスキップ

Next.jsを使用したマルチゾーンによるマイクロフロントエンドの構築方法

マルチゾーンは、ドメイン上の大規模なアプリケーションを、各々が一連のパスを提供する小さなNext.jsアプリケーションに分割するマイクロフロントエンドのアプローチです。これは、アプリケーション内の他のページと無関係なページのコレクションがある場合に便利です。これらのページを別のゾーン(つまり、別のアプリケーション)に移動することで、各アプリケーションのサイズを縮小し、ビルド時間を短縮し、特定のゾーンにのみ必要なコードを削除できます。アプリケーションが分離されているため、マルチゾーンはドメイン上の他のアプリケーションが独自のフレームワークを使用することも可能にします。

例えば、次のようなページセットを分割したいとします:

  • /blog/* すべてのブログ投稿
  • /dashboard/* ユーザーがダッシュボードにログインしているときのすべてのページ
  • /* 他のゾーンでカバーされていないウェブサイトの残りの部分

マルチゾーンのサポートにより、同じドメインで提供され、ユーザーには同じように見える3つのアプリケーションを作成できますが、各アプリケーションを独立して開発およびデプロイできます。

3つのゾーン:A、B、C。異なるゾーンからのルート間のハードナビゲーションと、同じゾーン内のルート間のソフトナビゲーションを示していま�す。3つのゾーン:A、B、C。異なるゾーンからのルート間のハードナビゲーションと、同じゾーン内のルート間のソフトナビゲーションを示しています。

同じゾーン内のページ間をナビゲートする場合、ページのリロードを必要としないソフトナビゲーションが行われます。例えば、この図では、/から/productsへのナビゲーションはソフトナビゲーションになります。

あるゾーンのページから別のゾーンのページにナビゲートする場合、例えば/から/dashboardへのナビゲーションは、現在のページのリソースをアンロードし、新しいページのリソースをロードするハードナビゲーションが行われます。頻繁に一緒に訪問されるページは、ハードナビゲーションを避けるために同じゾーンに配置するべきです。

ゾーンの定義方法

ゾーンは通常のNext.jsアプリケーションであり、他のゾーンのページや静的ファイルとの競合を避けるためにassetPrefixを設定します。

next.config.js
/** @type {import('next').NextConfig} */
const nextConfig = {
assetPrefix: '/blog-static',
}

Next.jsのアセット(JavaScriptやCSSなど)は、他のゾーンのアセットと競合しないようにassetPrefixでプレフィックスされます。これらのアセットは、各ゾーンの/assetPrefix/_next/...の下で提供されます。

他のより具体的なゾーンにルーティングされていないすべてのパスを処理するデフォルトのアプリケーションには、assetPrefixは必要ありません。

Next.js 15より古いバージョンでは、静的アセットを処理するために追加のリライトが必要な場合があります。これはNext.js 15では不要になりました。

next.config.js
/** @type {import('next').NextConfig} */
const nextConfig = {
assetPrefix: '/blog-static',
async rewrites() {
return {
beforeFiles: [
{
source: '/blog-static/_next/:path+',
destination: '/_next/:path+',
},
],
}
},
}

リクエストを正しいゾーンにルーティングする方法

マルチゾーンのセットアップでは、異なるアプリケーションによって提供されるため、パスを正しいゾーンにルーティングする必要があります。これを行うには、任意のHTTPプロキシを使用できますが、Next.jsアプリケーションの1つを使用してドメイン全体のリクエストをルーティングすることもできます。

Next.jsアプリケーションを使用して正しいゾーンにルーティングするには、rewritesを使用できます。異なるゾーンによって提供される各パスに対して、そのパスを他のゾーンのドメインに送信するリライトルールを追加します。例えば:

next.config.js
async rewrites() {
return [
{
source: '/blog',
destination: `${process.env.BLOG_DOMAIN}/blog`,
},
{
source: '/blog/:path+',
destination: `${process.env.BLOG_DOMAIN}/blog/:path+`,
}
];
}

destinationは、スキームとドメインを含むゾーンによって提供されるURLである必要があります。これはゾーンの本番ドメインを指すべきですが、ローカル開発ではlocalhostにリクエストをルーティングするためにも使用できます。

Good to know: URLパスはゾーンに固有であるべきです。例えば、2つのゾーンが/blogを提供しようとすると、ルーティングの競合が発生します。

ミドルウェアを使用したリクエストのルーティング

リクエストの遅延オーバーヘッドを最小限に抑えるために、rewritesを通じてリクエストをルーティングすることが推奨されますが、ルーティング時に動的な決定が必要な場合にはミドルウェアを使用することもできます。例えば、移行中にパスをどこにルーティングするかを決定するために機能フラグを使用している場合、ミドルウェアを使用できます。

middleware.js
export async function middleware(request) {
const { pathname, search } = req.nextUrl
if (pathname === '/your-path' && myFeatureFlag.isEnabled()) {
return NextResponse.rewrite(`${rewriteDomain}${pathname}${search}`)
}
}

ゾーン間のリンク

異なるゾーンのパスへのリンクは、Next.jsの<Link>コンポーネントではなく、aタグを使用するべきです。これは、Next.jsが<Link>コンポーネント内の相対パスをプリフェッチし、ソフトナビゲーションしようとするためであり、これはゾーンをまたいで機能しません。

コードの共有

異なるゾーンを構成するNext.jsアプリケーションは、任意のリポジトリに配置できます。しかし、これらのゾーンをモノレポに配置してコードをより簡単に共有することが便利な場合が多いです。異なるリポジトリに存在するゾーンの場合、コードは公開またはプライベートのNPMパッケージを使用して共有することもできます。

異なるゾーンでページが異なるタイミングでリリースされる可能性があるため、機能フラグは異なるゾーン間で機能を一斉に有効または無効にするのに役立ちます。

Next.js on Vercelアプリケーションでは、モノレポを使用して、影響を受けるすべてのゾーンを単一のgit pushでデプロイできます。

Server Actions

マルチゾーンでServer Actionsを使用する場合、ユーザー向けのドメインが複数のアプリケーションを提供する可能性があるため、ユーザー向けのオリジンを明示的に許可する必要があります。next.config.jsファイルに次の行を追加します:

next.config.js
const nextConfig = {
experimental: {
serverActions: {
allowedOrigins: ['your-production-domain.com'],
},
},
}

詳細はserverActions.allowedOriginsを参照してください。