前端架构师-week4-封装通用的npm包管理类Package

文章详细介绍了如何创建并管理npm模块,包括动态执行库exec模块的创建、Package类的属性和方法开发、模块的安装、更新和判断存在的功能。同时,文章讨论了如何处理不同操作系统的路径兼容问题以及利用npminstall库进行模块安装。此外,还涉及到了本地调试功能的实现,通过监听和环境变量来增强业务逻辑的解耦。
摘要由CSDN通过智能技术生成

目录

脚手架命令本地调试功能支持

动态执行库exec模块创建

创建 npm 模块通用类 Package

Package 类的属性、方法定义及构造函数逻辑开发

Package 类获取入口文件路径功能开发(pkg-dir应用+解决不同操作系统路径兼容问题) 

利用 npminstall 库安装 npm 模块

Package 类判断模块是否存在方法开发

Package 类更新模块逻辑开发

Package 类获取缓存模块入口文件功能改造

总结


脚手架命令本地调试功能支持

        环境变量是操作系统级别的,可以在Node多进程之间共享。做业务逻辑解耦。

        监听可以在业务逻辑执行之前执行。

function registerCommand() {
  program
    .name(Object.keys(pkg.bin)[0])
    .usage('<command> [options]')
    .version(pkg.version)
    .option('-d, --debug', '是否开启调试模式', false)
    .option('-tp, --targetPath <targetPath>', '是否指定本地调试文件路径', ''); //新增

  // 指定targetPath
  program.on('option:targetPath', function() {
    process.env.CLI_TARGET_PATH = program.targetPath;
  });

  program.parse(process.argv);
}


// imooc-cli-dev/commands/init/index.js 文件内容

'use strict'

function init(projectName, cmdObj) {
    console.log('init', projectName, cmdObj.force, process.env.CLI_TARGET_PATH)
}

module.exports = init

动态执行库exec模块创建

lerna create @imooc-cli-dev/exec

entry point: (lib/exec.js) lib/index.js #记得修改入口文件名
// 在 core/cli/package.json 中注册

"dependencies": {
    "@imooc-cli-dev/exec": "file:../exec",
    "colors": "^1.4.0"
}


// 在 core/cli 命令行中执行
npm link
// core/exec/index.js 文件内容

'use strict';

module.exports = exec;

function exec() {
    console.log('exec')
    console.log(process.env.CLI_TARGET_PATH)
    console.log(process.env.CLI_HOME_PATH)
}

创建 npm 模块通用类 Package

1. targetPath -> modulePath

2. modulePath -> Package ( npm 模块 )

3. Package.getRootFile(获取入口文件),无需把此步代码暴露在 exec 中

4. Package.update / Package.install

5. 封装 -> 复用

实现步骤:

lerna create @imooc-cli-dev/package

entry point: (lib/package.js) lib/index.js  #记得修改入口文件名

#把 package 移到 imooc-cli-dev/models 中
// 在 core/exec/package.json 中注册

"dependencies": {
    "@imooc-cli-dev/package": "file:../../models/package"
}


// 在 core/exec 命令行中执行
npm install
// models/package/lib/index.js 文件内容

'use strict'

class Package {
    constructor() {
        console.log('Package constructor')
    }
}

module.exports = Package
// core/exec/lib/index.js 中引用 package

const Package = require('@imooc-cli-dev/package')

function exec() {
    const pkg = new Package()
}

module.exports = exec

Package 类的属性、方法定义及构造函数逻辑开发

小知识点:

        lerna bootstrap 相当于 npm install 

        可变参数通过 arguments 来获取

         log.verbose() 调试时打印,生产环境不打印

        exec 引用 package,cli 模块也要进行 npm link

function isObject(o) {
  return Object.prototype.toString.call(o) === '[object Object]';
}
// models/package/lib/index.js 文件内容

'use strict';

const { isObject } = require('@imooc-cli-dev/utils');

