精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

又一個基于 Esbuild 的神器!

開發(fā) 開發(fā)工具
本文的主角是由 antfu 大佬開源的 esno 項目,接下來我將帶大家一起來揭開這個項目背后的秘密。

Node.js 并不支持直接執(zhí)行 TS 文件,如果要執(zhí)行 TS 文件的話,我們就可以借助 ts-node 這個庫。相信有些小伙伴在工作中也用過這個庫,關(guān)于 ts-node 這個庫的相關(guān)內(nèi)容我就不展開介紹了,因為本文的主角是由 antfu 大佬開源的 esno 項目,接下來我將帶大家一起來揭開這個項目背后的秘密。

閱讀完本文后,你將了解 esno 項目是如何執(zhí)行 TS 文件。此外,你還會了解如何劫持 Node.js 的 require 函數(shù)、如何為 ES Module 的 import 語句添加鉤子及如何自定義 https 加載器,以支持 import React from "https://esm.sh/react" 導(dǎo)入方式。

esno 是什么

esno 是基于 esbuild 的 TS/ESNext node 運(yùn)行時。該庫會針對不同的模塊化標(biāo)準(zhǔn),采用不同的方案:

  • esno - Node in CJS mode - by esbuild-register
  • esmo - Node in ESM mode - by esbuild-node-loader

使用 esno 的方式很簡單,你可以以全局或局部的方式來安裝它:

全局安裝

$ npm i -g esno

在安裝成功后,你就可以通過以下方式來直接執(zhí)行 TS 文件:

$ esno index.ts
$ esmo index.ts

局部安裝

$ npm i esno

而對于局部安裝的方式來說,一般情況下,我們會以 npm scripts 的方式來使用它:

{
"scripts": {
"start": "esno index.ts"
},
"dependencies": {
"esno": "0.14.0"
}
}

esno 是如何工作的

在開始分析 esno 的工作原理之前,我們先來熟悉一下該項目:

├── LICENSE
├── README.md
├── esmo.mjs
├── esno.js
├── package.json
├── pnpm-lock.yaml
├── publish.ts
└── tsconfig.json

觀察以上的項目結(jié)構(gòu)可知,該項目并不會復(fù)雜。在項目根目錄下的 package.json 文件中,我們看到了前面介紹的 esno 和 esmo 命令。

{
"bin": {
"esno": "esno.js",
"esmo": "esmo.mjs"
},
}

此外,在 package.json 的 scripts 字段中,我們發(fā)現(xiàn)了 release 命令。顧名思義,該命令用來發(fā)布版本。

{
"scripts": {
"release": "npx bumpp --tag --commit --push && node esmo.mjs publish.ts"
},
}

需要注意的是,在 publish.ts 文件中,使用到了 2021 年度 Github 上最耀眼的項目 zx,利用該項目我們可以輕松地編寫命令行腳本。寫作本文時,它的 Star 數(shù)已經(jīng)高達(dá) 27.5K,強(qiáng)烈推薦感興趣的小伙伴關(guān)注一下該項目。

簡單介紹了 esno 項目之后,接下來我們來分析 esno.js 文件:

#!/usr/bin/env node

const spawn = require('cross-spawn')
const spawnSync = spawn.sync

const register = require.resolve('esbuild-register')

const argv = process.argv.slice(2)

process.exit(spawnSync('node', ['-r', register, ...argv],
{ stdio: 'inherit' }).status)

由以上代碼可知,當(dāng)執(zhí)行 esno index.ts 命令后,會通過 spawnSync 來啟動 Node.js 程序執(zhí)行腳本。需要注意的是,在執(zhí)行時使用了 -r 選項,該選項的作用是預(yù)加載模塊:

-r, --require = ... module to preload (option can be repeated)

這里預(yù)加載的模塊是 esbuild-register,該模塊就是 esno 命令執(zhí)行 TS 文件的幕后英雄。

esbuild-register 是什么

esbuild-register 是一個基于 esbuild 來轉(zhuǎn)換 JSX、TS 和 esnext 特性的工具。你可以通過以下多種方式來安裝它:

