写点什么

私人定制 CLI 工具

  • 2019-09-27
  • 本文字数:5374 字

    阅读完需:约 18 分钟

私人定制CLI工具

1 前言

本文面向有 JavaScript 开发经验且希望入门 CLI 工具制作的开发人员。文中没有太多理论知识,通过从 0 到 1 开发一个简单的 CLI 工具,分享了 CLI 工具的开发、发布流程。看完本文,你可以学会如何"私人定制"一个 CLI 工具来提升开发效率。


总之,简单,实用,有趣。

2 什么是 CLI 工具

维基百科是这样定义 CLI 的


CLI(Command-line Interface),命令行界面,指通过输入字符组成的命令行进行操作的用户界面。


常见的 CLI 工具举例


系统命令:cd、mkdir、ifconfig


脚手架工具:create-react-app、vue-cli、yeoman


预处理器:less、sass、babel


测试工具:mocha、karma、wrk


构建工具:webpack、gulp、grunt

3 为什么要自己开发 CLI 工具

举例,如果想查看本机 ip 地址,通常我会使用 ifconfig 命令,如图:



ifconfig 执行结果


执行命令后系统输出了一堆信息,从这堆信息中找到ip地址大概需要花5秒时间,这不是我想要的。我希望的结果是这样的:



ip 执行结果


输入ip命令,直接得到本机ip地址,1秒搞定,简单快捷。


理想很丰满,现实很骨感。真实的情况是:当我在终端敲下 ip 命令时,系统给我的反馈是"command not found:ip"。



ip 执行结果


很显然,系统并没有为我提供ip这个命令,所以我们需要自己来实现它。

4 开始开发 CLI 工具

4.1 第一步:确保安装了 Node.js


node 版本检测 node 版本检测

4.2 第二步:编写脚本、测试运行

创建文件夹 ip-cli,并新建 index.js


mkdir ip-cli && cd ip-clitouch index.js
复制代码


编辑 index.js,内容如下


const os = require("os")const ip = os.networkInterfaces().en0[1].addressconst options = process.argv.slice(2)if (options[0] === '-v') {    console.log('v1.0.0')} else {    console.log(`your ip is: ${ip}`)}
复制代码


执行 node index.js,结果如下



执行脚本执行脚本

4.3 声明执行环境

执行 which env



which env


将返回的结果设置到index.js脚本的第一行,并指定脚本的解释器为node。


然后,代码就变成了下面这样:


#!/usr/bin/env node
const os = require("os")const ip = os.networkInterfaces().en0[1].addressconst options = process.argv.slice(2)if (options[0] === '-v') { console.log('v1.0.0')} else { console.log(`your ip is: ${ip}`)}
复制代码


代码中的 #!被称为 Shebang,是用来告诉编译器用什么命令执行文件的。Shebang 的一些具体用法罗列如下:


1、如果脚本文件中没有 #!这一行,那么执行时会默认采用当前 Shell 去解释这个脚本(即:

$ SHELL 环境变量)。

2、如果 #!之后的解释程序是一个可执行文件,那么执行这个脚本时,它就会把文件名及其参数一起作为参数传给那个解释程序去执行。

3、如果 #!指定的解释程序没有可执行权限,则会报错“bad interpreter: Permission

denied”。如果 #!指定的解释程序不是一个可执行文件,那么指定的解释程序会被忽略,转而交给当前的 SHELL 去执行这个脚本。

4、如果 #!指定的解释程序不存在,那么会报错“bad interpreter: No such file or directory”。注意:#!之后的解释程序,需要写其绝对路径(如:#!/bin/bash),它是不会自动到 $PATH 中寻找解释器的。

5、当然,如果你使用类似于”bash test.sh”这样的命令来执行脚本,那么 #!这一行将会被忽略掉,解释器当然是用命令行中显式指定的 bash。

6、脚本文件必须拥有可执行权限。


env 可以在系统的 PATH 目录中查找脚本解释器安装目录。


添加 #!/usr/bin/env node 是告诉系统,这个脚本使用 Node.js 来执行。这样我们就可以简化命令,执行 index.js 直接得到 ip 地址,不需要显式的调用 node index.js。

4.4 分配执行权限

在我们直接执行 index.js 后,结果如下:



执行 index.js


我们并没有得到 ip 地址,查看文件发现 index.js 没有可执行权限。



查看文件


执行 chmod +x index.js 给脚本分配可执行权限,然后 index.js 就可以执行了。



增加可执行权限

4.5 设置环境变量

