Yet another shell command executor (cmd, bash) and NodeJS fork with Inter Process Communication (IPC).
import { Shell } from 'shellbee'
// executes commmands and waits until they exit
let shell: Shell = await Shell.run(command: string | string[] | IShellParams);
// instatiate the new shell instance, attach all required event listeners, etc, and later call `run` to start executing
let shell = new Shell({ command: `git commit -am "foo"` });
interface IShellParams {
command?: string | IProcessSingleParams
commands?: (string | IProcessSingleParams)[]
detached?: boolean
cwd?: string
/** Watches the output and emits ready event when the child process prints expected text to the std */
matchReady?: RegExp
silent?: boolean
parallel?: boolean
// command should container js file to fork
fork?: boolean
// Defines if a child_process supports IPC
ipc?: boolean
timeoutsMs?: number
restartOnErrorExit?: boolean
}
interface IProcessSingleParams {
command: string
cwd?: string
detached?: boolean
matchReady?: RegExp
extract: {[ key: string ]: (output: string) => any }
}
interface IShell {
commands: ICommandOptions[]
results = [] as ProcessResult[]
std: string[] = [];
stderr: string[] = [];
stdout: string[] = [];
start: Date
end: Date
busy: boolean
/** Resolves after the all commands are settled */
promise: Promise<Shell>
run (): Promise<Shell>
kill()
onStart (cb: (data: { command: string }) => void): this
onStdout (cb: (data: { command: string, buffer: string }) => void): this
onStderr (cb: (data: { command: string, buffer: string }) => void): this
onExit (cb: (data: { command: string, code: number, result: ProcessResult }) => void): this
/** When rgxReady is specified the event will be called */
onReady (cb: ({ command: string }) => void): this
onComplete(cb: (shell: Shell) => void): this
}
// main.js
import { Shell } from 'shellbee'
let shell = new Shell({
command: 'bar.js',
fork: true,
ipc: true
})
shell.run();
let result = await shell.send('doFoo', { foo: 1 });
// bar.js
import { Shell } from 'shellbee'
Shell.ipc({
doFoo (...args) {
console.log(args);
return 'lorem';
}
});
- Fork a process as usual.
let shell = new Shell({ command: 'bar.js', fork: true });
shell.run();
- In the worker listen for a message
{
id: string,
method: string
args: any[]
}
The child process should process the work and sends a message with the id
and result back to parent process:
{
id: string,
data?: any,
error?: Error
}
Simple Forked file example:
// worker.js
process.on('message', async (message: { id, method, args }) => {
let result = await MyWorker.doWork(message.method, ...message.args);
process.send({
id: message.id,
data: result,
error: null
});
});
Execute a CLI task and makes sure it restarts on error or when output got stalled. For example you want to run a script from npm scripts forever, e.g. "npm run foo"
shellbee run npm run foo
--delay <ms_delay_between_restart>
--restart <max_restart_count_within_window>
--restart-window <ms_window_errors>
--stalled-output <restart_on_ms_no_output>
dalay
- Milliseconds before the process is started againrestart
- Amount of restarts within a time window. For example if the process has been crashed for 10 times within 30 seconds, do not restart any more.restart-window
- Milliseconds, a time-frame to count crashes in, when to many crashes occur (defined byrestart
)shellbee
stops restarting the process.stalled-output
- Milliseconds, if the process has no output (stdout
,stderr
) for this time, then we assume the process has stopped responding, soshellbee
will restart it