分析web应用内引用依赖的占比

背景

针对目前团队自己开发的组件库,对当前系统内引用组件库占比进行统计分析,以实现对当前进度的总结以及后续的覆盖度目标制定。

主要思路

目前找到的webpack分析插件,基本都是针对打包之后的分析打包之后的chunk进行分析,但是我希望的是分析每个页面中的import数,对比一下在所有页面中的import数中有多少是使用了组件库的。所以就在网上看了一些相关资料以及webpackapi文档。主要是利用webpackimportCallimportimportSpecifier三个钩子来实现,它们的作用直接跟着代码看一下。

完整代码实现

import fs from 'fs';
import path from 'path';
import resolve from 'enhanced-resolve';

let myResolve;

/**
 * 通过source获取真实文件路径
 * @param parser
 * @param source
 */
function getResource(parser, source) {
  if (!myResolve) {
    myResolve = resolve.create.sync(parser.state.options.resolve);
  }
  let result = '';
  try {
    result = myResolve(parser.state.current.context, source);
  } catch (err) {
    console.log(err);
  } finally {
    return result;
  }
}

class WebpackImportAnalysisPlugin {
  constructor(props) {
    this.pluginName = 'WebpackCodeDependenciesAnalysisPlugin';
    //  文件数组
    this.files = [];
    //  当前编译的文件
    this.currentFile = null;
    this.output = props.output;
  }

  apply(compiler) {
    compiler.hooks.compilation.tap(this.pluginName, (compilation, { normalModuleFactory }) => {
      const collectFile = parser => {
        const { rawRequest, resource } = parser.state.current;
        if (resource !== this.currentFile) {
          this.currentFile = resource;
          this.files.push({
            name: rawRequest,
            resource,
            children: []
          });
        }
      };
      const handler = parser => {
        // 用来捕获import(xxx)
        parser.hooks.importCall.tap(this.pluginName, expr => {
          collectFile(parser);
          let ast = {};
          const isWebpack5 = 'webpack' in compiler;
          // webpack@5 has webpack property, webpack@4 don't have the property
          if (isWebpack5) {
            // webpack@5
            ast = expr.source;
          } else {
            //webpack@4
            const { arguments: arg } = expr;
            ast = arg[0];
          }
          const { type, value } = ast;
          if (type === 'Literal') {
            const resource = getResource(parser, value);
            this.files[this.files.length - 1].children.push({
              name: value,
              resource,
              importStr: `import ('${value}')`
            });
          }
        });
        // 用来捕获 import './xxx.xx';
        parser.hooks.import.tap(this.pluginName, (statement, source) => {
          // 由于statement.specifiers.length大于0的时候同时会被importSpecifier钩子捕获,所以需要在这个地方拦截一下,这个地方只处理单独的引入。
          if (statement.specifiers.length > 0) {
            return;
          }
          collectFile(parser);
          this.files[this.files.length - 1].children.push({
            name: source,
            resource: getResource(parser, source),
            importStr: `import '${source}'`
          });
        });
        // 用来捕获 import xx from './xxx.xx';
        parser.hooks.importSpecifier.tap(
          this.pluginName,
          (statement, source, exportName, identifierName) => {
            collectFile(parser);
            let importStr = '';
            if (exportName === 'default') {
              importStr = `import ${identifierName} from '${source}'`;
            } else {
              if (exportName === identifierName) {
                importStr = `import { ${identifierName} } from '${source}'`;
              } else {
                importStr = `import { ${exportName}: ${identifierName} } from '${source}'`;
              }
            }
            this.files[this.files.length - 1].children.push({
              name: source,
              exportName,
              identifierName,
              importStr,
              resource: getResource(parser, source)
            });
          }
        );
      };

      normalModuleFactory.hooks.parser.for('javascript/auto').tap(this.pluginName, handler);
    });

    compiler.hooks.make.tap(this.pluginName, compilation => {
      compilation.hooks.finishModules.tap(this.pluginName, modules => {
        // 过滤掉深度遍历的node_modules中的文件,只分析业务代码中的文件
        const needFiles = this.files.filter(
          item => !item.resource.includes('node_modules') && !item.name.includes('node_modules')
        );
        fs.writeFile(this.output ?? path.resolve(__dirname, 'output.json'), JSOn.stringify(needFiles, null, 4), err => {
          if (!err) {
            console.log(`${path.resolve(__dirname, 'output.json')}写入完成`);
          }
        });
      });
    });
  }
}

