《vite技术揭秘、还原与实战》第6节--支持svite.config.ts配置文件

前言

上一节我们提供的svite.config.ts配置文件没有相应的TypeScript类型定义,这对开发者是不友好的

本节我们通过提供一个defineConfig函数来优化这个问题

源码获取

传送门

更新进度

公众号:更新至第16

博客:更新至第6

尝试

这似乎很简单,在packages\vite\src\node\config.ts中导出defineConfig函数,该函数用作向用户提供类型支持

export function defineConfig(config: UserConfig): UserConfig {
  return config;
}

为了达到这个目的,还需要在packages\vite\src\node\index.ts中导出UserConfig

export * from "./config";

接着在playground\config\svite.config.ts中导入并使用,理论上就大功告成了

import { defineConfig } from "svite";
export default defineConfig({
  server: {},
  root: "",
});

此时启用该用例,会发现如下报错

Dynamic require of "fs" is not supported

原因分析

该报错说明我们正在esm模块的执行过程中使用require语法,这似乎有点不可思议,因为我们在packages\vite\rollup.config.ts文件中定义的output.format确实是esm格式,打包结果如下所示

import { resolve } from 'node:path';
import { existsSync } from 'node:fs';
import { build } from 'esbuild';

const DEFAULT_CONFIG_FILES = ["svite.config.ts"];

async function buildBoundle(fileName) {
    ...
}
async function loadConfigFromBoundled(code, resolvedPath) {
    ...
}
function defineConfig(config) {
    return config;
}
async function parseConfigFile(conf) {
    ...
}
async function resolveConfig(userConf) {
    ...
}

export { defineConfig, parseConfigFile, resolveConfig };
//# sourceMappingURL=index.js.map

如上,我们的三个import也都是符合esm语法的,其中fspath??槭?code>node内置的,node 本身又支持esm,理论上来说不可能是它们导致的。那问题貌似出现在esbuild

我们找到node_modules下的esbuild文件夹,并根据package.json定位到入口为packages\vite\node_modules\esbuild\lib\main.js的文件,在其源码中发现了这两句代码

...
var fs = require("fs");
var os = require("os");
...

还记得我们上一节是如何处理svite.config.ts文件的吗?我们为了能在文件中引入外部???,使用esbuild进行了打包,而build行为会分析??橹械?code>import并将其打包进一个boundle,这就意味着var fs = require("fs");这行代码将会被打包到我们最终的boundleCode

import * as any from "some-pkg";
const fs = require("fs");

而对于boundleCode我们使用的是new Function的形式,至此,真相大白

const dynamicImport = new Function("file", "return import(file)");

如何解决

既然原因是对esbuild进行了分析打包,那是否可以跳过对esbuild包的build行为呢?

首先,我们找到在packages\vite\package.json中的dependencies,如下

"dependencies": {
    "esbuild": "^0.18.8",
    "magic-string": "^0.30.0",
    "rollup": "^3.21.0"
}

由于esbuilddependencies的一员,则意味着,我们一定能在用户项目的node_modules中找到该依赖包,这意味着使用import { defineConfig } from 'svite 时对入口的加载过程不会抛出错误

同时由于svite已经被打包过,因此如果我们能直接从node_modules引入则问题能够被解决,并且为了更安全的找到对应的包,我们使用绝对路径更为妥当

import { defineConfig } from 'XXX/node_modules/svite/dist/node/index.js'

那么问题就在于,svite如何被转换为XXX/node_modules/svite/dist/node/index.js呢?

源码分析

我们在原因分析中已经找到了问题出在打包处,并且也提出了解决方案:

1-将裸依赖从打包中排除

2-将裸依赖导出地址替换为绝对路径

故将代码定位到bundleConfigFile函数的名称为externalize-depsesbuild plugin中,源码简化如下

// packages/vite/src/node/config.ts