$ npm i esbuild esbuild-register -D
# Or Yarn
$ yarn add esbuild esbuild-register --dev
# Or pnpm
$ pnpm add esbuild esbuild-register -D

在成功安裝該模塊之后,就可以在命令行中,直接通過 node 應(yīng)用程序來執(zhí)行 ts 文件:

$ node -r esbuild-register file.ts

-r, --require = ... module to preload (option can be repeated)

-r 用于指定預(yù)加載的文件,即在執(zhí)行 file.ts 文件前,提前加載 esbuild-register 模塊

它將會使用 tsconfig.json 中的 jsxFactory,jsxFragmentFactory 和 target 配置項來執(zhí)行轉(zhuǎn)換操作。

esbuild-register 不僅可以在命令行中使用,而且還可以通過 API 的方式進(jìn)行使用:

const { register } = require('esbuild-register/dist/node')

const { unregister } = register({
// ...options
})

// Unregister the require hook if you don't need it anymore
unregister()

了解完 esbuild-register 的基本使用之后,接下來我們來分析它內(nèi)部是如何工作的。

esbuild-register 是如何工作的

esbuild-register 內(nèi)部利用了 pirates 這個庫來劫持 Node.js 的 require 函數(shù),從而讓你可以在命令行中,直接執(zhí)行 ts 文件。下面我們來看一下 esbuild-register 模塊中定義的 register 函數(shù):

// esbuild-register/src/node.ts
import { transformSync, TransformOptions } from 'esbuild'
import { addHook } from 'pirates'

export function register(esbuildOptions: RegisterOptions = {}) {
const {
extensions = DEFAULT_EXTENSIONS,
hookIgnoreNodeModules = true,
hookMatcher,
...overrides
} = esbuildOptions

// 利用 transformSync
const compile: COMPILE = function compile(code, filename, format) {
const dir = dirname(filename)
const options = getOptions(dir)
format = format ?? inferPackageFormat(dir, filename)

const {
code: js,
warnings,
map: jsSourceMap,
} = transformSync(code, {
sourcefile: filename,
sourcemap: 'both',
loader: getLoader(filename),
target: options.target,
jsxFactory: options.jsxFactory,
jsxFragment: options.jsxFragment,
format,
...overrides,
})
// 省略部分代碼
}

const revert = addHook(compile, {
exts: extensions,
ignoreNodeModules: hookIgnoreNodeModules,
matcher: hookMatcher,
})

return {
unregister() {
revert()
},
}
}

觀察以上的代碼可知,在 register 函數(shù)內(nèi)部是利用 esbuild 模塊提供的 transformSync API 來實現(xiàn) ts -> js 代碼的轉(zhuǎn)換。其實最關(guān)鍵的環(huán)節(jié),還是通過調(diào)用 pirates 這個庫提供的 addHook 函數(shù)來注冊編譯 ts 文件的鉤子。那么 addHook 函數(shù)內(nèi)部到底做了哪些處理呢?下面我們來看一下它的實現(xiàn):

// pirates-4.0.5/src/index.js
export function addHook(hook, opts = {}) {
let reverted = false;
const loaders = []; // 存放新的loader
const oldLoaders = []; // 存放舊的loader
let exts;

const originalJSLoader = Module._extensions['.js']; // 原始的JS Loader
// 省略部分代碼
exts.forEach((ext) => {
// 獲取已注冊的loader,若未找到,則默認(rèn)使用JS Loader
const oldLoader = Module._extensions[ext] || originalJSLoader;
oldLoaders[ext] = Module._extensions[ext];

loaders[ext] = Module._extensions[ext] = function newLoader(
mod, filename) {
let compile;
if (!reverted) {
if (shouldCompile(filename, exts, matcher, ignoreNodeModules)) {
compile = mod._compile;
mod._compile = function _compile(code) {
// 這里需要恢復(fù)成原來的_compile函數(shù),否則會出現(xiàn)死循環(huán)
mod._compile = compile;
// 在編譯前先執(zhí)行用戶自定義的hook函數(shù)
const newCode = hook(code, filename);
if (typeof newCode !== 'string') {
throw new Error(HOOK_RETURNED_NOTHING_ERROR_MESSAGE);
}

return mod._compile(newCode, filename);
};
}
}

oldLoader(mod, filename);
};
});
}

