npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

pixl-cli

v1.0.21

Published

Tools for building command-line apps for Node.js.

Downloads

3,683

Readme

Overview

This module provides utilities for creating command-line Node.js apps. Features include automatic parsing of command-line args into simple key/value pairs, prompting the user for information, and displaying graphical info boxes and progress bars.

Usage

Use npm to install the module:

npm install pixl-cli

Then use require() to load it in your code:

const cli = require('pixl-cli');

Basic Tools

Here are some of the more basic functions provided by the package.

Printing

A very simple cli.print() method is provided, which prints any string to STDOUT. This can be shortened to just print() if you import everything into the global namespace (see Importing Into Global below). Example:

const cli = require('pixl-cli');
cli.global();

print("Hello world!\n");

Unlike console.log() this does not add an EOL at the end of each string. It works just like the standard print() function from other languages. However, a println() method is also provided which does add an EOL.

Note that print() will be silent if --quiet mode is enabled. See Quiet Mode below.

An alternate function is provided for printing verbose output, called cli.verbose() (or just verbose() if imported into global). This function also prints a string to STDOUT, but only does so if --verbose mode is enabled. Example:

const cli = require('pixl-cli');
cli.global();

verbose("This will only be printed in verbose mode.\n");

See Verbose Mode below for details.

STDERR

To print something to STDERR you can use the cli.warn() method. This works similarly as cli.print() in that it honors Quiet Mode, does not include a trailing EOL, and gets imported to the global namespace if cli.global() is called. Example:

const cli = require('pixl-cli');
cli.global();

warn("This will be printed to STDERR.\n");

Dying

To print something to STDERR and exit immediately afterward, you can call cli.die() and pass in a message. This will exit with a non-zero code indicating that the process "failed". This method also gets imported to the global namespace if cli.global() is called. Example:

const cli = require('pixl-cli');
cli.global();

die("A fatal error occurred.\n");

Logging

To enable logging mode, so all calls to print(), verbose(), warn() and die() also get logged to a file, call cli.setLogFile() and pass in a path. The file need not exist, but the directory should. Example:

cli.setLogFile( "/let/log/myscript.log" );

Note that once the log file is set, everything printed is logged, even if quiet mode is enabled. Additionally, all calls to cli.verbose() are also logged, even if --verbose mode is not enabled.

The log file format is simply a date/time surrounded by square brackets, followed by a single space, followed by the raw text printed. All color is automatically stripped. Example log snippet:

[2019/05/18 18:49:12] Hello there!
[2019/05/18 18:49:13] Good bye!

You can also call cli.log() to log something directly without also printing it to the console. If you pass an object to cli.log() it is serialized to JSON.

Loading and Saving Files

Simple methods are provided to load and save files to/from strings. These are both synchronous calls. They are loadFile() which accepts a file path and returns the contents as a string, and saveFile() which accepts a file path and contents as a string. saveFile() writes to the specified file, replacing any existing content, and creating the file if necessary. Example of both functions:

const cli = require('pixl-cli');

let contents = cli.loadFile( "my-file.txt" );
cli.print( "File contents: " + contents + "\n" );

contents = "Replacing with new content!";
cli.saveFile( "my-file.txt", contents );

If you want to append to a file instead of replacing its contents, use cli.appendFile(). It accepts the same arguments as cli.saveFile(). Example:

contents = "Appending a new line!\n";
cli.appendFile( "my-file.txt", contents );

Note: These functions may throw errors if the underlying filesystem operations fail, so it is advisable to wrap them in a try/catch block.

TTY Detection

To detect whether your script's output is attached to a TTY (meaning, it is attached to a user terminal and not being piped to a file or script), you can call cli.tty(). This will return a Boolean true or false. Example:

if (cli.tty()) {
	cli.print("Hello user terminal!\n");
}

Specifically, this queries the STDOUT TTY. Meaning, the script's output is connected to a terminal, but its input may still be a pipe of some kind.

