Headers allow you to set custom HTTP headers for an incoming request path.
To set custom HTTP headers you can use the headers key in
blitz.config.js:
module.exports = {
async headers() {
return [
{
source: "/about",
headers: [
{
key: "x-custom-header",
value: "my custom header value",
},
{
key: "x-another-custom-header",
value: "my other custom header value",
},
],
},
]
},
}headers is an async function that expects an array to be returned
holding objects with source and headers properties:
source is the incoming request path pattern.headers is an array of header objects with the key and value
properties.basePath: false or undefined - if false the basePath won't be
included when matching, can be used for external rewrites only.locale: false or undefined - whether the locale should not be
included when matching.has is an array of has objects
with the type, key and value properties.Headers are checked before the filesystem which includes pages and
/public files.
If two headers match the same path and set the same header key, the last
header key will override the first. Using the below headers, the path
/hello will result in the header x-hello being world due to the last
header value set being world.
module.exports = {
async headers() {
return [
{
source: '/:path*',
headers: [
{
key: 'x-hello',
value: 'there',
},
],
},
{
source: '/hello',
headers: [
{
key: 'x-hello',
value: 'world',
},
],
},
],
},
}Path matches are allowed, for example /blog/:slug will match
/blog/hello-world (no nested paths):
module.exports = {
async headers() {
return [
{
source: '/blog/:slug',
headers: [
{
key: 'x-slug',
value: ':slug', // Matched parameters can be used in the value
},
{
key: 'x-slug-:slug', // Matched parameters can be used in the key
value: 'my other custom header value',
},
],
},
],
},
}To match a wildcard path you can use * after a parameter, for example
/blog/:slug* will match /blog/a/b/c/d/hello-world:
module.exports = {
async headers() {
return [
{
source: '/blog/:slug*',
headers: [
{
key: 'x-slug',
value: ':slug*', // Matched parameters can be used in the value
},
{
key: 'x-slug-:slug*', // Matched parameters can be used in the key
value: 'my other custom header value',
},
],
},
],
},
}To match a regex path you can wrap the regex in parenthesis after a
parameter, for example /blog/:slug(\\d{1,}) will match /blog/123 but
not /blog/abc:
module.exports = {
async headers() {
return [
{
source: '/blog/:post(\\d{1,})',
headers: [
{
key: 'x-post',
value: ':post',
},
],
},
],
},
}The following characters (, ), {, }, :, *, +, ? are used
for regex path matching, so when used in the source as non-special
values they must be escaped by adding \\ before them:
module.exports = {
async redirects() {
return [
{
// this will match `/english(default)/something` being requested
source: "/english\\(default\\)/:slug",
destination: "/en-us/:slug",
permanent: false,
},
]
},
}To only apply a header when either header, cookie, or query values also
match the has field can be used. Both the source and all has items
must match for the header to be applied.
has items have the following fields:
type: String - must be either header, cookie, host, or
query.key: String - the key from the selected type to match against.value: String or undefined - the value to check for, if undefined
any value will match. A regex like string can be used to capture a
specific part of the value, e.g. if the value first-(?<paramName>.*)
is used for first-second then second will be usable in the
destination with :paramName.module.exports = {
async headers() {
return [
// if the header `x-add-header` is present,
// the `x-another-header` header will be applied
{
source: "/:path*",
has: [
{
type: "header",
key: "x-add-header",
},
],
headers: [
{
key: "x-another-header",
value: "hello",
},
],
},
// if the source, query, and cookie are matched,
// the `x-authorized` header will be applied
{
source: "/specific/:path*",
has: [
{
type: "query",
key: "page",
// the page value will not be available in the
// header key/values since value is provided and
// doesn't use a named capture group e.g. (?<page>home)
value: "home",
},
{
type: "cookie",
key: "authorized",
value: "true",
},
],
headers: [
{
key: "x-authorized",
value: ":authorized",
},
],
},
// if the header `x-authorized` is present and
// contains a matching value, the `x-another-header` will be applied
{
source: "/:path*",
has: [
{
type: "header",
key: "x-authorized",
value: "(?<authorized>yes|true)",
},
],
headers: [
{
key: "x-another-header",
value: ":authorized",
},
],
},
// if the host is `example.com`,
// this header will be applied
{
source: "/:path*",
has: [
{
type: "host",
value: "example.com",
},
],
headers: [
{
key: "x-another-header",
value: ":authorized",
},
],
},
]
},
}When leveraging basePath support with
headers each source is automatically prefixed with the basePath unless
you add basePath: false to the header:
module.exports = {
basePath: "/docs",
async headers() {
return [
{
source: "/with-basePath", // becomes /docs/with-basePath
headers: [
{
key: "x-hello",
value: "world",
},
],
},
{
source: "/without-basePath", // is not modified since basePath: false is set
headers: [
{
key: "x-hello",
value: "world",
},
],
basePath: false,
},
]
},
}When leveraging i18n support with headers each
source is automatically prefixed to handle the configured locales
unless you add locale: false to the header. If locale: false is used
you must prefix the source with a locale for it to be matched correctly.
module.exports = {
i18n: {
locales: ["en", "fr", "de"],
defaultLocale: "en",
},
async headers() {
return [
{
source: "/with-locale", // automatically handles all locales
headers: [
{
key: "x-hello",
value: "world",
},
],
},
{
// does not handle locales automatically since locale: false is set
source: "/nl/with-locale-manual",
locale: false,
headers: [
{
key: "x-hello",
value: "world",
},
],
},
{
// this matches '/' since `en` is the defaultLocale
source: "/en",
locale: false,
headers: [
{
key: "x-hello",
value: "world",
},
],
},
{
// this gets converted to /(en|fr|de)/(.*) so will not match the top-level
// `/` or `/fr` routes like /:path* would
source: "/(.*)",
headers: [
{
key: "x-hello",
value: "worlld",
},
],
},
]
},
}Cache-Control headers set in blitz.config.js will be overwritten in
production to ensure that static assets can be cached effectively. If you
need to revalidate the cache of a page that has been
statically generated, you can do
so by setting revalidate in the page's
getStaticProps function.