仿vue的前端自定义cmd命令拉取项目脚手架

原文地址:https://github.com/screetBloom/wecat.js
含纯node或者commander实现自己的前端脚手架
文章码字分享不易,希望如果帮到您的话,帮忙github点个star

脚手架的实现

脚手架作用: 快速搭建一个我们预定义好的模板项目结构
我们这里就做了两件事(已经满足了我目前的需求,可以继续拓展)
1.自定义nodejs命令
2.在nodejs中执行shell命令(通俗说的命令行命令),拉取模板项目到本地

导言

我们平时经常会使用vue、angular、react等的脚手架,都可以达到如下效果

// 全局安装对应的脚手架  "xxx-cli"  (不全局安装的话,只能在当前安装包下使用)
npm install -g xxx-cli
// 接下来直接就如"vue init"就可以直接拉取一个模板项目到我们的当前文件夹
vue init

这个效果挺好用的,假如我积累了一套框架,我不想每次重开项目都拷贝到其他文件夹来用;当别人需要的时候,别人又要从我这拷贝一份;或者是我每次都给别人一串别人基本记不住的git的url链接,这个太麻烦了
我希望能有一套脚手架,能像这些成熟的框架的脚手架一样直接把我想要的模板项目用最简短而有效的命令拉取到任何我想要获取的电脑的文件夹中,再有需要了,我还能继续拓展

// 全局安装脚手架
npm install -g snowcat
// 拉取预定义模板
snowcat init

本地效果演示:



其它机器上演示:


脚手架具体实现过程

先上组织结构和代码,再从零开始讲实现方式和原理

0.0.1版本脚手架组织结构

实现0.0.1版本脚手架的完整代码

snowcat.js ==> 脚手架定义的所有命令的入口,这里暂时只有init命令

#!/usr/bin/env node
'use strict'
const program = require('commander')
program
    .version(require('../package').version )

program
    .command('init')
    .description('pull a new project')
    .alias('i')
    .action(() => {
        require('../command/init')()
    })

program.parse(process.argv)

if(!program.args.length){
    program.help()
}

init.js ==> init 命令的定义文件

'use strict'
const exec = require('child_process').exec
const projectUrl = 'https://github.com/screetBloom/wecat.js.git'

module.exports = () => {
    console.log('this is my first commander >>>>>> ')
    let cmdStr = `git clone `+projectUrl

    exec(cmdStr, (error, stdout, stderr) => {
        if (error) {
            console.log(error)
            process.exit()
        }
        console.log('pull我们的项目已经成功了')
        process.exit()
    })
}

package.json ==> 在package.json文件中声明整个文件包的可执行文件的位置

"bin": {
    "snowcat": "bin/snowcat.js"
  }

实现思路

上述的3个文件主要完成了2个最基本的事情
1.自定义nodejs命令。在nodejs原本肯定是没有"snowcat"这种命令的,这个是我们自定义的
2.用nodejs执行shell命令(通俗讲的命令行命令),这里主要是执行了git clone

那么我们现在先来尝试一下,如何自定义nodejs命令
在这里我们需要引入一个"commander.js"的npm包
先说明:不引入任何包都是可以完成我们上述的两件事,引入的主要原因有2个
1.有了这个npm包,可以简化我们命令行的开发,把我们主要精力还是回归到框架开发上
2.commander有大量的api,我们目前只是0.0.1版本,不依赖任何包来实现都是没有问题的,以后高版本1.0.0的拓展还是要用它的,这里我直接和大家说一下,也可以熟悉一下它的使用

我在这里补充一下不用任何依赖包的实现方式:

// PS.在nodejs中,可以直接用nodejs内置的全局变量process获取到你输入的命令的参数
// 现在我们直接就可以利用 process.argv来获取,如定义snowcat.js文件如下:
#!/usr/bin/env node
let run= function (para) {
    if(para[0] === '-test'){
        console.log('version is 1.0.0');
    }
    if(para[1] === '-host'){
        console.log('127.0.0.1');
    }
};
 console.log(process.argv)
