Headers are primarily defined with the route module headers
export. You can also set headers in entry.server.tsx
.
import { Route } from "./+types/some-route";
export function headers(_: Route.HeadersArgs) {
return {
"Content-Security-Policy": "default-src 'self'",
"X-Frame-Options": "DENY",
"X-Content-Type-Options": "nosniff",
"Cache-Control": "max-age=3600, s-maxage=86400",
};
}
You can return either a Headers
instance or HeadersInit
.
When the header is dependent on loader data, loaders and actions can also set headers.
data
import { data } from "react-router";
export async function loader({ params }: LoaderArgs) {
let [page, ms] = await fakeTimeCall(
await getPage(params.id)
);
return data(page, {
headers: {
"Server-Timing": `page;dur=${ms};desc="Page query"`,
},
});
}
headers
exportHeaders from loaders and actions are not sent in a hidden way, you must return them from the headers
export.
export function headers({
actionHeaders,
loaderHeaders,
}: HeadersArgs) {
return actionHeaders ? actionHeaders : loaderHeaders;
}
One notable exception is Set-Cookie
headers, which are automatically preserved from headers
, loader
, and action
in parent routes, even without exporting headers
from the child route.
Consider these nested routes
route("pages", "pages-layout-with-nav.tsx", [
route(":slug", "page.tsx"),
]);
If both route modules want to set headers, the headers from the deepest matching route will be sent.
When you need to keep both the parent and the child headers, you need to merge them in the child route.
The easiest way is to simply append to the parent headers. This avoids overwriting a header the parent may have set and both are important.
export function headers({ parentHeaders }: HeadersArgs) {
parentHeaders.append(
"Permissions-Policy: geolocation=()"
);
return parentHeaders;
}
Sometimes it's important to overwrite the parent header. Do this with set
instead of append
:
export function headers({ parentHeaders }: HeadersArgs) {
parentHeaders.set(
"Cache-Control",
"max-age=3600, s-maxage=86400"
);
return parentHeaders;
}
You can avoid the need to merge headers by only defining headers in "leaf routes" (index routes and child routes without children) and not in parent routes.
entry.server.tsx
The handleRequest
export receives the headers from the route module as an argument. You can append global headers here.
export default function handleRequest(
request,
responseStatusCode,
responseHeaders,
routerContext,
loadContext
) {
// set, append global headers
responseHeaders.set(
"X-App-Version",
routerContext.manifest.version
);
return new Response(await getStream(), {
headers: responseHeaders,
status: responseStatusCode,
});
}
If you don't have an entry.server.tsx
run the reveal
command:
react-router reveal