コンテンツにスキップ

ルーティング

Astroはファイルベースルーティングを使用して、プロジェクトのsrc/pages/ディレクトリのファイルレイアウトを元にビルドURLを生成します。

Astroでは、ルート間の移動に標準的なHTMLの<a>要素を使用します。フレームワーク固有の<Link>コンポーネントは提供されていません。

src/pages/index.astro
<p>Astroの<a href="/about/">概要</a>についてもっと読む!</p>
<!-- `base: "/docs"`が設定されている場合 -->
<p><a href="/docs/reference/">リファレンス</a>セクションでもっと学びましょう!</p>

src/pages/ディレクトリにある.astroページコンポーネント、MarkdownとMDXファイル(.md.mdx)は、自動的にウェブサイトのページとなります。各ページのルートは、src/pages/ディレクトリ内における自身のパスとファイル名に対応します。

# 静的ルーティングの例
src/pages/index.astro -> mysite.com/
src/pages/about.astro -> mysite.com/about
src/pages/about/index.astro -> mysite.com/about
src/pages/about/me.astro -> mysite.com/about/me
src/pages/posts/1.md -> mysite.com/posts/1

Astroページファイルのファイル名に動的ルートパラメータを指定すると、ファイルにマッチするページを複数生成できます。たとえばsrc/pages/authors/[author].astroは、ブログの各著者に対してプロフィールページを生成します。authorは、ページ内からアクセス可能なパラメータとなります。

Astroのデフォルトの静的出力モードでは、これらのページはビルド時に生成されるため、authorと対応するファイルを取得する場合、それらのリストを事前に決めておく必要があります。SSRモードでは、ルートにマッチしたリクエストに応じてページが生成されます。

すべてのルートをビルド時に決める必要があるため、動的ルートはgetStaticPaths()をエクスポートし、そこでparamsプロパティをもつオブジェクトの配列を返す必要があります。各オブジェクトは対応するルートを生成します。

[dog].astroはファイル名に動的なdogパラメータが定義されているため、getStaticPaths()から返されるオブジェクトのparamsにはdogを含める必要があります。Astro.paramsを使用してページからこのパラメータにアクセスできます。

src/pages/dogs/[dog].astro
---
export function getStaticPaths() {
return [
{params: {dog: 'clifford'}},
{params: {dog: 'rover'}},
{params: {dog: 'spot'}},
];
}
const { dog } = Astro.params;
---
<div>いい子だ、{dog}</div>

上のコードにより、/dogs/clifford/dogs/rover/dogs/spotという3つのページが生成され、各ページでは対応する犬の名前が表示されます。

ファイル名には複数のパラメータを含められますが、これらをすべてgetStaticPaths()paramsオブジェクトに含める必要があります。

src/pages/[lang]-[version]/info.astro
---
export function getStaticPaths () {
return [
{params: {lang: 'en', version: 'v1'}},
{params: {lang: 'fr', version: 'v2'}},
];
}
const { lang, version } = Astro.params;
---
...

上のコードは/en-v1/info/fr-v2/infoを生成します。

パラメータはパス内の異なる部分に設定できます。たとえば、上と同じgetStaticPaths()をもつsrc/pages/[lang]/[version]/info.astroファイルは、/en/v1/info/fr/v2/infoのルートを生成します。

getStaticPaths() (EN)についてもっと学ぶ。

getStaticPaths()関数に提供されるparamsはデコードされていません。パラメータ値をデコードする必要がある場合は、decodeURI関数を使用してください。

src/pages/[slug].astro
---
export function getStaticPaths() {
return [
{ params: { slug: decodeURI("%5Bpage%5D") }}, // "[page]"にデコードされます
]
}
---
getStaticPaths (EN)についてもっと詳しく見る。
関連レシピ: Add i18n features (EN)

より柔軟なURLルーティングが必要な場合は、.astroファイル名にレストパラメータ[...path])を使用することで、任意の深さのファイルパスにマッチさせられます。

src/pages/sequences/[...path].astro
---
export function getStaticPaths() {
return [
{params: {path: 'one/two/three'}},
{params: {path: 'four'}},
{params: {path: undefined }}
]
}
const { path } = Astro.params;
---
...

上のコードは/sequences/one/two/three/sequences/four/sequencesを生成します。(レストパラメータをundefinedに設定することで、トップレベルのページにマッチさせられます。)