class Package {
  constructor(options) {
    if (!options) {
      throw new Error('Package类的options参数不能为空!');
    }
    if (!isObject(options)) {
      throw new Error('Package类的options参数必须为对象!');
    }
    // package的目标路径
    this.targetPath = options.targetPath;
    // 缓存package的路径
    this.storeDir = options.storeDir;
    // package的name
    this.packageName = options.packageName;
    // package的version
    this.packageVersion = options.packageVersion;
  }

  // 判断当前Package是否存在
  async exists() {
  }

  // 安装Package
  async install() {
  }

  // 更新Package
  async update() {
  }

  // 获取入口文件的路径
  getRootFilePath() {
  }
}

module.exports = Package;
// core/exec/lib/index.js 文件内容

'use strict';

const Package = require('@imooc-cli-dev/package');
const log = require('@imooc-cli-dev/log');

const SETTINGS = {
  init: '@imooc-cli/init'
};


async function exec() {
  let targetPath = process.env.CLI_TARGET_PATH;
  const homePath = process.env.CLI_HOME_PATH;
  log.verbose('targetPath', targetPath);
  log.verbose('homePath', homePath);

  const cmdObj = arguments[arguments.length - 1];
  const cmdName = cmdObj.name();
  const packageName = SETTINGS[cmdName];
  const packageVersion = 'latest';

 
  pkg = new Package({
    targetPath,
    packageName,
    packageVersion,
  });
  console.log(pkg)
}

module.exports = exec;

Package 类获取入口文件路径功能开发(pkg-dir应用+解决不同操作系统路径兼容问题) 

思路: 

  获取入口文件的路径
  getRootFilePath() {

    1. 获取 package.json 所在目录 - pkg-dir
    2. 读取 package.json - require()
    3. 寻找main/lib - path
    4. 路径的兼容(macOS/windows)(macOS的path.sep路径分隔符为/,windows为\)
  }

步骤: 

# imooc-cli-dev/models/package 命令行中执行
npm install -S pkg-dir
const path = require('path');
const pkgDir = require('pkg-dir').sync;
const formatPath = require('@imooc-cli-dev/format-path');

// 获取入口文件的路径
  getRootFilePath() {
      // 1. 获取package.json所在目录
      const dir = pkgDir(targetPath);
      if (dir) {
        // 2. 读取package.json
        const pkgFile = require(path.resolve(dir, 'package.json'));
        // 3. 寻找main/lib
        if (pkgFile && pkgFile.main) {
          // 4. 路径的兼容(macOS/windows)
          return formatPath(path.resolve(dir, pkgFile.main));
        }
      }
      return null;
  }
# 在 imooc-cli-dev/utils 下新建包,修改入口文件名为 index.js
lerna create @imooc-cli-dev/format-path
// 在 models/package 中注册 utils/format-path
"dependencies": {
    "@imooc-cli-dev/format-path": "file:../../utils/format-path"
}

// 在 models/package 命令行中执行
npm install
// utils/format-path/lib/index.js 文件内容

'use strict';

const path = require('path');

module.exports = function formatPath(p) {
  if (p && typeof p === 'string') {
    const sep = path.sep;
    if (sep === '/') {
      return p;
    } else {
      return p.replace(/\\/g, '/');
    }
  }
  return p;
}

利用 npminstall 库安装 npm 模块

        这节实现 package 模块最重要的方法,安装依赖。

npminstall 库作用:

        Make npm install fast and handy.

// Usage:

const npminstall = require('npminstall');
const path = require('path');
const userHome = require('user-home');

npminstall({
  root: path.resolve(userHome, '.imooc-cli-dev'),
  storeDir: path.resolve(userHome, '.imooc-cli-dev', 'node_modules'),
  registry: 'https://registry.npmjs.org',
  pkgs: [  //要安装的包名
    { name: 'foo', version: '~1.0.0' },
  ],
});

// 结果:会在 user/.imooc-cli-dev/node_modules 中,安装 foo package

代码实现: 

npm install -S npminstall
// models/package/lib/index.js 文件内容

const npminstall = require('npminstall');
const { getDefaultRegistry, getNpmLatestVersion } = require('@imooc-cli-dev/get-npm-info');

class Package {

