欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

babel插件去除console示例詳解

 更新時間:2022年10月19日 15:07:35   作者:安穩(wěn)  
這篇文章主要為大家介紹了babel插件去除console示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

起因

已經頹廢了很久 因為實在不知道寫啥了 突然我某個同事對我說 寶哥 你看這個頁面好多console.log 不僅會影響性能 而且可能會被不法分子所利用 我覺得很有道理 所以我萌生了寫一個小插件來去除生產環(huán)境的console.log的想法

介紹

我們籠統(tǒng)的介紹下babel,之前我有一篇寫精度插件的babel文章,babel一共有三個階段:第一階段是將源代碼轉化為ast語法樹、第二階段是對ast語法樹進行修改,生成我們想要的語法樹、第三階段是將ast語法樹解析,生成對應的目標代碼。

窺探

我們的目的是去除console.log,我們首先需要通過ast查看語法樹的結構。我們以下面的console為例:

注意 因為我們要寫babel插件 所以我們選擇@babel/parser庫生成ast,因為babel內部是使用這個庫生成ast的

console.log("我會被清除"); 

初見AST

AST是對源碼的抽象,字面量、標識符、表達式、語句、模塊語法、class語法都有各自的AST。

我們這里只說下本文章中所使用的AST。

Program

program 是代表整個程序的節(jié)點,它有 body 屬性代表程序體,存放 statement 數組,就是具體執(zhí)行的語句的集合。

可以看到我們這里的body只有一個ExpressionStatement語句,即console.log。

ExpressionStatement

statement 是語句,它是可以獨立執(zhí)行的單位,expression是表達式,它倆唯一的區(qū)別是表達式執(zhí)行完以后有返回值。所以ExpressionStatement表示這個表達式是被當作語句執(zhí)行的。

ExpressionStatement類型的AST有一個expression屬性,代表當前的表達式。

CallExpression

expression 是表達式,CallExpression表示調用表達式,console.log就是一個調用表達式。

CallExpression類型的AST有一個callee屬性,指向被調用的函數。這里console.log就是callee的值。

CallExpression類型的AST有一個arguments屬性,指向參數。這里“我會被清除”就是arguments的值。

MemberExpression

Member Expression通常是用于訪問對象成員的。他有幾種形式:

a.b
a["b"]
new.target
super.b

我們這里的console.log就是訪問對象成員log。

  • 為什么MemberExpression外層有一個CallExpression呢?

實際上,我們可以理解為,MemberExpression中的某一子結構具有函數調用,那么整個表達式就成為了一個Call Expression。

MemberExpression有一個屬性object表示被訪問的對象。這里console就是object的值。

MemberExpression有一個屬性property表示對象的屬性。這里log就是property的值。

MemberExpression有一個屬性computed表示訪問對象是何種方式。computed為true表示[],false表示. 。

Identifier

Identifer 是標識符的意思,變量名、屬性名、參數名等各種聲明和引用的名字,都是Identifer。

我們這里的console就是一個identifier。

Identifier有一個屬性name 表示標識符的名字

StringLiteral

表示字符串字面量。

我們這里的log就是一個字符串字面量

StringLiteral有一個屬性value 表示字符串的值

公共屬性

每種 AST 都有自己的屬性,但是它們也有一些公共的屬性:

  • type:AST節(jié)點的類型
  • start、end、loc:start和end代表該節(jié)點在源碼中的開始和結束下標。而loc屬性是一個對象,有l(wèi)ine和column屬性分別記錄開始和結束的行列號
  • leadingComments、innerComments、trailingComments:表示開始的注釋、中間的注釋、結尾的注釋,每個 AST 節(jié)點中都可能存在注釋,而且可能在開始、中間、結束這三種位置,想拿到某個 AST 的注釋就通過這三個屬性。

如何寫一個babel插件?

babel插件是作用在第二階段即transform階段。

transform階段有@babel/traverse,可以遍歷AST,并調用visitor函數修改AST。

我們可以新建一個js文件,其中導出一個方法,返回一個對象,對象存在一個visitor屬性,里面可以編寫我們具體需要修改AST的邏輯。

+ export default () => {
+  return {
+    name: "@parrotjs/babel-plugin-console",
+    visitor,
+  };
+ };

構造visitor方法

path 是記錄遍歷路徑的 api,它記錄了父子節(jié)點的引用,還有很多增刪改查 AST 的 api

+ const visitor = { 
+   CallExpression(path, { opts }) {
+    //當traverse遍歷到類型為CallExpression的AST時,會進入函數內部,我們需要在函數內部修改
+  }
+ };

我們需要遍歷所有調用函數表達式 所以使用CallExpression。

去除所有console

我們將所有的console.log去掉

path.get 表示獲取某個屬性的path

path.matchesPattern 檢查某個節(jié)點是否符合某種模式

path.remove 刪除當前節(jié)點

CallExpression(path, { opts }) {
+  //獲取callee的path
+  const calleePath = path.get("callee"); 
+  //檢查callee中是否符合“console”這種模式
+  if (calleePath && calleePath.matchesPattern("console", true)) {
+       //如果符合 直接刪除節(jié)點  
+       path.remove();
+  }
},