执行 $PATH 查看环境变量



PATH


可以看到我电脑的 PATH 变量为


/usr/local/mongodb/bin:/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin:“/usr/libexec/java_home”::/platform-tools:/tools:“/usr/libexec/java_home”:“/usr/libexec/java_home”/bin:/usr/local/mongodb/bin:/Users/qianxuemin/bin/
复制代码


这里 usr 指 Unix System Resource,而不是 User, 通常/usr/bin 下面的都是系统预装的可执行程序,会随着系统升级而改变, /usr/local/bin 目录是给用户放置自己的可执行程序的地方,推荐放在这里,不会被系统升级而覆盖同名文件。如果两个目录下有相同的可执行程序,谁优先执行受到 PATH 环境变量的影响,比如我的电脑 PATH 变量中,/usr/local/bin 优先于/usr/bin。


接下来执行 ln index.js /usr/local/bin/ip,创建 index.js 的一个硬链接文件/usr/local/bin/ip,如图:



ln


然后就可以直接使用自定义的命令ip了 。



ip


如果只是自己使用,那么一个简单的CLI工具就开发完毕了,但作为一名喜欢分享的童鞋,我希望可以把这个CLI工具分享给他人。


这时候可以选择使用 npm 管理项目,与他人共享开发成果。为了演示在 npm 项目中开发 CLI 工具的流程,我们暂时先删除刚才的命令 ip



删除命令

5 使用 npm 管理项目

5.1 初始化 npm 项目

执行 npm init -y 将项目初始化为 npm 项目。



初始化


目录结构如下:


.├── index.js└── package.json
复制代码

5.2 设置 bin 字段

在 package.json 中设置 bin 字段,定义一个命令 ip 并指定该命令执行./index.js 文件。


{  "name": "ip-cli",  "version": "1.0.0",  "description": "",  "main": "index.js",  "bin": {    "ip": "./index.js"  },  "scripts": {    "test": "echo \"Error: no test specified\" && exit 1"  },  "keywords": [],  "author": "",  "license": "ISC"}
复制代码

5.3 测试 CLI 工具

执行 npm link ,在全局的 node_modules 下生成一个符号链接。



npm link


然后就可以在全局使用 package.json 中 bin 字段定义的命令 ip 了,执行 ip 命令结果如下



ip


输出了本机IP地址,测试成功。

5.4 发布 CLI 工具

执行 npm login 登录账号,登录成功后执行 npm whoami 可以看到 npm 用户名。



npm whoami


确认无误后执行 npm publish发布npm包。


发布包的时候需要确保包的名字没有被占用,如下是包名冲突的情况。



npm publish


遇到这种情况可以选择换个名字或者加 scope,如下我给包名增加 scope,将包名由 ip-cli 改为 @qianxuemin/ip-cli。


{  "name": "@qianxuemin/ip-cli",  "version": "1.0.2",  "description": "",  "main": "index.js",  "bin": {    "ip": "./index.js"  },  "scripts": {    "test": "echo \"Error: no test specified\" && exit 1"  },  "keywords": [],  "author": "",  "license": "ISC"}
复制代码


加了 scope 后,发布包的时候需要设置访问级别,我们设置为公开。



npm publish


发布成功后,其他开发人员就可以通过 npm install @qianxuemin/ip-cli -g 安装并使用了。



ip


至此,一个简单的 CLI 工具开发、发布完成。

6 注意事项

上面演示了 CLI 工具的整体开发、发布流程,没有对细节做太多说明。如下总结了在日常开发 CLI 工具中需要注意的事项与大家分享:


1、工具应当包含完整的README.md


2、工具应当包含 help 和 version 信息


3、工具应当包含运行指示,比如加载中


4、发布之前要到 https://www.npmjs.com 查找一下自己的包名有没有被占用,如果被占用需要先修改包名或加 scope


5、发布包时,可以通过在项目中添加.npmignore 或.gitignore 文件限制发布的文件内容,.npmignore 优先级高于.gitignore。或者在 package.json 中 files 字段设置发布哪些文件或目录,它的优先级高于 .npmignore 和 .gitignore。


7 开发 CLI 工具常用的库


相比原生 Node.js,使用一些开源模块能够简化 CLI 工具的开发,提高开发效率。如下是开发 CLI 工具常用的模块:


commander: 注册、解析命令行参数


Inquirer: 让命令行与用户进行交互


chalk: 给命令行字符加颜色


shelljs: 跨平台调用 shell 命令的 node 封装


Ora: 命令行提示图标