  // 安装Package
  async install() {

    return npminstall({
      root: this.targetPath,
      storeDir: this.storeDir,
      registry: getDefaultRegistry(),
      pkgs: [{
        name: this.packageName,
        version: this.packageVersion,
      }],
    });
  }
}
// models/package/package.json 文件中注册 get-npm-info
"dependencies": {
    "@imooc-cli-dev/get-npm-info": "file:../../utils/get-npm-info"
}

// models/package 命令行中执行
npm install
// core/exec/lib/index.js 文件内容

'use strict';

const path = require('path');
const Package = require('@imooc-cli-dev/package');
const log = require('@imooc-cli-dev/log');

const SETTINGS = {
  init: '@imooc-cli/init'
};

const CACHE_DIR = 'dependencies';

async function exec() {
  let targetPath = process.env.CLI_TARGET_PATH;
  const homePath = process.env.CLI_HOME_PATH;
  let storeDir = '';
  let pkg;
  log.verbose('targetPath', targetPath);
  log.verbose('homePath', homePath);

  const cmdObj = arguments[arguments.length - 1];
  const cmdName = cmdObj.name();
  const packageName = SETTINGS[cmdName];
  const packageVersion = 'latest';

  if (!targetPath) {
    targetPath = path.resolve(homePath, CACHE_DIR); // 生成缓存路径
    storeDir = path.resolve(targetPath, 'node_modules');
    log.verbose('targetPath', targetPath);
    log.verbose('storeDir', storeDir);
    pkg = new Package({
      targetPath,
      storeDir,
      packageName,
      packageVersion,
    });
    if (await pkg.exists()) {
      // 更新package
    } else {
      // 安装package
      await pkg.install();
    }
  } else {
    pkg = new Package({
      targetPath,
      packageName,
      packageVersion,
    });
  }
  const rootFile = pkg.getRootFilePath();
  if (rootFile) {
    require(rootFile).call(null, arguments});
  }
}

module.exports = exec;

Package 类判断模块是否存在方法开发

npm install path-exists -S
// models/package/lib/index.js 文件内容

const pathExists = require('path-exists').sync;
const { getDefaultRegistry, getNpmLatestVersion } = require('@imooc-cli-dev/get-npm-info');

class Package {
  constructor(options) {
    ......
    // package的缓存目录前缀
    this.cacheFilePathPrefix = this.packageName.replace('/', '_');
  }

  //安装时可以安装 latest,查询路径时还是要具体到版本号的。 
  //_@imooc-cli_init@1.1.2@@imooc-cli/ 磁盘上文件名格式
  //@imooc-cli/init 1.1.2
  get cacheFilePath() {
    return path.resolve(this.storeDir, `_${this.cacheFilePathPrefix}@${this.packageVersion}@${this.packageName}`);
  }

  async prepare() {
    if (this.packageVersion === 'latest') {
      this.packageVersion = await getNpmLatestVersion(this.packageName);
    }
  }

  // 判断当前Package是否存在
  async exists() {
    if (this.storeDir) {
      await this.prepare();
      return pathExists(this.cacheFilePath);
    } else {
      return pathExists(this.targetPath);
    }
  }

  // 安装Package
  async install() {
    await this.prepare();
    return npminstall({
      ......
    });
  }
}
// utils/get-npm-info/lib/index.js 文件内容

async function getNpmLatestVersion(npmName, registry) {
  let versions = await getNpmVersions(npmName, registry);
  if (versions) {
    return versions.sort((a, b) => semver.gt(b, a))[0];
  }
  return null;
}

Package 类更新模块逻辑开发

fs-extra 库作用: 

        fs-extra adds file system methods that aren't included in the native fs module and   adds promise support to the fs methods. It also uses graceful-fs to prevent EMFILE errors.     It should be a drop in replacement for fs.

fs-extra methods:

Async
copy emptyDir ensureFile ensureDir ensureLink ensureSymlink mkdirp
mkdirs move outputFile outputJson pathExists readJson remove writeJson 
Sync
copySync emptyDirSync ensureFileSync ensureDirSync ensureLinkSync
ensureSymlinkSync mkdirpSync mkdirsSync moveSync outputFileSync
outputJsonSync pathExistsSync readJsonSync removeSync writeJsonSync

