A collection of curated best practices on how to build successful, empathic and user-friendly Node.js Command Line Interface (CLI) applications. Node.js CLI Apps Best Practices
Why this guide?
A bad CLI can easily discourage users from interacting with it. Building successful CLIs requires attention to detail and empathy for the user in order to create a good user experience. It is very easy to get wrong.
In this guide I have compiled a list of best practices across areas of focus which aim to optimize for an ideal user experience when interacting with a CLI application.
Features:
✅ 29 best practices for building successful Node.js CLI applications✅ Read in a different language:🇨🇳 ,🇪🇸 , or help translate to other languages. Suggest new languages.🙏 Contributions are welcome
Why me?
Hi there, I'm Liran Tal and I'm addicted to building command line applications.
Some of my recent work, building Node.js CLIs, includes the following Open Source projects:
Dockly Immersive terminal interface for managing docker containers and services |
npq safely install packages with npm/yarn by auditing them as part of your install process |
lockfile-lint Lint an npm or yarn lockfile to analyze and detect security issues |
is-website-vulnerable finds publicly known security vulnerabilities in a website's frontend JavaScript libraries |
✨
The Team Thanks goes to these wonderful people (emoji key):
是你吖小刘 |
Terkel |
Jason Karns |
Dave Sag |
José J. Pérez Rivas |
Sureshraj |
Table of Contents
- 1 Command Line Experience
- 1.1 Respect POSIX args
- 1.2 Build empathic CLIs
- 1.3 Stateful data
- 1.4 Provide a colorful experience
- 1.5 Rich interactions
- 1.6 Hyperlinks everywhere
- 1.7 Zero configuration
- 1.8 Respect POSIX signals
- 2 Distribution
- 3 Interoperability
- 4 Accessibility
- 5 Testing
- 6 Errors
- 7 Development
- 7.1 Use a bin object
- 7.2 Use relative paths
- 7.3 Use the files field
- 8 Analytics
- 9 Appendix: CLI Frameworks
- 10 Appendix: CLI educational resources
1 Command Line Experience
This section deals with best practices concerned with creating beautiful and high-value user experience Node.js command line applications.
In this section:
- 1.1 Respect POSIX args
- 1.2 Build empathic CLIs
- 1.3 Stateful data
- 1.4 Provide a colorful experience
- 1.5 Rich interactions
- 1.6 Hyperlinks everywhere
- 1.7 Zero configuration
- 1.8 Respect POSIX signals
1.1 Respect POSIX args
Unix-like operating systems popularized the use of the command line and tools such as awk
, sed
. Such tools have effectively standardized the behavior of command line options (aka flags), options-arguments, and other operands.
Some examples of expected behavior:
- option-arguments or options can be notated in help or examples as square brackets (
[]
) to indicate they are optional, or with angle brackets (<>
) to indicate they are required. - allow short-form single letter arguments as aliases for long-form arguments (see reference from the GNU Coding Standards).
- options specified using the short form singular
-
may contain one alphanumeric character. - specifying multiple options with no values may be grouped, such as
myCli -abc
being equivalent tomyCli -a -b -c
.
Command line power-users will expect your command line application to have similar conventions as other Unix apps.
Reference to Open Source Node.js packages:
1.2 Build empathic CLIs
A command line interface for your program is no different than a web user interface in the sense of doing as much as you can as the program author to ensure that it is being used successfully.
Optimize for successful interactions by building empathic CLIs that support the user. As an example, let's explore the case of the curl
program that expects a URL as its primary data input, and the user failing to provide it. Such failure will lead to reading through a (hopefully) descriptive error messages or reviewing a curl --help
output. However, an empathic CLI would have presented an interactive prompt to capture input from the user, resulting in a successful interaction.
1.3 Stateful data
It may happen that you find yourself needing to provide storage persistence for your CLI application, such as remembering a username, email, API token, or other preferences between multiple invocations of the CLI. Use a configuration helper that allows the app to persist such user settings. Be sure to follow the XDG Base Directory Specification when reading/writing files (or choose a configuration helper that respects the spec). These keeps the user in control of where files are written and managed.
Reference projects:
1.4 Provide a colorful experience
Most terminals used today to interact with command line applications support colored text such as these enabled by specially crafted ANSI encoded characters.
A colorful display in your command line application output may further contribute to a richer experience and increased interaction. That said, unsupported terminals may experience a degraded output in the form of garbled information on the screen. Furthermore, a CLI may be used in a continuous integration build job which may not support colored output. Even outside of build servers, a CLI may be used through an IDE's console that may not handle certain characters. Manual opt-out must be available.
Reference projects:
Reference to Open Source Node.js packages:
1.5 Rich interactions
Rich interactivity can be introduced in the form of prompt inputs, which are more sophisticated than free text, such as dropdown select lists, radio button toggles, rating, auto-complete, or hidden password inputs.
Another type of rich interactivity is in the form of animated loaders and progress-bars which provide a better experience for users when asynchronous work is being performed.
Many CLIs provide default command line arguments without requiring any further interactive experience. Don't force your users to provide parameters that the app can work out for itself.
Reference to Open Source Node.js packages:
1.6 Hyperlinks everywhere
https://www.github.com
), as well as source code (e.g: src/Util.js:2:75
) - both of which a modern terminal is able to transform into a clickable link.
git.io/abc
which requires your user to copy and paste manually.
If you are sharing links to URLs, or pointing to a file and a specific line number and column in the file, you can provide properly formatted links to both of these examples that, once clicked, will open up the browser, or an IDE at the defined location.
Reference projects:
1.7 Zero configuration
Aim to provide a "works out of the box" experience when running the CLI application.
For example, POSIX defines a standard for environment variable configuration used for different purposes, such as: TMPDIR
, NO_COLOR
, DEBUG
, HTTP_PROXY
and others. Detect these automatically and prompt for confirmation when necessary.
Reference projects which are built around Zero configuration:
- The Jest JavaScript Testing Framework
- Parcel, a web application bundler
1.8 Respect POSIX signals
Especially for CLI applications, it is common to interact with user input and improperly managing keyboard events
may result in your app failing to respond to SIGINT interrupts, commonly used by users when they hit the CTRL+C
keys.
The problem of not respecting process signals worsens when the program is being orchestrated by non-human interaction. For example, a CLI that runs in a docker container but will not respond to software interrupt signals sent to it.
2 Distribution
This section deals with best practices concerned with distributing and packaging a Node.js command line application in an optimal matter for consumers.
In this section:
- 2.1 Prefer a small dependency footprint
- 2.2 Use the shrinkwrap, Luke
- 2.3 Cleanup configuration files
2.1 Prefer a small dependency footprint
A fast npm install
for Node.js CLIs invoked with npx
will provide a better user experience. This is made possible when the overall dependency, and transitive dependency, footprint is kept to a reasonable size.
A one-off global npm
installation of a slow-to-install npm
package will offer a one-off poor experience, but the use of npx
by users to invoke executable packages will make the degraded performance, due to npx
always fetching and installing packages from the registry, more significant and obvious.
Reference projects:
- Bundlephobia is a tool to help you find the cost of a npm package.
2.2 Use the shrinkwrap, Luke
npm-shrinkwrap.json
as a lockfile to ensure that pinned-down dependency versions (direct and transitive) propagate to your end users when they install your npm package.
npm
) will resolve them during installation, and transitive dependencies installed via version ranges may introduce breaking changes that you can't control, that may result in your Node.js CLI application failing to build or run.
Use the force shrinkwrap, Luke!
Typically, an npm package only defines its direct dependencies, and their version range, when being installed, and the npm package manager will resolve all the transitive dependencies' versions upon installation. Over time, the resolved versions of dependencies will vary, as new direct, and transitive dependencies will release new versions.
Even though Semantic Versioning is broadly accepted among maintainers, npm is known to introduce many dependencies for an average package being installed, which adds to the risk of a package introducing changes that may break your application.
The flip side of using npm-shrinkwrap.json
is the security implications you are forcing upon your consumers. The dependencies being installed are pinned to specific versions, so even if newer versions of these dependencies are released, they won't be installed. This moves the responsibility to you, the maintainer, to stay up-to-date with any security fixes in your dependencies, and release your CLI application regularly with security updates. Consider using Snyk Dependency Upgrade to automatically fix security issues across your dependency tree. Full disclosure: I am a developer advocate at Snyk.
👍 TipUse
npm shrinkwrap
command to generate the shrinkwrap lockfile, which is of the same format as that of apackage-lock.json
file.
References:
- Do you really know how a lockfile works for yarn and npm packages?
- Yarn docs: Should lockfiles be committed to the repository?
2.3 Cleanup configuration files
As mentioned in the stateful data section, if your CLI application uses persistent storage, such as to save configuration files, then the CLI application should also be responsible for removing its configuration files when it gets uninstalled.
Due to npm package manager not providing uninstall hook since npm v7, your program should include an uninstallation option, either via arguments (e.g. --uninstall
) or via rich interaction.
👍 TipOptionally you can provide
pre
andpost
uninstall script which will be automatically called if the npm version is 6 or lower. You can find a working example in this repository.
3 Interoperability
This section deals with best practices concerned with making your Node.js CLI seamlessly integrate with other command line tools, and following conventions that are natural for CLIs to operate in.
This section answers questions such as:
- Can I export the output of this CLI for easy parsing?
- Can I pipe the output of this CLI to the input of another command line tool?
- Can I pipe the result of another tool to this CLI?
In this section:
- 3.1 Accept input as STDIN
- 3.2 Enable structured output
- 3.3 Cross-platform etiquette
- 3.4 Support configuration precedence
3.1 Accept input as STDIN
$ curl -s "https://api.example.com/data.json" | your_node_cli
If the command line application works with data, such as performing some kind of task on a JSON file that is usually specified with --file <file.json>
command line argument.
An example that is based on the official Node.js API docs for the for readline module of how taking input from a command pipe is as follows:
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
rl.question("What do you think of Node.js? ", (answer) => {
// TODO: Log the answer in a database
console.log(`Thank you for your valuable feedback: ${answer}`);
rl.close();
});
Then pipe the input to the above Node.js application:
echo "Node.js is amazing" | node cli.js
3.2 Enable structured output
It is often useful for users of a command line application to parse the data and perform other tasks with it, such as using it to feed web dashboards, or email notifications.
Being able to easily extract the data of interest from a command line output provides a friendlier experience to users of your CLI.
3.3 Cross-platform etiquette
Even though, from a program's perspective, the functionality isn't being stripped down and should execute well in different operating systems, some missed nuances may render the program inoperable. Let's explore several cases where cross-platform ethics must be honored.
Wrongly spawning a command
You might need to spawn a process that runs a Node.js program. For example, you have the following script:
program.js
#!/usr/bin/env node
// the rest of your app code
This works:
const cliExecPath = 'program.js'
const process = childProcess.spawn(cliExecPath, [])
This is better:
const cliExecPath = 'program.js'
const process = childProcess.spawn('node', [cliExecPath])
Why is it better? The program.js
source code begins with the Unix-like Shebang notation, however Windows doesn't know how to interpret this due to the fact that Shebang isn't a cross-platform standard.
This is also true for package.json
scripts. Consider the following bad practice
of defining an npm run script:
"scripts": {
"postinstall": "myInstall.js"
}
This is due to the fact that the Shebang in myInstalls.js
will not help Windows
understand how to run this with the node
interpreter.
Instead, follow the best practice of:
"scripts": {
"postinstall": "node myInstall.js"
}
Shell interpreters vary
Not all characters are treated the same across different shell interpreters.
For example, the Windows command prompt doesn't treat a single quote the same as a double quote, as would be expected on a bash shell, and so it doesn't recognize that the whole string inside a single quote belongs to the same string group, which will lead to errors.
This will fail in a Windows Node.js environment that uses the Windows command prompt:
package.json
"scripts": {
"format": "prettier-standard '**/*.js'",
...
}
To fix this so that this npm run
script will indeed be cross-platform between Windows, macOS and Linux:
package.json
"scripts": {
"format": "prettier-standard \"**/*.js\"",
...
}
In this example we had to use double quotes and escape them with the JSON notation.
Avoid concatenating paths
Paths are constructed differently across different platforms. When they are built manually by concatenating strings they are bound not to be interoperable between different platforms.
Let's consider the following bad practice example:
const myPath = `${__dirname}/../bin/myBin.js`;
It assumes that forward slash is the path separator where on Windows, for example, a backslash is used.
Instead of manually building filesystem paths, defer to Node.js's own path
module to do this:
const myPath = path.join(__dirname, "..", "bin", "myBin.js");
Avoid chaining commands with semicolons
Linux shells are known to support a semicolon (;
) to chain commands to run
sequentially, such as: cd /tmp; ls
. However, doing the same on Windows will fail.
Avoid doing the following:
const process = childProcess.exec(`${cliExecPath}; ${cliExecPath2}`);
Instead, use the double ampersand or double pipe notations:
const process = childProcess.exec(`${cliExecPath} || ${cliExecPath2}`);
3.4 Support configuration precedence
Detect and support configuration setting using environment variables as this will be a common way in many toolchains to modify the behavior of the invoked CLI application.
Configuration order of precedence for command line applications should follow this:
- Command line arguments specified when the application is invoked.
- The spawned shell's environment variables, and any other environment variables available to the application.
- The project scope configuration, e.g: a local directory
.git/config
file. - The user scope configuration, e.g: the user's home directory configuration file:
~/.gitconfig
or its XDG equivalent:~/.config/git/config
. - The system scope configuration, e.g:
/etc/gitconfig
.
Reference projects:
4 Accessibility
This section deals with best practices concerned with making a Node.js CLI application available to users who wish to consume it, but who are lacking the environment for which the maintainer designed the application.
In this section:
- 4.1 Containerize the CLI
- 4.2 Graceful degradation
- 4.3 Node.js versions compatibility
- 4.4 Shebang autodetect the Node.js runtime
4.1 Containerize the CLI
npm
or npx
available, and so won't be able to run your CLI application.
Installing Node.js CLI applications from the npm registry will typically be done with Node.js native toolchain such as npm
or npx
. These are common across JavaScript and Node.js developers, and are expected to be referenced within install instructions.
However, if you are targeting a CLI application to be consumed by the general public, regardless of their familiarity with JavaScript, or availability of this tooling, then distributing your CLI application only in the form of an install from the npm registry will be restricting. If your CLI application is intended to be used in a build or CI environment then those may also be required to install Node.js related toolchain dependencies.
There are many ways to package and distribute an executable, and containerizing it as a Docker container that is pre-bundled with your CLI application is an easily consumable alternative and dependency-free (aside of requiring a Docker environment).
4.2 Graceful degradation
It is common to provide a rich terminal display in the form of colorful output, ascii charts, or even animation on the terminal and powerful prompt mechanism. These may contribute to a great user experience for those who have a supported terminal, however it may display garbled text or be completely inoperable for those without it.
To enable users with an unsupported terminal to properly use the Node.js CLI application, you may choose to:
- Auto-detect a terminal capability and evaluate during run-time whether to degrade the CLI interactivity
- Provide an opt-in for users to explicitly toggle a graceful degradation, such as by providing a
--json
command line argument to force it to output raw data.
👍 Tip
Supporting graceful degradation such as JSON output isn't only useful for
end-users and their unuspported terminals, but is also valuable for running
in continuous integration environments, as well as enabling users
the ability to connect your program's output with other program's input or
export data if needed.
4.3 Node.js versions compatibility
Sometimes it may be necessary to specifically target older Node.js versions that are missing new ECAMScript features. For example, if you are building a Node.js CLI that is mostly geared towards DevOps or IT, they may not have an ideal Node.js environment with an up to date runtime. As a reference, Debian Stretch (oldstable) ships with Node.js 8.11.1.
If you do need to target old versions of Node.js such as Node.js 8, 6 or 4, all of which are End of Life, prefer to use a transpiler such as Babel to make sure the generated code is compliant with the version of the V8 JavaScript engine and the Node.js run-time shipped with those versions.
Another workaround is to provide a containerized version of the CLI to avoid old targets. See Section (4.1) Containerize the CLI.
Don't dumb down your program code to use an older ECMAScript language specification that matches unmaintained or EOL Node.js versions as this will only lead to code maintenance issues.
If the CLI is invoked in an unsupported environment, attempt to detect it and exit with a descriptive error message to present a friendly and information error message. See this example for dockly.
4.4 Shebang autodetect the Node.js runtime
#!/usr/bin/env node
.
#!/usr/local/bin/node
is only specific to your own environment and may render the Node.js CLI inoperable in other environments where the location of Node.js is different.
It may be an easy start to develop a Node.js CLI by running the entry point file as node cli.js
, and later on adding #!/usr/local/bin/node
to the top of the cli.js
file, however the latter is still a flawed approach as that location of the node
executable is not guaranteed for other users' environments.
It should be noted that specifying #!/usr/bin/env node
as the best practice, is still making the assumption that the Node.js runtime is referenced as node
and not nodejs
or otherwise.
5 Testing
In this section:
5.1 Put no trust in locales
As you choose to test the CLI by running it and parsing output, you may be inclined to grep for specific features to ensure that they exist in the output such as properly providing examples when the CLI is ran with no arguments. e.g:
const output = execSync(cli);
expect(output).to.contain("Examples:"));
When tests will run on locales that aren't English-based, and if your CLI argument parsing library supports auto-detection of locales and adopting to it, then tests such as this will fail, due to language conversions from Examples
to the locale-equivalent language being set as the default locale in the system.
6 Errors
This section deals with best practices concerned with making a Node.js CLI application available to users who wish to consume it but are lacking an ideal environment for which the maintainer designed the application.
In essence, the goals of the best practices laid out in this section is to help users troubleshoot errors quickly and easily, without needing to consult documentation or source code to understand errors.
In this section:
- 6.1 Trackable errors
- 6.2 Actionable errors
- 6.3 Provide debug mode
- 6.4 Proper use of exit codes
- 6.5 Effortless bug reports
6.1 Trackable errors
If possible, extend trackable error codes with further information so these can be easily parsed and context is clear.
Ensure that, when error messages are returned, they include a reference number or specific error codes that can later be consulted. Much like HTTP status codes, so to do CLI applications require named or coded errors.
Example:
$ my-cli-tool --doSomething
Error (E4002): please provide an API token via environment variables
6.2 Actionable errors
Example:
$ my-cli-tool --doSomething
Error (E4002): please provide an API token via environment variables
6.3 Provide debug mode
Use environment variables as well as command line arguments to enable extended debug verbosity levels. Where it make sense in your code, plant debug messages that aid the user, and the maintainer, to understand the program flow, inputs and outputs, and other pieces of information that make problem solving easier.
Reference to Open Source Node.js packages:
6.4 Proper use of exit codes
Command line scripts often make use of the shell's $?
to infer a program's status code and act upon it. This is also utilized in continuous integration (CI) flows to determine whether a step completed successfully or not.
If your CLI always terminates with no specific status code, even on errors, then the shell and other programs that rely upon it have no way of knowing this. When an error happens that results in your program's termination, you should convey this meaning. For example:
try {
// something
} catch (err) {
// cleanup or otherwise
// then exit with proper status code
process.exit(1);
}
A short reference for exit codes:
- exit code 0 conveys a successful execution
- exit code 1 conveys a failure
You may also choose to use customized exit codes with semantics of your program, but if you do, be sure to document them properly.
Reference: A list of exit codes used by the BASH shell
6.5 Effortless bug reports
7 Development
This section deals with development and maintenance best practices of building a Node.js command line application.
In this section:
- 7.1 Use a bin object
- 7.2 Use relative paths
- 7.3 Use the files field
7.1 Use a bin object
The following package.json
shows an example of decoupling the name of the executable from the filename and its location in the project:
"bin": {
"myCli-is-cool": "./bin/myCli.js"
}
7.2 Use relative paths
process.cwd()
to access user input paths and use __dirname
to access project-based paths.
You may find yourself with the need to access files within the project's files scope, or to access files that are provided
from the user's input, such as log, JSON files or others. Confusing the use of process.cwd()
or __dirname
can lead
to errors, as well as not using neither of them.
How to properly access files:
process.cwd()
: use it when the file path that you need to access depends on the relative location of the Node.js CLI. A good example for this is when the CLI supports file paths to create logs, such as:myCli --outfile ../../out.json
. IfmyCli
is installed in/usr/local/node_modules/myCli/bin/myCli.js
thenprocess.cwd()
will not refer to that location, but rather to the current working directory, which is whichever the directory the user is at when the CLI was invoked.__dirname
: use it when you need to access a file from within the CLI's source code and refer to a file from the relevant location of the file which the code lies in. For example, when the CLI needs to access a JSON data file in another directory:fs.readFile(path.join(__dirname, '..', 'myDataFile.json'))
.
files
field
7.3 Use the files
field to only include necessary files in your published packages.
To keep the published package size small, we should only include files that are required to run our CLI application. See this post for more details.
The following files
field tells the npm CLI to include all the files inside the src directory except the spec files.
"files": [
"src",
"!src/**/*.spec.js"
],
8 Analytics
This section deals with analytics collections in Node.js command line applications.
In this section:
8.1 Strict Opt-in Analytics
Understandably, as a maintainer of a CLI application you would want to understand better how users are using it. However, stealthly and by-default "phone home" type of behavior without asking consent from users will be frawned upon.
Guidelines:
- Let the users know which data will be collected and what are you doing with it.
- Be mindful about privacy concerns and collecting potentially personal identifyable information.
- How, where and for which period of time is data stored.
References for other CLIs which collect analytics are Angular CLI, and Next.js CLI.
9 Appendix: CLI Frameworks
9.1 CLI Frameworks Table
Name | Description | npm | GitHub | Stars and downloads |
---|---|---|---|---|
oclif | A framework for building a command line interface. | Link to npm | Link to GitHub | |
inquirer | A collection of common interactive command line user interfaces. | Link to npm | Link to GitHub | |
ink | Ink provides the same component-based UI building experience that React offers in the browser, but for command-line apps. | Link to npm | Link to Github | |
pastel | Next.js-like framework for CLIs made with Ink. | Link to npm | Link to Github | |
ink UI | Collection of customizable UI components for CLIs made with Ink. | Link to npm | Link to Github | |
blessed | A curses-like library with a high level terminal interface API for node.js. | Link to npm | Link to GitHub | |
prompts | Lightweight, beautiful and user-friendly interactive prompts | Link to npm | Link to GitHub | |
vue-termui | A Vue.js based terminal UI framework that allows you to build modern terminal applications with ease. | Link to npm | Link to GitHub | |
clack | Effortlessly build beautiful command-line apps | Link to npm | Link to GitHub |
10 Appendix: CLI educational resources
- https://clig.dev/
- https://primer.style/cli/getting-started/principles
- @simonplend and @dolearning's workshop on crafting human friendly CLIs
Author
Node.js CLI Apps Best Practices © Liran Tal, Released under CC BY-SA 4.0 License.
This project follows the all-contributors specification. Contributions of any kind welcome!
License
This work is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.