@react-router/cloudflare
Advanced tools
Changelog
v7.3.0
Date: 2025-03-06
fetcherKey
as a parameter to patchRoutesOnNavigation
(#13061)react-router
- Detect and handle manifest-skew issues on new deploys during active sessions (#13061)
fetcher
calls to undiscovered routes, this mismatch will trigger a document reload of the current pathreact-router
- Skip resource route flow in dev server in SPA mode (#13113)react-router
- Fix single fetch _root.data
requests when a basename
is used (#12898)react-router
- Fix types for loaderData
and actionData
that contained Record
s (#13139)
unstable_SerializesTo
- see the note in the Unstable Changes
section below for more information@react-router/dev
- Fix support for custom client build.rollupOptions.output.entryFileNames
(#13098)@react-router/dev
- Fix usage of prerender
option when serverBundles
option has been configured or provided by a preset, e.g. vercelPreset
from @vercel/react-router
(#13082)@react-router/dev
- Fix support for custom build.assetsDir
(#13077)@react-router/dev
- Remove unused dependencies (#13134)@react-router/dev
- Stub all routes except root in "SPA Mode" server builds to avoid issues when route modules or their dependencies import non-SSR-friendly modules (#13023)@react-router/dev
- Remove unused Vite file system watcher (#13133)@react-router/dev
- Fix support for custom SSR build input when serverBundles
option has been configured (#13107)
future.unstable_viteEnvironmentApi
and serverBundles
options together, hyphens are no longer supported in server bundle IDs since they also need to be valid Vite environment names.@react-router/dev
- Fix dev server when using HTTPS by stripping HTTP/2 pseudo headers from dev server requests (#12830)@react-router/dev
- Lazy load Cloudflare platform proxy on first dev server request when using the cloudflareDevProxy
Vite plugin to avoid creating unnecessary workerd
processes (#13016)@react-router/dev
- Fix duplicated entries in typegen for layout routes and their corresponding index route (#13140)@react-router/express
- Update express
peerDependency
to include v5 (https://github.com/remix-run/react-router/pull/13064) (#12961)⚠️ Unstable features are not recommended for production use
react-router
- Add context
support to client side data routers (unstable) (#12941)react-router
- Support middleware on routes (unstable) (#12941)@react-router/dev
- Fix errors with future.unstable_viteEnvironmentApi
when the ssr
environment has been configured by another plugin to be a custom Vite.DevEnvironment
rather than the default Vite.RunnableDevEnvironment
(#13008)@react-router/dev
- When future.unstable_viteEnvironmentApi
is enabled and the ssr
environment has optimizeDeps.noDiscovery
disabled, define optimizeDeps.entries
and optimizeDeps.include
(#13007)context
(unstable)Your application clientLoader
/clientAction
functions (or loader
/action
in library mode) will now receive a context
parameter on the client. This is an instance of unstable_RouterContextProvider
that you use with type-safe contexts (similar to React.createContext
) and is most useful with the corresponding unstable_clientMiddleware
API:
import { unstable_createContext } from "react-router";
type User = {
/*...*/
};
const userContext = unstable_createContext<User>();
const sessionMiddleware: Route.unstable_ClientMiddlewareFunction = async ({
context,
}) => {
let user = await getUser();
context.set(userContext, user);
};
export const unstable_clientMiddleware = [sessionMiddleware];
export function clientLoader({ context }: Route.ClientLoaderArgs) {
let user = context.get(userContext);
let profile = await getProfile(user.id);
return { profile };
}
Similar to server-side requests, a fresh context
will be created per navigation (or fetcher
call). If you have initial data you'd like to populate in the context for every request, you can provide an unstable_getContext
function at the root of your app:
createBrowserRouter(routes, { unstable_getContext })
<HydratedRouter unstable_getContext>
This function should return an value of type unstable_InitialContext
which is a Map<unstable_RouterContext, unknown>
of context's and initial values:
const loggerContext = unstable_createContext<(...args: unknown[]) => void>();
function logger(...args: unknown[]) {
console.log(new Date.toISOString(), ...args);
}
function unstable_getContext() {
let map = new Map();
map.set(loggerContext, logger);
return map;
}
Middleware is implemented behind a future.unstable_middleware
flag. To enable, you must enable the flag and the types in your react-router.config.ts
file:
import type { Config } from "@react-router/dev/config";
import type { Future } from "react-router";
declare module "react-router" {
interface Future {
unstable_middleware: true; // 👈 Enable middleware types
}
}
export default {
future: {
unstable_middleware: true, // 👈 Enable middleware
},
} satisfies Config;
⚠️ Middleware is unstable and should not be adopted in production. There is at least one known de-optimization in route module loading for clientMiddleware
that we will be addressing this before a stable release.
⚠️ Enabling middleware contains a breaking change to the context
parameter passed to your loader
/action
functions - see below for more information.
Once enabled, routes can define an array of middleware functions that will run sequentially before route handlers run. These functions accept the same parameters as loader
/action
plus an additional next
parameter to run the remaining data pipeline. This allows middlewares to perform logic before and after handlers execute.
// Framework mode
export const unstable_middleware = [serverLogger, serverAuth]; // server
export const unstable_clientMiddleware = [clientLogger]; // client
// Library mode
const routes = [
{
path: "/",
// Middlewares are client-side for library mode SPA's
unstable_middleware: [clientLogger, clientAuth],
loader: rootLoader,
Component: Root,
},
];
Here's a simple example of a client-side logging middleware that can be placed on the root route:
const clientLogger: Route.unstable_ClientMiddlewareFunction = async (
{ request },
next
) => {
let start = performance.now();
// Run the remaining middlewares and all route loaders
await next();
let duration = performance.now() - start;
console.log(`Navigated to ${request.url} (${duration}ms)`);
};
Note that in the above example, the next
/middleware
functions don't return anything. This is by design as on the client there is no "response" to send over the network like there would be for middlewares running on the server. The data is all handled behind the scenes by the stateful router
.
For a server-side middleware, the next
function will return the HTTP Response
that React Router will be sending across the wire, thus giving you a chance to make changes as needed. You may throw a new response to short circuit and respond immediately, or you may return a new or altered response to override the default returned by next()
.
const serverLogger: Route.unstable_MiddlewareFunction = async (
{ request, params, context },
next
) => {
let start = performance.now();
// 👇 Grab the response here
let res = await next();
let duration = performance.now() - start;
console.log(`Navigated to ${request.url} (${duration}ms)`);
// 👇 And return it here (optional if you don't modify the response)
return res;
};
You can throw a redirect
from a middleware to short circuit any remaining processing:
import { sessionContext } from "../context";
const serverAuth: Route.unstable_MiddlewareFunction = (
{ request, params, context },
next
) => {
let session = context.get(sessionContext);
let user = session.get("user");
if (!user) {
session.set("returnTo", request.url);
throw redirect("/login", 302);
}
};
Note that in cases like this where you don't need to do any post-processing you don't need to call the next
function or return a Response
.
Here's another example of using a server middleware to detect 404s and check the CMS for a redirect:
const redirects: Route.unstable_MiddlewareFunction = async ({
request,
next,
}) => {
// attempt to handle the request
let res = await next();
// if it's a 404, check the CMS for a redirect, do it last
// because it's expensive
if (res.status === 404) {
let cmsRedirect = await checkCMSRedirects(request.url);
if (cmsRedirect) {
throw redirect(cmsRedirect, 302);
}
}
return res;
};
For more information on the middleware
API/design, please see the decision doc.
context
parameterWhen middleware is enabled, your application will use a different type of context
parameter in your loaders and actions to provide better type safety. Instead of AppLoadContext
, context
will now be an instance of ContextProvider
that you can use with type-safe contexts (similar to React.createContext
):
import { unstable_createContext } from "react-router";
import { Route } from "./+types/root";
import type { Session } from "./sessions.server";
import { getSession } from "./sessions.server";
let sessionContext = unstable_createContext<Session>();
const sessionMiddleware: Route.unstable_MiddlewareFunction = ({
context,
request,
}) => {
let session = await getSession(request);
context.set(sessionContext, session);
// ^ must be of type Session
};
// ... then in some downstream middleware
const loggerMiddleware: Route.unstable_MiddlewareFunction = ({
context,
request,
}) => {
let session = context.get(sessionContext);
// ^ typeof Session
console.log(session.get("userId"), request.method, request.url);
};
// ... or some downstream loader
export function loader({ context }: Route.LoaderArgs) {
let session = context.get(sessionContext);
let profile = await getProfile(session.get("userId"));
return { profile };
}
If you are using a custom server with a getLoadContext
function, the return value for initial context values passed from the server adapter layer is no longer an object and should now return an unstable_InitialContext
(Map<RouterContext, unknown>
):
let adapterContext = unstable_createContext<MyAdapterContext>();
function getLoadContext(req, res): unstable_InitialContext {
let map = new Map();
map.set(adapterContext, getAdapterContext(req));
return map;
}
unstable_SerializesTo
unstable_SerializesTo
added a way to register custom serialization types in Single Fetch for other library and framework authors like Apollo. It was implemented with branded type whose branded property that was made optional so that casting arbitrary values was easy:
// without the brand being marked as optional
let x1 = 42 as unknown as unstable_SerializesTo<number>;
// ^^^^^^^^^^
// with the brand being marked as optional
let x2 = 42 as unstable_SerializesTo<number>;
However, this broke type inference in loaderData
and actionData
for any Record
types as those would now (incorrectly) match unstable_SerializesTo
. This affected all users, not just those that depended on unstable_SerializesTo
. To fix this, the branded property of unstable_SerializesTo
is marked as required instead of optional.
For library and framework authors using unstable_SerializesTo
, you may need to add as unknown
casts before casting to unstable_SerializesTo
.
create-react-router
react-router
@react-router/architect
@react-router/cloudflare
@react-router/dev
@react-router/express
@react-router/fs-routes
@react-router/node
@react-router/remix-config-routes-adapter
@react-router/serve
Full Changelog: v7.2.0...v7.3.0