run(process.argv.slice(2));

顶部的"#!/usr/bin/env node"的意思是 显式的声明这个文件用node来执行
执行snowcat.js文件

node snowcat.js
/*
输出结果如下:
[ '/usr/local/bin/node',
  '/Users/chenwei/WebstormProjects/git_my/deep-in-vue/wim/test.js' ]
*/
node snowcat.js -test -host
/*
输出结果基本如下:
[ '/usr/local/bin/node',
  '/Users/chenwei/WebstormProjects/git_my/deep-in-vue/wim/test.js',
  '-test',
  '-host' ]
*/

这里我想告诉大家的就是process.argv,这个东西很关键,可以拿到用户输入的命令,然后你就可以根据输入执行对应的函数就行了

先把 console.log(process.argv)注释了, 再来自定义指令试一试

node snowcat.js -test
/*
输出如下:
version is 1.0.0
*/
node snowcat.js -host
 /*
 输出如下:
 127.0.0.1
 */

我写这个demo主要表示现在我们就可以直接根据参数来匹配对应执行的函数了,以上面的init.js文件为例
我们是先利用process.argv.slice(2) 获取到输入的参数,匹配一下执行对应的函数就行;纯node.js实现

argv返回的是一个不定长的数组,第一个是node.exe的路径,第二个是当前文件的路径,接下来是你命令后面跟的参数

nodejs中的process的官方说明文档在这
这里我们就顺着这个继续了,commander等等再说,对目前的我们来说也没到重要要偏说不可的地步

有没有注意到上面我们都是在js文件所在目录下直接"node snowcat.js -test"来执行js文件,我们该如何直接"snowcat -test"就执行js文件呢 ,也就是上面我们说的自定义nodejs命令
这个时候package.json就需要登场了

// 初始化一个package.json,相关信息自定义
npm init

在里面添加一行

"bin": {
    "snowcat": "snowcat.js"
  }

这个是什么意思呢: 简单说就是把命令名作为key,本地文件名作为value做一个映射。全局安装的时候,npm会把你定义的这个命令名"snowcat"对应的可执行文件安装到系统路径下,达到全局使用该命令的目的;本地安装的时候,会直接链接到'./node_modules/.bin/'

目前的配置信息应该基本如下:

{
  "name": "snowcat",
  "version": "0.0.1",
  "description": "my cli 0.0.1",
  "main": "init.js",
  "bin": {
    "snowcat": "snowcat.js"
  },
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "author": "wim_chen",
  "license": "ISC"
}

现在我们来尝试本地的全局的运行 "snowcat" 命令,这里需要用到 "npm link",这里注意:是我们自己全局使用,给别人全局用可以发布到npm仓库

// 在当前的package.json中输入该命令
npm link

这个"npm link"主要是在我们本机的全局的"node_modules"目录中,生成一个符号链接"a symbolic link"指向我们当前文件夹

又因为我们在package.json中定义了"bin",指出了全局安装的时候命令"snowcat"对应的js文件
现在我们在本机的任何一个地方输入"snowcat -test -host"都会输出下述结果:

[ '/usr/local/bin/node',
  '/Users/chenwei/Desktop/工作与兴趣/common_test/command/test_one/snowcat.js',
  '-test',
  '-host' ]
version is 1.0.0
127.0.0.1

想要进一步的达到如

npm install -g snowcat 
// 执行我们目前的脚手架
snowcat -test -host

现在只需要做如下几步:
1.注册一个npm账号,点击https://www.npmjs.com/ 直达npm官网
2.给你的package.json里面的name写一个大家都没用过名字(snowcat你是别想了,很明显已经名花有主了)
3.在你本地的package.json所在的路径下输入:"npm adduser",然后输入你的用户名、密码、邮箱
4.输入 "npm publish"就将你的npm包发送到了npm仓库
5.找个好朋友,让他安装一下你的包"npm install -g xxx",让他输入" snowcat -test -host"就可以打印出你写好的内容了,比如"我爱你"?