progress: 命令行进度条


blessed-contrib: 命令行可视化组件


download-git-repo:拉取 git 仓库源代码


由于文章篇幅有限,这里就不一一进行介绍了,使用时可以直接到 https://www.npmjs.com/搜索。

8 总结

本文通过一个简单的例子分享了 CLI 工具的开发、发布流程,并总结了一些开发注意事项和常用的库。旨在帮助 JavaScript 开发人员入门 CLI 工具开发。希望大家都能在掌握 CLI 工具开发流程后,发挥想象力,开发出一些实用、好玩的东西,让我们的开发变得高效、有趣。


作者介绍:


扣丁(企业代号名),目前负责贝壳新装修平台 B 端业务研发工作。


本文转载自公众号贝壳产品技术(ID:gh_9afeb423f390)。


原文链接:


https://mp.weixin.qq.com/s/I8XXSw1eMbKLjFIcp9KWdQ


2019-09-27 13:481071

评论

发布
暂无评论
发现更多内容

怎样让全彩LED显示屏变的节能环保?

Dylan

LED显示屏 全彩LED显示屏

真人踩过的坑,告诉你避免自动化测试新手常犯的10个错误

禅道项目管理

自动化 测试 自动化测试

2022年数据库审计产品排行榜-必看!

行云管家

数据库 数据库审计

2022亚洲国际物联网展会

AIOTE智博会

物联网展览会

Idea 连接 MySQL 数据库

攻城狮杰森

MySQL IDEA database 7月月更

K8S多集群管理很难?试试Karmada | K8S Internals系列第3期

BoCloud博云

容器 容器云 K8s 多集群管理

对话ACE第四期:分布式数据库未来发展的挑战和机遇

OceanBase 数据库

数据库 分布式数据库 oceanbase

科创人·观远数据CEO苏春园:让业务用起来,是BI行业推倒渗透率之墙的关键

科创人

大数据

今日分享| 阿里巴巴内部最新Docker精髓笔记

冉然学Java

Java Docker 阿里 #技术干货# Java 开发

长期的远程工作面临的几个问题和持续改进的组织自动化

Ryan Zheng

远程办公

仅需一个依赖给Swagger换上新皮肤,既简单又炫酷

程序知音

【等保小知识】等保整改是什么意思?整改内容包括哪些?

行云管家

等保测评 等级测评 等保整改

企业进行知识管理有哪些好处?

Geek_da0866

学界VS工业界:深度学习究竟能不能打破视频编解码天花板

小红书技术REDtech

深度学习 音视频 视频编解码 音视频技术

Linux 文件系统函数

贾献华

7月月更

SCA在得物DevSecOps平台上应用​

得物技术

安全 DevSecOps SCA

C# 使用Timer和ProgressBar控件制作一个倒计时器

IC00

C# 7月月更

设计消息队列存储消息数据的 MySQL 表格

Fan

架构实战营

strcmp() - 比较字符串

謓泽

7月月更

百度APP Android包体积优化实践(二)Dex行号优化

百度Geek说

Java Andriod

国产统信UOS系统运行小程序的探索

Geek_99967b

小程序

首发!这份字节大佬亲码算法面试大厂进阶宝典,让你轻松刷爆LeetCode!

了不起的程序猿

字节跳动 算法 java程序员

前两天面了个腾讯拿 38K 出来的,让我见识到了基础的天花板,今天share给大家~

程序知音

Java 程序员 java面试 后端技术 八股文

JAVA编程规范之异常处理

源字节1号

微信小程序 软件开发 前端开发 后端开发

如何搭建一个好的知识库管理系统?

Geek_da0866

NFTScan 与 Port3 在 NFT 数据领域达成战略合作

NFT Research

乘数科技云管控平台适配阿里云PolarDB,共促云原生数据库生态繁荣

阿里云数据库开源

数据库 阿里云 开源数据库 polarDB PolarDB for PostgreSQL

JS数组方法

bo

JavaScript 前端 7月月更

2022长三角工业自动化展会将于10月在南京国际展览中心召开

AIOTE智博会

工业自动化展会 工业机器人展会 江苏工博会

Python 爬虫 JS 逆向 X-Bogus,signature 加密算法,AST 理论篇

梦想橡皮擦

Python 爬虫 7月月更

Kyligence 出席华为全球智慧金融峰会,加速拓展全球市场

Kyligence

数据湖 数据分析 OLAP

私人定制CLI工具_文化 & 方法_扣丁_InfoQ精选文章