美文网首页
babel插件入门

babel插件入门

作者: 7revor | 来源:发表于2019-08-01 17:51 被阅读0次

    关于babel

    Babel 是一个 JavaScript 编译器

    Babel 是一个工具链,主要用于将 ECMAScript 2015+ 版本的代码转换为向后兼容的 JavaScript 语法,以便能够运行在当前和旧版本的浏览器或其他环境中。

    // Babel 输入: ES2015 箭头函数
    [1, 2, 3].map((n) => n + 1);
    
    // Babel 输出: ES5 语法实现的同等功能
    [1, 2, 3].map(function(n) {
      return n + 1;
    });
    

    babel做了什么

    babel 的转译过程也分为三个阶段,这三步具体是:

    解析 Parse

    将代码解析生成抽象语法树( 即AST )。

    转换 Transform

    对于 AST 进行变换一系列的操作,babel 接受得到 AST 并通过 babel-traverse 对其进行遍历,在此过程中进行添加、更新及移除等操作。

    生成 Generate

    将变换后的 AST 再转换为 JS 代码, 使用到的模块是 babel-generator。

    我们编写的 babel 插件则主要专注于第二步转换过程的工作,专注于对于代码的转化和拓展,解析与生成的偏底层相关操作则有对应的模块支持,在此我们理解它主要做了什么即可。

    准备工作

    正式开始之前,需要介绍两个概念:

    Visitors 访问者

    访问者是一个用于 AST 遍历的跨语言的模式。
    简单的说它们就是一个对象,定义了用于在一个树状结构中获取具体节点的方法。 这么说有些抽象,所以让我们来看一个例子

    const MyVisitor = {
      Identifier(path) {
        console.log("Im Identifier");
      },
      FunctionDeclaration(path){
        console.log("Im FunctionDeclaration");
      }
    };
    

    这是一个简单的访问者,把它用于AST遍历中时,每当在树中遇见一个 Identifier 的时候会调用 Identifier() 方法,遇见一个 FunctionDeclaration 的时候则会调用 FunctionDeclaration()
    方法。

    Paths 路径

    Visitors
    在遍历到每个节点的时候,
    都会给我们传入 path 参数,
    它包含了节点的信息以及节点和所在的位置,
    供我们对特定节点进行修改。
    之所以称之为 path 是其表示的是两个节点之间连接的对象,而非指当前的节点对象。

    更具体的API可以查看Babel插件手册

    插件格式

    一个完整的插件格式如下

    export default function({ types: t }) {
      return {
        pre(state) {  // 遍历之前 
             
        },
        visitor: { // 访问者
          VariableDeclaration(path) {
            // ... ...
          }
        },
        post(state) { // 遍历结束
          
        },
      };
    }
    

    注意

    这里有一个值得注意的问题,所有的babel插件会共享同一次遍历过程。

    也就是说,他们对节点的处理可能会相互影响。

    比如我们需要对所有的方法添加 try-catch ,就需要定义一个FunctionDeclaration访用来访问所有的函数。

    但是在其他插件里,比如babel-preset,它会生成一些辅助函数,这些辅助函数也会被我们的访问者访问。但我们只需要对源码进行处理。

    想要避免对这些不在原始代码中的节点进行访问,笔者现在也没找到一个最好的方法,有以下尝试:

    • 使用sourceMap,如果节点在sourceMap中找不到,则判断为生成的代码。
      但sourceMap需要借助于webpack获取(或许存在更好的方法,但笔者还没找到,欢迎指正),这样插件配置起来比较复杂,并且通用性不够好。
    • 借助path.node.loc。这个方法不准确,有些生成的节点也会含有location属性。
    • 在节点开始遍历之前手动添加一次额外的遍历,我们处理完成后再交由其他插件处理。也是笔者目前在用的方法。目前来看比较准确,但需要一次额外的遍历开销。

    开始

    首先定义 Visitor 来访问方法声明

    const funcVisitor = {
      FunctionDeclaration(path) {
        const functionBody = path.node.body; //获取方法的 body
        if (functionBody.type === 'BlockStatement') { // 含有block
          const body = functionBody.body; // 获取原来的block body
          path.get('body').replaceWith(wrapFunction({
            BODY: body,
            HANDLER:t.identifier('console.log')
          }))
        } 
      }
    }
    

    借助 babel-template 快速生成AST节点

    const wrapFunction = template(`{
      try {
        BODY
      } catch(err) {
        HANDLER(err)
      }
    }`);
    

    接下来组装

    const t = require('@babel/types');
    const wrapFunction = template(`{
      try {
        BODY
      } catch(err) {
        HANDLER(err)
      }
    }`);
    const funcVisitor = {
      FunctionDeclaration(path) {
        const functionBody = path.node.body; //获取方法的 body
        if (functionBody.type === 'BlockStatement') { // 含有block
          const body = functionBody.body; // 获取原来的block body
          path.get('body').replaceWith(wrapFunction({
            BODY: body,
            HANDLER:t.identifier('console.log')
          }))
        } 
      }
    }
    module.exports = function () {
      return {
        pre(file){ // 开始遍历之前
          file.path.traverse(funcVisitor); // 插入额外的遍历
        }
      };
    };
    

    使用

    webpack.config.js

    const myPlugin = require('xxx')
    module: {
        rules: [
          {
            test: /\.js$/,
            exclude:/node_modules/,//排除掉node_module目录
            loader:'babel-loader',
            options:{
              plugins:[myPlugin]
            },
          },
        ]
      },
    

    测试一下,输入代码

    function Foo(){
      console.log('Im foo')
    }
    

    输出为

    function Foo(){
      try{
       console.log('Im foo') 
      }catch (err) {
        console.error(err)
      }
    }
    

    一个简单的为方法声明增加try-catch的babel插件就开发完成了。但它也就仅仅能够应付测试中的简单情况。

    笔者已经写好了一个相对完善的插件,它可以为Promise添加.catch,也可以对 方法声明 | 类方法 注入捕获语句。配置也相对灵活,支持目录以及文件筛选。

    不完善的地方欢迎大家补充~

    源码传送门

    相关文章

      网友评论

          本文标题:babel插件入门

          本文链接:https://www.haomeiwen.com/subject/lpmcdctx.html