レストパラメータは他の名前付きパラメータと組み合わせて使用できます。たとえば、GitHubのファイルビューアは以下の動的ルートで表現できます。

/[org]/[repo]/tree/[branch]/[...file]

この例では、/withastro/astro/tree/main/docs/public/favicon.svgへのリクエストは、以下の名前付きパラメータへと分割されます。

{
org: 'withastro',
repo: 'astro',
branch: 'main',
file: 'docs/public/favicon.svg'
}

以下の例は、レストパラメータ([...slug])とgetStaticPaths()props (EN)機能を使用して、異なる深さのスラグに対してページを生成します。

src/pages/[...slug].astro
---
export async function getStaticPaths() {
const pages = [
{
slug: undefined,
title: "Astroストア",
text: "Astroにようこそ!",
},
{
slug: "products",
title: "Astroグッズ",
text: "たくさんの商品があります",
},
{
slug: "products/astro-handbook",
title: "究極のAstroハンドブック",
text: "Astroについて学びたければ、この本を読む必要があります",
},
];
return pages.map(({ slug, title, text }) => {
return {
params: { slug },
props: { title, text },
};
});
}
const { title, text } = Astro.props;
---
<html>
<head>
<title>{title}</title>
</head>
<body>
<h1>{title}</h1>
<p>{text}</p>
</body>
</html>

オンデマンドレンダリングでも、動的ルートの定義方法は同じです。ファイル名に[param][...path]などのブラケット表記を付ければ、任意の文字列やパスにマッチさせられます。ただし、これらのルートはビルド時に生成されないため、一致するURLへのリクエストごとにページが配信されます。静的ルートではないため、getStaticPaths()は使用しません。

オンデマンドレンダリング用のルートでは、スプレッド構文([...param])によるレストパラメータはファイル名につき 1つ だけ指定できます。たとえば src/pages/[locale]/[...slug].astrosrc/pages/[...locale]/[slug].astro は許可されますが、src/pages/[...locale]/[...slug].astro は許可されません。

src/pages/resources/[resource]/[id].astro
---
export const prerender = false; // 'server'モードでは不要
const { resource, id } = Astro.params;
---
<h1>{resource}: {id}</h1>

このページは、resources/users/1resources/colors/blueなど、任意のresourceidに対して提供されます。

SSRページではgetStaticPaths()を使用できないため、propsを受け取れません。しかし、オブジェクト内のslugパラメータの値を参照することで、前の例をSSRモードにも対応させられます。ルーティングが”/“の場合、slugパラメータはundefinedになります。以下の例では値がオブジェクトに存在しない場合は、404ページにリダイレクトしています。

src/pages/[...slug].astro
---
const pages = [
{
slug: undefined,
title: "Astroストア",
text: "Astroにようこそ!",
},
{
slug: "products",
title: "Astroグッズ",
text: "たくさんの商品があります",
},
{
slug: "products/astro-handbook",
title: "究極のAstroハンドブック",
text: "Astroについて学びたければ、この本を読む必要があります",
},
];
const { slug } = Astro.params;
const page = pages.find((page) => page.slug === slug);
if (!page) return Astro.redirect("/404");
const { title, text } = page;
---
<html>
<head>
<title>{title}</title>
</head>
<body>
<h1>{title}</h1>
<p>{text}</p>
</body>
</html>

読者を新しいページにリダイレクトしなければならない場合があります。サイトの構造が変更されたために永久的にリダイレクトする必要がある場合や、認証されたルートにログインするといったアクションに応じておこなう場合などです。

Astroの設定で、永久的に移動したページにユーザーをリダイレクトするルールを定義できます。また、ユーザーがサイトを利用した際に動的にリダイレクトすることもできます。

追加: astro@2.9.0

redirects値を使用して、Astroの設定から永久的なリダイレクトのマッピングを指定できます。

サイト内のリダイレクトを定義する場合、これは古いルートから新しいルートへのマッピングとなります。Astro v5.2.0以降では、httpまたはhttpsで始まる解析可能な外部URLへのリダイレクトも可能です。

astro.config.mjs
import { defineConfig } from 'astro/config';
export default defineConfig({
redirects: {
'/old-page': '/new-page',
'/blog': 'https://example.com/blog'
}
});