其實 addHook 函數(shù)的實現(xiàn)并不會復(fù)雜,該函數(shù)內(nèi)部就是通過替換 mod._compile 方法來實現(xiàn)鉤子的功能。即在調(diào)用原始的 mod._compile 方法進(jìn)行編譯前,會先調(diào)用 hook(code, filename)函數(shù)來執(zhí)行用戶自定義的 hook 函數(shù),從而對代碼進(jìn)行預(yù)處理。

而對于 esbuild-register 庫中的 register 函數(shù)來說,當(dāng) hook 函數(shù)執(zhí)行時,就會調(diào)用該函數(shù)內(nèi)部定義的 compile 函數(shù)來編譯 ts 代碼,然后再調(diào)用mod._compile 方法編譯生成的 js 代碼。

關(guān)于 esbuild-register 和 pirates 這兩個庫的內(nèi)容就先介紹到這里,如果你想詳細(xì)了解 pirates 這個庫是如何工作的,可以閱讀 如何為 Node.js 的 require 函數(shù)添加鉤子? 這篇文章。

現(xiàn)在我們已經(jīng)分析完 esno.js 文件,接下來我們來分析 esmo.mjs 文件。

esmo 是如何工作的

esmo 命令對應(yīng)的是 esmo.mjs 文件:

#!/usr/bin/env node

import spawn from 'cross-spawn'
import { resolve } from 'import-meta-resolve'
const spawnSync = spawn.sync

const argv = process.argv.slice(2)
resolve('esbuild-node-loader', import.meta.url).then((path) => {
process.exit(spawnSync('node', ['--loader', path, ...argv],
{ stdio: 'inherit' }).status)
})

由以上代碼可知,當(dāng)使用 node 應(yīng)用程序執(zhí)行 ES Module 文件時,會通過 --loader 選項來指定自定義的 ES Module 加載器。

--loader, --experimental-loader = ... use the specified module as a custom loader

需要注意的是,通過 --loader 選項指定的自定義加載器只適用于 ES Module 的 import 調(diào)用,并不適用于 CommonJS 的 require 調(diào)用。

那么自定義加載器有什么作用呢?在當(dāng)前最新的 Node.js v17.4.0 版本中,還不支持以 https://開頭的說明符。我們可以在自定義加載器中,利用 Node.js 提供的鉤子機(jī)制,讓 Node.js 可以使用import 導(dǎo)入以 https:// 協(xié)議開頭的 ES 模塊。

在分析如何自定義 https 資源加載器前,我們需要先介紹一下 import 說明符的概念。

import 說明符

import 語句的說明符是 from 關(guān)鍵字之后的字符串,例如 import { sep } from 'path' 中的 'path'。說明符也用于 export from 語句,并作為import() 表達(dá)式的參數(shù)。

責(zé)任編輯:武曉燕 來源: 全棧修仙之路
相關(guān)推薦

2023-05-14 23:38:43

Glarity用戶視頻

2020-02-18 20:28:23

AI人工智能

2020-01-20 14:40:39

工具代碼開發(fā)

2014-10-11 09:15:36

2022-11-30 10:59:20

2012-04-12 09:53:02

2009-04-22 15:16:30

2021-12-29 18:18:59

開源MedusaShopify

2017-08-31 10:32:35

交付技術(shù)

2021-01-29 09:07:39

數(shù)據(jù)保護(hù)信息安全數(shù)據(jù)隱私

2011-08-16 17:36:50

SolarisIllumos

2022-07-14 10:54:15

Python代碼Matplotlib

2012-06-25 10:20:22

敏捷開發(fā)

2014-12-01 12:57:46

亞馬遜天貓海淘

2018-09-30 08:00:15

區(qū)塊鏈碳排放氣候

2020-07-23 08:24:14

CSS偽類選擇器

2022-07-27 12:07:58

云計算公有云云支出

2009-08-17 08:54:56

2015-07-09 14:41:15

2012-02-13 09:42:41

備份服務(wù)器數(shù)據(jù)中心
點贊
收藏

