惠善一的博客:http://huishanyi.club/

通过命令行工具,初始化团队项目,并生成团队规范代码,一键创建项目,一键生成代码,一键生成功能模块···

解放双手,从 cli 开始, JSer 永不为奴!

创建项目目录

两种方法:

  1. 鼠标右键创建文件夹,文件夹名称即为项目名称;

  2. 打开系统命令面板(powershell、cmd···),输入命令:

    • mkdir 项目名称

初始化 node 项目

在项目文件目录运行命令:

npm init

运行命令后会出现如下提示(按照提示完成即可):

根据提示完成操作后,会在项目根目录生成项目描述文件package.json,可以回车跳过以下步骤,后续在package.json文件中完善项目信息。

This utility will walk you through creating a package.json file.
It only covers the most common items, and tries to guess sensible defaults. See `npm help init` for definitive documentation on these fields
and exactly what they do. Use `npm install <pkg>` afterwards to install a package and
save it as a dependency in the package.json file. Press ^C at any time to quit.
package name: 项目名称
version: 版本号
description: 项目描述
entry point: (index.js)入口
test command: 测试相关
git repository: git仓库地址
keywords: 关键词
author: 作者
license: 证书
About to write to E:\***\package.json: {
"name": "***",
"version": "***",
"description": "***",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "***",
"license": "ISC"
} Is this OK? (yes) yes

安装相关依赖

项目后续会使用以下依赖包。

运行命令:

npm i commander download-git-repo ora handlebars figlet clear chalk open

依赖包介绍:

  • commander(完整的 node.js 命令行解决方案)
  • download-git-repo(从节点下载并提取仓库(GitHub, GitLab, Bitbucket))
  • ora(终端进度条)
  • handlebars(轻量的语义化模板)
  • figlet(终端实现图形字体)
  • clear(终端清屏)
  • chalk(终端字符串样式)
  • open(跨平台级的,可打开 url、文件、可执行脚本)

创建 cli 入口文件

项目根目录创建文件夹bin,在bin文件夹中创建mycli.js(自定义)文件。

文件内容如下:

#!/usr/bin/env node
// 第一行为约定脚本解释器以 node 进行解析,必须放在文件第一行!行首不能有任何字符!

定义执行命令

打开package.json文件,添加相关代码(句首有+号的代码为新增代码):

{
"name": "***",
"version": "***",
"description": "***",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
+ "bin": {
+ "mycli": "./bin/mycli.js"
+ },
"author": "***",
"license": "ISC",
"dependencies": {
"chalk": "^4.1.0",
"clear": "^0.1.0",
"commander": "^6.2.0",
"download-git-repo": "^3.0.2",
"figlet": "^1.5.0",
"handlebars": "^4.7.6",
"open": "^7.3.0",
"ora": "^5.1.0"
}
}

映射命令至系统

package.json文件中刚添加的bin命令映射至系统中。

运行命令:

npm link

此时我们可以全局使用bin中的mycli命令了。

npm link机理:由于已经安装好 node 环境,故此我们可以直接全局使用nodenpm命令,而npm link则是将package.jsonbin下的命令通过npm系统环境变量桥接,所以我们可以全局使用bin中的命令。

至此我们实现了 cli 的第一步:可以全局使用我们自定义的命令了~~~


莫要问我为什么在这里用了彩色的字。。。


因为我比较————


骚!

定制命令行界面

在入口文件bin => mycli.js中进行命令行界面定制。

#!/usr/bin/env node
const program = require('commander') // 引入 commander program.version(require('../package.json').version) // 声明 cli 当前版本,如:输入mycli -V 命令,即可显示当前 cli 版本 program
.command('init <name>') // 定义使用命令,如:命令 mycli init <name>
.description('init project') // 命令描述
.action(name => { // 该命令对应的动作
console.log('init--------'+name);
}) program.parse(process.argv) // 将命令参数暴露出去,供使用者使用

定制初始化功能

在根目录中创建lib文件夹,在lib文件夹中创建init.js文件,用于编写初始化功能代码。

基础结构搭建

初始化功能基础代码结构。

lib => init.js文件代码如下:

