Article

Jamstackの衝撃:従来型CMSとの違いとメリット

高田晃太郎
Jamstackの衝撃:従来型CMSとの違いとメリット

Core Web Vitalsの“良好”しきい値は、LCP(Largest Contentful Paint:主要コンテンツの表示完了)≤2.5秒、CLS(Cumulative Layout Shift:レイアウトのズレ)≤0.1、INP(Interaction to Next Paint:操作から描画まで)≤200msと定義されています¹²。検索評価や事業KPIに直結するこれらの体験品質を安定して満たすには、静的プリレンダリング(あらかじめHTMLを生成)とCDN(コンテンツ配信ネットワーク)配信が物理的に有利です⁴。研究や業界レポートが示す通り、ユーザーに近いエッジからの配信はTTFB(Time to First Byte)を短縮しやすく⁴、一方でアプリサーバやデータベースが関与する動的生成は負荷変動の影響を受けやすい。Jamstackは“速いサイトをつくるフレームワーク”ではなく、配信モデルと変更管理を再設計するアーキテクチャです³。従来型CMSとの起因の違いを、経営・開発・運用という現場の視点で整理し、メリットを実務的な指標で語ります。

Jamstackは何を置き換えるのか

Jamstackの核は、ビルド時にページをプリレンダリングし、生成物をCDNの不変オブジェクトとして配布することにあります³。ページ生成はビルドの責務に、配信はCDNの責務に、動的処理は疎結合なAPIやサーバーレス関数(Functions)の責務に分離されます。対照的に、従来型CMSはリクエストごとにテンプレートとプラグイン、データベースを経てHTMLを組み立てるため、生成・配信・動的処理が同一ランタイムに集約され、オリジンがボトルネックになりやすいのが構造上の宿命です⁵。

この差はキャッシュ戦略にも直結します。Jamstackは“immutable deploy(不変デプロイ)”という考え方で、1デプロイ=1スナップショットとしてCDN上に固定化します³。これによりキャッシュキーが明確になり破棄も一括、ロールバックもデプロイ単位で秒単位に行いやすくなります。従来型はパージ対象の粒度がプラグインやDB更新に依存しがちで、キャッシュ整合性の運用負荷が跳ね上がる局面を避けにくい。

アーキテクチャの差分を“責務”で観る

JamstackはCMS自体を否定しません。むしろヘッドレスCMS(編集と配信を分離するCMS)を採用し、“編集はCMS、生成はビルド、配信はCDN、動的はAPI/Functions”と責務境界を明確化します³。これにより編集者はリッチエディタとプレビューに集中でき、開発者はUIのビルドと自動テストに集中でき、SREはCDNヒット率とエラーバジェットで配信を語れるようになります。従来型CMSは便利な統合体である一方、プラグインの相互依存が技術的負債を増やし、脆弱性対応やPHP/DBのライフサイクル管理が恒常的な運用タスクになります⁵。

配信モデルとキャッシュの違い

Jamstackは“事前計算して配る”モデルです³。SSG(静的サイト生成)やISR(インクリメンタル静的再生成:必要に応じて静的ページを更新)により、ホットなページはCDNに常駐し、コールドなページは初回アクセス時に生成して以降は静的化します。オフロード率(CDNで完結する割合)を高水準で現実的に達成でき、オリジンは更新と長尾の生成に専念できます。従来型もパーソナライズやキャッシュレイヤの工夫で高速化は可能ですが、アプリケーションの複雑性が増して“速さの再現性”が失われやすくなります⁵。

Jamstackのメリットを数値で語る

Jamstackの効果は抽象的な“速い・安全”に留まりません。性能、セキュリティ、スケーラビリティ、コスト、開発体験という5点で、組織と経営に伝わる指標で説明できます。性能面では、CDN配信されたプリレンダリングHTMLはグローバルでも低いTTFBを達成しやすく⁴、LCPは画像最適化と遅延読み込みの徹底で“良好”閾値(≤2.5秒)を現実的に狙えます¹。セキュリティ面はアタックサーフェスの縮小が本質で、PHPや管理画面を公開面から隔離し、パッチ適用もビルド時依存に集約できます³。スケーラビリティは“増えるほどCDNが得意”という逆説的な利点が働き、フラッシュトラフィック時でもエッジで吸収しやすくなります³。運用コストは、オートスケールやDBチューニングに投じていた時間を“ビルドの安定化”と“キャッシュヒット率の改善”に置換でき、障害対応の夜間呼び出しを減らせる可能性があります。開発体験は、API契約とスタブでローカル開発を完結させやすく、PRごとにプレビューURLが自動生成されるため、非同期なレビューが標準化します³。

パフォーマンスの実測設計とベンチマーク視点

中規模の製品マーケサイトを想定した比較事例(Next.jsのSSG+ISR構成と従来型CMSのSSR+ページキャッシュを同一CDN条件で比較)では、CDNヒット時のTTFB中央値やLCPがJamstack構成の方に有利に分布する傾向が報告されています。差分は“初期HTMLの即時性”と“配信の揺らぎの少なさ”に起因します。もちろん設計次第で従来型CMSも高速化可能ですが、再現性と運用難易度のバランスではJamstackが優位になりやすい⁵。実務では、Lighthouseのラボ計測とRUM(実ユーザ監視)を併用し、TTFB/LCP/INPの分布とCDNヒット率、ビルド時間をセットで追う設計が有効です。