これらのリダイレクトはファイルベースのルーティングと同じ優先順位ルールに従い、プロジェクト内に同じ名前の既存のページファイルがある場合は常に優先順位が低くなります。たとえば、プロジェクトにsrc/pages/old-page.astroファイルが含まれている場合、/old-page/new-pageにリダイレクトされません。

新しいルートと古いルートの両方に同じパラメータが含まれていれば、動的ルートも可能です。

{
"/blog/[...slug]": "/articles/[...slug]"
}

SSRまたは静的アダプターを使用するとオブジェクトを値として設定でき、そこでstatusコードや新しいdestinationを指定できます。

astro.config.mjs
import { defineConfig } from 'astro/config';
export default defineConfig({
redirects: {
'/old-page': {
status: 302,
destination: '/new-page'
},
'/news': {
status: 302,
destination: 'https://example.com/news'
}
}
});

astro buildの実行時に、Astroはデフォルトでmeta refreshタグを含むHTMLファイルを出力します。サポートされているアダプターの場合は、代わりにホストの設定ファイルにリダイレクトを含めて出力します。

ステータスコードはデフォルトで301です。HTMLファイルをビルドする場合、ステータスコードはサーバーによって使用されません。

AstroグローバルのAstro.redirectメソッドを使用すると、別のページに動的にリダイレクトできます。たとえばクッキーからセッションを取得してユーザーのログイン状態を確認した後などにこれをおこないます。

src/pages/account.astro
---
import { isLoggedIn } from '../utils';
const cookie = Astro.request.headers.get('cookie');
// ユーザーがログインしていない場合は、ログインページにリダイレクトします
if (!isLoggedIn(cookie)) {
return Astro.redirect('/login');
}
---

追加: astro@4.13.0

リライトを使用すると、ブラウザを別ページへリダイレクトせずに別のURLパスの内容を提供できます。アドレスバーには元のURLが表示されたままですが、実際にはAstro.rewrite() (EN)で指定したページが表示されます。