fs-extra Usage: 

const fs = require('fs-extra')

// Async with promises:
fs.copy('/tmp/myfile', '/tmp/mynewfile')
  .then(() => console.log('success!'))
  .catch(err => console.error(err))

// Async with callbacks:
fs.copy('/tmp/myfile', '/tmp/mynewfile', err => {
  if (err) return console.error(err)
  console.log('success!')
})

// Sync:
try {
  fs.copySync('/tmp/myfile', '/tmp/mynewfile')
  console.log('success!')
} catch (err) {
  console.error(err)
}

// Async/Await:
async function copyFiles () {
  try {
    await fs.copy('/tmp/myfile', '/tmp/mynewfile')
    console.log('success!')
  } catch (err) {
    console.error(err)
  }
}

步骤和实现代码: 

const fse = require('fs-extra'); //先 npm install -S

class Package {

  ......

  async prepare() {
    if (this.storeDir && !pathExists(this.storeDir)) {   //新增
      fse.mkdirpSync(this.storeDir);    //把没有创建好的目录都给创建好
    }
    if (this.packageVersion === 'latest') {
      this.packageVersion = await getNpmLatestVersion(this.packageName);
    }
  }

  getSpecificCacheFilePath(packageVersion) {
    return path.resolve(this.storeDir, `_${this.cacheFilePathPrefix}@${packageVersion}@${this.packageName}`);
  }
  
  // 更新Package
  async update() {
    await this.prepare();
    // 1. 获取最新的npm模块版本号
    const latestPackageVersion = await getNpmLatestVersion(this.packageName);
    // 2. 查询最新版本号对应的路径是否存在
    const latestFilePath = this.getSpecificCacheFilePath(latestPackageVersion);
    // 3. 如果不存在,则直接安装最新版本
    if (!pathExists(latestFilePath)) {
      await npminstall({
        root: this.targetPath,
        storeDir: this.storeDir,
        registry: getDefaultRegistry(),
        pkgs: [{
          name: this.packageName,
          version: latestPackageVersion,
        }],
      });
      this.packageVersion = latestPackageVersion; //健壮性体现,注意
    } else {
      this.packageVersion = latestPackageVersion; //健壮性体现,注意
    }
  }
}

Package 类获取缓存模块入口文件功能改造

class Package {

  // 获取入口文件的路径
  getRootFilePath() {
    function _getRootFile(targetPath) {
      // 1. 获取package.json所在目录
      const dir = pkgDir(targetPath);
      if (dir) {
        // 2. 读取package.json
        const pkgFile = require(path.resolve(dir, 'package.json'));
        // 3. 寻找main/lib
        if (pkgFile && pkgFile.main) {
          // 4. 路径的兼容(macOS/windows)
          return formatPath(path.resolve(dir, pkgFile.main));
        }
      }
      return null;
    }
    if (this.storeDir) {    //新增判断项,命令行中未传入targetPath
      return _getRootFile(this.cacheFilePath);
    } else {
      return _getRootFile(this.targetPath);
    }
  }
}

总结

        我们已经把 package 类封装好了。代表一个npm模块。支持直接使用本地已有的代码文件,和缓存npm库里面的文件两种功能。实现了 npm 模块安装、更新、判断存在、获取入口文件路径等一系列逻辑。对class也有比较多的应用,构造函数、get、方法。逻辑还是比较复杂的。 

        此章我们共同完成了 imooc-cli 脚手架动态命令加载 的功能,这块功能的难度还是相当大的。我们实现了一整套动态加载的机制,我们将动态命令抽象为一个package,既可以通过缓存去实时从 npm 下载这个命令,也可以直接从本地去映射这个命令。核心是获取一个入口文件地址,将这个入口文件直接以 require 方式进行调用。我们还差一步,就是将 require 方式调用改为 node 子进程方式调用。可以额外获得更多的 CPU 资源,以便获得更高的执行性能。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

chengbo_eva

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值