const { promisify } = require('util')		// node 核心模块 util,其中的 promisify 方法可以将异步回调方法改造成返回 Promise 实例的方法
const figlet = promisify(require('figlet')) // 以 promise 形式引入 figlet 模块,figlet 可以式代码以图形出现在终端
const clear = require('clear') // 终端清屏
const chalk = require('chalk'); // 终端字符串样式定制
const colorLog = (content, color) => console.log(chalk[color||'green'](content));// 封装日志输出,可定制颜色 module.exports = async name => {
clear();
const data = await figlet('Gisuni Welcome');
colorLog(data,'red')
}

连接功能与命令

bin => mycli.jsinit方法进行引入并使用,代码如下:

#!/usr/bin/env node
const program = require('commander') // 引入 commander program.version(require('../package.json').version) // 声明 cli 当前版本,如:输入mycli -V 命令,即可显示当前 cli 版本 program
.command('init <name>') // 定义使用命令,如:命令 mycli init <name>
.description('init project') // 命令描述
- .action(name => { // 该命令对应的动作
- console.log('init--------'+name);
- })
+ .action(require('../lib/init')) program.parse(process.argv) // 将命令参数暴露出去,供使用者使用

下拉仓库功能

将仓库中的代码下拉至本地。

lib文件夹中创建download.js文件,其代码如下:

const { promisify } = require('util')	// node 核心模块 util,其中的 promisify 方法可以将异步回调方法改造成返回 Promise 实例的方法

module.exports.clone = async (repo, desc) => {
const download = promisify(require('download-git-repo')) // 以 promise 形式引入 download-git-repo 模块,download-git-repo 可以下拉仓库代码
const ora = require('ora') // 引入终端进度条代码
const process = ora(`下载中··· ${repo}`) // 进度条提示 process.start() // 进度条开始
await download(repo, desc) // 下拉仓库代码
process.succeed() // 进度条结束
}

引入下拉功能

将下拉功能引入至初始化模块中。

lib => init.js文件代码如下:

const { promisify } = require('util')		// node 核心模块 util,其中的 promisify 方法可以将异步回调方法改造成返回 Promise 实例的方法
const figlet = promisify(require('figlet')) // 以 promise 形式引入 figlet 模块,figlet 可以式代码以图形出现在终端
const clear = require('clear') // 终端清屏
const chalk = require('chalk') // 终端字符串样式定制
const colorLog = (content, color) => console.log(chalk[color||'green'](content)) // 封装日志输出,可定制颜色
+const { clone } = require('./download') // 引入下拉功能模块 module.exports = async name => {
clear();
const data = await figlet('Gisuni Welcome');
colorLog(data,'red');
+ colorLog(`创建项目:${name}`);
+ await clone('github:用户名/项目名称', name); // 如果项目克隆地址为:https://github.com/ShanYi-Hui/react-template.git,则此处应改造为:github:ShanYi-Hui/react-template
}

自动安装依赖

下拉的项目将自动安装依赖包。

lib => init.js文件代码如下:

const { promisify } = require('util')		// node 核心模块 util,其中的 promisify 方法可以将异步回调方法改造成返回 Promise 实例的方法
const figlet = promisify(require('figlet')) // 以 promise 形式引入 figlet 模块,figlet 可以式代码以图形出现在终端
const clear = require('clear') // 终端清屏
const chalk = require('chalk') // 终端字符串样式定制
const colorLog = (content, color) => console.log(chalk[color||'green'](content)) // 封装日志输出,可定制颜色
const { clone } = require('./download') // 引入下拉功能模块 +const spawn = async (...args) => {
+ const { spawn } = require('child_process') // node 核心模块 child_process,spawn 方法会异步地衍生子进程,且不阻塞 Node.js 事件循环
+ return new Promise(resolve => {
+ const proc = spawn(...args);
+ proc.stdout.pipe(process.stdout);
+ proc.stderr.pipe(process.stderr);
+ proc.on('close', () => {
+ resolve();
+ })
+ })
+} module.exports = async name => {
clear();
const data = await figlet('Gisuni Welcome');
colorLog(data,'red');
colorLog(`创建项目:${name}`);
await clone('github:用户名/项目名称', name); // 如果项目克隆地址为:https://github.com/ShanYi-Hui/react-template.git,则此处应改造为:github:ShanYi-Hui/react-template
+ colorLog('安装依赖中···');
+ await spawn(process.platform === 'win32'?'npm.cmd':'npm', ['install'], { cwd:`./${name}` }); // 安装依赖
+ colorLog(`
+安装完成!
+
+项目将自动启动,若未启动请执行如下命令:
+==============================
+ cd ${name}
+ npm run serve
+==============================
+ `);
+}

