使用 Node 构建命令行应用

简介: > * 原文地址:[How to create a real-world Node CLI app with Node](https://medium.freecodecamp.org/how-to-create-a-real-world-node-cli-app-with-node-391b727bbed3) > * 原文作者:[Timber.io](https://medium.freeco

使用 Node 构建命令行应用

JavaScript 的开发领域内,命令行应用还尚未获得足够的关注度。事实上,大部分开发工具都应该提供命令行界面来给像我们一样的开发者使用,并且用户体验应该与精心创建的 Web 应用程序相当,比如一个漂亮的设计,易用的菜单,清晰的错误反馈,加载提示和进度条等。

目前并没有太多的实际教程来指导我们使用 Node 构建命令行界面,所以本文将是开篇之作,基于一个基本的 hello world 命令应用,逐步构建一个名为 outside-cli 的应用,它可以提供当前的天气并预测未来 10 天任何地方的天气情况。

提示:有不少的库可以帮助你构建复杂的命令行应用,例如 oclifyargscommander,但是为了你更好地理解背后的原理,我们会保持外部依赖尽可能的少。当然,我们假设你已经拥有了 JavaScriptNode 的基础知识。

入门

与其他的 JavaScript 项目一样,最佳实践便是创建 package.json 和一个空的入口文件,目前还不需要任何依赖,保持简单。

package.json

{
  "name": "outside-cli",
  "version": "1.0.0",
  "license": "MIT",
  "scripts": {},
  "devDependencies": {},
  "dependencies": {}
}
AI 代码解读

index.js

module.exports = () => {
  console.log('Welcome to the outside!')
}
AI 代码解读

我们将使用 bin 文件来运行这个新程序,并且会把 bin 文件添加到系统目录里,使其在任何地方都可以被调用。

#!/usr/bin/env node
require('../')()
AI 代码解读

是不是之前从未见过 #!/usr/bin/env node ? 它被称为 shebang)。它告知系统这不是一个 shell 脚本并指明应该使用不同的解释程序。

bin 文件需要保持简单,因为它的本意仅是用来调用主函数,我们所有的代码都应当放置在此文件之外,这样才可以保证模块化和可测试,同时也可以实现未来在其他的代码里被调用。

为了能够直接运行 bin 文件,我们需要赋予正确的文件权限,如果你是在 UNIX 环境下,你只需要执行 chomd +x bin/outsideWindows 用户就只能靠自己了,建议使用 Linux 子系统。

接下来,我们将添加 bin 文件到 package.json 里,随后当我们全局安装此包时( npm install -g outside-cli ),bin 文件会被自动添加到系统目录内。

package.json

{
  "name": "outside-cli",
  "version": "1.0.0",
  "license": "MIT",
  "bin": {
    "outside": "bin/outside"
  },
  "scripts": {},
  "devDependencies": {},
  "dependencies": {}
}
AI 代码解读

现在我们输入 ./bin/outside ,就可以直接运行了,欢迎消息将会被打印出来,在你的项目根目录执行 npm link,它将会在系统路径和你的二进制文件之间建立软连接,这样 outside 命令便可以在任何地方运行了。

CLI 应用程序由参数和指令构成,参数(或「标志」)是指前缀为一个或两个连字符构成的值(例如 -d--debug--env production ),它对应用来说非常有用。指令是指没有标志的其他所有值。

与指令不同,参数并不要求特定的顺序,举个例子,运行 outside today Brooklyn,必须约定第二个指令只能代表地域,使用 -- 则不然,运行 outside today --location Brooklyn,可以方便地添加更多的选项。

为了使应用更加实用,我们需要解析指令和参数,然后转换为字面量对象,我们可以使用 process.argv 来手动实现,但是现在我们要安装项目的第一个依赖 minimist ,让它来帮我们搞定这些事儿。

npm install --save minimist
AI 代码解读

index.js

const minimist = require('minimist')

module.exports = () => {
  const args = minimist(process.argv.slice(2))
  console.log(args)
}
AI 代码解读

提示:因为 process.argv 的前两个参数分别是解释器和二进制文件名,所以我们使用 .slice(2) 移除掉前两个参数,只关心传递进来的其他命令。

现在执行 outside today 将会输出 { _: ['today'] }。执行 outside today --location "Brooklyn, NY",将会输出 { _: ['today'], location: 'Brooklyn, NY' }。不过现在我们不用进一步深挖参数的用法,等到实际使用 location的时候再继续深入,目前了解的已经足够我们实现第一个指令了。

参数语法

可以通过这篇文章帮助你更好地理解参数语法。基本上,一个参数可以有一个或者两个连字符,然后紧跟着是它对应的值,在不填写时它的值默认为 true, 单连字符参数还可以使用缩写的格式( -a -b -c 或者 -abc 都对应着 { a: true, b: true, c: true } )。

如果参数值包含特殊字符或者空格,则必须使用引号包裹着。例如 --foo bar 对应着 { : ['baz'], foo: 'bar' }--foo "bar baz" 对应 { foo: 'bar baz' }

分割每个指令的代码,在其被调用时再加载至内存是一个最佳实践,这有助于缩短启动时间,避免不必要的加载。在主指令代码里简单地使用 switch 就可以实现此实践了。在这种设置下,我们需要把每个指令写到独立的文件里,并且导出一个函数,与此同时,我们把参数传递给每个指令函数用以在后期使用。

index.js

const minimist = require('minimist')

module.exports = () => {
  const args = minimist(process.argv.slice(2))
  const cmd = args._[0]

  switch (cmd) {
    case 'today':
      require('./cmds/today')(args)
      break
    default:
      console.error(`"${cmd}" is not a valid command!`)
      break
  }
}
AI 代码解读

cmds/today.js

module.exports = (args) => {
  console.log('today is sunny')
}
AI 代码解读

现在如果执行 outside today,你会看到输出 today is sunny,如果执行 outside foobar,会输出 "foobar" is not a valid command。目前的原型已经很不错了,接下来我们需要通过 API 来获取天气的真实数据。

有一些命令和参数是我们希望在每个命令行应用中都包含的:help--help-h 用来展示帮助清单;--version-v 用来显示当前应用的版本信息。当指令没有指定时,我们也应当默认展示帮助清单。

Minimist 会自动解析参数为键值对,因此运行 outside --version 会使得 args.version 等于 true。那么在程序里通过设置 cmd 变量来保存 helpversion 参数的判定结果,然后在 switch 语句中添加两个处理语句,就可以实现上述功能了。

const minimist = require('minimist')

module.exports = () => {
  const args = minimist(process.argv.slice(2))

  let cmd = args._[0] || 'help'

  if (args.version || args.v) {
    cmd = 'version'
  }

  if (args.help || args.h) {
    cmd = 'help'
  }

  switch (cmd) {
    case 'today':
      require('./cmds/today')(args)
      break

    case 'version':
      require('./cmds/version')(args)
      break

    case 'help':
      require('./cmds/help')(args)
      break

    default:
      console.error(`"${cmd}" is not a valid command!`)
      break
  }
}
AI 代码解读

实现新指令时,格式需要和 today 指令保持一致。

cmds/version.js

const { version } = require('../package.json')

module.exports = (args) => {
  console.log(`v${version}`)
}
AI 代码解读

cmds/help.js

const menus = {
  main: `
    outside [command] <options>

    today .............. show weather for today
    version ............ show package version
    help ............... show help menu for a command`,

  today: `
    outside today <options>

    --location, -l ..... the location to use`,
}

module.exports = (args) => {
  const subCmd = args._[0] === 'help'
    ? args._[1]
    : args._[0]

  console.log(menus[subCmd] || menus.main)
}
AI 代码解读

现在如果执行 outside help todayoutside toady -h,你便会看到 today 指令的帮助信息了,执行 outsideoutside -h 亦是如此。

目前的项目设定是令人愉悦的,因为当你需要添加一个新指令时,你只需要创建一个新指令文件,把它添加到 switch 语句中,再设置一个帮助信息便可以了。

cmds/forecast.js

module.exports = (args) => {
  console.log('tomorrow is rainy')
}
AI 代码解读

index.js

*// ...*
    case 'forecast':
      require('./cmds/forecast')(args)
      break
*// ...*
AI 代码解读

cmds/help.js

const menus = {
  main: `
    outside [command] <options>

    today .............. show weather for today
    forecast ........... show 10-day weather forecast
    version ............ show package version
    help ............... show help menu for a command`,

  today: `
    outside today <options>

    --location, -l ..... the location to use`,

  forecast: `
    outside forecast <options>

    --location, -l ..... the location to use`,
}

// ...
AI 代码解读

有些指令执行起来可能需要很长时间。如果你会执行从 API 获取数据,内容生成,将文件写入磁盘,或者其他需要花费超过几毫秒的程序,那么便需要向用户提供一些反馈来表明你的程序仍在响应中。你可以使用进度条来展示操作的进度,也可以直接显示一个进度指示器。

对当前的应用来说,我们无法获知 API 请求的进度,所以我们使用一个简单的 spinner 来表达程序仍在运行中就可以了。我们接下来安装两个依赖,axios 用于网络请求,ora 来实现 spinner

npm install --save axios ora
AI 代码解读

从 API 获取数据

现在我们先创建一个使用雅虎天气 API 来获得某个地域天气情况的工具函数。

提示:雅虎 API 使用非常简洁的 YQL 语法,我们不需要刻意理解它,直接拷贝使用即可。另外,它也是唯一一个我发现不需要提供 API key 的天气 API 了。

utils/weather.js

const axios = require('axios')

module.exports = async (location) => {
  const results = await axios({
    method: 'get',
    url: 'https://query.yahooapis.com/v1/public/yql',
    params: {
      format: 'json',
      q: `select item from weather.forecast where woeid in
        (select woeid from geo.places(1) where text="${location}")`,
    },
  })

  return results.data.query.results.channel.item
}
AI 代码解读

cmds/today.js

const ora = require('ora')
const getWeather = require('../utils/weather')

module.exports = async (args) => {
  const spinner = ora().start()

  try {
    const location = args.location || args.l
    const weather = await getWeather(location)

    spinner.stop()

    console.log(`Current conditions in ${location}:`)
    console.log(`\t${weather.condition.temp}° ${weather.condition.text}`)
  } catch (err) {
    spinner.stop()

    console.error(err)
  }
}
AI 代码解读

现在当你执行 outside today --location "Brooklyn, NY" 后,你首先会看到一个快速旋转的 spinner 出现在应用发起请求期间,随后便会展示天气信息了。

当请求发生得很快时,我们是难以看到加载指示的,如果你想人为地减慢速度,你可以在请求天气工具函数前加上这一句:await new Promise(resolve => setTimeout(resolve, 5000))

非常棒!接下来我们复制下上面的代码来实现 forecast 指令,然后简单修改下输出格式。

cmds/forecast.js

const ora = require('ora')
const getWeather = require('../utils/weather')

module.exports = async (args) => {
  const spinner = ora().start()

  try {
    const location = args.location || args.l
    const weather = await getWeather(location)

    spinner.stop()

    console.log(`Forecast for ${location}:`)
    weather.forecast.forEach(item =>
      console.log(`\t${item.date} - Low: ${item.low}° | High: ${item.high}° | ${item.text}`))
  } catch (err) {
    spinner.stop()

    console.error(err)
  }
}
AI 代码解读

现在当你执行 outside forecast --location "Brooklyn, NY" 后,你会看到未来 10 天的天气预测结果了。接下来我们再锦上添花下,当 location 没有指定时,使用我们编写的一个工具函数来实现自动根据 IP 地址获取所处位置。

utils/location.js

const axios = require('axios')

module.exports = async () => {
  const results = await axios({
    method: 'get',
    url: 'https://api.ipdata.co',
  })

  const { city, region } = results.data
  return `${city}, ${region}`
}
AI 代码解读

cmds/today.js & cmds/forecast.js

*// ...*
const getLocation = require('../utils/location')

module.exports = async (args) => {
  *// ...*
    const location = args.location || args.l || await getLocation()
    const weather = await getWeather(location)
  *// ...*
}
AI 代码解读

现在当你不添加 location 参数执行指令后,你将会看到当前地域对应的天气信息。

错误处理

本篇文章我们并不会详细介绍错误处理的最佳方案(后面的教程里会介绍),但是最重要的是要记住使用正确的退出码。

如果你的命令行应用出现了严重错误,你应当使用 process.exit(1),终端会感知到程序并未完全执行,此时便可以通过 CI 程序来对外通知。

接下来我们创建一个工具函数来实现当运行一个不存在的指令时,程序会抛出正确的退出码。

utils/error.js

module.exports = (message, exit) => {
  console.error(message)
  exit && process.exit(1)
}
AI 代码解读

index.js

*// ...*
const error = require('./utils/error')

module.exports = () => {
  *// ...*
    default:
      error(`"${cmd}" is not a valid command!`, true)
      break
  *// ...*
}
AI 代码解读

收尾

最后一步是将我们编写的库发布到远程包管理平台上,由于我们使用 JavaScriptNPM 再合适不过了。现在,我们需要额外填一些儿信息到 package.json 里。

{
  "name": "outside-cli",
  "version": "1.0.0",
  "description": "A CLI app that gives you the weather forecast",
  "license": "MIT",
  "homepage": "https://github.com/timberio/outside-cli#readme",
  "repository": {
    "type": "git",
    "url": "git+https://github.com/timberio/outside-cli.git"
  },
  "engines": {
    "node": ">=8"
  },
  "keywords": [
    "weather",
    "forecast",
    "rain"
  ],
  "preferGlobal": true,
  "bin": {
    "outside": "bin/outside"
  },
  "scripts": {},
  "devDependencies": {},
  "dependencies": {
    "axios": "^0.18.0",
    "minimist": "^1.2.0",
    "ora": "^2.0.0"
  }
}
AI 代码解读
  • 设置 engine 可以确保使用者拥有一个较新的 Node 版本。因为我们未经编译直接使用了 async/await,所以我们要求 Node 版本 必须在 8.0 及以上。
  • 设置 preferGlobal 将会在安装时提示使用者本库最好全局安装而非作为局部依赖安装。

目前就这些内容了,现在你便可以通过 npm publish 发布至远端来供他人下载了。如果你想更进一步,发布到其他包管理工具(例如 Homebrew )上,你可以了解下 pkgnexe,它们可以帮助你把应用打包到一个独立的二进制文件里。

总结

本篇文章介绍的代码目录结构是 Timber 上所有的命令行应用都遵循的,它有助于保持组织和模块化。

对于速读的读者,我们也提供了一些本教程的关键要点

  • Bin 文件是整个命令行应用的入口,它的职责仅是调用主函数。
  • 指令文件在未执行时不应该被加载到主函数里。
  • 始终包含 helpversion 指令。
  • 指令文件需要保持简单,它们的主要职责是调用其他工具函数,随后展示信息给用户。
  • 始终包含一些运行指示给到用户。
  • 应用退出时应当使用正确的退出码。

我希望你现在能够更好地了解如何使用 Node 创建和组织命令行应用。本文只是开篇之作,随后我们会继续深入理解如何优化设计,生成 ascii art 和添加色彩等。本文的源码可以在 GitHub 上获取到。

也树
+关注
目录
打赏
0
0
0
0
1142
分享
相关文章
深入浅出Node.js:从零开始构建后端服务
【10月更文挑战第42天】在数字时代的浪潮中,掌握一门后端技术对于开发者来说至关重要。Node.js,作为一种基于Chrome V8引擎的JavaScript运行环境,允许开发者使用JavaScript编写服务器端代码,极大地拓宽了前端开发者的技能边界。本文将从Node.js的基础概念讲起,逐步引导读者理解其事件驱动、非阻塞I/O模型的核心原理,并指导如何在实战中应用这些知识构建高效、可扩展的后端服务。通过深入浅出的方式,我们将一起探索Node.js的魅力和潜力,解锁更多可能。
如何使用内存监控工具来定位和解决Node.js应用中的性能问题?
总之,利用内存监控工具结合代码分析和业务理解,能够逐步定位和解决 Node.js 应用中的性能问题,提高应用的运行效率和稳定性。需要耐心和细致地进行排查和优化,不断提升应用的性能表现。
196 77
如何优化Node.js应用的内存使用以提高性能?
通过以上多种方法的综合运用,可以有效地优化 Node.js 应用的内存使用,提高性能,提升用户体验。同时,不断关注内存管理的最新技术和最佳实践,持续改进应用的性能表现。
143 62
深入浅出:使用Node.js构建RESTful API
在这个数字时代,API已成为软件开发的基石之一。本文旨在引导初学者通过Node.js和Express框架快速搭建一个功能完备的RESTful API。我们将从零开始,逐步深入,不仅涉及代码编写,还包括设计原则、最佳实践及调试技巧。无论你是初探后端开发,还是希望扩展你的技术栈,这篇文章都将是你的理想指南。
如何使用内存监控工具来优化 Node.js 应用的性能
需要注意的是,不同的内存监控工具可能具有不同的功能和特点,在使用时需要根据具体工具的要求和操作指南进行正确使用和分析。
80 31
深入浅出Node.js:从零开始构建RESTful API
在数字化时代的浪潮中,后端开发作为连接用户与数据的桥梁,扮演着至关重要的角色。本文将引导您步入Node.js的奇妙世界,通过实践操作,掌握如何使用这一强大的JavaScript运行时环境构建高效、可扩展的RESTful API。我们将一同探索Express框架的使用,学习如何设计API端点,处理数据请求,并实现身份验证机制,最终部署我们的成果到云服务器上。无论您是初学者还是有一定基础的开发者,这篇文章都将为您打开一扇通往后端开发深层知识的大门。
52 12
深入理解Node.js事件循环及其在后端开发中的应用
本文旨在揭示Node.js的核心特性之一——事件循环,并探讨其对后端开发实践的深远影响。通过剖析事件循环的工作原理和关键组件,我们不仅能够更好地理解Node.js的非阻塞I/O模型,还能学会如何优化我们的后端应用以提高性能和响应能力。文章将结合实例分析事件循环在处理大量并发请求时的优势,以及如何避免常见的编程陷阱,从而为读者提供从理论到实践的全面指导。
|
2月前
|
如何使用内存快照分析工具来分析Node.js应用的内存问题?
需要注意的是,不同的内存快照分析工具可能具有不同的功能和操作方式,在使用时需要根据具体工具的说明和特点进行灵活运用。
56 3
深入浅出Node.js:从零开始构建RESTful API
在数字化时代的浪潮中,后端开发如同一座灯塔,指引着数据的海洋。本文将带你航行在Node.js的海域,探索如何从一张白纸到完成一个功能完备的RESTful API。我们将一起学习如何搭建开发环境、设计API结构、处理数据请求与响应,以及实现数据库交互。准备好了吗?启航吧!