增加env api

一般去除console.log都是在生產環(huán)境執(zhí)行 所以增加env參數

AST的第二個參數opt中有插件傳入的配置

+  const isProduction = process.env.NODE_ENV === "production";
CallExpression(path, { opts }) {
....
+  const { env } = opts;
+  if (env === "production" || isProduction) {
       path.remove();
+  }
....
},

增加exclude api

我們上面去除了所有的console,不管是error、warning、table都會清除,所以我們加一個exclude api,傳一個數組,可以去除想要去除的console類型

....
+ const isArray = (arg) => Object.prototype.toString.call(arg) === "[object Array]";
- const { env } = opts;
+ const { env,exclude } = opts;
if (env === "production" || isProduction) {
- path.remove();  
+ //封裝函數進行操作
+ removeConsoleExpression(path, calleePath, exclude);
}
+const removeConsoleExpression=(path, calleePath, exclude)=>{
+  if (isArray(exclude)) { 
+    const hasTarget = exclude.some((type) => {
+      return calleePath.matchesPattern("console." + type);
+    });
+    //匹配上直接返回不進行操作
+    if (hasTarget) return;
+  }
+  path.remove();
+}

增加commentWords api

某些時候 我們希望一些console 不被刪除 我們可以給他添加一些注釋 比如

//no remove
console.log("測試1");
console.log("測試2");//reserse
//hhhhh
console.log("測試3")

如上 我們希望帶有no remove前綴注釋的console 和帶有reserse后綴注釋的console保留不被刪除

之前我們提到 babel給我們提供了leadingComments(前綴注釋)和trailingComments(后綴注釋)我們可以利用他們 由AST可知 她和CallExpression同級,所以我們需要獲取他的父節(jié)點 然后獲取父節(jié)點的屬性

path.parentPath 獲取父path

path.node 獲取當前節(jié)點

