Skip to content
/ cwd Public

Run CLI commands with Node. A wrapper for Node's "child_process" module.

License

Notifications You must be signed in to change notification settings

taitulism/cwd

Repository files navigation

run-in-cwd

License: MIT Build Status

Run CLI commands with Node.

Table Of Contents

Get Started

  1. Install:
    $ npm install run-in-cwd
  2. Require:
    const createCwd = require('run-in-cwd')
  3. Get a Cwd Instance:
    const projectDir = createCwd('/path/to/target/folder')
  4. Run a command:
    // run to completion
    projectDir.runCmd('git status').then((results) => {...})
    
    // run with more control
    projectDir.spawn('git status')
        .on('line/out', (line) => {...})
        .on('close', (exitCode) => {...})

 

WARNING: If you let your users pass in the command and/or any of its arguments - make sure they are safe.

 

Default Instance

run-in-cwd exports a default instance ready to use if your target folder is in which the current process runs in. Meaning, what process.cwd() returns.

const cwd = require('run-in-cwd');

cwd.runCmd('git status').then(...)

It's the same instance you would get if you run createCwd('./') or even createCwd()

NOTE: For context's sake, do not extract cwd methods:

// DO NOT:
const {runCmd} = require('run-in-cwd')

Instance API


.runCmd( cmd, [args, [options] ] )


Executes a command and returns a completion promise (command exits).
Similar to Node's child_process.exec and child_process.execFile (see docs).

Don't forget to handle errors with promise.catch() or a try-catch wrapper.

Arguments:

  • cmd (Required) - A command string (e.g. 'npm')
  • args - An array of the command's arguments (e.g. ['-flag', 'key=value']). Could also be a string or a number for a single argument.

NOTE: cmd string could also include the arguments e.g. .runCmd('npm install')

  • options - spawn options object. See Node's docs.
    Additional option:
    • maxCacheSize - Limit the command cache in MegaBytes. The limit is the total output for both stdout & stderr streams. Default value is 10. An exception is thrown when max size is exceeded.
      cwd.runCmd('cat longFile.txt', {maxCacheSize: 20}) // 20 MB limit

Returns: A promise for a command results object.

A result object has the following properties:

  • exitCode - Number - The command exit code.
  • isOk - Boolean - true if command exit code is 0. false otherwise.
  • stdout - String - The commands's stdout output string (utf-8 encoded).
  • stderr - String - The commands's stderr output string (utf-8 encoded).
  • output - String - Both streams' output string (utf-8 encoded).
  • stdoutLines - Array - The commands's stdout output, split to lines.
  • stderrLines - Array -The commands's stderr output, split to lines.
  • outputLines - Array - Both streams' output, split to lines.

Examples:
Promise style:

cwd.runCmd('npm install')
    .then({isOk, stderr} => {
        // ...
    })
    .catch((err) => {
        // handle exception...
    });

Async-Await style:

(async () => {
    try {
        const {isOk, stdoutLines} = await projectDir.runCmd('npm install')

        // ...
    }
    catch (err) {
        // handle exception...
    }
})();

The difference between !isOk, stderr and error:

A CLI command creates a process that has two output channels it utilize to communicate with its parent process: one for the command's standard output (stdout), and one for its errors (stderr). Every command also finishes with an exit code. The consensus is exit code 0 for success (a kind of an HTTP "200 OK" status code) and non-zero otherwise. Different commands use those channels and exit codes differently, even "incorrectly" as there is no standard other than subjective common sense...

So when a command fails it will probably send some details through its stderr channel and exit with a non-zero exit code but it will be completed. No errors will be thrown in the Node process that executed the command.

An exception is thrown when there was a problem with the command execution in our environment. For example when the command itself is not found (e.g. a typo like ggit).

 

 


.spawn( cmd, [args, [options] ] )


Runs a command and returns a child process.

It is highly recommended to handle errors:

childProc.on('error', (err) => { /* handle error */ })

Arguments:

  • cmd (Required) - A command string (e.g. 'npm')
  • args - An array of the command's arguments (e.g. ['-flag', 'cache=500']). Could also be a string or a number for a single argument.

NOTE: cmd string could also include the arguments e.g. .spawn('npm install')

Returns: <child_process>

Examples:
Calling .spawn() is only the first part of spawning a process. We then need to handle the returned process's lifecycle events.

First, spawn a child process:

const childProc = cwd.spawn('npm install --save')

 

