Skip to content

从 v7 迁移

如果你是从 rolldown-vite 迁移,这是为 v6 和 v7 集成 Rolldown 的 Rolldown 技术预览版,只有标题中带 NRV 的章节适用。

默认浏览器目标变更 NRV

build.target 的默认浏览器值以及 'baseline-widely-available' 已更新为更新的浏览器版本:

  • Chrome 107 → 111
  • Edge 107 → 111
  • Firefox 104 → 114
  • Safari 16.0 → 16.4

这些浏览器版本与截至 2026-01-01 的 Baseline Widely Available 功能集保持一致。换句话说,它们都是在大约两年前发布的。

Rolldown

Vite 8 使用基于 RolldownOxc 的工具,而不是 esbuildRollup

渐进式迁移

rolldown-vite 包实现了带有 Rolldown 的 Vite 7,没有其他 Vite 8 的变更。这可以作为迁移到 Vite 8 的中间步骤。请参阅 Vite 7 文档中的 Rolldown 集成指南 以从 Vite 7 切换到 rolldown-vite

对于从 rolldown-vite 迁移到 Vite 8 的用户,你可以撤销 package.json 中的依赖变更并更新到 Vite 8:

json
{
  "devDependencies": {
    "vite": "npm:rolldown-vite@7.2.2"
    "vite": "^8.0.0"
  }
}

依赖优化器现在使用 Rolldown

Rolldown 现在用于依赖优化,而不是 esbuild。Vite 仍然支持 optimizeDeps.esbuildOptions 以保持向后兼容,方法是将其自动转换为 optimizeDeps.rolldownOptionsoptimizeDeps.esbuildOptions 现已弃用,将在未来移除,我们鼓励你迁移到 optimizeDeps.rolldownOptions

以下选项会自动转换:

你可以从 configResolved 钩子中获取兼容性层设置的选项:

js
const plugin = {
  name: 'log-config',
  configResolved(config) {
    console.log('options', config.optimizeDeps.rolldownOptions)
  },
},

通过 Oxc 进行 JavaScript 转换

Oxc 现在用于 JavaScript 转换,而不是 esbuild。Vite 仍然支持 esbuild 选项以保持向后兼容,方法是将其自动转换为 oxcesbuild 现已弃用,将在未来移除,我们鼓励你迁移到 oxc

以下选项会自动转换:

esbuild.supported 选项不受 Oxc 支持。如果你需要此选项,请参阅 oxc-project/oxc#15373

你可以从 configResolved 钩子中获取兼容性层设置的选项:

js
const plugin = {
  name: 'log-config',
  configResolved(config) {
    console.log('options', config.oxc)
  },
},