51CTO技術(shù)棧公眾號

中文在线观看av| 中文字幕精品视频在线| 麻豆传媒视频在线观看免费| 国产福利一区二区| 欧美亚洲在线视频| 肉色超薄丝袜脚交69xx图片| 日日夜夜精品视频| 日韩欧美一区二区在线| 最新视频 - x88av| 免费在线观看污视频| 久久精品国产网站| 91国产在线精品| 久艹在线观看视频| 中文字幕中文字幕精品| 欧美精品亚洲一区二区在线播放| 国产日韩第一页| 天堂91在线| 美国一区二区三区在线播放| 97精品国产97久久久久久| 毛片久久久久久| 亚洲婷婷伊人| 精品国产1区二区| 伊人色在线观看| 美女一区网站| 亚洲成人av免费| 糖心vlog在线免费观看| 成人在线观看免费| 99久久99久久精品国产片果冻| 国产精品最新在线观看| 欧美啪啪小视频| 激情91久久| 久热精品视频在线免费观看| 天天躁日日躁aaaa视频| 欧美黑人巨大videos精品| 欧美一级在线视频| 一本一道久久a久久综合蜜桃| 91精品论坛| 亚洲成人精品一区二区| 国产91沈先生在线播放| 免费黄色在线网站| 国产精品欧美一区喷水| 日本亚洲自拍| 成人午夜影视| 久久青草欧美一区二区三区| 国产中文一区二区| 色香蕉在线视频| 激情图片小说一区| 成人网欧美在线视频| 亚洲天堂狠狠干| 捆绑紧缚一区二区三区视频| 国产精品久久9| 一级特黄免费视频| 免费成人av在线| 国产精品视频在线观看| 伊人网综合在线| 蜜臀av一级做a爰片久久| 国产精品久久久久久av福利| 91黑人精品一区二区三区| 老司机午夜免费精品视频| 青草成人免费视频| 四虎影院在线免费播放| 日本亚洲欧美天堂免费| 国产精品亚洲欧美导航| 国产精品久久久久精| 国产剧情一区在线| 超碰97人人人人人蜜桃| 理论片中文字幕| 91在线porny国产在线看| 精品视频一区二区三区四区| 亚洲人成色777777老人头| 久久免费看少妇高潮| 日本亚洲欧洲精品| 成人在线影视| 午夜激情一区二区三区| www黄色av| 国外成人福利视频| 日韩免费观看高清完整版 | 99热99精品| 欧美午夜精品久久久久久蜜| 风间由美一区| 亚洲在线观看免费视频| 国产午夜伦鲁鲁| 国产精品久久久久久妇女| 欧美电影一区二区| 中文字幕免费高清视频| 欧美日韩亚洲在线观看| 欧美xxxx做受欧美.88| 日本中文字幕免费| 日本欧美一区二区三区| 99r国产精品视频| 天堂а√在线8种子蜜桃视频| 国产精品入口麻豆九色| 久久www视频| 国产激情视频在线| 一区精品在线播放| 国产黄视频在线| av在线亚洲一区| 日韩精品在线观看网站| 老司机成人免费视频| 国产日韩综合| 亚洲精品欧美日韩| 韩国三级在线观看久| 一区二区三区av电影| 国产福利一区视频| 1313精品午夜理伦电影| 中文在线资源观看视频网站免费不卡| 中文字幕五月天| 日日噜噜夜夜狠狠视频欧美人 | 国产毛片一区二区三区| 欧美第一黄色网| 最新中文字幕第一页| 不卡一区二区三区四区| 中文字幕99| 色老太综合网| 亚洲国产私拍精品国模在线观看| 五月天色婷婷丁香| 日韩影院免费视频| 久久国产主播精品| 欧美黄色视屏| 欧美一区欧美二区| 色www亚洲国产阿娇yao| 久久久久国产精品午夜一区| 国产日韩精品久久| 伊人222成人综合网| 欧美日韩中字一区| av电影网站在线观看| 欧美日韩综合| 91精品视频免费| 在线看av的网址| 色综合久久久久久久久久久| 亚洲av成人精品一区二区三区| 久久香蕉国产| 国产精品久久久久秋霞鲁丝| 牛牛澡牛牛爽一区二区| 精品久久久一区二区| 中国极品少妇xxxx| 欧美视频一区| 91九色蝌蚪嫩草| 成人video亚洲精品| 欧美日本精品一区二区三区| 女人黄色一级片| 日本一区中文字幕| 亚洲欧美精品在线观看| 美女写真久久影院| 中日韩美女免费视频网址在线观看 | 色吊丝在线永久观看最新版本| 亚洲成人动漫av| 五月天激情小说| 亚洲人成久久| 久久亚洲综合网| 亚洲成人短视频| 国产一区二区三区久久精品| 波多野结衣日韩| 国产精品嫩草99a| 在线黄色免费看| 亚洲一本二本| 狠狠色综合欧美激情| 老色鬼在线视频| 亚洲人成电影网站色xx| 中文无码精品一区二区三区| 中文字幕一区二区三区在线播放 | 色综合老司机第九色激情 | av在线播放一区二区三区| 国产午夜福利100集发布| 亚洲69av| 国产乱人伦真实精品视频| 麻豆视频在线| 精品欧美黑人一区二区三区| 亚洲久久在线观看| 国产欧美精品一区二区色综合| a在线观看免费视频| 99国产精品一区二区| 51午夜精品| 在线观看爽视频| 色阁综合伊人av| 成人毛片在线免费观看| 高跟丝袜一区二区三区| www.99热| 国产精品中文字幕日韩精品 | 一区视频免费观看| 99久久精品99国产精品| 婷婷六月天在线| 亚洲电影在线一区二区三区| 国产麻豆乱码精品一区二区三区 | 国内一区在线| jvid一区二区三区| 欧美激情精品久久久久久变态| 神马午夜电影一区二区三区在线观看 | 成人爱爱电影网址| 国产小视频精品| 黄色在线一区| 宅男av一区二区三区| 精品资源在线| 国产精品视频一区二区三区四| 午夜影院免费在线| 在线观看日韩av| 免费av网站观看| 欧美日本在线一区| 九九热精品视频在线| 国产精品成人网| 国产视频久久久久久| 日韩av一二三| 久久久久久久久久久久久久一区| 国产高清精品二区| 国产成人精品免费视频| 激情影院在线| 日韩亚洲欧美中文高清在线| 欧美一区二区公司| 91麻豆精品国产91久久久久久 | 欧美日韩亚洲一区二区三区在线观看| 国产精品**亚洲精品| 日韩av电影国产| www.综合网.com| 久久精品99久久久久久久久| 欧美另类自拍| 亚洲精品电影网| 精品国产区一区二| 欧美美女黄视频| 亚洲精品成人在线视频| 亚洲国产日韩精品| 色偷偷www8888| 国产欧美一区二区三区鸳鸯浴| 国产精品无码电影| 成人av网站免费观看| www.污污视频| 欧美aaaaaa午夜精品| 国内外成人激情视频| 亚洲激情黄色| 免费拍拍拍网站| 欧美影视一区| 国产一二三四区在线观看| 成人女性视频| 手机成人在线| 激情综合网五月| 欧美性天天影院| 一区二区三区日本久久久| 国产一区免费| 综合欧美亚洲| 国产精品免费一区二区三区四区 | 中文精品视频一区二区在线观看| 精品国产乱码久久久久久果冻传媒| 精品视频一区二区三区四区| 任你弄精品视频免费观看| 国产偷国产偷亚洲高清97cao| 日韩视频一二区| 99久久伊人精品影院| 日韩免费一级| 97久草视频| 国产成人澳门| 久久大片网站| 色综合www| 日产精品高清视频免费| 欧美手机视频| 亚洲欧美日韩另类精品一区二区三区| 欧美日韩亚洲在线观看| 中文字幕中文字幕一区三区| 国产精品二区不卡| 91精品一区二区三区四区| 午夜日韩福利| www在线观看免费| 久久一二三四| 日本中文字幕影院| 国产精品一区二区男女羞羞无遮挡| www.久久com| 成人动漫中文字幕| 欧美做受高潮6| 亚洲欧洲三级电影| 日操夜操天天操| 日本高清视频一区二区| 亚洲一级黄色大片| 精品国产免费一区二区三区香蕉| 亚洲av激情无码专区在线播放| 亚洲视频欧洲视频| 黄色精品免费看| 97国产精品视频人人做人人爱| 日韩三级影视| 亚洲最大成人网色| 美女精品一区最新中文字幕一区二区三区| 欧美婷婷久久| 国产尤物精品| 日本va中文字幕| 国产精品一品二品| 泷泽萝拉在线播放| 亚洲日本va午夜在线影院| 日韩乱码人妻无码中文字幕| 在线精品视频一区二区| 国产黄色av片| 国产性猛交xxxx免费看久久| 在线观看电影av| 庆余年2免费日韩剧观看大牛| 国产成人免费| 九九99久久| 99九九热只有国产精品| 日本a在线免费观看| 美女视频网站久久| av直播在线观看| 一区二区三区免费看视频| 波多野结衣黄色| 精品日韩成人av| 亚洲精品传媒| 日韩av第一页| 欧美精品中文字幕亚洲专区| 亚洲激情一区二区三区| 在线视频精品| 久草福利在线观看| 国产女同性恋一区二区| 国产在线视频你懂的| 欧美美女一区二区三区| 日韩欧美在线观看一区二区| 欧美成年人在线观看| 精品免费av一区二区三区| 国产欧美在线一区二区| 在线国产一区二区| 奇米影音第四色| 久久夜色精品国产欧美乱极品| 久久久久久久久久91| 69av一区二区三区| a天堂在线资源| 日产精品99久久久久久| 香蕉久久夜色精品国产使用方法| 国产成人一二三区| 激情久久五月天| 特级西西人体高清大胆| 91久久精品午夜一区二区| 午夜在线视频免费| 久久久欧美一区二区| 麻豆精品在线| 视色,视色影院,视色影库,视色网| 日韩精品一级中文字幕精品视频免费观看 | 国产a级片视频| 亚洲激情男女视频| 国产三级在线观看视频| 北条麻妃在线一区二区| 日本成人一区二区| 亚洲视频电影| 麻豆国产91在线播放| 99久久99久久精品免费| 欧美视频一区二区| 97在线观看免费观看高清 | 中文字幕一区二区三区蜜月| 性高潮视频在线观看| 尤物精品国产第一福利三区| 欧美日韩激情电影| 色99中文字幕| 麻豆精品久久久| 老司机成人免费视频| 欧美一区二区三区四区高清| 在线看三级电影| 成人黄视频免费| 精品91在线| 蜜桃精品成人影片| 色天天综合色天天久久| 国产精品四虎| 国产主播精品在线| 欧美午夜不卡| 国产精品无码一区二区三| 日韩欧美有码在线| 99视频在线观看地址| 91精品啪在线观看麻豆免费| 欧美精品1区| 无码人妻aⅴ一区二区三区 | 2021国产精品视频| 国产欧美日韩一区二区三区四区| 国产一级片黄色| 中文字幕五月欧美| 国产 日韩 欧美 综合| 91成人性视频| 日韩在线观看| 四虎国产精品永久免费观看视频| 亚洲国产精品一区二区www在线 | 久久精品人人做人人爽| 粉嫩久久久久久久极品| 欧美成人黑人猛交| 欧美国产一区二区在线观看| 99久久久国产精品无码网爆| 久久人人爽国产| 国产精品美女久久久久久不卡| 岛国av免费在线| 欧美日韩国产一区在线| 中文字幕在线观看日本| 国产精品一国产精品最新章节| 亚洲男女自偷自拍| 中文字幕美女视频| 亚洲第一在线视频| 精品久久在线| 黄色大片在线免费看| 中文字幕国产一区| 人妻少妇精品无码专区久久| 国产精品高清在线| 极品少妇一区二区三区| 国产黄色大片免费看| 精品欧美一区二区久久| 久久久久久一区二区三区四区别墅| 成人免费网站入口| 国产精品视频九色porn| 水中色av综合| 成人黄视频免费| 激情图片小说一区| 久久精品视频2| 高清一区二区三区四区五区|