<child_process>

.spawn's return object is Node's native ChildProcess with additional events.

  • Event: 'line'

    Is triggered for each line of both stdout and stderr streams. Text is split to lines by newline characters.

  • Event: 'line/out'

    Same as line event, but for stdout only.

  • Event: 'line/err'

    Same as line event, but for stderr only.

const cwd = require('run-in-cwd');
const childProc = cwd.spawn('git status') 

let isClean = false;

childProc.on('line/out', (line) => {
    if (line.includes('nothing to commit')) {
        isClean = true;
    }
})

childProc.on('close', (exitCode) => {
    if (exitCode === 0 && isClean) { // 0 is like 200 OK. 
        console.log('Branch is clean')
    }
    else {
        console.log('Stage & Commit')
    }
})

 


.runShellCmd( cmd, [args, [options] ] )

.spawnShell( cmd, [args, [options] ] )


Those are the shelled versions of .runcCmd() and .spawn(), respectivly. Meaning, the option {shell: true} is used. Use the shelled versions when you need to chain commands, redirect I/O, file globbing patterns and other shell behvior.

For example:

cwd.runShellCmd('git commit -m "nice feature" && git status 1>last-status.txt')

 


.parentProcess


Returns a cwd instance that redirects all of its commands' I/O to its parent process. It utilizes Node's {stdio:inherit} option. Read more.

cwd.parentProcess.runCmd('git status')

Is the equivalent of:

cwd.runCmd('git status', {stdio: 'inherit'})

NOTE: All cwd instances (including the default instance) have the .parentProcess prop.

When the parent process is Node's global process object, it usually means write output to the terminal screen and read input from the keyboard. In this case the above code acts just like running git status from the terminal yourself.

 


What is CWD?

CWD stands for: Current Working Directory.
When working with a Command Line Interface (CLI) you execute commands from within a certain directory, your code folder, for example:

# Windows
C:\path\code\>

# Linux
~/path/code $

This basic concept of running commands from within a certain folder was the main trigger to create run-in-cwd.

 


run-in-cwd vs. Node's child_process

Command arguments

With Node, when you want to run a simple command with arguments like: git status you would normally either (1) pass a command string and an arguments array (even for a single argument) or (2) add the {shell: true} option.

const childProc = require('child_process')

childProc.spawn('git', ['status'])
childProc.spawn('git status', {shell: true})

run-in-cwd takes care of the arguments for you

const cwd = require('run-in-cwd')

cwd.spawn('git status')

And when you really need a shell:

cwd.spawnShell('git status')

 

Setting the working directory

Both ways use process.cwd() as the default working directory for running commands.

When you need to run commands on the same directory but it's not your Current Working Directory, with Node, you will find yourself repetitively using the "cwd" option.

Node's child process:

const childProc = require('child_process')

childProc.spawn('git', ['status'], {cwd: '../path-to/my-folder'})
childProc.spawn('git', ['commit'], {cwd: '../path-to/my-folder'})
childProc.spawn('git', ['push', 'origin', 'master'], {cwd: '../path-to/my-folder'})

With run-in-cwd you only do it once:

const createCwd = require('run-in-cwd')
const cwd = createCwd('../path-to/my-folder')

cwd.spawn('git status')
cwd.spawn('git commit')
cwd.spawn('git push origin master')

 

Data Events

To read a command's output we need to listen to the command streams 'data' events.

Data events emit chunks:

cp.stdout.on('data', (chunk) => {
    console.log(chunk)
    // <Buffer 68 65 6c 6c 6f 20 77 6f 72 6c 64>
})

Data chunks, besides being buffers, as random pieces of data cannot gurantee you get a whole word or a full sentence.

run-in-cwd provides some higher-level events that emit text (UTF-8 strings), split into lines.

So the equivalent of the native way:

// Native Node
const childProc = require('child_process')
const cp = childProc.spawn('git', ['status']);

cp.stdout.on('data', (chunk) => ...)
cp.stderr.on('data', (chunk) => ...)

would be:

// run-in-cwd
const cwd = require('run-in-cwd')
const cp = cwd.spawn('git status');

cp.on('line/out', (line) => ...)
cp.on('line/err', (line) => ...)

// emits for both
cp.on('line', (line) => ...)

About

Run CLI commands with Node. A wrapper for Node's "child_process" module.

Resources

License

Stars

Watchers

Forks

Packages

No packages published