To detect the width of the current TTY (user's terminal), you can call cli.width(). If a TTY is attached, this will return the current number of characters that will fit in one single line. Example:

let width = cli.width();
cli.print("Your terminal is " + width + " characters wide.\n");

Other Tools

The following utility functions from the pixl-tools module are also imported, so you can use them directly:

| Function Name | Description | |---------------|-------------| | cli.getTextFromBytes() | Convert a byte count into a human readable string, e.g. 5 MB. | | cli.commify() | Apply commas to a positive integer using US-style formatting, e.g. 1,000,000. | | cli.shortFloat() | Trim floating point decimal to 2-digit precision, unless digits are zeros. | | cli.pct() | Return percentage string given arbitrary value and a maximum limit, e.g. '55%'. | | cli.zeroPad() | Pad an integer with zeros on the left side, up to a specified max. | | cli.getTextFromSeconds() | Convert a number of seconds into a human-readable string, e.g. 3 hours. | | cli.getNiceRemainingTime() | Calculate estimated remaining time, given progress and start time. | | cli.pluralize() | Apply English language pluralization to a word, based on a specified value. | | cli.ucfirst() | Upper-case the first character of a string, lower-case the rest. |

In fact, the entire pixl-tools module is made available to you as cli.Tools, so you don't have to import it separately.

Command-Line Arguments

Any arguments passed to your script on the command-line are parsed using the pixl-args module, and provided as simple key/value pairs in cli.args. Example:

node my-script.js --name "Joseph Huckaby" --city San\ Mateo

Then cli.args will contain:

{
	"name": "Joseph Huckaby",
	"city": "San Mateo"
}

Several different kinds of arguments are available. You can use single or double-dashes, strings which appear to be integers or floats are parsed as such, any switch without a value is set to true, any repeated switches are converted to an array of values, and any values provided without a switch are appended to a special other array, which can come before or after all the switches. Here is an example of all these things:

node my-script.js file1.txt file2.txt --name "Joe" --amount 50 --freq 0.5 -z 1 --verbose --add thing1 --add thing2

Then cli.args will contain:

{
	"other": ["file1.txt", "file2.txt"],
	"name": "Joe",
	"amount": 50,
	"freq": 0.5,
	"z": 1,
	"verbose": true,
	"add": ["thing1", "thing2"]
}

Please see the pixl-args module documentation for more details.

Argument Aliases

You can support argument aliases (e.g. -v for --verbose) by providing a map to the mapArgs() function. Example:

const cli = require('pixl-cli');
cli.mapArgs({
	'v': 'verbose',
	'q': 'quiet',
	'd': 'debug'
});

Verbose Mode

The --verbose switch has a special meaning. It is used by the cli.verbose() function, and will control whether it outputs anything or not. cli.verbose() works just like cli.print() except that it only prints if the --verbose switch is present on the command-line. Example use:

const cli = require('pixl-cli');
cli.verbose("This will only be printed in verbose mode.\n");

Quiet Mode

The --quiet switch also has a special meaning. If passed on the command-line, it will mute all calls to cli.print() and of course cli.verbose(), meaning they will not output anything at all. This is useful for giving your script a special silent mode, possibly for running in a crontab.

Prompting The User

To prompt the user for input, you can call cli.prompt(). Pass in a string to prompt them with, a default answer, and a callback function which will be fired and passed their answer. This is an asynchronous operation, so beware of code flow. Example:

const cli = require('pixl-cli');

cli.prompt("What is your name?", "", function(name) {
	cli.print("Hello " + name + "!\n");
} );

If a default string is provided, it is displayed just after the question in square brackets, which is the typical way of hinting a default answer. If the user presses Enter without typing in a value, your default answer will be sent to your callback.

Note that if your script is running without a TTY (i.e. without an attached terminal, as in piped to a file or another script), no prompt is displayed, and your callback is fired immediately and passed your default answer.

Yes/No Questions

If you just need to ask the user a simple yes/no question, a convenience method is provided. Just call cli.yesno(), provide a string to prompt them with, a default answer (y or n), and a callback. This time your callback is passed Boolean true if the user answered yes (or y or any word that contains a y), or false for no or anything non-yes. Example:

cli.yesno("Do you like bacon?", "y", function(yes) {
	if (yes) cli.print("You can be my friend.\n");
	cli.print("You are dead to me.\n");
} );

Note that if your script is running without a TTY (i.e. without an attached terminal, as in piped to a file or another script), no prompt is displayed, and your callback is fired immediately and passed true or false based on your default answer.

Displaying Info Boxes

Call cli.box() to render a string (or paragraph) of text surrounded by an ASCII art border. Example:

const cli = require('pixl-cli');

cli.print( 
	cli.box("The quick brown fox jumped over the lazy dog.") + "\n" 
);

This would produce:

┌───────────────────────────────────────────────┐
│ The quick brown fox jumped over the lazy dog. │
└───────────────────────────────────────────────┘

Absent any options, the box will be sized to fit your text string, with exactly one space of horizontal space on either side. You can customize this and several other options by passing in an object as the 2nd argument to the cli.box() method. Here are all the possible properties:

| Property Name | Description | |---------------|-------------| | width | An optional fixed width. If specified and the string exceeds the width, it will be word-wrapped. | | styles | An array of chalk styles or functions to apply to the box border graphics, defaults to ["gray"]. | | hspace | An optional number of spaces to insert on the left and right sides of the text (defaults to 1). | | vspace | An optional number of vertical lines to insert above and below the text (defaults to 0). | | indent | Number of characters to indent the box by (defaults to 0). |

Example:

let text = "Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.";

cli.print(
	cli.box( text, {
		width: 40,
		styles: ["magenta", function( text ) { return text.toUpperCase() }],
		hspace: 2,
		vspace: 1,
		indent: 4
	} ) + "\n"
);

This will produce:

	┌────────────────────────────────────────────┐
	│                                            │
	│  Lorem ipsum dolor sit amet, consectetur   │
	│  adipisicing elit, sed do eiusmod tempor   │
	│  incididunt ut labore et dolore magna      │
	│  aliqua. Ut enim ad minim veniam, quis     │
	│  nostrud exercitation ullamco laboris      │
	│  nisi ut aliquip ex ea commodo consequat.  │
	│  Duis aute irure dolor in reprehenderit    │
	│  in voluptate velit esse cillum dolore eu  │
	│  fugiat nulla pariatur. Excepteur sint     │
	│  occaecat cupidatat non proident, sunt in  │
	│  culpa qui officia deserunt mollit anim    │
	│  id est laborum.                           │
	│                                            │
	└────────────────────────────────────────────┘

Centering Text

To center align text within a fixed width, you can use the cli.center() method. This accepts a string which may be single or multiple lines, and an optional width in characters. If you omit the width, it uses the length of the longest line. Example:

cli.print(
	cli.center(
		"This Is A Title\n" +
		"The quick brown fox jumped over the lazy dog."
	) + "\n"
);

This would produce:

┌───────────────────────────────────────────────┐
│                This Is A Title                │
│ The quick brown fox jumped over the lazy dog. │
└───────────────────────────────────────────────┘

Of course, you can also just use cli.center() on its own, without wrapping it in a call to cli.box().

Word-Wrapping Text

To word-wrap text when using cli.box() simply specify a width property in your options object. Your string will be word-wrapped to fit into the specified width. However, there is also a low-level cli.wrap() function you can call directly, which simply wraps text at a specified width. Example use:

cli.print(
	cli.wrap( text, 40 ) + "\n"
);

This would produce:

Lorem ipsum dolor sit amet, consectetur
adipisicing elit, sed do eiusmod tempor
incididunt ut labore et dolore magna
aliqua. Ut enim ad minim veniam, quis
nostrud exercitation ullamco laboris
nisi ut aliquip ex ea commodo consequat.
Duis aute irure dolor in reprehenderit
in voluptate velit esse cillum dolore eu
fugiat nulla pariatur. Excepteur sint
occaecat cupidatat non proident, sunt in
culpa qui officia deserunt mollit anim
id est laborum.

Displaying Tables

Table Example

Calling cli.table() produces a nicely formatted ASCII table with a grid of data. It expects an array of rows, and each element should be an array of columns. The first row will be rendered as the "header" and use a different style. Example:

const cli = require('pixl-cli');

let rows = [
	[ "Username", "Full Name", "Email Address", "Status" ],
	[ "jhuckaby", "Joseph Huckaby", "[email protected]", "Administrator" ],
	[ "tsmith", "Tom Smith", "[email protected]", "Active" ],
	[ "dcook", "David Cook", "[email protected]", "Suspended" ]
];

cli.print( 
	cli.table(rows) + "\n" 
);

This would produce the following output:

┌──────────┬────────────────┬───────────────────┬───────────────┐
│ Username │ Full Name      │ Email Address     │ Status        │
├──────────┼────────────────┼───────────────────┼───────────────┤
│ jhuckaby │ Joseph Huckaby │ [email protected] │ Administrator │
│ tsmith   │ Tom Smith      │ [email protected]   │ Active        │
│ dcook    │ David Cook     │ [email protected] │ Suspended     │
└──────────┴────────────────┴───────────────────┴───────────────┘

Of course, if your Terminal supports ANSI color and font styles, the header would be bold and yellow, the text would be cyan, and the borders would be gray. You can customize the styles by passing an object as the second argument to cli.table() with the following keys:

| Property Name | Description | |---------------|-------------| | headerStyles | An array of chalk styles or functions to apply to the header cell text. | | textStyles | An array of chalk styles or functions to apply to the table cell text. | | borderStyles | An array of chalk styles or functions to apply to the border graphics. | | indent | Number of characters to indent the table by (defaults to 0). | | autoFit | Automatically "fit" table into the available terminal width, if it is too wide. This will add ellipsis to longer columns as required. |

Here is an example specifying all the possible options:

cli.print( 
	cli.table(rows, {
		headerStyles: ["bold", "yellow"],
		textStyles: ["cyan", function( text ) { return text.toUpperCase() }],
		borderStyles: ["gray"],
		indent: 0,
		autoFit: true
	}) + "\n" 
);

Each of the three style keys should be set to an array of style names supported by the chalk module or a custom function to support different text transformations. These are arrays because each component may contain multiple styles. For example, the headerStyles is rendered with both yellow and bold by default.

Note that each text cell must only be one line (multi-line is not supported).

Graphical Progress Bars

Progress Bar Example

To display a graphical ASCII progress bar, first call cli.progress.start() to display the progress bar and begin animation. Then, as you make progress on your operation, call cli.progress.update() and pass in a number between 0.0 and 1.0 representing the total progress (the maximum is configurable, see below). When your operation is complete, call cli.progress.end(). That's it!

Here is a simple example:

const cli = require('pixl-cli');

let amount = 0;
cli.progress.start();

setInterval( function() {
	amount += 0.0005;
	cli.progress.update( amount );
	
	if (amount >= 1.0) {
		cli.progress.end();
		cli.print("\tThe end!\n\n");
		process.exit(0);
	}	
}, 10 );

Note that the progress bar will redraw itself every 100ms (configurable), so you don't need to worry about calling cli.progress.update() constantly. You only really need to call it when the amount changes.

Note that if your script is running without a TTY (i.e. without an attached terminal, as in piped to a file or another script), the progress bar is completely no-op'ed. Meaning, it doesn't display anything, and all the API calls do nothing and don't throw any errors.

Configuration

Calls to cli.progress.start() and cli.progress.update() accept an object containing any of the following properties:

| Property Name | Description | |---------------|-------------| | amount | Initial amount to display in progress bar (between 0 and max). Defaults to 0.0. | | max | Maximum amount which represents 100% of bar being filled. Defaults to 1.0. | | text | Optional custom text string to display on right side of progress bar. | | width | The width of the actual progress bar in characters (defaults to 30). | | indent | Number of characters to indent the progress bar by (defaults to 0). | | freq | The animation update frequency in milliseconds (defaults to 100). | | color | Boolean (true), specifies whether to use chalk colors or not. | | unicode | Boolean (true), specifies whether to use Unicode characters or basic ASCII. | | remain | Boolean (true), specifies whether to show the estimated remaining time or not. | | spinner | An array of characters to use as the spinning animation. | | braces | An array of two characters to use as the left and right braces around the bar. | | filling | An array of characters to use as the bar chunk filling progression. | | filled | A single character representing one filled bar chunk, defaults to "⣿". | | styles | A set of color styles to use (see Changing Color Styles below). | | pct | Show percentage (defaults to true). |

If you call cli.progress.update() and pass in a number, the library assumes you are just updating the amount. However, if you pass in an object, all the specified properties are updated.

Temporarily Erasing The Bar

If you need to temporarily erase the progress bar, presumably to write some lines of output to the console, you can call the cli.progress.erase() method. To reduce flickering, you can force a redraw just after outputting your lines by calling cli.progress.draw().

However, if you use the built-in CLI print(), println(), verbose() or verboseln() functions, the progress bar is automatically hidden and redrawn for you. You should only need to manually erase and redraw it if you are calling some other code that outputs to STDOUT or STDERR. Example:

cli.progress.erase();
console.log( some_large_object );
cli.progress.draw();

Customizing the Look

The progress bar uses specific Unicode characters to achieve its look. You can customize these by passing in the following keys to your call to cli.progress.start():

cli.progress.start({
	spinner: ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'],
	braces: ['⟦', '⟧'],
	filling: [' ', '⡀', '⡄', '⡆', '⡇', '⣇', '⣧', '⣷'],
	filled: '⣿'
});

The spinner array contains all the characters used for the 1-character spinning animation at the left side of the bar. They are picked in sequence for each tick (freq milliseconds). To disable the spinner, set this to an array containing one empty space character.

The braces array should contain exactly two elements: one for the left side, and one for the right side of the bar. By default these are stylized Unicode brackets, but can be replaced with any characters you want.

The filling array contains progressive characters representing a partial bar chunk as it fills up, before it is replaced with the filled character. To disable the partial chunk blocks, set this to an array containing one empty space character.

Finally, the filled is a single character (not an array) which represents a filled chunk of the progress bar. This is what replaces the filling animation once it completes a block.

If you switch to plain ASCII mode by setting the unicode property to false, this is the same as passing the following:

cli.progress.start({
	spinner: ['|', '/', '-', "\\"],
	braces: ['[', ']'],
	filling: [' ', '.', ':'],
	filled: '#'
});

Indeterminite Progress

For indeterminate progress, simply set the amount to the max. If these two values are identical, the progress bar is shown in an "indeterminate" state (e.g. the filled portion of the bar is drawn in a different color, default gray). This is to handle cases where a job "sits at 100%" but isn't quite complete, and also handle deliberate intermintate jobs (i.e. your code can just set the amount to the max to show an interminate bar). This also hides the time remaining.

For an additional UI hint, if you know your job is going to be indeterminate from the start, set the pct option to false to hide the percentage display. Otherwise it'll show "100%" during indeterminacy.

Changing Color Styles

The progress bar uses a set of color styles from the chalk module or a custom function for the spinner, braces, bar chunks, percentage display, remaining time, and your custom text (if applicable). To customize these, pass in a styles object with the following keys:

cli.progress.start({
	styles: {
		spinner: ['bold', 'green'],
		braces: ['gray'],
		bar: ['bold', 'cyan'],
		indeterminate: ['gray'],
		pct: ['bold', 'yellow'],
		remain: ['green'],
		text: [function( text ) { return text.toUpperCase() }]
	}
});

Each key should be set to an array of styles supported by the chalk module or a function. These are arrays because each component may contain multiple styles. For example, by default the spinner, bar and pct are styled with both a color and bold.

The indeterminate style is applied to the filled portion of the bar when the amount is exactly equal to the max.

Automatic Width

To set the progress bar size automatically to take up the full width of your Terminal window, you can use the cli.width() function. Just subtract 27 characters from it (possibly more if you use indent or text) to account for the spinner, braces, percentage and remaining time:

cli.progress.start({
	width: cli.width() - 27
});

To handle dynamically resizing Terminal windows, just reset the width in each of your calls to cli.progress.update():

cli.progress.update({
	amount: MY_CURRENT_AMOUNT,
	width: cli.width() - 27
});

Keep progress bar visible

To keep the progress bar visible after it reaches 100%, just pass the value false to the cli.progress.end() function. This will prevent it from being erased from the screen.

Unicode or ASCII

By default the progress bar is rendered using various Unicode characters:

⠼ ⟦⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿              ⟧ 53% (11 sec remain)

However, if you set the unicode parameter to false, these will be replaced with standard ASCII equivalents:

/ [################:             ] 55% (10 sec remain)

You can change the exact characters used as well. See the Customizing the Look section.

Hiding the Cursor

While the progress bar is active, your terminal cursor is hidden, and then restored when cli.progress.end() is called, or when your process exits. This is all handled automatically, except when your process dies unexpectedly, such as via an uncaught Ctrl-C (SIGINT), uncaught termination signal, or a hard crash. When these happen, the progress bar has no chance to restore the cursor, so it ends up remaining hidden until you close and reopen your terminal window.

To fix this, you have two options. You can either catch the signal or crash yourself, and call cli.progress.end(), or you can let the library handle it for you:

When calling cli.progress.start() there are some optional arguments which activate special behavior for the terminal cursor restoration. They are:

| Property Name | Description | |---------------|-------------| | catchInt | When set to true, this will hook SIGINT and call cli.progress.end() to restore the cursor. | | catchTerm | When set to true, this will hook SIGTERM and call cli.progress.end() to restore the cursor. | | catchCrash | When set to true, this will hook uncaughtException and call cli.progress.end() to restore the cursor. | | exitOnSig | When set to true, this will also call process.exit() (with the appropriate status code) after handling the SIGINT or SIGTERM signal. |

Example:

cli.progress.start({
	catchInt: true,
	catchTerm: true,
	catchCrash: true,
	exitOnSig: true
});

Note: You should only need exitOnSig if you are not already handling the signal in your own code, and calling process.exit() yourself.

Chalk

All the style methods from the wonderful chalk module are automatically imported, so you can use them like this:

const cli = require('pixl-cli');
cli.print( cli.bold.red("This is bold and red!") + "\n" );

Importing Into Global

You can optionally import some of the most commonly used methods into the global namespace, so you can use them from anywhere without having to prefix them. The method list includes print(), verbose(), a bunch of others (see below), as well as all the chalk style methods. Example:

const cli = require('pixl-cli');
cli.global();

print( box( bold.red("This is bold, red and in a box!") ) + "\n" );

The full list of methods and objects that are imported are:

  • print()
  • println()
  • verbose()
  • warn()
  • die()
  • loadFile()
  • saveFile()
  • appendFile()
  • prompt()
  • yesno()
  • table()
  • box()
  • wrap()
  • center()
  • commify()
  • shortFloat()
  • pct()
  • pluralize()
  • ucfirst()
  • progress
  • args

License

The MIT License

Copyright (c) 2016 - 2019 Joseph Huckaby.

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.