A Rsbuild plugin that provides seamless integration with React Router, supporting both client-side routing and server-side rendering (SSR).
- 🚀 Zero-config setup with sensible defaults
- 🔄 Automatic route generation from file system
- 🖥️ Server-Side Rendering (SSR) support
- 📱 Client-side navigation with SPA mode (
ssr: false) - 📄 Static prerendering for hybrid static/dynamic sites
- 🛠️ TypeScript support out of the box
- 🔧 Customizable configuration
- 🎯 Support for route-level code splitting
- ☁️ Cloudflare Workers deployment support
- 🔗 Module Federation support (experimental)
npm install rsbuild-plugin-react-router
# or
yarn add rsbuild-plugin-react-router
# or
pnpm add rsbuild-plugin-react-routerFor the federation examples and Playwright e2e tests, use Node 22 and the repo-pinned pnpm version:
nvm install
nvm use
corepack enable
corepack prepare pnpm@9.15.3 --activateAdd the plugin to your rsbuild.config.ts:
import { defineConfig } from '@rsbuild/core';
import { pluginReact } from '@rsbuild/plugin-react';
import { pluginReactRouter } from 'rsbuild-plugin-react-router';
export default defineConfig({
plugins: [
pluginReactRouter({
// options here
}),
pluginReact(),
],
});React Router application settings live in react-router.config.*. The Rsbuild
plugin only needs options for Rsbuild-specific behavior.
pluginReactRouter({
customServer: false,
serverOutput: 'module',
lazyCompilation: undefined,
logPerformance: false,
parallelRouteTransform: undefined,
onRouteTopologyChange: undefined,
federation: false,
});| Option | Default | Description |
|---|---|---|
customServer |
false |
Disables the built-in development SSR middleware. Enable this when an app owns the server with createDevServer() or an adapter. |
serverOutput |
'module' |
Emitted Rsbuild server format: 'module' or 'commonjs'. When omitted, React Router's serverModuleFormat selects the format ('esm' -> 'module', 'cjs' -> 'commonjs'); setting serverOutput overrides it. |
lazyCompilation |
undefined |
Optional Rsbuild dev lazy-compilation config. When enabled here or through dev.lazyCompilation, React Router hydration-critical modules stay eager so the browser manifest and route modules are not replaced by lazy proxies. |
logPerformance |
false |
Logs structured React Router plugin timing information through the Rsbuild logger. |
parallelRouteTransform |
undefined |
Controls worker-thread route transforms. undefined auto-enables workers for 256+ routes, true forces the default worker count, a positive integer sets the worker count, and false keeps transforms inline. |
onRouteTopologyChange |
undefined |
Notification for programmatic/custom dev servers. Recreate the Rsbuild server when route files are added, removed, or moved. The callback is not awaited. |
federation |
false |
Enables the plugin's experimental Module Federation integration. |
When federation is enabled, configure the Module Federation plugin with
experiments.asyncStartup: true. The dev server resolves async server build
exports automatically; production custom servers or adapters should resolve
async exports before passing the build to React Router's request handler.
Put React Router framework settings in react-router.config.*:
import type { Config } from '@react-router/dev/config';
export default {
ssr: true,
buildDirectory: 'build',
appDirectory: 'app',
basename: '/',
splitRouteModules: true,
subResourceIntegrity: false,
} satisfies Config;Commonly used options:
| Option | Default | Notes |
|---|---|---|
ssr |
true |
Set false for SPA mode. SPA mode still runs a build-time server render to create build/client/index.html. |
buildDirectory |
'build' |
Output root. Client assets go in <buildDirectory>/client; server output goes in <buildDirectory>/server. |
appDirectory |
'app' |
Directory containing root, routes, and optional entry.client / entry.server files. |
basename |
'/' |
Base URL used for routing, prerender requests, and manifest asset paths. |
serverBuildFile |
'index.js' |
Server build file name. It must end in .js. |
serverModuleFormat |
'esm' |
React Router server module format: 'esm' or 'cjs'. serverOutput can override the emitted Rsbuild server format. |
serverBundles |
undefined |
Advanced server bundle splitting by route branch. Disabled when ssr: false. |
routeDiscovery |
React Router | Defaults to lazy discovery for SSR and initial discovery for SPA mode. routeDiscovery.mode: 'lazy' is invalid for SPA. |
prerender |
undefined |
true, an array of paths, a function, or { paths, concurrency } / { paths, unstable_concurrency }. |
splitRouteModules |
true |
Splits client route module exports. The legacy future.v8_splitRouteModules flag is also accepted. |
subResourceIntegrity |
false |
Emits SRI metadata for browser scripts. The legacy future.unstable_subResourceIntegrity flag is normalized to this key. |
buildEnd |
undefined |
Hook called after the build with the React Router build manifest and resolved config. |
The plugin will look for react-router.config with any supported JS/TS extension, in this order:
react-router.config.tsxreact-router.config.tsreact-router.config.mtsreact-router.config.jsxreact-router.config.jsreact-router.config.mjs
If none are found, it falls back to defaults.
React Router "Framework Mode" is implemented as a Vite plugin, but this Rsbuild plugin aims to provide equivalent framework-mode behaviors (typegen, Route Module API types, route module splitting, SPA/SSR/prerender strategies) on top of Rsbuild/Rspack.
In practice, you should be able to use the @react-router/dev/* config + routes
APIs, import generated ./+types/* in route modules, and use the standard
entry.client/entry.server entrypoints like you would in the official setup.
This plugin is a lightweight adapter to run React Router on Rsbuild. It does not aim to replace ModernJS or its higher-level framework features. If your goal is a full framework or advanced microfrontend support, ModernJS may be a better fit.
React Router's SPA Mode still requires a build-time server render of the root route to generate a hydratable index.html (this is how the official React Router Vite plugin works).
When ssr: false:
- The plugin builds both
webandnodeinternally. - It generates
build/client/index.htmlby running the server build once (requestingbasenamewith theX-React-Router-SPA-Mode: yesheader). - It removes
build/serverafter generatingindex.html, so the output is deployable as static assets.
Important: In SPA mode, use clientLoader instead of loader for data loading since there's no server at runtime.
For static sites with multiple pages, you can prerender specific routes at build time:
// react-router.config.ts
import type { Config } from '@react-router/dev/config';
export default {
ssr: false,
prerender: [
'/',
'/about',
'/docs',
'/docs/getting-started',
'/docs/advanced',
'/projects',
],
} satisfies Config;When prerender is specified:
- Each path in the array is rendered at build time
- Static HTML files are generated for each route (e.g.,
/about→build/client/about/index.html) - The server build is removed after prerendering for static deployment
- Non-prerendered routes fall back to client-side routing
You can also use prerender: true to prerender all static routes automatically.
prerender can also be a function:
export default {
ssr: false,
prerender: ({ getStaticPaths }) =>
getStaticPaths().filter(path => path !== '/admin'),
} satisfies Config;Prerendering defaults to one path at a time, matching React Router. Use
concurrency for larger sites; unstable_concurrency is still accepted for
older configs:
export default {
ssr: false,
prerender: {
paths: ['/', '/about'],
concurrency: 4,
},
} satisfies Config;For builds with 256+ routes, detailed file-size reporting is compacted to totals
by default to avoid gzipping and printing thousands of assets. Set
performance.printFileSize to an object to customize that output.
Route transform source maps are generated in development only. If you enable
Rsbuild source maps for faster local debugging, prefer a cheap JS map:
output.sourceMap: { js: 'cheap-module-source-map', css: false }.
Routes can be defined in app/routes.ts using the helper functions from @react-router/dev/routes:
import {
type RouteConfig,
index,
layout,
prefix,
route,
} from '@react-router/dev/routes';
export default [
// Index route for the home page
index('routes/home.tsx'),
// Regular route
route('about', 'routes/about.tsx'),
// Nested routes with a layout
layout('routes/docs/layout.tsx', [
index('routes/docs/index.tsx'),
route('getting-started', 'routes/docs/getting-started.tsx'),
route('advanced', 'routes/docs/advanced.tsx'),
]),
// Routes with dynamic segments
...prefix('projects', [
index('routes/projects/index.tsx'),
layout('routes/projects/layout.tsx', [
route(':projectId', 'routes/projects/project.tsx'),
route(':projectId/edit', 'routes/projects/edit.tsx'),
]),
]),
] satisfies RouteConfig;The plugin provides several helper functions for defining routes:
index()- Creates an index routeroute()- Creates a regular route with a pathlayout()- Creates a layout route with nested childrenprefix()- Adds a URL prefix to a group of routes
Route components support the following exports:
default- The route componentErrorBoundary- Error boundary componentHydrateFallback- Loading component during hydrationLayout- Layout componentclientLoader- Client-side data loadingclientAction- Client-side form actionsclientMiddleware- Client-side middlewarehandle- Route handlelinks- Prefetch linksmeta- Route meta datashouldRevalidate- Revalidation control
loader- Server-side data loadingaction- Server-side form actionsmiddleware- Server-side middlewareheaders- HTTP headers
- Files ending in
.client.*are treated as client-only. Their exports are stubbed toundefinedin the server build, so they are safe to import from route components for browser-only behavior. - Files ending in
.server.*are server-only. If they are imported by code compiled for the web environment, the build will fail with a clear error. Keep.serverimports in server entrypoints or other server-only code.
If you configure output.assetPrefix in Rsbuild, the plugin uses that value
for the React Router browser manifest and server build publicPath so asset
URLs resolve correctly when serving from a CDN or sub-path.
The plugin supports two ways to handle server-side rendering:
-
Default Server Setup: By default, the plugin automatically sets up the necessary middleware for SSR.
-
Custom Server Setup: For more control, you can disable the automatic middleware setup by enabling custom server mode:
// rsbuild.config.ts
import { defineConfig } from '@rsbuild/core';
import { pluginReactRouter } from 'rsbuild-plugin-react-router';
import { pluginReact } from '@rsbuild/plugin-react';
export default defineConfig(() => {
return {
plugins: [
pluginReactRouter({
customServer: true,
}),
pluginReact(),
],
};
});If the server is created programmatically with createDevServer(), pass
onRouteTopologyChange and use it to recreate that server. Rsbuild's
reload-server watcher is owned by the CLI and is not installed by the
programmatic API. The callback is a notification and is not awaited, so it can
safely start a serialized replacement task. Always await the active server's
close() before calling createDevServer() again; the plugin rejects overlapping
or out-of-order replacement instead of closing one server from inside another
server's startup hooks. If startup fails before returning a server, or if
close() rejects, restart the process before retrying unless you can externally
prove and force complete teardown; a fresh Rsbuild instance alone is not
sufficient. Do not launch concurrent createDevServer() calls.
Create one server entry point (server.js) and let it own the React Router
request handler in both development and production. Only the build provider
changes between modes:
import { createRsbuild, loadConfig } from '@rsbuild/core';
import { createRequestHandler } from '@react-router/express';
import {
loadReactRouterServerBuild,
resolveReactRouterServerBuild,
} from 'rsbuild-plugin-react-router';
import express from 'express';
import path from 'path';
import { fileURLToPath } from 'url';
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);
const app = express();
const isDev = process.env.NODE_ENV !== 'production';
async function startServer() {
let devServer;
let build;
if (isDev) {
const config = await loadConfig();
const rsbuild = await createRsbuild({
rsbuildConfig: config.content,
});
const currentDevServer = await rsbuild.createDevServer();
devServer = currentDevServer;
app.use(currentDevServer.middlewares);
build = () => loadReactRouterServerBuild(currentDevServer);
} else {
app.use(
express.static(path.join(__dirname, 'build/client'), {
index: false,
})
);
build = await resolveReactRouterServerBuild(
import('./build/server/static/js/app.js')
);
}
app.use(
createRequestHandler({
build,
mode: isDev ? 'development' : 'production',
getLoadContext() {
return {
// Add custom loader/action context here.
};
},
})
);
const port = Number.parseInt(process.env.PORT || '3000', 10);
const server = app.listen(port, () => {
const mode = isDev ? 'Development' : 'Production';
console.log(`${mode} server is running on http://localhost:${port}`);
devServer?.afterListen();
});
devServer?.connectWebSocket({ server });
}
startServer().catch(console.error);loadReactRouterServerBuild waits for a complete React Router development
generation. During rebuilds it returns the last successfully evaluated server
build, whose embedded manifest is paired with the selected web compilation.
A failed or incomplete candidate does not replace that last-good pair. The
built-in development middleware uses the same path. Calling
devServer.environments.node.loadBundle() directly bypasses this guarantee.
When serverBundles is configured, pass its exact Rsbuild entry name as the
optional second argument (for example, bundle-a/index). The default build
and every configured bundle are
evaluated and published as one generation; one failing bundle keeps the whole
previous generation active.
resolveReactRouterServerBuild accepts an imported production server module,
normalizes ESM and CommonJS namespace shapes, resolves supported asynchronous
build exports, and validates the result before it reaches React Router.
This guarantee covers the eagerly evaluated server entry object and its embedded manifest. It does not snapshot deferred server chunks, make emitted client assets immutable, or delay Rsbuild's WebSocket success notification. Same-path server or client chunks can change before the matching framework generation commits. Closing that publication gap requires a supported Rsbuild graph-settled hook plus immutable or staged outputs.
Then update your package.json scripts:
{
"scripts": {
"dev": "NODE_ENV=development NODE_OPTIONS=\"--experimental-vm-modules\" node server.js",
"build": "rsbuild build",
"start": "NODE_ENV=production node server.js"
}
}To deploy your React Router app to Cloudflare Workers:
- Configure Rsbuild (
rsbuild.config.ts):
import { defineConfig } from '@rsbuild/core';
import { pluginReact } from '@rsbuild/plugin-react';
import { pluginReactRouter } from 'rsbuild-plugin-react-router';
export default defineConfig({
environments: {
node: {
performance: {
chunkSplit: { strategy: 'all-in-one' },
},
tools: {
rspack: {
experiments: { outputModule: true },
externalsType: 'module',
output: {
chunkFormat: 'module',
chunkLoading: 'import',
workerChunkLoading: 'import',
wasmLoading: 'fetch',
library: { type: 'module' },
module: true,
},
resolve: {
conditionNames: [
'workerd',
'worker',
'browser',
'import',
'require',
],
},
},
},
},
},
plugins: [pluginReactRouter({ customServer: true }), pluginReact()],
});- Configure Wrangler (
wrangler.toml):
workers_dev = true
name = "my-react-router-worker"
compatibility_date = "2024-11-18"
main = "./build/server/static/js/app.js"
assets = { directory = "./build/client/" }
[vars]
VALUE_FROM_CLOUDFLARE = "Hello from Cloudflare"
# Optional build configuration
# [build]
# command = "npm run build"
# watch_dir = "app"- Create Worker Entry (
server/index.ts):
import { createRequestHandler } from 'react-router';
declare global {
interface CloudflareEnvironment extends Env {}
interface ImportMeta {
env: {
MODE: string;
};
}
}
declare module 'react-router' {
export interface AppLoadContext {
cloudflare: {
env: CloudflareEnvironment;
ctx: ExecutionContext;
};
}
}
// @ts-expect-error - virtual module provided by React Router at build time
import * as serverBuild from 'virtual/react-router/server-build';
const requestHandler = createRequestHandler(serverBuild, import.meta.env.MODE);
export default {
fetch(request, env, ctx) {
return requestHandler(request, {
cloudflare: { env, ctx },
});
},
} satisfies ExportedHandler<CloudflareEnvironment>;- Update Package Dependencies:
{
"dependencies": {
"@react-router/node": "^7.1.3",
"@react-router/serve": "^7.1.3",
"react-router": "^7.1.3"
},
"devDependencies": {
"@cloudflare/workers-types": "^4.20241112.0",
"@react-router/cloudflare": "^7.1.3",
"@react-router/dev": "^7.1.3",
"wrangler": "^3.106.0"
}
}- Setup Deployment Scripts (
package.json):
{
"scripts": {
"build": "rsbuild build",
"deploy": "npm run build && wrangler deploy",
"dev": "rsbuild dev",
"start": "wrangler dev"
}
}- The
workers_dev = truesetting enables deployment to workers.dev subdomain mainpoints to your Worker's entry point in the build outputassetsdirectory specifies where your static client files are located- Environment variables can be set in the
[vars]section - The
compatibility_dateshould be kept up to date - TypeScript types are provided via
@cloudflare/workers-types - Development can be done locally using
wrangler dev - Deployment is handled through
wrangler deploy
-
Local Development:
# Start local development server npm run dev # or npm start
-
Production Deployment:
# Build and deploy npm run deploy
The plugin automatically:
- Runs type generation during development and build
- Sets up development server with live reload
- Handles route-based code splitting
- Manages client and server builds
React Router "Framework Mode" wraps Data Mode using a Vite plugin. This Rsbuild plugin aims to match the important behaviors without depending on Vite:
- Typegen + Route Module API types (
./+types/*) - Route module splitting (
splitRouteModules) - SPA mode (
ssr: false), SSR mode, and static prerendering (prerender)
Some Vite-specific integrations (for example Vite's environment API + critical CSS endpoint) are not supported 1:1.
The repository includes several examples demonstrating different use cases:
| Example | Description | Port | Command |
|---|---|---|---|
| default-template | Standard SSR setup with React Router | 3000 | pnpm dev |
| spa-mode | Single Page Application (ssr: false) |
3001 | pnpm dev |
| prerender | Static prerendering for multiple routes | 3002 | pnpm dev |
| custom-node-server | Custom Express server with SSR | 3003 | pnpm dev |
| cloudflare | Cloudflare Workers deployment | 3004 | pnpm dev |
| client-only | .client modules with SSR hydration |
3010 | pnpm dev |
| epic-stack | Full-featured Epic Stack example | 3005 | pnpm dev |
| federation/epic-stack | Module Federation host | 3006 | pnpm dev |
| federation/epic-stack-remote | Module Federation remote | 3007 | pnpm dev |
Each example has unique ports configured to allow running multiple examples simultaneously.
# Install dependencies
pnpm install
# Build the plugin
pnpm build
# Run any example
cd examples/default-template
pnpm devEach example includes Playwright e2e tests:
cd examples/default-template
pnpm test:e2eMIT
