Skip to content

rstackjs/rsbuild-plugin-react-router

Repository files navigation

rsbuild-plugin-react-router

Rsbuild Logo

A Rsbuild plugin that provides seamless integration with React Router, supporting both client-side routing and server-side rendering (SSR).

Features

  • 🚀 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)

Installation

npm install rsbuild-plugin-react-router
# or
yarn add rsbuild-plugin-react-router
# or
pnpm add rsbuild-plugin-react-router

Local development

For 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 --activate

Usage

Add 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(),
  ],
});

Configuration

React Router application settings live in react-router.config.*. The Rsbuild plugin only needs options for Rsbuild-specific behavior.

Plugin Options

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.

React Router Config

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.

Config File Resolution

The plugin will look for react-router.config with any supported JS/TS extension, in this order:

  • react-router.config.tsx
  • react-router.config.ts
  • react-router.config.mts
  • react-router.config.jsx
  • react-router.config.js
  • react-router.config.mjs

If none are found, it falls back to defaults.

Framework Mode

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.

FAQ

rsbuild-plugin-react-router vs ModernJS

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.

SPA Mode (ssr: false)

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 web and node internally.
  • It generates build/client/index.html by running the server build once (requesting basename with the X-React-Router-SPA-Mode: yes header).
  • It removes build/server after generating index.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.

Static Prerendering

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., /aboutbuild/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 }.

Route Configuration

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 route
  • route() - Creates a regular route with a path
  • layout() - Creates a layout route with nested children
  • prefix() - Adds a URL prefix to a group of routes

Route Components

Route components support the following exports:

Client-side Exports

  • default - The route component
  • ErrorBoundary - Error boundary component
  • HydrateFallback - Loading component during hydration
  • Layout - Layout component
  • clientLoader - Client-side data loading
  • clientAction - Client-side form actions
  • clientMiddleware - Client-side middleware
  • handle - Route handle
  • links - Prefetch links
  • meta - Route meta data
  • shouldRevalidate - Revalidation control

Server-side Exports

  • loader - Server-side data loading
  • action - Server-side form actions
  • middleware - Server-side middleware
  • headers - HTTP headers

Client/Server-only Modules

  • Files ending in .client.* are treated as client-only. Their exports are stubbed to undefined in 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 .server imports in server entrypoints or other server-only code.

Asset Prefix

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.

Custom Server Setup

The plugin supports two ways to handle server-side rendering:

  1. Default Server Setup: By default, the plugin automatically sets up the necessary middleware for SSR.

  2. 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"
  }
}

Cloudflare Workers Deployment

To deploy your React Router app to Cloudflare Workers:

  1. 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()],
});
  1. 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"
  1. 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>;
  1. 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"
  }
}
  1. Setup Deployment Scripts (package.json):
{
  "scripts": {
    "build": "rsbuild build",
    "deploy": "npm run build && wrangler deploy",
    "dev": "rsbuild dev",
    "start": "wrangler dev"
  }
}

Key Configuration Notes:

  • The workers_dev = true setting enables deployment to workers.dev subdomain
  • main points to your Worker's entry point in the build output
  • assets directory specifies where your static client files are located
  • Environment variables can be set in the [vars] section
  • The compatibility_date should 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

Development Workflow:

  1. Local Development:

    # Start local development server
    npm run dev
    # or
    npm start
  2. Production Deployment:

    # Build and deploy
    npm run deploy

Development

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

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.

Examples

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.

Running Examples

# Install dependencies
pnpm install

# Build the plugin
pnpm build

# Run any example
cd examples/default-template
pnpm dev

Running E2E Tests

Each example includes Playwright e2e tests:

cd examples/default-template
pnpm test:e2e

License

MIT

About

A Rsbuild plugin that provides seamless integration with React Router

Topics

Resources

License

Contributing

Stars

Watchers

Forks

Contributors