以下はISRを使った安定運用のための最小構成例です。ビルド時に主要ページを生成し、長尾はアクセスに応じて再生成、以後は静的化します。

// pages/[slug].tsx
import { GetStaticPaths, GetStaticProps } from 'next';
import React from 'react';
import { getPostSlugs, getPostBySlug } from '../lib/cms';

export const getStaticPaths: GetStaticPaths = async () => {
  const slugs = await getPostSlugs({ limit: 2000 });
  return { paths: slugs.map((s) => ({ params: { slug: s } })), fallback: 'blocking' };
};

export const getStaticProps: GetStaticProps = async ({ params }) => {
  const slug = params?.slug as string;
  const post = await getPostBySlug(slug);
  if (!post) return { notFound: true };
  return { props: { post }, revalidate: 60 }; // 60秒で再検証
};

export default function Post({ post }: { post: any }) {
  return <article><h1>{post.title}</h1><div dangerouslySetInnerHTML={{ __html: post.html }} /></article>;
}

ヘッドレスCMSからのフェッチもビルド境界に閉じ込めます。APIキーはビルド時のシークレットとして扱い、実行時に流出しない設計が基本です。

// lib/cms.ts
import fetch from 'node-fetch';

const ENDPOINT = process.env.CMS_ENDPOINT!;
const TOKEN = process.env.CMS_TOKEN!;

export async function getPostSlugs(opts?: { limit?: number }) {
  const res = await fetch(`${ENDPOINT}/posts?fields=slug&limit=${opts?.limit ?? 100}` , {
    headers: { Authorization: `Bearer ${TOKEN}` }
  });
  const data = await res.json();
  return data.items.map((i: any) => i.slug);
}

export async function getPostBySlug(slug: string) {
  const res = await fetch(`${ENDPOINT}/posts/${slug}`, {
    headers: { Authorization: `Bearer ${TOKEN}` }
  });
  if (res.status === 404) return null;
  return res.json();
}

運用とセキュリティの現実

Jamstackでは管理画面はヘッドレスCMSに隔離され、公開面のオリジンは原則としてCDNのオブジェクトストレージになります。WAF(Web Application Firewall)やRASP(Runtime Application Self-Protection)は依然として価値がありますが、侵入口の数が構造的に減るため、プラグイン由来の重大脆弱性やSQLインジェクションのリスクは大きく低下します³。パッチ適用は“ライブラリアップデート→再ビルド→デプロイ”のパイプラインに集約され、監査もGitの履歴で一元化できます。ログの観点でも、配信はCDNログで、生成はCIログで、動的はFunctionsのログで分離でき、原因切り分けの時間を短縮できます。

Jamstackの“現実的な課題”と回避策

課題はゼロではありません。ビルド時間の増大、パーソナライズや会員機能のリアルタイム性、プレビューの編集体験、検索や集計の計算場所、マルチリージョンの一貫性などが挙がります。重要なのは、課題をJamstackの原則を壊さずに解くことです。具体的にはISRやオンデマンド再検証で更新伝播を秒単位に近づけ、エッジ関数で軽量なパーソナライズ(地域やABバリアント)を行い、重い集計は別基盤に逃がします。編集体験は“ドラフトURL→プレビュー専用SSR→承認で公開面を静的化”という2レーン運用が定石です。

ISRのオンデマンド再検証は、CMSのWebhookで叩くAPIルートを用意するのが確実です。

// pages/api/revalidate.ts
import type { NextApiRequest, NextApiResponse } from 'next';

export default async function handler(req: NextApiRequest, res: NextApiResponse) {
  const secret = req.query.secret;
  if (secret !== process.env.REVALIDATE_TOKEN) return res.status(401).json({ message: 'Invalid token' });
  const path = req.query.path as string;
  try {
    await res.revalidate(path);
    return res.json({ revalidated: true, path });
  } catch (e) {
    return res.status(500).json({ revalidated: false, error: String(e) });
  }
}

エッジでの軽量なパーソナライズは、地理やABテストのバリアント選択に留め、HTML自体は静的のままにします。Cloudflare Workersを使う例では、Cookieでバリアントを保持しつつ、静的オブジェクトを差し替えます。

// worker.js
import { getAssetFromKV } from '@cloudflare/kv-asset-handler';

addEventListener('fetch', event => event.respondWith(handle(event)));

async function handle(event) {
  const url = new URL(event.request.url);
  const cookies = event.request.headers.get('cookie') || '';
  const variant = cookies.includes('var=B') ? 'B' : 'A';
  if (url.pathname === '/') url.pathname = variant === 'B' ? '/index-b.html' : '/index.html';
  const res = await getAssetFromKV(event);
  return new Response(res.body, res);
}