现在我们来按要求拉取我们的项目,用一开始的文件结构举例


先创建一下对应的文件

首先"npm init"我们的package.json文件,并设置"commander.js"依赖和合适的bin

{
  "name": "snowcat",
  "version": "0.0.1",
  "description": "my js cli 0.0.1",
  "main": "index.js",
  "bin": {
    "snowcat": "bin/snowcat.js"
  },
  "dependencies": {
    "commander": "^2.9.0"
  },
  "author": "wim_chen",
  "license": "ISC"
}

这里先说明一下 commander.js的语法

program
    .command('init')       // 命令是 init
    .description('pull a new project')    // 命令的描述
    .alias('i')    // 命令别名,用init和i都行
    .action(() => {
      require('../command/init')()  // 执行init命令时要做什么,这里是执行init文件里导出的函数
  })

开始是编写我们的命令行入口文件,bin文件夹下的snowcat.js,也很简单

// 头部添加显示声明:本文件用node来执行
#!/usr/bin/env node
// 严格模式
'use strict'
// 引入 commander,用于处理自定义nodejs命令
const program = require('commander')
// 引用package.json里面的版本号来定义当前版本
program
    .version(require('../package').version )

// 定义init命令,同时定义init命令的简化命令 i,包括命令的脚本文件所在路径
program
    .command('init')
    .description('pull a new project')
    .alias('i')
    .action(() => {
        require('../command/init')()
    })

// 这一句必不可少,作用是解析命令行参数argv,这里的process.argv是nodejs全局对象的属性
program.parse(process.argv)

// 如果用户只是输入了 "snowcat"没带参数,就给他展示他能输入的所有命令
if(!program.args.length){
    program.help()
}

我们先来试一下,执行snowcat命令

node ./bin/snowcat.js 

显示如下,就是正确的,说明我们commander.js使用的很顺利

03.png

编写我们的 init.js

'use strict'
// 这个是node自带调用自窗口执行shell命令的方法,等会用
const exec = require('child_process').exec
module.exports = () => {
    console.log('this is my first commander >>>>>> ')
}

现在我们输入

node ./bin/snowcat.js  init

接下来我们来利用git来拉取我们的项目
编写init.js

'use strict'
const exec = require('child_process').exec
const projectUrl = 'https://github.com/screetBloom/wecat.js.git'

module.exports = () => {
    console.log('this is my first commander >>>>>> ')

    // git命令,远程拉取项目并自定义项目名
    let cmdStr = `git clone `+projectUrl

    // 在nodejs中执行shell命令,第一个参数是命令,第二个是具体的回调函数
    exec(cmdStr, (error, stdout, stderr) => {
        if (error) {
            console.log(error)
            process.exit()
        }
        console.log('pull我们的项目已经成功了')
        process.exit()
    })

}

现在我们再输入

node ./bin/snowcat.js  init

此时项目已经可以正确的拉取下来了,接下来我们来进行本地全局安装,在当前的package.json路径下输入"npm link"(可以本地全局使用了)
在其它路径下拉取项目

成功,那么现在我们把它放到npm仓库里,如果上一次你已经放进去0.0.1版本了,这次就需要修改版本号了,操作如下:

// 比如对最后一位进行修改:增1,命令,回车:
npm version patch    
// 比如对第二位进行了修改:增1,命令:
npm version minor    
// 比如对第一位进行了修改:增1,命令:
npm version major  
     
// 查看所有的版本号:
npm view snowcat versions

最后我们再在其它机器上测试

// 全局安装脚手架
npm install -g snowcat
// 拉取预定义模板
snowcat init

0.0.1版本的脚手架分享到此结束

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容