This library provides a set of Webpack loaders and plugins required for building a React Server Components (RSC) application bundle for the browser, as well as for the server, with server-side rendering (SSR).
The server bundle can be deployed to any serverless, edge, or Node.js-based environment.
@mfng/webpack-rsc
can be used standalone as an RSC bundling solution or in
conjunction with
@mfng/core
.
Disclaimer: There are many moving parts involved in creating an RSC app that also handles SSR, without using a framework like Next.js. This library only provides the necessary parts for bundling the app. For a fully working integration, see https://github.com/unstubbable/mfng.
To use this library in your React Server Components project, follow these steps:
- Install the library as a dev dependency:
npm install --save-dev @mfng/webpack-rsc
- Update your
webpack.config.js
to include the loaders and plugins provided by this library. See the example configuration below for reference.
The following example demonstrates how to use the loaders and plugins in a Webpack configuration:
import {
WebpackRscClientPlugin,
WebpackRscServerPlugin,
createWebpackRscClientLoader,
createWebpackRscServerLoader,
createWebpackRscSsrLoader,
webpackRscLayerName,
} from '@mfng/webpack-rsc';
const clientReferencesMap = new Map();
const serverReferencesMap = new Map();
const rscServerLoader = createWebpackRscServerLoader({
clientReferencesMap,
serverReferencesMap,
});
const rscSsrLoader = createWebpackRscSsrLoader({serverReferencesMap});
const rscClientLoader = createWebpackRscClientLoader({serverReferencesMap});
const serverConfig = {
name: 'server',
// ...
module: {
rules: [
{
// Match the entry modules that should end up in the RSC layer:
resource: [/\/server\/rsc\//, /\/app\.tsx$/],
layer: webpackRscLayerName,
},
{
// Match the modules that should end up in a shared layer (RSC & SSR):
resource: /\/server\/shared\//,
layer: 'shared',
},
{
issuerLayer: webpackRscLayerName,
resolve: {conditionNames: ['react-server', '...']},
},
{
oneOf: [
{
issuerLayer: webpackRscLayerName,
test: /\.tsx?$/,
use: [rscServerLoader, 'swc-loader'],
},
{
test: /\.tsx?$/,
use: [rscSsrLoader, 'swc-loader'],
},
{
issuerLayer: webpackRscLayerName,
test: /\.m?js$/,
use: rscServerLoader,
},
{
test: /\.m?js$/,
use: rscSsrLoader,
},
],
},
],
},
plugins: [
new WebpackRscServerPlugin({clientReferencesMap, serverReferencesMap}),
],
experiments: {layers: true},
// ...
};
const clientConfig = {
name: 'client',
dependencies: ['server'],
// ...
module: {
rules: [
{
test: /\.tsx?$/,
use: [rscClientLoader, 'swc-loader'],
},
{
test: /\.m?js$/,
use: rscClientLoader,
},
],
},
plugins: [new WebpackRscClientPlugin({clientReferencesMap})],
// ...
};
export default [serverConfig, clientConfig];
Note: It's important to specify the names and dependencies of the configs as shown above, so that the plugins work in the correct order, even in watch mode.
This library provides the following Webpack loaders and plugins:
A function to create the RSC server loader use
item for the server entry
webpack config. It should be used if the issuerLayer
is webpackRscLayerName
.
This loader is responsible for replacing client components in a use client
module with client references (objects that contain meta data about the client
components), and removing all other parts of the client module. It also
populates the given clientReferencesMap
.
In addition, the loader handles server references for React server actions by
adding meta data to all exported functions of a use server
module, and also
populates the given serverReferencesMap
.
A function to create the RSC SSR loader use
item for the server entry webpack
config. It should be used if the issuerLayer
is not webpackRscLayerName
.
This loader is responsible for replacing server actions in a use server
module
that are imported from client modules with stubs. The stubs are functions that
throw an error, since React does not allow calling server actions during SSR (to
avoid waterfalls). All other parts of the server module are removed. It also
populates the given serverReferencesMap
.
The server plugin resolves the client references that the server loader has created, and adds them as additional includes to the bundle, so that they are available for server-side rendering (SSR).
The plugin also generates the server manifest that is needed for validating the
server references of server actions (also known as mutations) that are sent back
from the client. It also adds module IDs to the given serverReferencesMap
.
A function to create the RSC client loader use
item for the client entry
webpack config. This loader is responsible for replacing server actions in a
use server
module with server references (based on the given
serverReferencesMap
), and removing all other parts of the server module, so
that the server module can be imported from a client module.
Note: Importing server actions from a client module requires that
callServer
can be imported from a module. Per default
@mfng/core/client/browser
is used as import source, but this can be customized
with the callServerImportSource
option.
The client plugin resolves the client references that were saved by the server
loader in clientReferencesMap
into separate client chunks that can be loaded
by the browser. This plugin also generates the React client manifest file that
is needed for creating the RSC stream with
ReactServerDOMServer.renderToReadableStream()
, as well as the React SSR
manifest that is needed for creating the HTML stream (SSR) with
ReactServerDOMClient.createFromReadableStream()
.