项目自启动

依赖安装完成后将自动打开浏览器,并在指定端口(默认 8080 端口)运行服务。

lib => init.js文件代码如下:

const { promisify } = require('util')		// node 核心模块 util,其中的 promisify 方法可以将异步回调方法改造成返回 Promise 实例的方法
const figlet = promisify(require('figlet')) // 以 promise 形式引入 figlet 模块,figlet 可以式代码以图形出现在终端
const clear = require('clear') // 终端清屏
const chalk = require('chalk') // 终端字符串样式定制
const colorLog = (content, color) => console.log(chalk[color||'green'](content)) // 封装日志输出,可定制颜色
const { clone } = require('./download') // 引入下拉功能模块
+const open = require('open') // 打开链接 const spawn = async (...args) => {
const { spawn } = require('child_process') // node 核心模块 child_process,spawn 方法会异步地衍生子进程,且不阻塞 Node.js 事件循环
return new Promise(resolve => {
const proc = spawn(...args);
proc.stdout.pipe(process.stdout); // 子进程中的日志通过管道传输至主进程
proc.stderr.pipe(process.stderr); // 子进程中的错误捕获通过管道输出至主进程
proc.on('close', () => { // 关闭子进程
resolve(); // promise 状态切换至执行完成状态
})
})
} module.exports = async name => {
clear();
const data = await figlet('Gisuni Welcome');
colorLog(data,'red');
colorLog(`创建项目:${name}`);
await clone('github:用户名/项目名称', name); // 如果项目克隆地址为:https://github.com/ShanYi-Hui/react-template.git,则此处应改造为:github:ShanYi-Hui/react-template
colorLog('安装依赖中···');
await spawn(process.platform === 'win32'?'npm.cmd':'npm', ['install'], { cwd:`./${name}` }); // 安装依赖
colorLog(`
安装完成! 项目将自动启动,若未启动请执行如下命令:
==============================
cd ${name}
npm run serve
==============================
`);
+ open('http://localhost:8080'); // 打开指定端口
+ await spawn(process.platform === 'win32'?'npm.cmd':'npm', ['run','serve'], { cwd:`./${name}` }); // 启动项目
}

其实到这里,一个最基本的 cli 已经制作完成了~~~


那么接下来将开始丰富它的功能!

更新中···

