Перенаправления (redirects)
Редиректы позволяют перенаправлять входящий путь запроса на другой целевой путь.
Для использования редиректов вы можете использовать ключ redirects в next.config.js:
module.exports = {
async redirects() {
return [
{
source: '/about',
destination: '/',
permanent: true,
},
]
},
}redirects — это асинхронная функция, которая ожидает возврата массива объектов со свойствами source, destination и permanent:
source— шаблон пути входящего запроса.destination— путь, на который нужно перенаправить.permanent—trueилиfalse. Еслиtrue, будет использоваться код состояния 308, который указывает клиентам/поисковым системам кэшировать редирект навсегда. Еслиfalse, будет использоваться код 307, который является временным и не кэшируется.
Почему Next.js использует 307 и 308? Традиционно 302 использовался для временных редиректов, а 301 — для постоянных, но многие браузеры изменяли метод запроса редиректа на
GET, независимо от исходного метода. Например, если браузер делал запросPOST /v1/users, который возвращал код состояния302с местоположением/v2/users, последующий запрос мог бытьGET /v2/usersвместо ожидаемогоPOST /v2/users. Next.js использует коды состояния 307 (временный редирект) и 308 (постоянный редирект), чтобы явно сохранить использованный метод запроса.
basePath:falseилиundefined— еслиfalse,basePathне будет включаться при сопоставлении, может использоваться только для внешних редиректов.locale:falseилиundefined— указывает, следует ли исключать локаль при сопоставлении.has— массив объектов has со свойствамиtype,keyиvalue.missing— массив объектов missing со свойствамиtype,keyиvalue.
Редиректы проверяются до файловой системы, включая страницы и файлы /public.
Редиректы не применяются к клиентской маршрутизации (Link, router.push), если только не присутствует Middleware и соответствует пути.
При применении редиректа любые значения запроса, предоставленные в запросе, будут переданы в целевой путь редиректа. Например, рассмотрим следующую конфигурацию редиректа:
{
source: '/old-blog/:path*',
destination: '/blog/:path*',
permanent: false
}При запросе /old-blog/post-1?hello=world клиент будет перенаправлен на /blog/post-1?hello=world.
Сопоставление путей
Разрешено сопоставление путей, например /old-blog/:slug совпадёт с /old-blog/hello-world (без вложенных путей):
module.exports = {
async redirects() {
return [
{
source: '/old-blog/:slug',
destination: '/news/:slug', // Сопоставленные параметры могут использоваться в destination
permanent: true,
},
]
},
}Сопоставление путей с подстановочными знаками
Для сопоставления пути с подстановочными знаками можно использовать * после параметра, например /blog/:slug* совпадёт с /blog/a/b/c/d/hello-world:
module.exports = {
async redirects() {
return [
{
source: '/blog/:slug*',
destination: '/news/:slug*', // Сопоставленные параметры могут использоваться в destination
permanent: true,
},
]
},
}Сопоставление путей с регулярными выражениями
Для сопоставления пути с регулярным выражением можно обернуть regex в скобки после параметра, например /post/:slug(\\d{1,}) совпадёт с /post/123, но не с /post/abc:
module.exports = {
async redirects() {
return [
{
source: '/post/:slug(\\d{1,})',
destination: '/news/:slug', // Сопоставленные параметры могут использоваться в destination
permanent: false,
},
]
},
}Следующие символы (, ), {, }, :, *, +, ? используются для сопоставления путей с регулярными выражениями, поэтому при использовании в source как неспециальные значения они должны быть экранированы добавлением \\ перед ними:
module.exports = {
async redirects() {
return [
{
// это совпадёт с запросом `/english(default)/something`
source: '/english\\(default\\)/:slug',
destination: '/en-us/:slug',
permanent: false,
},
]
},
}Сопоставление заголовков, кук и запросов
Чтобы редирект применялся только при совпадении значений заголовков, кук или запросов, можно использовать поле has или missing. И source, и все элементы has должны совпадать, а все элементы missing не должны совпадать для применения редиректа.
Элементы has и missing могут иметь следующие поля:
type:String— должно бытьheader,cookie,hostилиquery.key:String— ключ из выбранного типа для сопоставления.value:Stringилиundefined— значение для проверки. Еслиundefined, будет совпадать любое значение. Можно использовать строку, похожую на regex, для захвата определённой части значения, например, если значениеfirst-(?<paramName>.*)используется дляfirst-second, тоsecondможно будет использовать в destination как:paramName.
module.exports = {
async redirects() {
return [
// если присутствует заголовок `x-redirect-me`,
// этот редирект будет применён
{
source: '/:path((?!another-page$).*)',
has: [
{
type: 'header',
key: 'x-redirect-me',
},
],
permanent: false,
destination: '/another-page',
},
// если присутствует заголовок `x-dont-redirect`,
// этот редирект НЕ будет применён
{
source: '/:path((?!another-page$).*)',
missing: [
{
type: 'header',
key: 'x-do-not-redirect',
},
],
permanent: false,
destination: '/another-page',
},
// если совпадают source, query и cookie,
// этот редирект будет применён
{
source: '/specific/:path*',
has: [
{
type: 'query',
key: 'page',
// значение page не будет доступно в destination,
// так как value предоставлено и не использует именованную группу захвата, например (?<page>home)
value: 'home',
},
{
type: 'cookie',
key: 'authorized',
value: 'true',
},
],
permanent: false,
destination: '/another/:path*',
},
// если заголовок `x-authorized` присутствует и
// содержит соответствующее значение, этот редирект будет применён
{
source: '/',
has: [
{
type: 'header',
key: 'x-authorized',
value: '(?<authorized>yes|true)',
},
],
permanent: false,
destination: '/home?authorized=:authorized',
},
// если хост — `example.com`,
// этот редирект будет применён
{
source: '/:path((?!another-page$).*)',
has: [
{
type: 'host',
value: 'example.com',
},
],
permanent: false,
destination: '/another-page',
},
]
},
}Редиректы с поддержкой basePath
При использовании basePath с редиректами каждый source и destination автоматически дополняется basePath, если только вы не добавите basePath: false к редиректу:
module.exports = {
basePath: '/docs',
async redirects() {
return [
{
source: '/with-basePath', // автоматически становится /docs/with-basePath
destination: '/another', // автоматически становится /docs/another
permanent: false,
},
{
// не добавляет /docs, так как установлено basePath: false
source: '/without-basePath',
destination: 'https://example.com',
basePath: false,
permanent: false,
},
]
},
}Редиректы с поддержкой i18n
При использовании i18n с редиректами каждый source и destination автоматически дополняется для обработки настроенных locales, если только вы не добавите locale: false к редиректу. Если используется locale: false, вы должны добавить локаль к source и destination для правильного сопоставления.
module.exports = {
i18n: {
locales: ['en', 'fr', 'de'],
defaultLocale: 'en',
},
async redirects() {
return [
{
source: '/with-locale', // автоматически обрабатывает все локали
destination: '/another', // автоматически передаёт локаль
permanent: false,
},
{
// не обрабатывает локали автоматически, так как установлено locale: false
source: '/nl/with-locale-manual',
destination: '/nl/another',
locale: false,
permanent: false,
},
{
// совпадает с '/', так как `en` — defaultLocale
source: '/en',
destination: '/en/another',
locale: false,
permanent: false,
},
// возможно сопоставить все локали, даже если установлено locale: false
{
source: '/:locale/page',
destination: '/en/newpage',
permanent: false,
locale: false,
},
{
// преобразуется в /(en|fr|de)/(.*), поэтому не совпадёт с корневыми
// маршрутами `/` или `/fr`, как это сделал бы /:path*
source: '/(.*)',
destination: '/another',
permanent: false,
},
]
},
}В некоторых редких случаях может потребоваться назначить пользовательский код состояния для правильного перенаправления старых HTTP-клиентов. В таких случаях можно использовать свойство statusCode вместо permanent, но не оба одновременно. Для обеспечения совместимости с IE11 автоматически добавляется заголовок Refresh для кода состояния 308.
Другие редиректы
- Внутри API Routes можно использовать
res.redirect(). - Внутри
getStaticPropsиgetServerSidePropsможно перенаправлять определённые страницы во время запроса.
История версий
| Версия | Изменения |
|---|---|
v13.3.0 | Добавлено missing. |
v10.2.0 | Добавлено has. |
v9.5.0 | Добавлено redirects. |