插件系统
Rsbuild 提供了一套轻量强大的插件系统,用以实现自身的大多数功能,并允许用户进行扩展。
开发者编写的插件能够修改 Rsbuild 的默认行为,并添加各类额外功能,包括但不限于:
- 修改 Rspack 配置
- 处理新的文件类型
- 修改或编译文件
- 部署产物
开发插件
插件提供类似 (options?: PluginOptions) => RsbuildPlugin 的函数作为入口。
插件示例
pluginFoo.ts
import type { RsbuildPlugin } from '@rsbuild/core';
export type PluginFooOptions = {
  message?: string;
};
export const pluginFoo = (options: PluginFooOptions = {}): RsbuildPlugin => ({
  name: 'plugin-foo',
  setup(api) {
    api.onAfterStartDevServer(() => {
      const msg = options.message || 'hello!';
      console.log(msg);
    });
  },
});
注册插件:
rsbuild.config.ts
import { pluginFoo } from './pluginFoo';
export default {
  plugins: [pluginFoo({ message: 'world!' })],
};
插件结构
函数形式的插件可以 接受选项对象 并 返回插件实例,并通过闭包机制管理内部状态。
其中各部分的作用分别为:
- name属性用于标注插件名称。
- setup作为插件逻辑的主入口。
- api对象包含了各类钩子和工具函数。
命名规范
插件的命名规范如下:
- 插件的函数命名为 pluginXXX,并通过具名导出。
- 插件的 name采用scope:foo-bar或plugin-foo-bar格式,添加scope:可以避免和其他插件产生命名冲突。
下面是一个例子:
pluginFooBar.ts
import type { RsbuildPlugin } from '@rsbuild/core';
export const pluginFooBar = (): RsbuildPlugin => ({
  name: 'xxx:foo-bar',
  setup() {},
});
TIP
Rsbuild 官方插件的 name 统一使用 rsbuild: 作为前缀,比如 rsbuild:react 对应 @rsbuild/plugin-react。
模板仓库
rsbuild-plugin-template 是一个最小的 Rsbuild 插件模板仓库,你可以基于该仓库来开发你的 Rsbuild 插件。
生命周期钩子
Rsbuild 在内部按照约定的生命周期进行任务调度,插件可以通过注册钩子来介入工作流程的任意阶段,并实现自己的功能。
Rsbuild 生命周期钩子的完整列表参考 API 文档。
Rsbuild 不会接管底层 Rspack 的生命周期,相关生命周期钩子的使用方式见对应文档:Rspack Plugin API。
使用配置项
自行编写的插件通常使用初始化时传入函数的参数作为配置项即可,开发者可以随意定义和使用函数的入参。
但某些情况下插件可能需要读取 / 修改 Rsbuild 公用的配置项,这时就需要了解 Rsbuild 内部对配置项的生产和消费流程:
- 读取、解析配置并合并默认值
- 插件通过 api.modifyRsbuildConfig(...)回调修改配置项
- 归一化配置项并提供给插件后续消费,此后无法再修改配置项
整套流程可以通过这个简单的插件体现:
export const pluginUploadDist = (): RsbuildPlugin => ({
  name: 'plugin-upload-dist',
  setup(api) {
    api.modifyRsbuildConfig((config) => {
      // 尝试关闭代码压缩,需要自己填充默认值
      config.output ||= {};
      config.output.disableMinimize = true;
      // 轮到其它插件修改配置...
    });
    api.onBeforeBuild(() => {
      // 读取最终的配置
      const config = api.getNormalizedConfig();
      if (!config.output.disableMinimize) {
        // 其它插件又启用了压缩则报错
        throw new Error(
          'You must disable minimize to upload readable dist files.',
        );
      }
    });
    api.onAfterBuild(() => {
      const config = api.getNormalizedConfig();
      const distRoot = config.output.distPath.root;
      // 上传 `distRoot` 目录下所有产物文件...
    });
  },
});
插件中有三种方式使用配置项对象:
- api.modifyRsbuildConfig(config => {})在回调中修改配置
- api.getRsbuildConfig()获取配置项
- api.getNormalizedConfig()获取归一化后的配置项
归一化的配置项会再次合并默认值,并移除大部分可选类型,对于 PluginUploadDist 的例子,其部分类型定义为:
api.modifyRsbuildConfig((config: RsbuildConfig) => {});
api.getRsbuildConfig() as RsbuildConfig;
type RsbuildConfig = {
  output?: {
    disableMinimize?: boolean;
    distPath?: { root?: string };
  };
};
api.getNormalizedConfig() as NormalizedConfig;
type NormalizedConfig = {
  output: {
    disableMinimize: boolean;
    distPath: { root: string };
  };
};
getNormalizedConfig() 的返回值类型与 RsbuildConfig 的略有不同、相比文档其它地方描述的类型进行了收窄,
在使用时无需自行判空、填充默认值。
因此使用配置项的最佳方式应该是:
- 通过 api.modifyRsbuildConfig(config => {})来修改配置
- 在其后的生命周期中读取 api.getNormalizedConfig()作为插件实际使用的配置
修改 Rspack 配置
Rsbuild 插件可以通过多种方式修改 Rspack 的配置项。
- api.modifyRspackConfig(config => {})修改最终的 Rspack 配置。
- api.modifyBundlerChain(chain => {})修改- bundler-chain,用法与 webpack-chain 相同。
- api.onAfterCreateCompiler(compiler => {})直接操作 Rspack 的 compiler 实例。
参考范例
修改 Loader
Loader 可以读取和处理不同类型的文件模块,具体参考 concepts 和 loaders。
import type { RsbuildPlugin } from '@rsbuild/core';
export const pluginTypeScriptExt = (): RsbuildPlugin => ({
  name: 'plugin-typescript-ext',
  setup(api) {
    api.modifyBundlerChain(async (chain) => {
      // 设置 ts-loader 将更多的文件识别为 typescript 模块
      chain.module.rule(CHAIN_ID.RULE.TS).test(/\.(ts|mts|cts|tsx|tss|tsm)$/);
    });
  },
});
添加模块入口
import type { RsbuildPlugin } from '@rsbuild/core';
export const pluginAdminPanel = (): RsbuildPlugin => ({
  name: 'plugin-admin-panel',
  setup(api) {
    api.modifyBundlerChain(async (chain) => {
      config.entry('admin-panel').add('src/admin/panel.js');
    });
  },
});
注册 Rspack 插件
你可以在 Rsbuild 插件中注册 Rspack 插件,比如注册 eslint-webpack-plugin:
import type { RsbuildPlugin } from '@rsbuild/core';
import ESLintPlugin from 'eslint-webpack-plugin';
export const pluginEslint = (options?: Options): RsbuildPlugin => ({
  name: 'plugin-eslint',
  setup(api) {
    // Use bundler-chain to register a bundler plugin.
    api.modifyBundlerChain(async (chain) => {
      chain.plugin('eslint-webpack-plugin').use(ESLintPlugin, [
        {
          // plugins options
        },
      ]);
    });
  },
});