babel插件入门

关于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,也可以对 方法声明 | 类方法 注入捕获语句。配置也相对灵活,支持目录以及文件筛选。

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

源码传送门

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

推荐阅读更多精彩内容

  • 目录 Babel简介 Babel运行原理 AST解析 AST转换 写一个Babel插件 Babel简介 Babel...
    圆儿圈圈阅读 4,004评论 0 3
  • 前言 [实践系列] 主要是让我们通过实践去加深对一些原理的理解。 [实践系列]前端路由 有兴趣的同学可以关注 [实...
    null仔阅读 1,336评论 0 1
  • babel官网 babel 介绍 Babel 是一个通用的多用途 JavaScript 编译器。通过 Babel ...
    锋享前端阅读 1,806评论 0 10
  • 在处理我的 Webflow/React transpiler 时,在我脑海中突然出现写下这篇文章的想法。我想做的就...
    Howie126313阅读 1,127评论 0 0
  • 相信目前常与 ES6 代码打交道的同学对 Babel 应该不会陌生,在 ES6 代码被编译转化为 ES5 代码的过...
    DC_er阅读 611评论 0 0