本文首發於 Ahonn's Blog: 如何實現一個 Babel Macrosjavascript
經過 babel 插件,咱們很容易的就在編譯時將某些代碼轉換成其餘代碼以實現某些優化。例如 babel-plugin-lodash 能夠幫咱們將直接 import 的 lodash 替換成可以進行 tree shaking 的代碼;經過 babel-plugin-preval 在編譯時執行腳本並使用返回值原位替換。前端
一切看起來都很美好,但實際上在使用 babel 插件時咱們還須要對 .babelrc
或者 babel.config.js
進行配置。java
{
"plugins": ["preval"]
}
複製代碼
在暴露 babel 配置文件的項目下或許還可以接受,但在 create-react-app 下就不得不破壞原來的和諧, eject 一下配置再進行相關的配置了。react
有沒有什麼更好的方式呢?有的,咱們能夠用 babel-plugin-macroswebpack
babel-plugin-macros 顯而易見是一個 babel 插件,它提供了一種零配置編譯時替換代碼的方式。咱們只須要在 babel 配置裏添加 babel-plugin-macros 插件配置就可使用了。顯然這個 「零配置」 是把自身除外的。但別擔憂,create-react-app 已經內置了這個插件,能夠開箱即用。git
{
"plugins": ["macros"]
}
複製代碼
而後就能夠開始真正的零配置體驗,引入咱們須要的 macro 直接使用。github
// 編譯前
import preval from 'preval.macro';
const one = preval`module.exports = 1 + 2 - 1 - 1`;
// 編譯後
const one = 3;
複製代碼
與 babel-plugin-preval 相比,咱們不在須要再進行額外的配置,而是經過 import macro 來使用對應的功能。babel 在編譯期會讀取以 .macro 結尾的包,並執行對應的邏輯來替換代碼,這種方式比插件來的更加直觀,咱們不再會出現 「這個 preval 是哪裏引進來?」 的疑問了。web
那麼怎麼實現一個 babel macros 呢?npm
假設咱們有這麼一個場景:咱們的項目中包括先後端的代碼,後端的 Node.js 經過 dotenv 讀取項目根目錄下的 .env
獲取某些配置,如今咱們有一些前端 JavaScript 代碼也須要使用到 .env
裏到某些配置,但不能把全部的配置都暴露到 JavaScript 中。json
通常狀況下,咱們能夠將 .env 中的某些配置傳入 webpack 的 DefinePlugin 插件中,前端代碼經過讀取全局變量的方式進行訪問。如今咱們經過 Babel macros 的方式來實現以下效果:
# .env
NAME=ahonn
NUMBER=123
複製代碼
// 編譯前
import dotenv from 'dotenv.macro';
const NAME = dotenv('NAME');
const NUMBER = dotenv('NUMBER');
// 編譯後
const NAME = "ahonn";
const NUMBER = "123";
複製代碼
babel-plugin-macros 會把引入的 .macro 或者 .macro.js 當成宏進行處理,全部首先咱們須要建立一個名爲 dotenv.macro.js 的文件,而且這個文件導出的應該是一個經過 createMacro
包裝後的函數。
若是沒有經過 createMacro
進行包裝的話,執行 babel
就會提示:The macro imported from "../../dotenv.macro" must be wrapped in "createMacro" which you can get from "babel-plugin-macros".
const { createMacro } = require('babel-plugin-macros');
module.exports = createMacro(({ references, state, babel }) => {
// TODO
});
複製代碼
傳入 createMacro
的函數接受三個參數:
require(‘@babel/core’)
相同在咱們的例子中 references 的值是 { default: [ NodePath {...} ] }
,這裏的 default
中的 NodePath 便是上面編譯前代碼中 dotenv
調用在 AST 中的節點。 (若是對 AST 或者 babel 插件開發不太熟悉的話,推薦閱讀 babel-handbook/plugin-handbook.md)
拿到對應的 AST 節點(後面稱爲 path)以後,咱們須要對調用形式進行判斷來肯定如何轉換代碼,這裏咱們經過判斷 path.parentPath
的節點類型來判斷。
咱們能夠經過傳入 createMacro
的函數的第三個參數 babel 來獲取一些用於判斷節點類型的函數,babel.types
等價於 @babel/types。
babel.types.isCallExpression
來判斷是否爲函數形式調用babel.types.isTaggedTemplateExpression
來判斷是否爲模版字符串形式調用咱們只對函數形式調用處理:
const { createMacro } = require('babel-plugin-macros');
module.exports = createMacro(({ references, state, babel }) => {
references.default.forEach((path) => {
if (path.parentPath && babel.types.isCallExpression(path.parentPath)) {
// TODO
}
});
});
複製代碼
作完前置的條件判斷以後,如今咱們就能夠經過 dotenv
來獲取 .env
中配置的值,而後將對應的值替換對應的 AST 節點,從而使得編譯後的代碼在 macro 引用位置被替換爲目標值。
const dotenv = require('dotenv');
const { createMacro } = require('babel-plugin-macros');
module.exports = createMacro(({ references, state, babel }) => {
const env = dotenv.config();
references.default.forEach((path) => {
if (path.parentPath && babel.types.isCallExpression(path.parentPath)) {
const args = path.parentPath.get('arguments');
const key = args[0].evaluate().value;
const value = env.parsed[key]; // ahonn
}
});
});
複製代碼
咱們經過 path.parentPath.get('arguments')
獲取到父節點(即節點類型爲 CallExpression 的節點)中的 arguments 屬性(即函數調用參數列表)。而後經過 args[0].evaluate().value
來獲取第一個參數的值,即爲 dotenv('NAME')
中的 'NAME'
。最後從 dotenv 解析的 env 對象中獲取目標值 'ahonn'
。
最後一步,咱們須要判斷上一步獲取的目標值的類型,而後根據不一樣的類型進行 AST 轉換。以咱們上面的例子來講就是:
const NAME = dotenv('NAME');
轉換爲 const NAME = 'ahonn';
const NUMBER = dotenv('NUMBER');
轉換爲 const NUMBER = 123;
const dotenv = require('dotenv');
const { createMacro } = require('babel-plugin-macros');
module.exports = createMacro(({ references, state, babel }) => {
const env = dotenv.config();
references.default.forEach((path) => {
if (path.parentPath && babel.types.isCallExpression(path.parentPath)) {
const args = path.parentPath.get('arguments');
const key = args[0].evaluate().value;
const value = env.parsed[key];
if (typeof value === 'number') {
path.parentPath.replaceWith(babel.types.numericLiteral(value));
} else {
path.parentPath.replaceWith(babel.types.stringLiteral(value));
}
}
});
});
複製代碼
經過 typeof value
判斷目標值的類型,這裏只處理數字與字符串,非數字的值都當成字符串處理。而後再一次的經過 babel.types
中提供的 numericLiteral
與 stringLiteral
來建立對應的 AST 節點。最後將 path.parentParh
替換爲生產的節點。
到這裏,一個讀取 .env 中對應的值並在編譯時替換相應的代碼的 macro 就完成了。上面咱們提到的 preval.macro
的實現也與上面相似。
爲何是替換掉 path.parentPath ? A: 由於咱們拿到的 references 中的引用只是對應的宏的 AST 節點,而通常 Babel macros 中咱們經過函數調用或者模版字符串形式進行調用,所以須要往上一層進行替換。
能夠經過 Babel macros 拓展 JavaScript 語法麼? 不行,由於 Babel 只可以識別合法的 JavaScript 語法,即便使用 babel-plugin-macros 也沒法改變這一事實。若是想要拓展 JavaScript 語法的話須要修改 babel-parser。具體怎麼作,能夠查看這篇文章:Creating custom JavaScript syntax with Babel | Tan Li Hau
看到這裏,能夠發現實現一個 Babel macros 的過程與開發 Babel 插件的流程相似,都是對 AST 進行操做。babel-plugin-macro 只是提供一個在「外部」進行 AST 修改的方式,經過這種方式可以靈活的對 Babel 編譯時進行拓展。但話又說回來,這種方式用多了會不會令代碼變得很差維護呢?歡迎留言討論。