async function bundleConfigFile(
  fileName: string,
  isESM: boolean,
): Promise<{ code: string; dependencies: string[] }> {
  ...
  const result = await build({
    ...
    plugins: [
      {
        name: 'externalize-deps',
        setup(build) {
          ...
          // 排除裸依赖
          build.onResolve(
            { filter: /^[^.].*/ },
            async ({ path: id, importer, kind }) => {
              ...
              return {
                path: idFsPath,
                external: true,
              }
            },
          )
        },
      },
      ...
    ],
  })
  const { text } = result.outputFiles[0]
  return {
    code: text,
    ...
  }
}

如上,vite通过onResolve钩子来完成对??槁肪兜奶婊缓湍?榈呐懦ぷ?,这分别对应着返回对象的pathexternal属性,当为external设置为true后,esbuild将会自动将模块从打包结果中排除,因此,我们的重点是分析??槁肪短婊皇窃趺词迪值?/strong>,也即pathvalue值是如何生成的

// 获取裸??榈谋镜匚募肪?idFsPath = resolveByViteResolver(id, importer, !isImport)
// 将本地文件路径转换为可用于网络访问的URL
idFsPath = pathToFileURL(idFsPath).href

进入resolveByViteResolver函数,并按顺序找到tryNodeResolve函数,如下是笔者简化后的代码

// packages/vite/src/node/plugins/resolve.ts

export function tryNodeResolve(...): PartialResolvedId | undefined {
  const { preserveSymlinks, packageCache,... } = options
  ...
  // 获取目录,即vite.config.ts所在的目录,一般为用户项目根目录
  const basedir = path.dirname(importer)
  // 获取裸模块包的信息
  const pkg = resolvePackageData(pkgId, basedir, preserveSymlinks, packageCache)
  ...
  // ??榈既牒?  const resolveId = deepMatch ? resolveDeepImport : resolvePackageEntry
  ...
  // 从模块信息中分析入口文件地址
  const resolved = resolveId(unresolvedId, pkg, targetWeb, options)
  ...
  return resolved
}

进入resolvePackageData函数,可以看到,vitenpm包的package.json开始查找,一般来说,在第一次while循环过程中就能找到,若实在找不到就依次到上一级

// packages/vite/src/node/packages.ts
export function resolvePackageData(...): PackageData | null {
  ...
  const originalBasedir = basedir
  while (basedir) {
    ...
    // 找到导入npm包的package.json文件
    const pkg = path.join(basedir, 'node_modules', pkgName, 'package.json')
    try {
      if (fs.existsSync(pkg)) {
        // 读取package.json
        const pkgPath = preserveSymlinks ? pkg : safeRealpathSync(pkg)
        const pkgData = loadPackageData(pkgPath)
        ...
        return pkgData
      }
    } catch {}
    // 进入上一级目录查找
    const nextBasedir = path.dirname(basedir)
    if (nextBasedir === basedir) break
    basedir = nextBasedir
  }

  return null
}

pkgPath这一行,根据preserveSymlinks取值来决定是否使用符号链接,符号链接其实类似于一种别名,通过读取A可直接获取源文件B,而非符号链接则必须找到实际的源文件B的路径才能进行内容的读取

safeRealpathSync的实现又根据操作系统的不同有差异

// packages/vite/src/node/utils.ts
export let safeRealpathSync = isWindows
  ? fs.realpathSync
  : fs.realpathSync.native

总而言之,vite获取到了一个指向npm包的绝对路径,下一步使用loadPackageData来进行文件内容的读取,如下,其本质上就是fs的文件读取操作

export function loadPackageData(pkgPath: string): PackageData {
  const data = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'))
  ...
  return data
}

现在我们返回tryNodeResolve函数

对于resolveId的取值,则分别对应如下两种导入方式

import xxx from 'x'
import yyy from 'x/y'

笔者不打算在此处对这两个函数进行展开,因为在svite的实现中笔者并不打算完全采用该方式。感兴趣的读者可以自己按提示找到对应的文件定位到代码处进行查看。实际上其实现思路也很简单:找到package.json中的文件导出字段如exports,然后进行匹配拼接即可

?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容