- const { exclude, env } = opts;
+ const { exclude, commentWords, env } = opts;
+ const isFunction = (arg) =>Object.prototype.toString.call(arg) === "[object Function]";
+ // 判斷是否有前綴注釋 
+ const hasLeadingComments = (node) => {
+  const leadingComments = node.leadingComments;
+  return leadingComments && leadingComments.length;
+ };
+ // 判斷是否有后綴注釋 
+ const hasTrailingComments = (node) => {
+  const trailingComments = node.trailingComments;
+  return trailingComments && trailingComments.length;
+ };
+ //判斷是否有關鍵字匹配 默認no remove || reserve 且如果commentWords和默認值是相斥的
+ const isReserveComment = (node, commentWords) => {
+ if (isFunction(commentWords)) {
+   return commentWords(node.value);
+ }
+ return (
+    ["CommentBlock", "CommentLine"].includes(node.type) &&
+    (isArray(commentWords)
+      ? commentWords.includes(node.value)
+      : /(no[t]? remove\b)|(reserve\b)/.test(node.value))
+  );
+};
- const removeConsoleExpression = (path, calleePath, exclude) => {
+ const removeConsoleExpression = (path, calleePath, exclude,commentWords) => {
+ //獲取父path
+ const parentPath = path.parentPath;
+ const parentNode = parentPath.node;
+ //標識是否有前綴注釋
+ let leadingReserve = false;
+ //標識是否有后綴注釋
+ let trailReserve = false;
+ if (hasLeadingComments(parentNode)) {
+    //traverse 
+    parentNode.leadingComments.forEach((comment) => {
+      if (isReserveComment(comment, commentWords)) {
+        leadingReserve = true;
+      }
+    });
+  }
+ if (hasTrailingComments(parentNode)) {
    //traverse 
+   parentNode.trailingComments.forEach((comment) => {
+     if (isReserveComment(comment, commentWords)) {
+       trailReserve = true;
+     }
+   });
+ } 
+ //如果沒有前綴節(jié)點和后綴節(jié)點 直接刪除節(jié)點
+ if (!leadingReserve && !trailReserve) {
+    path.remove();
+  }
} 

細節(jié)完善

我們大致完成了插件 我們引進項目里面進行測試

console.log("測試1");
//no remove
console.log("測試2"); 
console.log("測試3");//reserve
console.log("測試4");
//新建.babelrc 引入插件
{
    "plugins":[["../dist/index.cjs",{
        "env":"production"
    }]]
}

理論上應該移除測試1、測試4,但是我們驚訝的發(fā)現(xiàn) 竟然一個console沒有刪除??!經過排查 我們大致確定了問題所在

因為測試2的前綴注釋同時也被AST納入了測試1的后綴注釋中了,而測試3的后綴注釋同時也被AST納入了測試4的前綴注釋中了

所以測試1存在后綴注釋 測試4存在前綴注釋 所以測試1和測試4沒有被刪除

那么我們怎么判斷呢?

對于后綴注釋

我們可以判斷后綴注釋是否與當前的調用表達式處于同一行,如果不是同一行,則不將其歸納為后綴注釋

 if (hasTrailingComments(parentNode)) {
+    const { start:{ line: currentLine } }=parentNode.loc;
    //traverse
    // @ts-ignore
    parentNode.trailingComments.forEach((comment) => { 
+      const { start:{ line: currentCommentLine } }=comment.loc;
+      if(currentLine===currentCommentLine){
+        comment.belongCurrentLine=true;
+      }
+     //屬于當前行才將其設置為后綴注釋
-      if (isReserveComment(comment, commentWords))
+      if (isReserveComment(comment, commentWords) && comment.belongCurrentLine) {
        trailReserve = true;
      }
    });
  } 

我們修改完進行測試 發(fā)現(xiàn)測試1 已經被刪除

對于前綴注釋

那么對于前綴注釋 我們應該怎么做呢 因為我們在后綴注釋的節(jié)點中添加了一個變量belongCurrentLine,表示該注釋是否是和節(jié)點屬于同一行。

那么對于前綴注釋,我們只需要判斷是否存在belongCurrentLine,如果存在belongCurrentLine,表示不能將其當作前綴注釋。

 if (hasTrailingComments(parentNode)) {
+    const { start:{ line: currentLine } }=parentNode.loc;
    //traverse
    // @ts-ignore
    parentNode.trailingComments.forEach((comment) => { 
+      const { start:{ line: currentCommentLine } }=comment.loc;
+      if(currentLine===currentCommentLine){
+        comment.belongCurrentLine=true;
+      }
+     //屬于當前行才將其設置為后綴注釋
-      if (isReserveComment(comment, commentWords))
+      if (isReserveComment(comment, commentWords) && comment.belongCurrentLine) {
        trailReserve = true;
      }
    });
  } 

發(fā)布到線上

我現(xiàn)已將代碼發(fā)布到線上

安裝

yarn add @parrotjs/babel-plugin-console

使用

舉個例子:新建.babelrc

{
    "plugins":[["../dist/index.cjs",{
        "env":"production"
    }]]
}

git地址

以上就是babel插件去除console示例詳解的詳細內容,更多關于babel插件去除console的資料請關注腳本之家其它相關文章!

相關文章

  • fabric.js實現(xiàn)diy明信片功能

    fabric.js實現(xiàn)diy明信片功能

    這篇文章主要為大家詳細介紹了fabric.js實現(xiàn)diy明信片功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-03-03
  • D3.js實現(xiàn)散點圖和氣泡圖的方法詳解

    D3.js實現(xiàn)散點圖和氣泡圖的方法詳解

    這篇文章將會給大家介紹了另外兩種可視化圖表,利用D3.js實現(xiàn)散點圖和氣泡圖,文章通過多個方面介紹的非常詳細,下面來一起看看吧。
    2016-09-09
  • C#程序員入門學習微信小程序的筆記

    C#程序員入門學習微信小程序的筆記

    這篇文章主要給大家分享了一位C#程序員入門學習微信小程序的筆記,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
    2019-03-03
  • 微信小程序實現(xiàn)點擊圖片旋轉180度并且彈出下拉列表

    微信小程序實現(xiàn)點擊圖片旋轉180度并且彈出下拉列表

    這篇文章主要為大家詳細介紹了微信小程序實現(xiàn)點擊圖片旋轉180度并且彈出下拉列表,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-11-11
  • 微信小程序外賣選購頁實現(xiàn)切換分類與數量加減功能案例

    微信小程序外賣選購頁實現(xiàn)切換分類與數量加減功能案例

    這篇文章主要介紹了微信小程序外賣選購頁實現(xiàn)切換分類與數量加減功能,結合具體實例形式分析了微信小程序狀態(tài)記錄、判定及數值運算相關操作技巧,需要的朋友可以參考下
    2019-01-01
  • 利用原生js和jQuery實現(xiàn)單選框的勾選和取消操作的方法

    利用原生js和jQuery實現(xiàn)單選框的勾選和取消操作的方法

    下面小編就為大家?guī)硪黄迷鷍s和jQuery實現(xiàn)單選框的勾選和取消操作的方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-09-09
  • 公眾號SVG動畫交互實戰(zhàn)代碼

    公眾號SVG動畫交互實戰(zhàn)代碼

    這篇文章主要介紹了公眾號SVG動畫交互實戰(zhàn)代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-05-05
  • 利用Echarts如何實現(xiàn)多段圓環(huán)圖

    利用Echarts如何實現(xiàn)多段圓環(huán)圖

    這篇文章主要給大家介紹了關于利用Echarts如何實現(xiàn)多段圓環(huán)圖的相關資料,文中通過實例代碼代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2022-03-03
  • javascript中獲取class的簡單實現(xiàn)

    javascript中獲取class的簡單實現(xiàn)

    下面小編就為大家?guī)硪黄猨avascript中獲取class的簡單實現(xiàn)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-07-07
  • JavaScript作用域、閉包、對象與原型鏈概念及用法實例總結

    JavaScript作用域、閉包、對象與原型鏈概念及用法實例總結

    這篇文章主要介紹了JavaScript作用域、閉包、對象與原型鏈,結合實例形式總結分析了javascript中變量與函數的作用域、閉包、對象、原形鏈相關概念、用法及注意事項,需要的朋友可以參考下
    2018-08-08

最新評論