リライトは、複数のパスで同じコンテンツを表示する場合(例:/products/shoes/men//products/men/shoes/)に便利で、2つの異なるソースファイルを別々にメンテナンスする必要がなくなります。

リライトはSEOとユーザー体験の両面で有効です。リダイレクトせずに済むケースや、本来 404 を返すURLでもコンテンツを表示できます。リライトの一般的な使用例は、言語の異なるバリアントに対して同じローカライズされたコンテンツを表示することです。

次の例では、/es-CU/(キューバスペイン語)にアクセスするとリライトにより/es/バージョンのページを表示します。/es-cu/articles/introductionへのリクエストでは、src/pages/es/articles/introduction.astroのコンテンツが表示されます。

src/pages/es-cu/articles/introduction.astro
---
return Astro.rewrite("/es/articles/introduction");
---

エンドポイントファイルでcontext.rewrite()を使用して、別のページにルーティングできます。

src/pages/api.js
export function GET(context) {
if (!context.locals.allowed) {
return context.rewrite("/");
}
}

Astro.rewrite()に渡されたURLが実行時エラーを発生させる場合、Astroは開発環境ではオーバーレイエラーを表示し、本番環境では500ステータスコードを返します。URLがプロジェクトに存在しない場合は、404ステータスコードが返されます。

意図的にリライトを使って/404ページをレンダリングすることもできます。例えば、eコマースショップの商品が利用できなくなったことを示す場合、以下のようにHTTPレスポンスステータスに基づいて条件付きでリライトできます。

src/pages/[item].astro
---
const { item } = Astro.params;
if (!itemExists(item)) {
return Astro.rewrite("/404");
}
---

また、例えば存在しないURLにアクセスした際にサイトの特定のページを表示する場合は、以下のようにパスを指定できます。

src/middleware.mjs
export const onRequest = async (context, next) => {
const response = await next();
if (response.status === 404) {
return context.rewrite("/");
}
return response;
}

指定されたリライトパスからコンテンツを表示する前に、Astro.rewrite()関数は新しい完全なレンダリングフェーズをトリガーします。これにより、新しいルーティング/リクエストに対するミドルウェアが再実行されます。

Astro.rewrite()APIリファレンス (EN)についてもっと詳しく見る。

複数のルートが同じURLパスをビルドする可能性があります。たとえば、以下のルートはすべて/posts/createをビルドできます。

  • ディレクトリsrc/pages/
    • […slug].astro
    • ディレクトリposts/
      • create.astro
      • [page].astro
      • [pid].ts
      • […slug].astro

Astroは、ページをビルドするためにどのルートを使用すべきかを知る必要があります。そのために、以下のルールを順に適用してルートの順番を決定します。

  • Astroの予約ルート
  • より多くのパスセグメントを持つルートが、詳細度が低いルートよりも優先される。上の例では、/posts/配下のすべてのルートが、/[...slug].astroよりも優先される。
  • パスパラメータを持たない静的ルートは、動的ルートより優先される。例えば、/posts/create.astroは、他のすべてのルートよりも優先される。
  • 名前付きパラメータを使用する動的ルーティングは、レストパラメータよりも優先される。例えば、/posts/[page].astro/posts/[...slug].astroよりも優先される。
  • 事前レンダリングされた動的ルートは、サーバーの動的ルートよりも優先される。
  • エンドポイントはページよりも優先される。
  • ファイルベースのルートはリダイレクトよりも優先される。
  • ルートを上記のルールで解決できない場合、Nodeのデフォルトロケールに基づいてアルファベット順に解決される。

上記のようにファイルが配置されている場合に、リクエストされたURLと、HTMLをビルドするために使用されるルートがどのようにマッチングされるかの例をいくつか見てみましょう。

  • pages/posts/create.Astro - /posts/createだけをビルドします
  • pages/posts/[pid].ts - /posts/abc/posts/xyzなどをビルドします。しかし、/posts/createはビルドしません
  • pages/posts/[page].astro - /posts/1/posts/2などをビルドします。しかし、/posts/create/posts/abc/posts/xyzはビルドしません
  • pages/posts/[...slug].astro - /posts/1/2/posts/a/b/cなどをビルドします。しかし、/posts/create/posts/1/posts/abcなどはビルドしません
  • pages/[...slug].astro - /abc/xyz/abc/xyzなどをビルドします。しかし、/posts/create/posts/1/posts/abcなどはビルドしません

内部ルートは、Astroの機能を正しく動作させるため、ユーザー定義またはインテグレーション定義のルートよりも優先されます。以下はAstroの予約ルートです。

  • _astro/:CSS文書、バンドルされたクライアントスクリプト、最適化された画像、およびその他のViteアセットなど、すべての静的アセットをクライアントに提供します。
  • _server_islands/サーバーアイランドで遅延実行される動的コンポーネントを提供します。
  • _actions/:定義されたアクションを提供します。

Astroは、複数のページに分割する必要がある大規模なデータコレクションのために、ページネーションを組み込みでサポートしています。Astroは、前ページと次ページのURL、総ページ数など、一般的なページネーションプロパティを生成します。

ページネーションされるルート名には、標準的な動的ルートと同じ[ブラケット]構文を使用する必要があります。たとえば、ファイル名 /astronauts/[page].astro/astronauts/1, /astronauts/2 などのルートを生成し、[page]は生成されるページ番号となります。

paginate()関数を使用すると、次のように値の配列に対してこれらのページを生成できます。

src/pages/astronauts/[page].astro
---
export async function getStaticPaths({ paginate }) {
const astronautPages = [{
astronaut: 'ニール・アームストロング',
}, {
astronaut: 'バズ・オルドリン',
}, {
astronaut: 'サリー・ライド',
}, {
astronaut: 'ジョン・グレン',
}];
// 宇宙飛行士の配列から、1ページに2人づつ入るようにページを生成します
return paginate(astronautPages, { pageSize: 2 });
}
// ページネーションされたデータは、すべて"page"プロパティとして渡されます
const { page } = Astro.props;
---
<!--現在のページ番号を表示します。Astro.params.pageも使用可能です!-->
<h1>{page.currentPage}ページ</h1>
<ul>
<!--宇宙飛行士情報の配列を列挙します-->
{page.data.map(({ astronaut }) => <li>{astronaut}</li>)}
</ul>

これで、1ページに2つのアイテムが配置された、以下のページが生成されます。

  • /astronauts/1 - 1ページ目には「ニール・アームストロング」と「バズ・オルドリン」を表示します
  • /astronauts/2 - 2ページ目には「サリー・ライド」と「ジョン・グレン」を表示します

paginate()関数を使用すると、各ページのデータはpageプロパティで渡されます。pageプロパティは多くの便利なプロパティを持っています。

interface Page<T = any> {
/** paginate() 関数に渡されたページのデータを含む配列 */
data: T[];
/** メタデータ */
/** 0から始まる、ページ上の最初のアイテムのインデックス */
start: number;
/** 0から始まる、ページ上の最後のアイテムのインデックス */
end: number;
/** 結果の総数 */
total: number;
/** 1から始まる、現在のページ番号 */
currentPage: number;
/** 1ページあたりのアイテム数(デフォルトは10) */
size: number;
/** 最終ページ番号 */
lastPage: number;
url: {
/** 現在のページのURL */
current: string;
/** 前のページのURL(もしあれば) */
prev: string | undefined;
/** 次のページのURL(もしあれば) */
next: string | undefined;
/** 最初のページのURL(現在のページが最初のページでない場合) */
first: string | undefined;
/** 最後のページのURL(現在のページが最後のページでない場合) */
last: string | undefined;
};
}

以下の例では、ページの現在の情報とページ間を移動するためのリンクを表示しています。

src/pages/astronauts/[page].astro
---
// 前の例と同じように、{ astronaut } オブジェクトのリストをページネーションします
export function getStaticPaths({ paginate }) { /* ... */ }
const { page } = Astro.props;
---
<h1>{page.currentPage}ページ</h1>
<ul>
{page.data.map(({ astronaut }) => <li>{astronaut}</li>)}
</ul>
{page.url.first ? <a href={page.url.first}>最初</a> : null}
{page.url.prev ? <a href={page.url.prev}>前へ</a> : null}
{page.url.next ? <a href={page.url.next}>次へ</a> : null}
{page.url.last ? <a href={page.url.last}>最後</a> : null}
ページネーションのpage (EN)についてもっと学ぶ。

ページネーションのより高度なユースケースはネストされたページネーションです。これは、ページネーションを他の動的ルーティングパラメータと組み合わせたものです。ネストされたページネーションを使用すると、ページネーションされたコレクションを何らかのプロパティやタグでグループ化できます。

たとえば、ページネーションされたMarkdownの投稿を何らかのタグでグループ化したい場合、以下のURLにマッチする/src/pages/[tag]/[page].Astroページを作成してネストされたページネーションを使用します。

  • /red/1 (tag=red)
  • /red/2 (tag=red)
  • /blue/1 (tag=blue)
  • /green/1 (tag=green)

ネストされたページネーションは、paginate()の結果をグループごとに配列としてgetStaticPaths()から返すことで動作します。

以下の例では、上記のURLを作成するために、ネストされたページネーションを実装しています。

src/pages/[tag]/[page].astro
---
export function getStaticPaths({paginate}) {
const allTags = ['red', 'blue', 'green'];
const allPosts = await Astro.glob('../../posts/*.md');
// すべてのタグに対して、paginate()の結果を返します。
// その結果がどのタググループに対するものかAstroに伝えるために、
// `{params: {tag}}`を必ず`paginate()`に渡してください。
return allTags.flatMap((tag) => {
const filteredPosts = allPosts.filter((post) => post.frontmatter.tag === tag);
return paginate(filteredPosts, {
params: { tag },
pageSize: 10
});
});
}
const { page } = Astro.props;
const params = Astro.params;

アンダースコア(_)を接頭辞としてファイル名に付けることで、ページやディレクトリをビルドから除外できます。アンダースコアで始まるファイルはルーターによって認識されず、dist/ディレクトリにも配置されません。

これを使用すると、一時的にページを無効にしたり、テストやユーティリティ、コンポーネントを関連するページと同じフォルダーに配置したりできます。

以下の例では、src/pages/index.astrosrc/pages/posts/post1.mdのみがページルートとHTMLファイルとしてビルドされます。

  • ディレクトリsrc/pages/
    • ディレクトリ_hidden-directory/
      • page1.md
      • page2.md
    • _hidden-page.astro
    • index.astro
    • ディレクトリposts/
      • _SomeComponent.astro
      • _utils.js
      • post1.md
貢献する コミュニティ スポンサー