理解 Babel 及其插件机制

Aug 3, 2018 00:00 · 1378 words · 3 minute read tech js

简介

Babel 是 JavaScript 编译器,更确切地说是源码到源码的编译器,通常也叫做“转换编译器(transpiler)”。 意思是说你为 Babel 提供一些 JavaScript 代码,Babel 更改这些代码,然后返回给你新生成的代码。

Babel 处理的三个步骤

  • 解析 parse
  • 转换 transform
  • 生成 generate

相关包和 API

  • @babel/parser (babylon)
  • babel-core
  • babel-types
  • babel-traverse
  • babel-generator

解析

该步骤接收代码并输出 AST。 这个步骤分为两个阶段:词法分析(Lexical Analysis) 和  语法分析(Syntactic Analysis)

const parser = require('@babel/parser')
const code = 'function square(n) {return n * n;}';
const node = parser.parse(code) // 解析代码返回AST树

要方便的查看代码的 AST 结构,可以访问 http://astexplorer.net/ 进行测试。Babel 使用一个基于 ESTree 并修改过的 AST,它的内核说明文档可以在 这里 找到。

function square(n) {
  return n * n;
}

该函数的 AST 结构如下 (简化了一些属性)

{
  type: "FunctionDeclaration",
  id: {
    type: "Identifier",
    name: "square"
  },
  params: [{
    type: "Identifier",
    name: "n"
  }],
  body: {
    type: "BlockStatement",
    body: [{
      type: "ReturnStatement",
      argument: {
        type: "BinaryExpression",
        operator: "*",
        left: {
          type: "Identifier",
          name: "n"
        },
        right: {
          type: "Identifier",
          name: "n"
        }
      }
    }]
  }
}

每一个节点都有如下所示的接口(Interface):

interface Node {
  type: string;
}

AST 就是由多层嵌套的 Node 结合而成。

转换

转换步骤接收 AST 并对其进行遍历,在此过程中对节点进行添加、更新及移除等操作。 这是 Babel 或是其他编译器中最复杂的过程 同时也是插件将要介入工作的部分。

babel-core 向外暴露出 babel.transform 接口

const result = babel.transform(code, {
    plugins: [
        myPluginName
    ]
})

生成

代码生成步骤把最终(经过一系列转换之后)的 AST 转换成字符串形式的代码,同时还会创建源码映射(source maps)。. 代码生成其实很简单:深度优先遍历整个 AST,然后构建可以表示转换后代码的字符串。利用 babel-generator 将 AST 输出为转码后的代码字符串

babel-traverse

Babel Traverse(遍历)模块维护了整棵树的状态,并且负责替换、移除和添加节点。

我们可以和 Babylon 一起使用来遍历和更新节点:

import * as babylon from "@babel/parser";
import traverse from "babel-traverse";

const code = `function square(n) {
  return n * n;
}`;

const ast = babylon.parse(code);

traverse(ast, {
  enter(path) {
    if (
      path.node.type === "Identifier" &&
      path.node.name === "n"
    ) {
      path.node.name = "x";
    }
  }
});

babel-types

Babel Types模块是一个用于 AST 节点的 Lodash 式工具库,它包含了构造、验证以及变换 AST 节点的方法。 该工具库包含考虑周到的工具方法,对编写处理AST逻辑非常有用。

简单的说就是封装了一系列方法,方便操作变换 AST 的节点。

import traverse from "babel-traverse";
import * as t from "babel-types";

traverse(ast, {
  enter(path) {
    if (t.isIdentifier(path.node, { name: "n" })) {
      path.node.name = "x";
    }
  }
});

写一个简单的 babel 插件

使用 Visitor 模式遍历访问 Node 节点

利用深度遍历每个节点的时候,包含两个时机:进入节点enter 和离开节点 exit,类似 koa 的一个洋葱模型。我们也应该在这两个时机的时候去对 AST 树做对应的转换。以下是Visitor 访问的例子。

const MyVisitor = {
  // 简化 enter 的形式
  Identifier() {
    console.log("Called!");
  }
  // 完整形式
  //Identifier() {
  //  enter() {...}
  //  exit() {...}
  //}
};

// 对于这个函数
function square(n) {
  return n * n;
}

// 遍历包含了四个 Identifier (1个 square 和 3个 n)
path.traverse(MyVisitor);
// 输出四次结果
Called!
Called!
Called!
Called!

Path

path 的概念是为了简化对 AST 树的操作。Path 是表示两个节点之间连接的对象。 在某种意义上,路径是一个节点在树中的位置以及关于该节点各种信息的响应式 Reactive 表示。 当你调用一个修改树的方法后,路径信息也会被更新。 Babel 帮你管理这一切,从而使得节点操作简单,尽可能做到无状态。

首先编写插件

这个插件的作用就是简单的把所有变量名为 foo 的变量改为 bar,具体实现如下:

module.exports = function myPlugin(babel) {
  return {
    visitor: {
      Identifier(path) {
        if (path.node.name === 'foo') {
          path.node.name = 'bar';
        }
      }
    }
  };
};

调用插件

const babel = require('babel-core');
const myPlugin = require('./myPlugin');
const example = 'const foo = 1; console.log(foo);'
const res = babel.transform(example, {plugins: [myPlugin]});
// res.code = "const bar = 1; console.log(bar);"

总结

简单介绍了 babel 转换代码的基本内容,实现了一个简单的插件。在具体开发插件的过程中,最重要的还是对 AST 的转换,所以需要对 AST 的操作和相关 API 有更加具体的理解。详细信息可以看babel-notebook的介绍,本文也有很大一部分参考了 babel-notebook。

tweet Share