最佳实践初始化项目公用cli

package.json入口实现

js 复制代码
{
  "name": "xxx-cli",
  "version": "1.0.0",
  "description": "Command line interface for rapid project development",
  "bin": {
    "xxx-cli": "bin/cli.mjs"
  },
  "repository": {
    "type": "git",
    "url": "git+https://git地址"
  },
  "keywords": [
    "cli"
  ],
  "author": "",
  "license": "MIT",
  "bugs": {
    "url": "https://git地址/issues"
  },
  "engines": {
    "node": ">= 14.18.0"
  },
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "dependencies": {
    "chalk": "^5.0.1",
    "child_process": "^1.0.2",
    "commander": "^9.4.0",
    "compressing": "^1.7.0",
    "figlet": "^1.5.2",
    "fs-extra": "^10.1.0",
    "inquirer": "^9.1.0",
    "ora": "^6.1.2",
    "semver": "^7.3.7",
    "shelljs": "^0.8.5"
  }
}

bin/cli.mjs内部实现

js 复制代码
#!/usr/bin/env node
// `#!/usr/bin/env node`:指定使用 Node.js 执行此脚本。

// 导入需要的模块
import { readFile } from 'fs/promises'; // 异步文件读取
import chalk from 'chalk'; // 控制台文本样式
import semverSatisfies from 'semver/functions/satisfies.js'; // 版本比较
import { program } from 'commander'; // 命令行参数解析
import create from '../lib/create.mjs'; // 创建项目命令

const CLI_NAME = 'xxx-cli';

// 读取 package.json 获取 Node 版本要求和 CLI 版本
const { engines: { node: requiredVersion }, version } = JSON.parse(
    await readFile(
        new URL('../package.json', import.meta.url)
    )
);

// 检查 Node 版本是否满足要求
function checkNodeVersion(wanted, id) {
    if (!semverSatisfies(process.version, wanted, { includePrerelease: true })) {
        console.log(chalk.red(
            'You are using Node ' + process.version + ', but this version of ' + id +
            ' requires Node ' + wanted + '.\nPlease upgrade your Node version.'
        ));
        process.exit(1); // 退出程序
    }
}

// 执行 Node 版本检查
checkNodeVersion(requiredVersion, CLI_NAME);

// 初始化 commander 程序
program
    .name(CLI_NAME) // CLI 工具名称
    .version(version); // CLI 工具版本

// 定义 create 命令
program
    .command('create <app-name>') // 命令名称和参数
    .description('create a new project powered by xxx-cli') // 命令描述
    .option("-f, --force", "overwrite target directory if it exists") // 命令选项
    .action((name, options) => { // 命令执行函数
        console.log(chalk.cyan(`Creating project: ${name}`));
        create(name, options); // 调用 create 函数
    });

// 解析命令行参数
program.parse(process.argv);

lib/create.mjs内部实现

js 复制代码
import ora from 'ora';
import chalk from 'chalk'
import fse from 'fs-extra'
import shelljs from 'shelljs'
import path from 'path'
import inquirer from 'inquirer'

const PROJECT_TYPES = [ 
  { name: 'pc', value: 'pc' }, 
  { name: 'h5', value: 'h5' }, 
]; 
const PROJECT_MAP = { 
  pc: 'pc最佳实践git地址', 
  h5: 'h5最佳实践git地址', 
};

async function create(projectName, options) {
  const progress = ora('创建中...')
  const targetDirectory = path.join(process.cwd(), projectName)
  try {
    // 判断是否存在git,不存在则提示并退出
    if (!shelljs.which('git')) {
      console(chalk.red('Sorry, xxx-cli requires git'));
      return
    }
    const isExist = await fse.pathExists(targetDirectory)
    // 判断目录下是否存在同名文件夹
    if (isExist) {
      // 如果有 --force 选项,则删除已存在的目录
      if (options.force) {
        await fse.remove(targetDirectory);
      } else {
        const { isOverwrite } = await new inquirer.prompt([
          {
            name: "isOverwrite", // 与返回值对应
            type: "list",
            message: "目标目录已存在,请选择操作:",
            choices: [
              { name: "Overwrite", value: true },
              { name: "Cancel", value: false },
            ],
          },
        ]);
        // 移除同名文件夹
        if (isOverwrite) {
          console.log('remove existing directory...')
          await fse.remove(targetDirectory);
        } else {
          return;
        }
      }
    }
    // 项目类型
    const { projectType } = await new inquirer.prompt([
      {
        name: "projectType", // 与返回值对应
        type: "list",
        message: "Please select project type:",
        choices: PROJECT_TYPES,
      },
    ]);

    // 安装依赖项目
    shelljs.exec(`git clone ${PROJECT_MAP[projectType]} ${projectName}`, async (code, stdout, stderr) => {
      if (code === 0) {
        progress.start()
        try {
          // 删除原有.git
          await fse.remove(path.join(process.cwd(), projectName, '.git'))
        } catch (error) {
          console.log(error)
        }
        progress.succeed()
        console.log(`\r\nSuccessfully created project ${chalk.cyan(projectName)}`);
        console.log(`\r\n  cd ${chalk.cyan(projectName)}`);
        console.log("  git init");
        console.log("  pnpm install");
          console.log("  pnpm dev");
      }
    })
  } catch (error) {
    progress.fail(chalk.red('创建项目失败:', error));
    console.log(error);
  }
}

export default create
相关推荐
庸俗今天不摸鱼34 分钟前
【万字总结】前端全方位性能优化指南(十)——自适应优化系统、遗传算法调参、Service Worker智能降级方案
前端·性能优化·webassembly
黄毛火烧雪下41 分钟前
React Context API 用于在组件树中共享全局状态
前端·javascript·react.js
Apifox1 小时前
如何在 Apifox 中通过 CLI 运行包含云端数据库连接配置的测试场景
前端·后端·程序员
一张假钞1 小时前
Firefox默认在新标签页打开收藏栏链接
前端·firefox
高达可以过山车不行1 小时前
Firefox账号同步书签不一致(火狐浏览器书签同步不一致)
前端·firefox
m0_593758101 小时前
firefox 136.0.4版本离线安装MarkDown插件
前端·firefox
掘金一周1 小时前
金石焕新程 >> 瓜分万元现金大奖征文活动即将回归 | 掘金一周 4.3
前端·人工智能·后端
三翼鸟数字化技术团队1 小时前
Vue自定义指令最佳实践教程
前端·vue.js
Jasmin Tin Wei2 小时前
蓝桥杯 web 学海无涯(axios、ecahrts)版本二
前端·蓝桥杯
圈圈编码2 小时前
Spring Task 定时任务
java·前端·spring