手写cli的更多相关文章

  1. 【Win 10 应用开发】手写识别

    记得前面(忘了是哪天写的,反正是前些天,请用力点击这里观看)老周讲了一个14393新增的控件,可以很轻松地结合InkCanvas来完成涂鸦.其实,InkCanvas除了涂鸦外,另一个大用途是墨迹识别, ...

  2. JS / Egret 单笔手写识别、手势识别

    UnistrokeRecognizer 单笔手写识别.手势识别 UnistrokeRecognizer : https://github.com/RichLiu1023/UnistrokeRecogn ...

  3. 如何用卷积神经网络CNN识别手写数字集?

    前几天用CNN识别手写数字集,后来看到kaggle上有一个比赛是识别手写数字集的,已经进行了一年多了,目前有1179个有效提交,最高的是100%,我做了一下,用keras做的,一开始用最简单的MLP, ...

  4. 【转】机器学习教程 十四-利用tensorflow做手写数字识别

    模式识别领域应用机器学习的场景非常多,手写识别就是其中一种,最简单的数字识别是一个多类分类问题,我们借这个多类分类问题来介绍一下google最新开源的tensorflow框架,后面深度学习的内容都会基 ...

  5. caffe_手写数字识别Lenet模型理解

    这两天看了Lenet的模型理解,很简单的手写数字CNN网络,90年代美国用它来识别钞票,准确率还是很高的,所以它也是一个很经典的模型.而且学习这个模型也有助于我们理解更大的网络比如Imagenet等等 ...

  6. 使用神经网络来识别手写数字【译】(三)- 用Python代码实现

    实现我们分类数字的网络 好,让我们使用随机梯度下降和 MNIST训练数据来写一个程序来学习怎样识别手写数字. 我们用Python (2.7) 来实现.只有 74 行代码!我们需要的第一个东西是 MNI ...

  7. 手写原生ajax

    关于手写原生ajax重要不重要,各位道友自己揣摩吧, 本着学习才能进步,分享大家共同受益,自己也在自己博客里写一下 function createXMLHTTPRequest() { //1.创建XM ...

  8. springmvc 动态代理 JDK实现与模拟JDK纯手写实现。

    首先明白 动态代理和静态代理的区别: 静态代理:①持有被代理类的引用  ② 代理类一开始就被加载到内存中了(非常重要) 动态代理:JDK中的动态代理中的代理类是动态生成的.并且生成的动态代理类为$Pr ...

  9. 为sproto手写了一个python parser

    这是sproto系列文章的第三篇,可以参考前面的<为sproto添加python绑定>.<为python-sproto添加map支持>. sproto是云风设计的序列化协议,用 ...

随机推荐

  1. Beyond Compare 3, 简体中文版 安装

    转载: 1.https://www.scootersoftware.com/download.php 2.http://www.scootersoftware.com/download.php 下载地 ...

  2. matlab中strcmpi比较字符串(不区分大小写)

    来源:https://ww2.mathworks.cn/help/matlab/ref/strcmpi.html?searchHighlight=strcmpi&s_tid=doc_srcht ...

  3. 用于ASP.net的MVC模块

    下载MVCBricks_SRC - 492.58 KB 表的内容 介绍系统要求游戏闪屏的最终考虑历史 介绍 自从我写上一篇关于ASP的文章以来,已经有很长时间了.净的话题.这次我决定写一些关于它的东西 ...

  4. day64:nginx模块之限制连接&状态监控&Location/用nginx+php跑项目/扩展应用节点

    目录 1.nginx模块:限制连接 limit_conn 2.nginx模块:状态监控 stub_status 3.nginx模块:Location 4.用nginx+php跑wordpress项目 ...

  5. python框架day01

    一.注意事项 # 如何让你的计算机能够正常的启动django项目 1.计算机的名称不能有中文 2.一个pycharm窗口只开一个项目 3.项目里面所有的文件也尽量不要出现中文 4.python解释器尽 ...

  6. SolrCloud搭建

    什么是SolrCloud? SolrCloud是基于 solr 和 zookeeper 的分布式搜索方案,它的主要思想是使用zookeeper作为SolrCloud集群的配置信息中心,统一管理Solr ...

  7. 多测试讲解_009肖sirRF自动化框架安装教程

    robot framework:自动化测试框架 Python3.7 RIDE(可视化界面).  Wxpython  pip(在线下载) . setuptools(在线安装) . 第三方库 第三方库:s ...

  8. 【小白学PyTorch】21 Keras的API详解(下)池化、Normalization层

    文章来自微信公众号:[机器学习炼丹术].作者WX:cyx645016617. 参考目录: 目录 1 池化层 1.1 最大池化层 1.2 平均池化层 1.3 全局最大池化层 1.4 全局平均池化层 2 ...

  9. day32 Pyhton 模块02复习 序列化

    一. 什么是序列化 在我们存储数据或者网络传输数据的时候. 需要对我们的对象进行处理. 把对象处理成方便存储和传输的数据格式. 这个过程叫序列化 不同的序列化, 结果也不同. 但是目的是一样的. 都是 ...

  10. 安装Node,创建vue项目,运行及打包

    1.安装node js 下载地址:http://nodejs.cn/download/ 2.安装完成后运行Node.js command prompt(node -v查看安装版本) 3.安装npm(由 ...