Skip to content

Latest commit

 

History

History
194 lines (159 loc) · 6.39 KB

README.md

File metadata and controls

194 lines (159 loc) · 6.39 KB

Webpack RSC Integration

⚠️ Experimental

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.

Getting Started

To use this library in your React Server Components project, follow these steps:

  1. Install the library as a dev dependency:
npm install --save-dev @mfng/webpack-rsc
  1. Update your webpack.config.js to include the loaders and plugins provided by this library. See the example configuration below for reference.

Example Webpack Configuration

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.

Webpack Loaders and Plugins

This library provides the following Webpack loaders and plugins:

createWebpackRscServerLoader

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.

createWebpackRscSsrLoader

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.

WebpackRscServerPlugin

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.

createWebpackRscClientLoader

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.

WebpackRscClientPlugin

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().