Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/DiegoZoracKy/magicli
Automagically generates command-line interfaces (CLI) for any module. Expected options and help sections are created automatically based on parameters names, with support to async.
https://github.com/DiegoZoracKy/magicli
cli command-line command-line-interface nodejs
Last synced: 16 days ago
JSON representation
Automagically generates command-line interfaces (CLI) for any module. Expected options and help sections are created automatically based on parameters names, with support to async.
- Host: GitHub
- URL: https://github.com/DiegoZoracKy/magicli
- Owner: DiegoZoracKy
- Created: 2017-11-05T23:33:49.000Z (about 7 years ago)
- Default Branch: master
- Last Pushed: 2021-01-25T02:50:51.000Z (almost 4 years ago)
- Last Synced: 2024-09-19T01:58:28.116Z (about 2 months ago)
- Topics: cli, command-line, command-line-interface, nodejs
- Language: JavaScript
- Size: 17.6 KB
- Stars: 179
- Watchers: 5
- Forks: 7
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# MagiCLI
[![Build Status](https://api.travis-ci.org/DiegoZoracKy/magicli.svg)](https://travis-ci.org/DiegoZoracKy/magicli) [![npm](https://img.shields.io/npm/v/magicli.svg)]() [![npm](https://img.shields.io/npm/l/magicli.svg)]()
Automagically generates command-line interfaces (CLI), for any module.
Just `require('magicli')();` and your module is ready to be executed via CLI.The main goal is to have any module prepared to be executed via CLI (installed globally with `-g`, or by using **npx**):
To see why I believe you should plug it on your module, even if you don't need a CLI (it probably will serve someone on the community), read here: [Introducing MagiCLI: Automagically generates a command-line interface (CLI) for any module
](https://hackernoon.com/introducing-magicli-automagically-generates-a-command-line-interface-cli-for-any-module-49543e50f86d)**It can be installed globally, in order to *execute* any module or .js file via CLI.**
## Goals
* Minimal setup (*one line*)
* Automatic options names based on functions parameters
* Out of the box support to async functions (`Promises`, or any *thenable* lib)
* A specific help section for each nested property (*"subcommands"*)
* *Name*, *Description* and *Version* extracted from package.json
* Simple API to hook into the execution flow (*stdin*, *before*, *after*)
* Cover all possible cases of module.exports (*Function*, *Object* with nested properties, Destructuring parameters)
* Provide a CLI to be used to execute any given module or .js file via CLI## Usage (the most simple and minimal way)
* `npm install magicli`
* Add the property **bin** to your package.json containing the value **./bin/magicli.js**
* Create the file **./bin/magicli.js** with the following content:```javascript
#!/usr/bin/env noderequire('magicli')();
```**Done!** Install your module with `-g`, or use it via **[npx](http://blog.npmjs.org/post/162869356040/introducing-npx-an-npm-package-runner)**, and run it with `--help` to see the result. The `--version` option will show the same value found at *package.json*. In the same way you can just run `node ./bin/magicli.js --help` to test it quickly, without installing it.
Let's suppose that **your-module** exports the function:
```javascript
module.exports = function(param1, param2) {
return param1 + param2;
}
```When calling it via CLI, with `--help`, you will get:
```bash
Description:Same description found at package.json
Usage:
$ your-module [options]
Options:
--param1
--param2
```The program will be expecting options with the same name as the parameters declared at the exported function, and it doesn't need to follow the same order. Example:
`$ your-module --param2="K" --param1="Z"` would result in: `ZK`.
Important: MagiCLI requires the module in order to analyse it, and provide the command-line interface for it. Keep that in mind in case your module does something just by being required.
## Usage via CLI
In order to **execute** any module or .js file via CLI, install it globally:
```bash
$ npm install magicli -g
```Then just pass in as the first argument, the path to a module or a .js file. Examples:
* `$ magicli . --help`
* `$ magicli ./path/to-some-module --help`
* `$ magicli ./path/to-a-file.js --help`Or use it via **[npx](http://blog.npmjs.org/post/162869356040/introducing-npx-an-npm-package-runner)** without the need to install it.
Let's suppose that you have a simple .js file as this one:
```javascript
module.exports = {
sum: (n1, n2) => n1 + n2,
ec: {
ho: str => `${str} !!!`
}
}
```Just execute **magicli** on it, as `$ magicli ./path/to-the-file-above.js --help` and you will get:
```bash
Commands:
sum
ec-ho
````$ magicli ./path/to-the-file-above.js sum --help` will give you:
```bash
Usage:
$ sum [options]Options:
--n1
--n2
```
and `$ magicli ./path/to-the-file-above.js sum --n1=4 --n2=2` will result in `6`### How it works
MagiCLI is capable of handling many styles of `exports`, like:
* Functions
* Object Literal
* Nested properties
* Class with static methodsAnd also any kind of parameters declaration (*Destructuring Parameters*, *Rest Parameters*).
If **your-module** were like this:
```javascript
// An Arrow function with Destructuring assignment and Default values
const mainMethod = ([p1, [p2]] = ['p1Default', ['p2Default']], { p3 = 'p3Default' } = {}) => `${p1}-${p2}-${p3}`;// Object Literal containing a nested method
module.exports = {
mainMethod,
nested: {
method: param => `nested method param value is: "${param}`
}
};
````$ your-module --help` would result in:
```bash
Description:Same description found at package.json
Usage:
$ your-module
Commands:
mainMethod
nested-method
````$ your-module mainMethod --help` would be:
```bash
Usage:$ your-module mainMethod [options]
Options:
--p1
--p2
--p3
````$ your-module nested-method --help` returns:
```bash
Usage:$ your-module nested-method [options]
Options:
--param
```Calling *mainMethod* without any parameter:
`$ your-module mainMethod`results in:
` p1Default-p2Default-p3Default`While defining the parameter for *nested-method*:
`$ your-module mainMethod nested-method --param=paramValue`would return:
` nested method param value is: "paramValue"`Note: Nested methods/properties will be turned into commands separated by `-`, and it can be configurable via options (`subcommandDelimiter`).
## Usage Options
`magicli({ commands = {}, validateRequiredParameters = false, help = {}, version = {}, pipe = {}, enumerability = 'enumerable', subcommandDelimiter = '-'})`Options are provided to add more information about commands and its options, and also to support a better control of a command execution flow, without the need to change the source code of the module itself (for example, to `JSON.stringify` an `Object Literal` that is returned).
### enumerability
By default, only the enumerable nested properties will be considered. The possible values are: `'enumerable'` (default), `'nonenumerable'` or `'all'`.
### validateRequiredParameters
MagiCLI can validate the required parameters for a command and show the help in case some of them are missing. The default value is `false`.### help
**help.option**
To define a different option name to show the help section. For example, if `'modulehelp'` is chosen, `--modulehelp` must be used instead of `--help` to show the help section.**help.stripAnsi**
Set to `true` to strip all ansi escape codes (colors, underline, etc.) and output just a raw text.### version
**version.option**
To define a different option name to show the version. For example, if `'moduleversion'` is chosen, `--moduleversion` must be used instead of `--version` to show the version number.### pipe (stdin, before and after)
The pipeline of a command execution is:
**stdin** (command.pipe.stdin || magicliOptions.pipe.stdin) =>
**magicliOptions.pipe.before** =>
**command.pipe.before** =>
**command.action** (the method in case) =>
**command.pipe.after** =>
**magicliOptions.pipe.after** =>
**stdout**
Where each of these steps can be handled if needed.
As it can be defined on *commands* option, for each command, **pipe** can also be defined in *options* to implement a common handler for all commands. The expected properties are:
**pipe.stdin**
`(stdinValue, args, positionalArgs, argsAfterEndOfOptions)`Useful to get a value from *stdin* and set it to one of the expected *args*.
**pipe.before**
`(args, positionalArgs, argsAfterEndOfOptions)`To transform the data being input, before it is passed in to the main command action.
**pipe.after**
`(result, parsedArgs, positionalArgs, argsAfterEndOfOptions)`Note: **stdin** and **before** must always return *args*, and **after** must always return *result*, as these values will be passed in for the next function in the pipeline.
### commands
The options are effortlessly extracted from the parameters names, however it is possible to give more information about a command and its options, and also give instructions to the options parser.**commands** expects an `Object Literal` where each key is the command name. It would be the module's name for the main function that is exported, and the command's name as it is shown at the *Commands:* section of `--help`. For example:
```javascript
commands: {
'mainmodulename': {},
'some-nested-method': {}
}
```For each command the following properties can be configurable:
#### options
Is an *Array* of *Objects*, where each contains:**name** (*required*)
The name of the parameter that will be described**required**
To tell if the parameter is required.**description**
To give hints or explain what the option is about.**type**
To define how the parser should treat the option (Array, Object, String, Number, etc.). Check [yargs-parser](https://github.com/yargs/yargs-parser) for instructions about *type*, as it is the engine being used to parse the options.**alias**
To define an alias for the option.#### pipe (stdin, before and after)
The pipeline of a command execution is:
**stdin** (command.pipe.stdin || magicliOptions.pipe.stdin) =>
**magicliOptions.pipe.before** =>
**command.pipe.before** =>
**command.action** (the method in case) =>
**command.pipe.after** =>
**magicliOptions.pipe.after** =>
**stdout**
Where each of these steps can be handled if needed.
As it can be defined on *options* to implement a common handler for all commands, **pipe** can also be defined for each command.
**pipe.stdin**
`(stdinValue, args, positionalArgs, argsAfterEndOfOptions)`Useful to get a value from *stdin* and set it to one of the expected *args*.
**pipe.before**
`(args, positionalArgs, argsAfterEndOfOptions)`To transform the data being input, before it is passed in to the main command action.
**pipe.after**
`(result, parsedArgs, positionalArgs, argsAfterEndOfOptions)`Note: **stdin** and **before** must always return *args*, and **after** must always return *result*, as these values will be passed in for the next function in the pipeline.
If needed, a more thorough guide about this section can be found at [cliss](https://github.com/DiegoZoracKy/cliss) (as this is the module under the hood to handle that)
A full featured use of the module would look like:
```javascript
magicli({
commands,
enumerability,
subcommandDelimiter,
validateRequiredParameters,
help: {
option,
stripAnsi
},
version: {
option
},
pipe: {
stdin: (stdinValue, args, positionalArgs, argsAfterEndOfOptions) => {},
before: (args, positionalArgs, argsAfterEndOfOptions) => {},
after: (result, parsedArgs, positionalArgs, argsAfterEndOfOptions) => {}
}
});
```## Example
To better explain with an example, let's get the following module and configure it with MagiCLI to:
* Define **p1** as `String` (*mainMethod*)
* Write a description for **p2** (*mainMethod*)
* Define **p3** as required (*mainMethod*)
* Get **p2** from stdin (*mainMethod*)
* Use **before** (command) to upper case **param** (*nested-method*)
* Use **after** (command) to JSON.stringify the result of (*nested-method*)
* Use **after** (options) to decorate all outputs (*nested-method*)**module** ("main" property of package.json)
```javascript
'use strict';module.exports = {
mainMethod: (p1, p2, { p3 = 'p3Default' } = {}) => `${p1}-${p2}-${p3}`,
nested: {
method: param => {// Example of a Promise being handled
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve({ param });
}, 2000);
});
}
}
};
```**magicli.js** ("bin" property of package.json)
```javascript
#!/usr/bin/env noderequire('../magicli')({
commands: {
'mainMethod': {
options: [{
name: 'p1',
description: 'Number will be converted to String',
type: 'String'
}, {
name: 'p2',
description: 'This parameter can be defined via stdin'
}, {
name: 'p3',
required: true
}],
pipe: {
stdin: (stdinValue, args, positionalArgs, argsAfterEndOfOptions) => {
args.p2 = stdinValue;
return args;
}
}
},
'nested-method': {
options: [{
name: 'param',
description: 'Wait for it...'
}],
pipe: {
before: (args, positionalArgs, argsAfterEndOfOptions) => {
if (args.param) {
args.param = args.param.toUpperCase();
}
return args;
},after: JSON.stringify
}
}
},
pipe: {
after: (result, positionalArgs, argsAfterEndOfOptions) => `======\n${result}\n======`
}
});
```## Tests
There is another repository called [MagiCLI Test Machine](https://github.com/DiegoZoracKy/magicli-test-machine), where many real published modules are being successfully tested. As the idea is to keep increasing the number of real modules tested, it made more sense to maintain a separated repository for that, instead of being constantly increasing the size of MagiCLI itself over time. I ask you to contribute with the growing numbers of those tests by adding your own module there via a pull request.
If you find some case that isn't being handled properly, please open an *issue* or feel free to create a PR ;)