目前,Oxc 转换器不支持降级原生装饰器,因为我们正在等待规范进展,请参阅 (oxc-project/oxc#9170)。

降级原生装饰器的变通方案

你可以暂时使用 BabelSWC 来降级原生装饰器。

使用 Babel:

bash
$ npm install -D @rolldown/plugin-babel @babel/plugin-proposal-decorators
bash
$ yarn add -D @rolldown/plugin-babel @babel/plugin-proposal-decorators
bash
$ pnpm add -D @rolldown/plugin-babel @babel/plugin-proposal-decorators
bash
$ bun add -D @rolldown/plugin-babel @babel/plugin-proposal-decorators
bash
$ deno add -D npm:@rolldown/plugin-babel npm:@babel/plugin-proposal-decorators
vite.config.ts
ts
import { defineConfig } from 'vite'
import babel from '@rolldown/plugin-babel'

function decoratorPreset(options: Record<string, unknown>) {
  return {
    preset: () => ({
      plugins: [['@babel/plugin-proposal-decorators', options]],
    }),
    rolldown: {
      // 仅当文件包含装饰器时才运行此转换。
      filter: {
        code: '@',
      },
    },
  }
}

export default defineConfig({
  plugins: [babel({ presets: [decoratorPreset({ version: '2023-11' })] })],
})

使用 SWC:

bash
$ npm install -D @rollup/plugin-swc @swc/core
bash
$ yarn add -D @rollup/plugin-swc @swc/core
bash
$ pnpm add -D @rollup/plugin-swc @swc/core
bash
$ bun add -D @rollup/plugin-swc @swc/core
bash
$ deno add -D npm:@rollup/plugin-swc npm:@swc/core
js
import { defineConfig, withFilter } from 'vite'

export default defineConfig({
  // ...
  plugins: [
    withFilter(
      swc({
        swc: {
          jsc: {
            parser: { decorators: true, decoratorsBeforeExport: true },
            transform: { decoratorVersion: '2023-11' },
          },
        },
      }),
      // 仅当文件包含装饰器时才运行此转换。
      { transform: { code: '@' } },
    ),
  ],
})

esbuild 回退方案

esbuild 不再被 Vite 直接使用,现在是可选依赖。如果你使用的插件使用了 transformWithEsbuild 函数,你需要将 esbuild 安装为 devDependencytransformWithEsbuild 函数已弃用,将在未来移除。我们建议迁移到新的 transformWithOxc 函数。

通过 Oxc 进行 JavaScript 压缩

Oxc Minifier 现在用于 JavaScript 压缩,而不是 esbuild。你可以使用已弃用的 build.minify: 'esbuild' 选项切换回 esbuild。此配置选项将在未来移除,并且你需要将 esbuild 安装为 devDependency,因为 Vite 不再直接依赖 esbuild。

如果你之前使用 esbuild.minify* 选项来控制压缩行为,现在可以使用 build.rolldownOptions.output.minify。如果你之前使用 esbuild.drop 选项,现在可以使用 build.rolldownOptions.output.minify.compress.drop* 选项

Oxc 不支持属性混淆及其相关选项 (mangleProps, reserveProps, mangleQuoted, mangleCache)。如果你需要这些选项,请参阅 oxc-project/oxc#15375

esbuild 和 Oxc Minifier 对源代码的假设略有不同。如果你怀疑压缩器导致代码损坏,可以在此处比较这些假设:

请报告你在 JavaScript 应用中发现的任何与压缩相关的问题。

通过 Lightning CSS 进行 CSS 压缩

Lightning CSS 现在默认用于 CSS 压缩。你可以使用 build.cssMinify: 'esbuild' 选项切换回 esbuild。注意你需要将 esbuild 安装为 devDependency

Lightning CSS 支持更好的语法降级,你的 CSS 打包大小可能会略微增加。

一致的 CommonJS 互操作

来自 CommonJS (CJS) 模块的 default 导入现在以一致的方式处理。

如果满足以下条件之一,default 导入是被导入者 CJS 模块的 module.exports 值。否则,default 导入是被导入者 CJS 模块的 module.exports.default 值:

  • 导入者是 .mjs.mts
  • 导入者最近的 package.jsontype 字段设置为 module
  • 被导入者 CJS 模块的 module.exports.__esModule 值未设置为 true。
之前的行为

在开发环境中,如果满足以下条件之一,default 导入是被导入者 CJS 模块的 module.exports 值。否则,default 导入是被导入者 CJS 模块的 module.exports.default 值:

  • 导入者包含在依赖优化中 且为 .mjs.mts
  • 导入者包含在依赖优化中 且导入者最近的 package.jsontype 字段设置为 module
  • 被导入者 CJS 模块的 module.exports.__esModule 值未设置为 true。

在生产构建中,条件是:

  • 被导入者 CJS 模块的 module.exports.__esModule 值未设置为 true。
  • module.exportsdefault 属性不存在

(假设 build.commonjsOptions.defaultIsModuleExports 未从默认值 'auto' 更改)

有关此问题的更多详细信息,请参阅 Rolldown 的文档:来自 CJS 模块的模糊 default 导入 - 打包 CJS | Rolldown

此变更可能会破坏一些现有的导入 CJS 模块的代码。你可以使用已弃用的 legacy.inconsistentCjsInterop: true 选项暂时恢复之前的行为。如果你发现受此变更影响的包,请向包作者报告或向他们发送拉取请求。确保链接到上面的 Rolldown 文档,以便作者了解上下文。

移除了使用格式嗅探的模块解析

package.json 中同时存在 browsermodule 字段时,Vite 过去会根据文件内容解析字段,并且会为浏览器选择 ESM 文件。引入这是因为有些包使用 module 字段指向 Node.js 的 ESM 文件,而有些其他包使用 browser 字段指向浏览器的 UMD 文件。鉴于现代 exports 字段解决了这个问题并且现在被许多包采用,Vite 不再使用这种启发式方法,而是始终尊重 resolve.mainFields 选项的顺序。如果你依赖此行为,可以使用 resolve.alias 选项将字段映射到所需的文件,或使用包管理器应用补丁(例如 patch-packagepnpm patch)。

外部化模块的 Require 调用

外部化模块的 require 调用现在保留为 require 调用,而不转换为 import 语句。这是为了保留 require 调用的语义。如果你想将它们转换为 import 语句,可以使用 Rolldown 内置的 esmExternalRequirePlugin,它从 vite 重新导出。

js
import { defineConfig, esmExternalRequirePlugin } from 'vite'

export default defineConfig({
  // ...
  plugins: [
    esmExternalRequirePlugin({
      external: ['react', 'vue', /^node:/],
    }),
  ],
})

有关更多详细信息,请参阅 Rolldown 的文档:require 外部模块 - 打包 CJS | Rolldown

UMD / IIFE 中的 import.meta.url

import.meta.url 不再在 UMD / IIFE 输出格式中进行 polyfill。默认情况下它将替换为 undefined。如果你更喜欢之前的行为,可以使用 define 选项配合 build.rolldownOptions.output.intro 选项。有关更多详细信息,请参阅 Rolldown 的文档:众所周知的 import.meta 属性 - 非 ESM 输出格式 | Rolldown

移除了 build.rollupOptions.watch.chokidar 选项

build.rollupOptions.watch.chokidar 选项已被移除。请迁移到 build.rolldownOptions.watch.watcher 选项。

移除了对象形式的 build.rollupOptions.output.manualChunks 并弃用函数形式

对象形式的 output.manualChunks 选项不再支持。函数形式的 output.manualChunks 已弃用。Rolldown 拥有更灵活的 codeSplitting 选项。有关 codeSplitting 的更多详细信息,请参阅 Rolldown 的文档:手动代码分割 - Rolldown

build() 抛出 BundleError

此变更仅影响 JS API 用户。

build() 现在抛出 BundleError 而不是插件中抛出的原始错误。BundleError 的类型为 Error & { errors?: RolldownError[] },它将单个错误包装在 errors 数组中。如果你需要单个错误,需要访问 .errors

js
try {
  await build()
} catch (e) {
  if (e.errors) {
    for (const error of e.errors) {
      console.log(error.code) // 错误代码
    }
  }
}

模块类型支持和自动检测

此变更仅影响插件作者。

Rolldown 实验性支持 模块类型,类似于 esbuild 的 loader 选项。因此,Rolldown 会根据解析后的 id 的扩展名自动设置模块类型。如果你在 loadtransform 钩子中将其他模块类型的内容转换为 JavaScript,可能需要在返回值中添加 moduleType: 'js'

js
const plugin = {
  name: 'txt-loader',
  load(id) {
    if (id.endsWith('.txt')) {
      const content = fs.readFile(id, 'utf-8')
      return {
        code: `export default ${JSON.stringify(content)}`,
        moduleType: 'js', 
      }
    }
  },
}

其他相关的弃用项

以下选项已弃用,将在未来移除:

  • build.rollupOptions:重命名为 build.rolldownOptions
  • worker.rollupOptions:重命名为 worker.rolldownOptions
  • build.commonjsOptions:现在是无操作
  • build.dynamicImportVarsOptions.warnOnError:现在是无操作
  • resolve.alias[].customResolver:改用带有 resolveId 钩子和 enforce: 'pre' 的自定义插件

已移除的废弃功能 NRV

  • 不再支持将 URL 传递给 import.meta.hot.accept。请改为传递一个 id。 (#21382)

高级

这些破坏性变更预计只会影响少数用例:

  • Extglobs 尚不支持 (rolldown-vite#365)
  • TypeScript legacy namespace 仅部分支持。请参阅 Oxc Transformer 的相关文档 以获取更多详情。
  • define 不为对象共享引用:当你将一个对象作为值传递给 define 时,每个变量都将拥有该对象的一个独立副本。请参阅 Oxc Transformer 的相关文档 以获取更多详情。
  • bundle 对象变更(bundle 是在 generateBundle / writeBundle 钩子中传递的对象,由 build 函数返回):
    • 不支持赋值给 bundle[foo]。Rollup 也不鼓励这样做。请改用 this.emitFile()
    • 引用在各钩子之间不共享 (rolldown-vite#410)
    • structuredClone(bundle) 会报错 DataCloneError: #<Object> could not be cloned。不再支持此操作。请使用 structuredClone({ ...bundle }) 进行克隆。 (rolldown-vite#128)
  • Rollup 中的所有并行钩子现在作为顺序钩子工作。请参阅 Rolldown 的文档 以获取更多详情。
  • "use strict"; 有时不会被注入。请参阅 Rolldown 的文档 以获取更多详情。
  • 不支持使用 plugin-legacy 转换到 ES5 及以下版本 (rolldown-vite#452)
  • 将同一浏览器的多个版本传递给 build.target 选项现在会报错:esbuild 会选择该浏览器的最新版本,这可能不是你的本意。
  • Rolldown 缺少支持:以下功能不受 Rolldown 支持,Vite 也不再支持。
  • parseAst / parseAstAsync 函数现已弃用,改为使用功能更丰富的 parseSync / parse 函数。
  • renderChunk 钩子之前会移除注释,而不是在 renderChunk 钩子之后移除
  • 除了 此处 列出的注释之外,其余注释都会被移动,而 Rollup 只有在相邻代码被移除时才会移除注释

从 v6 迁移

请先查看 Vite v7 文档中的 从 v6 迁移指南,了解将应用移植到 Vite 7 所需的更改,然后再继续本页面上的更改。