export default WebpackImportAnalysisPlugin;
// 以文件为基准,扁平化输出所有的import
[
  {
    "name": "./src/routes",
    "resource": "/src/routes.tsx",
    "children": [
      {
        "name":"react",
        "exportName":"lazy",
        "identifierName":"lazy",
        "importStr":"import { lazy } from 'react'",
        "resource":"/node_modules/.pnpm/react@17.0.2/node_modules/react/index.js"
      },
    ...  
    ]
  },
  ...
]

后续

上面拿到的数据是扁平化的数据,如果针对需要去分析整体的树状结构,可以直接将扁平化数据处理一下,定义一个主入口去寻找它的子级,这样可以自己去生成一颗树状的import关系图。

原文链接:https://www.cnblogs.com/aloneMing/p/17316100.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:分析web应用内引用依赖的占比 - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • Angular实现的table表格排序功能完整示例

    让我为你详细讲解“Angular实现的table表格排序功能完整示例”的完整攻略。 什么是Angular实现的table表格排序功能 在Angular中,我们可以通过使用ngFor指令循环渲染table表格中的数据,并在表格头部添加按钮进行排序,达到对表格数据排序的目的。这种方法可以在应用中节省代码量,并提高数据可读性。 如何实现Angular实现的tabl…

    JavaScript 2023年6月10日
    00
  • 探讨js字符串数组拼接的性能问题

    探讨JS字符串数组拼接的性能问题 在开发中,我们经常需要对字符串进行拼接操作,特别是基于HTML标签的文本拼接,因此对于拼接操作的性能问题需谨慎对待,当操作次数较小时,性能影响可忽略,但当操作次数较多时,性能问题将显著影响代码的执行速度。本文将着重分析字符串数组的拼接性能问题,并提供一些优化解决方案。 字符串数组拼接(Array.prototype.join…

    JavaScript 2023年5月28日
    00
  • javascript实现将数字转成千分位的方法小结【5种方式】

    下面是讲解“JavaScript实现将数字转成千分位的方法小结【5种方式】”的完整攻略。 什么是千分位? 千分位是指将数字每隔三位加一个逗号表示的形式,比如:“1,234,567”。 为什么要使用千分位? 使用千分位可以使数字更加易读,尤其是对于大的数字更加方便观察。 实现方式 以下是五种JavaScript实现将数字转成千分位的方法: 方法一:toFixe…

    JavaScript 2023年5月28日
    00
  • 无语,javascript居然支持中文(unicode)编程!

    当我看到 “无语,JavaScript居然支持中文(Unicode)编程!” 这句话时,我相信说的是JavaScript支持使用Unicode字符作为标识符。这意味着您可以在JavaScript编程时使用中文或其他unicode字符,这对特定项目或程序员可能很有用。 下面是使用JavaScript中文(Unicode)标识符的完整攻略。 使用Unicode字…

    JavaScript 2023年5月19日
    00
  • js实现三角形粒子运动

    当我们需要实现三角形形式的粒子运动效果时,可以使用JavaScript来实现。下面是实现的完整攻略。 步骤一:准备工作 首先要准备好基本的HTML和CSS代码,用来在页面上展示三角形和粒子运动效果。 其中HTML需要包含一个canvas元素,用来在页面上绘制三角形和粒子,代码如下: <canvas id="canvas">&l…

    JavaScript 2023年6月11日
    00
  • JS实现n秒后自动跳转的两种方法

    下面我将针对“JS实现n秒后自动跳转的两种方法”进行详细讲解。 方法一:使用setTimeout()方法 我们可以使用JS的setTimeout()方法来实现n秒后自动跳转,具体操作步骤如下: 在页面中添加JS代码,定义计时器,并使用setTimeout()方法来实现需要跳转的URL地址。 <script> // 设置跳转的URL地址 var t…

    JavaScript 2023年5月27日
    00
  • 在vue项目中利用popstate处理页面返回的操作介绍

    在Vue项目中,可以利用 popstate 事件来处理页面返回的操作。下面详细介绍利用 popstate 的具体步骤。 1. 理解popstate事件 popstate 事件是 HTML5 History API 的一部分,可以在浏览器的后退或前进按钮被点击时进行传递。当浏览器历史发生变化时, popstate 事件将被触发。 2. 注册popstate事件…

    JavaScript 2023年6月11日
    00
  • 简化版的vue-router实现思路详解

    简化版的vue-router实现思路详解 前言 Vue.js 是一个非常流行的前端框架,其专注于视图层的渲染。而 Vue-router 是 Vue.js 的一个关键插件,它管理着 Vue.js 应用程序中的路由,可以帮助我们更好地管理前端路由。在本篇文章中,我将为大家介绍一个简化版的 Vue-router 实现思路。 设计思路 Vue-router 的设计思…

    JavaScript 2023年6月11日
    00
合作推广
合作推广
分享本页
返回顶部