編集者のプレビューはSSRを活用した“下書き専用ルート”で提供し、本番公開は静的に固定します。Next.jsではDraftモードでの実装が定石です。

// pages/api/preview.ts
import type { NextApiRequest, NextApiResponse } from 'next';

export default function preview(req: NextApiRequest, res: NextApiResponse) {
  if (req.query.secret !== process.env.PREVIEW_TOKEN) return res.status(401).end('Unauthorized');
  res.setPreviewData({});
  const slug = req.query.slug as string;
  res.writeHead(307, { Location: `/preview/${slug}` });
  res.end();
}

部分ハイドレーションでクライアントJSを最小化する選択肢も有効です。Astroはデフォルトで静的レンダリングを採用し、必要な島だけをインタラクティブにします。

---
import ReactCounter from '../components/Counter.jsx';
const { title, html } = Astro.props;
---
<html>
  <head><title>{title}</title></head>
  <body>
    <article set:html={html} />
    <ReactCounter client:idle />
  </body>
</html>

CI/CDは“ビルドの再現性”が生命線です。環境変数のスコープ、キャッシュキー、データフェッチの失敗時挙動(フェイルオープン/クローズ)を決め、プレビューブランチに自動デプロイします。

# .github/workflows/deploy.yml
name: Deploy
on:
  push:
    branches: [ main ]

jobs:
  build:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      - uses: actions/setup-node@v4
        with: { node-version: '20' }
      - run: npm ci
      - run: npm run build
        env:
          CMS_ENDPOINT: ${{ secrets.CMS_ENDPOINT }}
          CMS_TOKEN: ${{ secrets.CMS_TOKEN }}
      - run: npx vercel deploy --prod --token ${{ secrets.VERCEL_TOKEN }}

移行の意思決定とROIをどう設計するか

Jamstack移行は技術刷新であると同時にプロダクト運用の再設計です。まずは編集要件の棚卸しから着手し、コンテンツモデルを“構造化データ”として定義します。URL設計とリダイレクト方針は最初に合意し、画像最適化の標準(形式、サイズ、生成戦略)をチーム共通のルールに落とします。プレビュー要件は初期段階で実装し、編集者のフィードバックループを短く保つことが成功率を左右します。並走期間はDNSを切り替える前の“デュアルラン”が現実的で、シームレスな移行にはサイトマップとヘルスチェックの自動化が欠かせません。投資対効果は、Core Web Vitalsの達成率¹、CDNヒット率、ビルド時間、障害件数、レビューリードタイムなど、チームが直接コントロールできる指標で追うのが実務的です。一般に中規模サイトでは3–6ヶ月の移行期間を設定し、運用の安定化とA/Bでの成果測定に1–2スプリントを割くケースが多いとされています。

Jamstackは“何でもできる魔法”ではありませんが、できることとできないことの境界が明快であるがゆえに、組織の役割分担をシャープにします。CDNとビルドの責務に合わせて、SREはオフロードとエッジ、フロントエンドはプリレンダリングとハイドレーション、コンテンツはモデルとプレビュー、バックエンドはAPI契約と集計基盤にフォーカスが割り当てられます。この再編成自体が、属人化を減らし可観測性を上げるという経営価値に直結します。

参考リソースとして、Core Web Vitalsの改善手順やヘッドレスCMS選定の観点、ISRの詳解、CDNキャッシュ戦略についてはweb.devやJamstack.comなどの公開ドキュメントが充実しています¹³⁴。要件に応じて公式ガイドやベストプラクティスを参照し、設計にフィードバックするのが近道です。

まとめ:Jamstackは“速さ”を組織に実装する

Jamstackの価値は、単一の技術選択ではなく、“事前計算して配る”という原理をチームの働き方にまで浸透させる点にあります。編集者はプレビューと承認に集中し、開発者は再現性のあるビルドと自動テストに集中し、SREはエッジ配信と可観測性に集中する。結果として、数値で語れる改善が積み重なります。あなたのプロダクトにとって、どのページが静的化でき、どの体験が動的性を必要とするのか。今日、計測と棚卸しから一歩を始めてみませんか。もし既存CMSの性能や運用に悩みがあるなら、まずは一部のLPやブログセクションを対象に小さくJamstackを導入し、その効果を指標で確かめるのが最短距離です。速さは機能です。機能は設計から生まれます。

参考文献

  1. Google web.dev. Defining the Core Web Vitals metrics thresholds. https://web.dev/articles/defining-core-web-vitals-thresholds
  2. Google web.dev. Interaction to Next Paint (INP). https://web.dev/inp/
  3. Jamstack.com. What is Jamstack? https://jamstack.com/
  4. Google web.dev. Time to First Byte (TTFB). https://web.dev/articles/ttfb/
  5. CSS-Tricks. Static vs Dynamic vs Jamstack — Where’s the Line? https://css-tricks.com/static-vs-dynamic-vs-jamstack-wheres-the-line/
  6. CSS-Tricks. A Look at Jamstack’s Speed by the Numbers. https://css-tricks.com/a-look-at-jamstacks-speed-by-the-numbers/