diff --git a/.editorconfig b/.editorconfig
new file mode 100644
index 0000000..237a029
--- /dev/null
+++ b/.editorconfig
@@ -0,0 +1,12 @@
+root = true
+
+[*]
+indent_style = space
+indent_size = 2
+end_of_line = lf
+charset = utf-8
+trim_trailing_whitespace = true
+insert_final_newline = true
+
+[*.md]
+indent_size = 2
diff --git a/.eslintrc.json b/.eslintrc.json
new file mode 100644
index 0000000..f0fd227
--- /dev/null
+++ b/.eslintrc.json
@@ -0,0 +1,14 @@
+{
+ "extends": "airbnb-base",
+ "plugins": [
+ "import"
+ ],
+ "rules": {
+ "comma-dangle": [
+ "error",
+ "never"
+ ],
+ "no-param-reassign": 0,
+ "no-console": 0
+ }
+}
diff --git a/.jscs.json b/.jscs.json
deleted file mode 100644
index 7ff7197..0000000
--- a/.jscs.json
+++ /dev/null
@@ -1,72 +0,0 @@
-{
- "requireCurlyBraces": [
- "if",
- "else",
- "for",
- "while",
- "do",
- "try",
- "catch"
- ],
- "requireSpaceAfterKeywords": [
- "if",
- "else",
- "for",
- "while",
- "do",
- "switch",
- "case",
- "return",
- "try",
- "catch",
- "function",
- "typeof"
- ],
- "requireSpaceBeforeBlockStatements": true,
- "requireParenthesesAroundIIFE": true,
- "requireSpacesInConditionalExpression": true,
- "disallowSpacesInNamedFunctionExpression": {
- "beforeOpeningRoundBrace": true
- },
- "disallowSpacesInFunctionDeclaration": {
- "beforeOpeningRoundBrace": true
- },
- "requireSpaceBetweenArguments": true,
- "requireMultipleVarDecl": "onevar",
- "requireVarDeclFirst": true,
- "requireBlocksOnNewline": true,
- "disallowEmptyBlocks": true,
- "disallowSpacesInsideArrayBrackets": true,
- "disallowSpacesInsideParentheses": true,
- "requireCommaBeforeLineBreak": true,
- "disallowSpaceAfterPrefixUnaryOperators": true,
- "disallowSpaceBeforePostfixUnaryOperators": true,
- "disallowSpaceBeforeBinaryOperators": [
- ","
- ],
- "requireSpacesInForStatement": true,
- "requireSpacesInAnonymousFunctionExpression": {
- "beforeOpeningRoundBrace": true,
- "beforeOpeningCurlyBrace": true
- },
- "requireSpaceBeforeBinaryOperators": true,
- "requireSpaceAfterBinaryOperators": true,
- "disallowKeywords": [
- "with",
- "continue"
- ],
- "validateIndentation": 4,
- "disallowMixedSpacesAndTabs": true,
- "disallowTrailingWhitespace": true,
- "disallowTrailingComma": true,
- "disallowKeywordsOnNewLine": [
- "else"
- ],
- "requireLineFeedAtFileEnd": true,
- "requireCapitalizedConstructors": true,
- "requireDotNotation": true,
- "disallowNewlineBeforeBlockStatements": true,
- "disallowMultipleLineStrings": true,
- "requireSpacesInsideObjectBrackets": "allButNested",
- "requireSpaceBeforeObjectValues": true
-}
diff --git a/.jshintignore b/.jshintignore
deleted file mode 100644
index fce512f..0000000
--- a/.jshintignore
+++ /dev/null
@@ -1,3 +0,0 @@
-Gruntfile.js
-node_modules
-app/public/**
diff --git a/.jshintrc b/.jshintrc
deleted file mode 100644
index 6ae46bc..0000000
--- a/.jshintrc
+++ /dev/null
@@ -1,28 +0,0 @@
-{
- "esnext": true,
- "bitwise": true,
- "curly": true,
- "eqeqeq": true,
- "forin": true,
- "immed": true,
- "indent": 4,
- "latedef": "nofunc",
- "newcap": true,
- "noarg": true,
- "nonew": true,
- "quotmark": "single",
- "undef": true,
- "unused": true,
- "trailing": true,
- "maxlen": 160,
- "node": true,
- "globals": {
- "_require": true,
- "describe": false,
- "it": false,
- "before": false,
- "beforeEach": false,
- "after": false,
- "afterEach": false
- }
-}
\ No newline at end of file
diff --git a/Gruntfile.js b/Gruntfile.js
index 4c89247..d0f2a05 100644
--- a/Gruntfile.js
+++ b/Gruntfile.js
@@ -1,51 +1,12 @@
-'use strict';
-
-
require('dotenv').load();
-module.exports = grunt => {
-
- grunt.initConfig({
- jshint: {
- options: {
- jshintrc: true
- },
- whir: [
- '.'
- ]
- },
- jscs: {
- src: '.',
- options: {
- config: '.jscs',
- fix: true
- }
- },
- nodemon: {
- dev: {
- script: 'app/index.js',
- options: {
- args: ['--trace', '--trace-sync-io'],
- callback: nodemon => {
- nodemon.on('log', event => {
- console.log(event.colour);
- });
- },
- env: {
- NODE_ENV: grunt.option('env') || 'local',
- PORT: process.env.PORT
- },
- cwd: __dirname,
- ignore: ['node_modules/**'],
- ext: 'js',
- delay: 500
- }
- }
- }
- });
- grunt.loadNpmTasks('grunt-jscs');
- grunt.loadNpmTasks('grunt-nodemon');
- grunt.loadNpmTasks('grunt-contrib-jshint');
+module.exports = (grunt) => {
+ grunt.initConfig({
+ eslint: {
+ target: ['./app/**/*.js']
+ }
+ });
- grunt.registerTask('hint', 'Hinting...', ['jshint:whir', 'jscs']);
+ grunt.loadNpmTasks('grunt-eslint');
+ grunt.registerTask('default', ['eslint']);
};
diff --git a/LICENSE b/LICENSE
index eebc1ef..43060d6 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,6 +1,6 @@
MIT License
-Copyright (c) 2016 WhirIO
+Copyright (c) 2017 Stefan Aichholzer
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
diff --git a/README.md b/README.md
index 38a9102..7155f35 100644
--- a/README.md
+++ b/README.md
@@ -1,28 +1,26 @@
+[![Dependency status](https://gemnasium.com/badges/github.com/WhirIO/Client.svg)](https://gemnasium.com/github.com/WhirIO/Client)
+[![Alpha](https://img.shields.io/badge/status-alpha-8456AC.svg)](https://github.com/WhirIO/Client)
+
-
+
-
-
-
+### Installation
```
-This is still under development.
-It might be unstable until this notice is removed.
+$> npm i -g whir.io
```
-### Getting started:
-```
-$> npm install -g whir.io
-```
### Options:
-- `--user || -u`: Your username for a particular channel. `Required`
-- `--channel || -c`: The channel to join. Default: `whir generated name`
-- `--host || -h`: The server running Whir. Default: `chat.whir.io`
-- `--mute || -m`: Mute the conversation. Does not require a value.
+- `--user || -u`: Your username (per channel) `Required`
+- `--pass || -p`: Password, for private channels.
+- `--channel || -c`: Channel you are joining (or creating) `Default: random name`
+- `--host || -h`: Whir's host `Default: chat.whir.io`
+- `--mute || -m`: Mute the conversation.
+
-### Example uses:
+### Chat:
```
$> whir.io --user=stefan --channel=friends
```
@@ -30,20 +28,36 @@ $> whir.io --user=stefan --channel=friends
or
```
-$> whir.io -u=stefan -c=friends
+$> whir.io -u stefan -c friends
```
or
```
-$> whir.io -u=stefan -m
-// You will be connected to a randomly named channel.
+$> whir.io -u stefan -m
+// Connected to a random channel.
// The conversation is muted.
```
+or
+
+```
+$> whir.io -u stefan -c friends -h myawesomedomain.chat
+// Running whir on your own server/domain.
+```
+
+
+### Notes
+You can also setup and run your own **whir** server.
+Here's how to do that: [https://github.com/WhirIO/Server](https://github.com/WhirIO/Server)
+
+
+### Contribute
+```
+fork https://github.com/WhirIO/Server
+```
+
### License
[MIT](https://github.com/WhirIO/Client/blob/master/LICENSE)
-
-### Enjoy!
diff --git a/app/core/components.js b/app/core/components.js
new file mode 100644
index 0000000..6fb8d50
--- /dev/null
+++ b/app/core/components.js
@@ -0,0 +1,182 @@
+const blessed = require('blessed');
+const Emitter = require('events').EventEmitter;
+
+class Components extends Emitter {
+
+ constructor(options) {
+ super();
+
+ this.screen = blessed.screen(options);
+ this.screen.title = options.screenTitle;
+
+ this.scroll = [];
+ this.scrollIndex = 0;
+ }
+
+ render(status = null) {
+ if (status === 'no_history') {
+ return;
+ }
+
+ if (!this.input.detached) {
+ this.input.focus();
+ }
+
+ this.screen.render();
+ }
+
+ title() {
+ this.title = blessed.text({
+ screen: this.screen,
+ top: 0,
+ width: '100%',
+ height: 3,
+ padding: 1,
+ style: {
+ bg: 'green',
+ fg: 'black'
+ }
+ });
+
+ this.title.setText(this.screen.title);
+ return this.title;
+ }
+
+ users() {
+ this.users = blessed.list({
+ screen: this.screen,
+ width: '25%',
+ top: 3,
+ keys: true,
+ border: 'line',
+ interactive: false,
+ padding: {
+ top: 0,
+ right: 0,
+ bottom: 1,
+ left: 1
+ },
+ style: {
+ border: {
+ fg: 'white'
+ },
+ selected: {
+ bg: 'green',
+ fg: 'black'
+ }
+ }
+ });
+
+ return this.users;
+ }
+
+ timeline() {
+ this.timeline = blessed.box({
+ screen: this.screen,
+ mouse: true,
+ top: 3,
+ left: '25%-1',
+ height: '100%-7',
+ border: 'line',
+ scrollable: true,
+ alwaysScroll: true,
+ scrollbar: true,
+ padding: {
+ top: 1,
+ right: 0,
+ bottom: 1,
+ left: 2
+ },
+ style: {
+ border: {
+ fg: 'white'
+ },
+ scrollbar: {
+ bg: 'white',
+ fg: 'black'
+ }
+ }
+ });
+
+ return this.timeline;
+ }
+
+ input() {
+ this.input = blessed.textbox({
+ screen: this.screen,
+ content: '',
+ border: 'line',
+ padding: {
+ top: 1,
+ right: 2,
+ bottom: 1,
+ left: 2
+ },
+ style: {
+ fg: 'default',
+ bg: 'default',
+ border: {
+ fg: 'white',
+ bg: 'default'
+ }
+ },
+ left: '25%-1',
+ height: 5,
+ top: '100%-5',
+ keys: true,
+ mouse: true,
+ inputOnFocus: true
+ });
+
+ /**
+ * Enable scrolling through the conversation with the arrow keys.
+ * @see this.input.key('up', scroll);
+ * @see this.input.key('down', scroll);
+ */
+ const scroll = (char, key) => {
+ const condition = () => {
+ if (key.name === 'up') {
+ return this.scrollIndex < this.scroll.length;
+ }
+
+ return this.scrollIndex > 1;
+ };
+
+ if (this.scroll.length) {
+ let found = false;
+ while (!found) {
+ if (condition()) {
+ found = true;
+ this.scrollIndex += key.name === 'up' ? 1 : -1;
+ const data = this.scroll[this.scroll.length - this.scrollIndex];
+ this.input.setValue(data.message);
+ return this.render();
+ }
+ found = true;
+ }
+ }
+
+ return true;
+ };
+
+ this.input.key(['up', 'down'], scroll);
+ this.input.key(['C-c'], () => {
+ this.input.clearValue();
+ return this.render();
+ });
+
+ this.input.on('submit', (value) => {
+ value = value.trim();
+ if (!value) {
+ return this.render();
+ }
+
+ this.input.clearValue();
+ return this.emit('message', value.trim());
+ });
+
+ return this.input;
+ }
+}
+
+module.exports = Components;
diff --git a/app/core/screen.js b/app/core/screen.js
index cfe9258..dd52434 100644
--- a/app/core/screen.js
+++ b/app/core/screen.js
@@ -1,340 +1,183 @@
-'use strict';
-
-
-const blessed = require('blessed');
const chalk = require('chalk');
-const string = _require('library/string');
+const Components = require('./components');
+const moment = require('moment');
+const string = require('../library/string');
class Screen {
- constructor (whir) {
-
- this.whir = whir;
- this.screen = blessed.screen({
- smartCSR: true,
- dockBorders: true,
- fullUnicode: true
- });
- this.screen.title = 'Whir.io';
-
- /*
- this.screen.key('tab', () => {
- this.screen.focusNext();
- this.render();
- });
- */
-
- this.destroy = this.destroy.bind(this, whir);
- this.screen.key(['escape', 'C-c'], this.destroy);
-
- this.screen.append(this.title());
- this.screen.append(this.users());
- this.screen.append(this.timeline());
- this.screen.append(this.input());
- this.render();
+ constructor(whir, { user = null, scrollSize = 250 }) {
+ this.whir = whir;
+
+ this.components = new Components({
+ smartCSR: true,
+ dockBorders: true,
+ fullUnicode: true,
+ screenTitle: 'Whir.io'
+ });
+
+ this.user = user;
+ this.scrollSize = scrollSize;
+
+ this.components.on('message', message => this.whir.send(message));
+ this.components.screen.key(['escape'], this.destroy.bind(this, true));
+ this.components.screen.append(this.components.title());
+ this.components.screen.append(this.components.users());
+ this.components.screen.append(this.components.timeline());
+ this.components.screen.append(this.components.input());
+ this.components.render();
+ }
+
+ /**
+ * This is the history for the current session only.
+ * It is non-atomic and it will be cleared when the application closes.
+ * It can be accessed using the arrow keys.
+ * @param item
+ */
+ scroll(item) {
+ if (item.user === this.user) {
+ if (this.components.scroll.length >= this.scrollSize) {
+ this.components.scroll.shift();
+ }
+
+ this.components.scroll.push(item);
+ this.components.scrollIndex = 0;
}
-
- title () {
-
- this.title = blessed.text({
- screen: this.screen,
- top: 0,
- width: '100%',
- height: 4,
- padding: {
- top: 1,
- right: 1,
- bottom: 1,
- left: 1
- },
- style: {
- bg: 'green',
- fg: 'black',
- }
- });
-
- this.title.setText('Whir.io');
- return this.title;
+ }
+
+ print(data, { sender = 'whir', render = true } = {}) {
+ /**
+ * Notification sound on incoming messages, when mute = false
+ */
+ if (sender !== 'me' && !this.muteChannel) {
+ process.stdout.write('\u0007');
}
- users () {
-
- this.users = blessed.list({
- screen: this.screen,
- width: '25%',
- top: 3,
- keys: true,
- border: 'line',
- interactive: false,
- padding: {
- top: 0,
- right: 0,
- bottom: 1,
- left: 1
- },
- style: {
- border: {
- fg: 'white'
- },
- selected: {
- bg: 'green',
- fg: 'black',
- }
- }
- });
-
- return this.users;
+ /**
+ * A blank line between messages from different users.
+ * Might be removed if a "floating-box" approach is adopted.
+ */
+ if (this.lastSender !== data.user && !data.command) {
+ this.components.timeline.pushLine('');
}
- timeline () {
-
- this.timeline = blessed.box({
- screen: this.screen,
- keys: true,
- top: 3,
- left: '25%-1',
- height: '100%-7',
- border: 'line',
- scrollable: true,
- alwaysScroll: true,
- scrollbar: true,
- padding: {
- top: 1,
- right: 0,
- bottom: 1,
- left: 2
- },
- style: {
- border: {
- fg: 'white'
- },
- scrollbar: {
- bg: 'white',
- fg: 'black'
- }
- }
- });
-
- return this.timeline;
+ /**
+ * Add or remove users from the user panel.
+ * Skip this step when loading a user's history.
+ * @see this.components.users()
+ */
+ if (!data.fromHistory) {
+ if (data.action === 'join') {
+ this.components.users.addItem(data.user);
+ } else if (data.action === 'leave') {
+ this.components.users.removeItem(data.user);
+ }
}
- loadHistory () {
- this.whir.history.forEach(data => {
- data.mute = true;
- data.channel = this.whir.channel;
- this.echo(data, data.user, false);
- });
-
- this.render();
+ /**
+ * When establishing a connection, all users are sent back.
+ * This takes the data sent by the server and merges it with the
+ * existing users, sorts them (alphabetically) and re-populates
+ * the users list.
+ */
+ if (data.currentUsers) {
+ data.currentUsers = data.currentUsers
+ .concat(this.components.users.children)
+ .filter((x, i, a) => a.indexOf(x) === i)
+ .sort();
+
+ this.components.users.setItems(data.currentUsers);
}
- input () {
-
- this.input = blessed.textbox({
- content: '',
- screen: this.screen,
- border: 'line',
- padding: {
- top: 1,
- right: 2,
- bottom: 1,
- left: 2
- },
- style: {
- fg: 'default',
- bg: 'default',
- border: {
- fg: 'white',
- bg: 'default'
- }
- },
- left: '25%-1',
- height: 5,
- top: '100%-5',
- keys: true,
- mouse: true,
- inputOnFocus: true
- });
-
- /**
- * Enable scrolling through the conversation with the arrow keys.
- * @see this.input.key('up', history);
- * @see this.input.key('down', history);
- */
- const history = (char, key) => {
-
- const condition = () => key.name === 'up' ?
- this.whir.historyIndex < this.whir.history.length :
- this.whir.historyIndex > 1;
-
- if (this.whir.history.length) {
- let found = false;
- while (!found) {
- if (condition()) {
- this.whir.historyIndex += key.name === 'up' ? 1 : -1;
- let data = this.whir.history[this.whir.history.length - this.whir.historyIndex];
- if (data.user === this.whir.user) {
- found = true;
- this.input.setValue(data.message);
- this.render();
- }
- continue;
- }
- found = true;
- }
- }
- };
-
- this.input.key('up', history);
- this.input.key('down', history);
-
- this.input.on('submit', value => {
- value = value.trim();
- if (!value) {
- return this.render();
- }
-
- this.input.clearValue();
- this.whir.send(value.trim());
+ /**
+ * Replacements; underline, bold, italics, etc.
+ * Find and replace any emoji (as per: http://www.fileformat.info/info/emoji/list.htm)
+ * Format the line to be rendered.
+ * Render any additional payload sent by the server.
+ * Scroll the timeline to the bottom.
+ */
+ data.message = data.message.replace(/_([\w\s.]+)_/gi, chalk.green.underline('$1'));
+ data.message = data.message.replace(/-([\w\s.]+)-/gi, chalk.white('$1'));
+ data.message = string.emojinize(data.message);
+
+ if (data.payload && data.payload.showTitle) {
+ this.components.timeline.pushLine(data.message);
+ } else if (!data.command) {
+ data.timestamp = moment(data.timestamp).format('HH:mm');
+ data.timestamp = `${chalk.black.bgGreen(data.timestamp)} `;
+ const user = data.user ? chalk.green(`${data.timestamp}${data.user}: `) : '';
+ if (data.alert) {
+ data.message = data.message.split('\n');
+ data.message = data.message.map((message) => {
+ message = chalk.white.bgRed(message);
+ return message;
});
-
- return this.input;
+ data.message = data.message.join('\n');
+ }
+ this.components.timeline.pushLine(user + data.message);
}
- echo (data, sender = 'whir', render = true) {
-
- if (data.command === 'exit') {
- return this.destroy();
- }
-
- /**
- * Notification sound on incoming messages, when mute = false
- */
- if (sender !== 'me' && !data.mute) {
- process.stdout.write('\u0007');
- }
-
- /**
- * A blank line between messages from different users.
- * Might be removed if a "floating-boxes" approach is adopted.
- */
- if (this.lastSender !== data.user && !data.command) {
- this.timeline.pushLine('');
- }
-
- /**
- * Add or remove users from the user panel.
- * @see this.users()
- */
- if (data.action) {
- let method = data.action.method === 'join' ? 'addItem' :
- data.action.method === 'leave' ? 'removeItem' :
- null;
-
- if (method) {
- this.users[method](data.action.user);
- }
- }
-
- /**
- * When establishing a connection, all users are sent back.
- * This takes the data sent by the server, merges it with the
- * existing users, sorts them (alphabetically) and re-populates
- * the list with the new array of users.
- */
- if (data.currentUsers) {
- data.currentUsers = data.currentUsers
- .concat(this.users.children)
- .filter((x, i, a) => a.indexOf(x) === i)
- .sort();
-
- this.users.setItems(data.currentUsers);
- }
+ /**
+ * The response (payload) is flexible in order to accommodate
+ * various operations based on whatever the server returns.
+ * Currently only the "date" is in use.
+ */
+ if (data.payload) {
+ let padding = null;
+ if (typeof data.payload.pad === 'number') {
+ padding = data.payload.pad;
+ } else if (data.payload.pad) {
+ padding = 0;
+ Object.entries(data.payload.items).forEach(([key]) => {
+ padding = key.length > padding ? key.length : padding;
+ });
- /**
- * Replacements; underline, bold, italics, etc.
- * Find and replace any emoji (as per: http://www.fileformat.info/info/emoji/list.htm)
- * Format the line to be rendered.
- * In case an extra payload was sent by the server, render that as well.
- * Scroll the timeline to the bottom.
- */
- data.message = data.message.replace(/_(\w.+)_/gi, chalk.green.underline('$1'));
- data.message = string.emojinize(data.message);
- if (!data.payload && !data.command) {
- let line = chalk.green(`${data.user}:`) + ' ' + data.message;
- this.timeline.pushLine(line);
- } else if (data.payload) {
- for (let item in data.payload) {
- let line = chalk.green(`/${string.pad(item)}`) + data.payload[item];
- this.timeline.pushLine(line);
- }
+ const match = data.payload.pad.match(/\+([\d])+/i);
+ if (match) {
+ padding += parseInt(match[1], 10);
}
- this.timeline.setScrollPerc(100);
-
- /**
- * Keep track of the user who send the last message, just for rendering
- * purposes and update the connected number of users.
- * @see this.users
- */
- this.lastSender = data.user;
- this.title.setText(`Channel: ${data.channel} | User: ${this.whir.user} | Users: ${this.users.children.length + 1}`);
-
- if (render) {
- this.render();
+ }
+
+ Object.entries(data.payload.items).forEach(([key, item]) => {
+ let passedItem;
+ switch (item.type) {
+ case 'date':
+ passedItem = moment(item.value).fromNow();
+ break;
+ default:
+ passedItem = item.value;
}
- return this;
- }
-
- error (data) {
-
- const error = blessed.box({
- top: 'center',
- left: 'center',
- width: '50%',
- height: '30%',
- padding: {
- top: 2,
- right: 3,
- bottom: 2,
- left: 3
- },
- align: 'center',
- valign: 'middle',
- content: (data.message || data) + '\n\nPress `esc` to close the application.',
- tags: true,
- border: {
- type: 'line'
- },
- style: {
- fg: 'default',
- bg: 'default',
- border: {
- fg: 'white'
- }
- }
- });
- this.screen.append(error);
- this.render();
+ const line = `\u258B ${string.pad(key, 'right', padding)}${chalk.white(passedItem)}`;
+ this.components.timeline.pushLine(line);
+ });
}
-
- render () {
-
- this.input.focus();
- this.screen.render();
+ this.components.timeline.setScrollPerc(100);
+
+ /**
+ * Keep track of the user who send the last message, just for rendering
+ * purposes and update the connected number of users.
+ * @see this.components.users
+ */
+ this.lastSender = data.user;
+ const channel = `Channel: ${data.channel}`;
+ const user = `User: ${this.whir.user}`;
+ const users = `${this.components.users.children.length + 1}`;
+ this.components.title.setText(`${this.muteChannel ? '\uD83D\uDD07' : '\uD83D\uDD09'} ${channel} | ${user} | Users: ${users}`);
+
+ if (render) {
+ this.components.render();
}
- destroy (whir) {
+ return this;
+ }
- whir.saveHistory()
- .then(error => {
- this.screen.destroy();
- if (error) {
- console.error(`\n > ${error}\n`);
- }
- return process.exit();
- });
+ destroy(exit = false) {
+ this.components.screen.destroy();
+ if (exit) {
+ console.error(`\n 👋 See you soon, ${this.user}!\n`);
+ process.exit(0);
}
+ }
}
module.exports = Screen;
diff --git a/app/core/whir.js b/app/core/whir.js
index cacbc8d..2814cec 100644
--- a/app/core/whir.js
+++ b/app/core/whir.js
@@ -1,113 +1,194 @@
-'use strict';
-
-
+const crypto = require('../library/crypto');
+const Emitter = require('events').EventEmitter;
const fs = require('fs');
+const lineReader = require('readline');
+const Screen = require('./screen');
const WS = require('ws');
-const path = require('path');
-const crypto = _require('library/crypto');
-const Emitter = require('events');
-
-class Whir extends Emitter {
- constructor (argv = {}) {
-
- super();
- this.history = [];
- this.historyIndex = 0;
- this.historyLoaded = false;
- this.historyPath = path.normalize(`${__dirname}/../../history`);
- this.host = argv.host || 'chat.whir.io';
- this.getHeaders(argv)
- .then(headers => {
- this.user = argv.user;
- this.mute = argv.mute || false;
- this.socket = new WS(`ws://${this.host}`, headers);
- this.socket
- .on('open', () => {})
- .on('message', data => {
- data = JSON.parse(data.toString('utf8'));
- this.channel = data.channel || argv.channel;
- data.mute = this.mute;
-
- if (!this.historyLoaded) {
- this.loadHistory(data, this.emit.bind(this, 'history'));
- } else {
- this.appendHistory(data);
- }
-
- this.emit('received', data);
- })
- .on('close', (code, data) => this.emit('close', { user: 'whir', message: data }));
- })
- .catch(error => console.error(error));
+const getHeaders = async ({ user, channel, pass, store }) => {
+ const headers = data => ({
+ headers: {
+ 'x-whir-session': data.session,
+ 'x-whir-channel': channel || '',
+ 'x-whir-user': user,
+ 'x-whir-pass': pass || ''
}
+ });
+
+ try {
+ const data = fs.readFileSync(`${store}/${user}.whir`, 'utf8');
+ return headers(JSON.parse(data));
+ } catch (error) {
+ const session = crypto.hash(await crypto.bytes(128), 'RSA-SHA256');
+ try {
+ fs.appendFileSync(`${store}/${user}.whir`, JSON.stringify({ session }), { flag: 'a' });
+ return headers({ session });
+ } catch (writeError) {
+ return { error: writeError };
+ }
+ }
+};
- send (message) {
+class Whir extends Emitter {
- const data = {
- user: this.user,
- channel: this.channel,
- message: message
- };
+ constructor(argv = {}, unsecure = false) {
+ super();
+
+ this.host = argv.host;
+ this.user = argv.user;
+ this.channel = argv.channel;
+ this.scrollSize = argv.scrollSize;
+ this.store = argv.store;
+ this.muteChannel = argv.mute || false;
+
+ this.protocol = `ws${unsecure ? '' : 's'}`;
+ getHeaders(argv)
+ .then((headers) => {
+ if (headers.error) {
+ throw headers.error;
+ }
+
+ return this.connect(headers);
+ })
+ .catch(error => this.emit('error', error));
+ }
+
+ connect(headers) {
+ try {
+ this.socket = new WS(`${this.protocol}://${this.host}`, headers);
+ } catch (error) {
+ return this.emit('error', error);
+ }
- this.socket.send(JSON.stringify(data), { binary: true, mask: true });
- if (data.message.match(/^\/[\w]/)) {
- data.command = data.message.replace(/^\//g, '');
+ return this.socket
+ .on('open', async () => {
+ this.screen = new Screen(this, { user: this.user, scrollSize: this.scrollSize });
+ this.screen.muteChannel = true;
+
+ this.screen.components.render(await this.loadHistory());
+ this.screen.muteChannel = this.muteChannel;
+ })
+ .on('message', async (data) => {
+ try {
+ data = JSON.parse(data.toString('utf8'));
+ this.channel = data.channel || this.channel;
+ data.timestamp = (new Date()).getTime();
+
+ await this.writeHistory(data);
+ return this.emit('received', data);
+ } catch (error) {
+ return this.emit('alert', error);
}
+ })
+ .on('error', error => this.emit('error', error))
+ .on('close', (code, data) => this.emit('close', data));
+ }
+
+ send(message) {
+ const data = {
+ user: this.user,
+ channel: this.channel,
+ message,
+ timestamp: (new Date()).getTime()
+ };
+
+ let localCommand = false;
+ if (data.message.match(/^\/[\w]+/)) {
+ data.command = data.message.replace(/^\//g, '');
+ switch (data.command) {
+ case 'exit': return this.screen.destroy(true);
+ case 'clear':
+ localCommand = true;
+ this.screen.components.timeline.getLines().forEach((lines, index) => {
+ this.screen.components.timeline.deleteLine(index);
+ });
+ break;
+ case 'mute':
+ localCommand = true;
+ this.screen.muteChannel = true;
+ break;
+ case 'unmute':
+ localCommand = true;
+ this.screen.muteChannel = false;
+ break;
+ default:
+ }
+ }
- this.historyIndex = 0;
- this.appendHistory(data);
- this.emit('sent', data);
+ if (!localCommand) {
+ this.socket.send(JSON.stringify(data), { binary: true, mask: true });
}
- async getHeaders (argv) {
+ this.writeHistory(data);
+ this.screen.scroll(data);
- const headers = {
- 'x-whir-session': crypto.hash(await crypto.bytes(128), 'RSA-SHA256')
- };
+ return this.emit('sent', data);
+ }
- if (argv.channel) {
- headers['x-whir-channel'] = argv.channel;
- }
+ writeHistory(data) {
+ return new Promise((yes) => {
+ if (!data.user) {
+ return yes();
+ }
- if (argv.user) {
- headers['x-whir-user'] = argv.user;
+ const file = `${this.store}/${this.user}.${this.channel}.whir`;
+ return fs.appendFile(file, `${JSON.stringify(data)}\n`, (error) => {
+ if (error) {
+ return this.emit('alert', 'Your conversation could not be saved.');
}
- return { headers: headers };
+ return yes();
+ });
+ });
+ }
+
+ loadHistory() {
+ const history = `${this.store}/${this.user}.${this.channel}.whir`;
+ return new Promise((yes, no) => lineReader.createInterface({
+ input: fs.createReadStream(history)
+ .on('error', yes.bind(null, 'no_history'))
+ .on('end', yes)
+ })
+ .on('line', (line) => {
+ try {
+ const data = JSON.parse(line);
+ data.fromHistory = true;
+ this.screen.print(data, { render: false });
+ if (data.user === this.user) {
+ this.screen.scroll(data);
+ }
+ return true;
+ } catch (error) {
+ return no(error);
+ }
+ }));
+ }
+
+ error(data) {
+ try {
+ if (!data) {
+ data = {};
+ } else {
+ data = JSON.parse(data);
+ }
+ } catch (error) {
+ // data is not JSON or is empty
}
- appendHistory (data) {
-
- this.history.push({
- user: data.user,
- message: data.message,
- timestamp: (new Date()).getTime()
- });
+ if (data.code === 'ECONNREFUSED') {
+ data.message = ` 😖 It was not possible to connect to the server.\n (${data.message})
+ \n Make sure your whir.io server is listening.`;
+ } else {
+ data.message = ` 😞 ${data.message || 'The server terminated your connection.'}`;
}
- loadHistory (data, callback) {
-
- fs.readFile(`${this.historyPath}/${this.user}.${this.channel}.json`, (error, history) => {
- if (!error) {
- this.history = JSON.parse(history);
- this.appendHistory(data);
- }
-
- this.historyLoaded = true;
- callback();
- });
+ if (this.screen) {
+ this.screen.destroy();
}
- saveHistory () {
-
- return new Promise(yes => {
- fs.writeFile(`${this.historyPath}/${this.user}.${this.channel}.json`, JSON.stringify(this.history), error => {
- error = error ? 'An error has occurred; your conversation could not be saved.' : null;
- return yes(error);
- });
- });
- }
+ console.error(`\n${data.message}\n`);
+ process.exit(1);
+ }
}
module.exports = Whir;
diff --git a/app/index.js b/app/index.js
index 80048f1..76b6bb5 100644
--- a/app/index.js
+++ b/app/index.js
@@ -1,36 +1,33 @@
#!/usr/bin/env node
-'use strict';
+const path = require('path');
+const Whir = require('./core/whir');
+const yargs = require('yargs');
-global._require = module => require(`${__dirname}/${module}`);
-const Whir = _require('core/whir');
-const Screen = _require('core/screen');
-const argv = require('yargs')
- .options({
- user: { alias: 'u', describe: 'Username.', demand: true },
- channel: { alias: 'c', describe: 'Channel.', default: null },
- host: { alias: 'h', describe: 'Whir.io server.', default: 'chat.whir.io' },
- mute: { alias: 'm', describe: 'Mute the conversation.' }
- })
- .usage('\nUsage: whir.io --user=[user]')
- .example('whir.io --user=stefan')
- .example('whir.io --user=stefan --channel=box')
- .epilogue('For more information, visit https://whir.io')
- .argv;
+const expect = {
+ user: { alias: 'u', describe: 'Username.', demand: true },
+ pass: { alias: 'p', describe: 'Password.', default: null },
+ channel: { alias: 'c', describe: 'Channel.', default: null },
+ host: { alias: 'h', describe: 'Whir.io server.', default: 'chat.whir.io' },
+ mute: { alias: 'm', describe: 'Mute the conversation.' },
+ store: { alias: 's', describe: 'Where to store application data.', default: path.normalize(`${__dirname}/../store`) },
+ scrollSize: { alias: 'ss', describe: 'Lines to keep in scroll history.', default: 100 }
+};
+const argv = yargs.options(expect)
+ .usage('\nUsage: whir.io --user=[user]')
+ .example('whir.io --user=stefan')
+ .example('whir.io -u stefan -c friends')
+ .epilogue('For more information, visit https://whir.io')
+ .argv;
+const whir = new Whir(argv, process.env.UNSECURE_SOCKET === 'true');
-try {
- const whir = new Whir(argv);
- const screen = new Screen(whir);
-
- whir.on('sent', data => screen.echo(data, 'me'))
- .on('received', data => screen.echo(data))
- .on('close', data => screen.error(data))
- .on('error', data => screen.error(data))
- .on('history', () => screen.loadHistory());
-
-} catch (error) {
- console.error('\n' + error.message);
- console.error(error.stack + '\n');
-
- process.exit(1);
-}
+/**
+ * Emitting events makes the architecture more plug-able.
+ * It's easy to implement custom logic -or extended the existing one-
+ * for each emitted event.
+ */
+whir.on('sent', data => whir.screen.print(data, { sender: 'me' }))
+ .on('received', data => whir.screen.print(data))
+ .on('alert', data => whir.error(data))
+ .on('close', data => whir.error(data))
+ .on('error', data => whir.error(data));
diff --git a/app/library/crypto.js b/app/library/crypto.js
index f925a28..ed46af3 100644
--- a/app/library/crypto.js
+++ b/app/library/crypto.js
@@ -1,25 +1,22 @@
-'use strict';
-
-
const crypto = require('crypto');
module.exports = {
- bytes: (length, encoding = 'hex') => new Promise((yes, no) => {
- crypto.randomBytes(length, (error, bytes) => {
- if (error || !bytes) {
- return no(null);
- }
+ bytes: (length, encoding = 'hex') => new Promise((yes, no) => {
+ crypto.randomBytes(length, (error, bytes) => {
+ if (error || !bytes) {
+ return no(null);
+ }
- yes(bytes.toString(encoding));
- });
- }),
+ return yes(bytes.toString(encoding));
+ });
+ }),
- hash: (data, algorithm = 'RSA-SHA512', encoding = 'hex') => {
- if (typeof data !== 'string') {
- data = JSON.stringify(data);
- }
-
- return crypto.createHash(algorithm).update(data, 'utf8').digest(encoding);
+ hash: (data, algorithm = 'RSA-SHA512', encoding = 'hex') => {
+ if (typeof data !== 'string') {
+ data = JSON.stringify(data);
}
+
+ return crypto.createHash(algorithm).update(data, 'utf8').digest(encoding);
+ }
};
diff --git a/app/library/string.js b/app/library/string.js
index 8b2093d..9fc0314 100644
--- a/app/library/string.js
+++ b/app/library/string.js
@@ -1,28 +1,15 @@
-'use strict';
-
-
-const emoji = _require('support/emoji.json');
+const emoji = require('../support/emoji.json');
module.exports = {
- emojinize: input => input.replace(/:([\w]+):/g, (match, icon) => {
- return emoji[icon] || match;
- }),
-
- pad: (string, side = 'right', char = ' ', total = 10) => {
- if (!string || string.length >= total) {
- return string;
- }
+ emojinize: input => input.replace(/:([\w]+):/g, (match, icon) => emoji[icon] || match),
- const pad = (total - string.length) / char.length;
- for (let i = 0; i < pad; i++) {
- if (side === 'left') {
- string = char + string;
- } else if (side === 'right') {
- string += char;
- }
- }
-
- return string;
+ pad: (string, side = 'right', padding = null, char = ' ') => {
+ if (!string || !padding || string.length >= padding) {
+ return string;
}
+
+ const pad = char.repeat(padding - string.length);
+ return side === 'right' ? string + pad : pad + string;
+ }
};
diff --git a/media/whir.png b/media/whir.png
index c79e3a0..77e669a 100644
Binary files a/media/whir.png and b/media/whir.png differ
diff --git a/package-lock.json b/package-lock.json
new file mode 100644
index 0000000..987bb87
--- /dev/null
+++ b/package-lock.json
@@ -0,0 +1,1838 @@
+{
+ "name": "whir.io",
+ "version": "1.3.0",
+ "lockfileVersion": 1,
+ "dependencies": {
+ "abbrev": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.0.tgz",
+ "integrity": "sha1-0FVMIlZjbi9W58LlrRg/hZQo2B8=",
+ "dev": true
+ },
+ "acorn": {
+ "version": "5.0.3",
+ "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.0.3.tgz",
+ "integrity": "sha1-xGDfCEkUY/AozLguqzcwvwEIez0=",
+ "dev": true
+ },
+ "acorn-jsx": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz",
+ "integrity": "sha1-r9+UiPsezvyDSPb7IvRk4ypYs2s=",
+ "dev": true,
+ "dependencies": {
+ "acorn": {
+ "version": "3.3.0",
+ "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz",
+ "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo=",
+ "dev": true
+ }
+ }
+ },
+ "ajv": {
+ "version": "4.11.8",
+ "resolved": "https://registry.npmjs.org/ajv/-/ajv-4.11.8.tgz",
+ "integrity": "sha1-gv+wKynmYq5TvcIK8VlHcGc5xTY=",
+ "dev": true
+ },
+ "ajv-keywords": {
+ "version": "1.5.1",
+ "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-1.5.1.tgz",
+ "integrity": "sha1-MU3QpLM2j609/NxU7eYXG4htrzw=",
+ "dev": true
+ },
+ "ansi-escapes": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-1.4.0.tgz",
+ "integrity": "sha1-06ioOzGapneTZisT52HHkRQiMG4=",
+ "dev": true
+ },
+ "ansi-regex": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+ "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8="
+ },
+ "ansi-styles": {
+ "version": "2.2.1",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz",
+ "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4="
+ },
+ "aproba": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.1.1.tgz",
+ "integrity": "sha1-ldNgDwdxCqDpKYxyatXs8urLq6s="
+ },
+ "are-we-there-yet": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.4.tgz",
+ "integrity": "sha1-u13KOCu5TwXhUZQ3PRb9O6HKEQ0="
+ },
+ "argparse": {
+ "version": "1.0.9",
+ "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.9.tgz",
+ "integrity": "sha1-c9g7wmP4bpf4zE9rrhsOkKfSLIY=",
+ "dev": true
+ },
+ "array-find-index": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/array-find-index/-/array-find-index-1.0.2.tgz",
+ "integrity": "sha1-3wEKoSh+Fku9pvlyOwqWoexBh6E=",
+ "dev": true
+ },
+ "array-union": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz",
+ "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=",
+ "dev": true
+ },
+ "array-uniq": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz",
+ "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=",
+ "dev": true
+ },
+ "arrify": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz",
+ "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=",
+ "dev": true
+ },
+ "async": {
+ "version": "1.5.2",
+ "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz",
+ "integrity": "sha1-7GphrlZIDAw8skHJVhjiCJL5Zyo=",
+ "dev": true
+ },
+ "babel-code-frame": {
+ "version": "6.22.0",
+ "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.22.0.tgz",
+ "integrity": "sha1-AnYgvuVnqIwyVhV05/0IAdMxGOQ=",
+ "dev": true
+ },
+ "balanced-match": {
+ "version": "0.4.2",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-0.4.2.tgz",
+ "integrity": "sha1-yz8+PHMtwPAe5wtAPzAuYddwmDg=",
+ "dev": true
+ },
+ "bindings": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.2.1.tgz",
+ "integrity": "sha1-FK1hE4EtLTfXLme0ystLtyZQXxE="
+ },
+ "bl": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/bl/-/bl-1.2.1.tgz",
+ "integrity": "sha1-ysMo977kVzDUBLaSID/LWQ4XLV4="
+ },
+ "blessed": {
+ "version": "0.1.81",
+ "resolved": "https://registry.npmjs.org/blessed/-/blessed-0.1.81.tgz",
+ "integrity": "sha1-+WLWh+wsNpVwrnGvhDJW5tDKESk="
+ },
+ "brace-expansion": {
+ "version": "1.1.7",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.7.tgz",
+ "integrity": "sha1-Pv/DxQ4ABTH7cg6v+A8K6O8jz1k=",
+ "dev": true
+ },
+ "browser-stdout": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.0.tgz",
+ "integrity": "sha1-81HTKWnTL6XXpVZxVCY9korjvR8=",
+ "dev": true
+ },
+ "buffer-shims": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/buffer-shims/-/buffer-shims-1.0.0.tgz",
+ "integrity": "sha1-mXjOMXOIxkmth5MCjDR37wRKi1E="
+ },
+ "bufferutil": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/bufferutil/-/bufferutil-3.0.0.tgz",
+ "integrity": "sha1-r7uDHEcimszwsfIH1KmUKEGwqw8="
+ },
+ "builtin-modules": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz",
+ "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8="
+ },
+ "caller-path": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz",
+ "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=",
+ "dev": true
+ },
+ "callsites": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz",
+ "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=",
+ "dev": true
+ },
+ "camelcase": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz",
+ "integrity": "sha1-1UVjW+HjPFQmScaRc+Xeas+uNN0="
+ },
+ "camelcase-keys": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/camelcase-keys/-/camelcase-keys-2.1.0.tgz",
+ "integrity": "sha1-MIvur/3ygRkFHvodkyITyRuPkuc=",
+ "dev": true,
+ "dependencies": {
+ "camelcase": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-2.1.1.tgz",
+ "integrity": "sha1-fB0W1nmhu+WcoCys7PsBHiAfWh8=",
+ "dev": true
+ }
+ }
+ },
+ "chalk": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz",
+ "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg="
+ },
+ "chownr": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.0.1.tgz",
+ "integrity": "sha1-4qdQQqlVGQi+vSW4Uj1fl2nXkYE="
+ },
+ "circular-json": {
+ "version": "0.3.1",
+ "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.1.tgz",
+ "integrity": "sha1-vos2rvzN6LPKeqLWr8B6NyQsDS0=",
+ "dev": true
+ },
+ "cli-cursor": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-1.0.2.tgz",
+ "integrity": "sha1-ZNo/fValRBLll5S9Ytw1KV6PKYc=",
+ "dev": true
+ },
+ "cli-width": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.1.0.tgz",
+ "integrity": "sha1-sjTKIJsp72b8UY2bmNWEewDt8Ao=",
+ "dev": true
+ },
+ "cliui": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz",
+ "integrity": "sha1-EgYBU3qRbSmUD5NNo7SNWFo5IT0="
+ },
+ "co": {
+ "version": "4.6.0",
+ "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz",
+ "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=",
+ "dev": true
+ },
+ "code-point-at": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
+ "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c="
+ },
+ "coffee-script": {
+ "version": "1.10.0",
+ "resolved": "https://registry.npmjs.org/coffee-script/-/coffee-script-1.10.0.tgz",
+ "integrity": "sha1-EpOLz5vhlI+gBvkuDEyegXBRCMA=",
+ "dev": true
+ },
+ "colors": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/colors/-/colors-1.1.2.tgz",
+ "integrity": "sha1-FopHAXVran9RoSzgyXv6KMCE7WM=",
+ "dev": true
+ },
+ "commander": {
+ "version": "2.9.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-2.9.0.tgz",
+ "integrity": "sha1-nJkJQXbhIkDLItbFFGCYQA/g99Q=",
+ "dev": true
+ },
+ "concat-map": {
+ "version": "0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+ "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
+ "dev": true
+ },
+ "concat-stream": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.0.tgz",
+ "integrity": "sha1-CqxmL9Ur54lk1VMvaUeE5wEQrPc=",
+ "dev": true
+ },
+ "console-control-strings": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz",
+ "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4="
+ },
+ "contains-path": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz",
+ "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=",
+ "dev": true
+ },
+ "core-util-is": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
+ "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
+ },
+ "cross-spawn": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-4.0.2.tgz",
+ "integrity": "sha1-e5JHYhwjrf3ThWAEqCPL45dCTUE="
+ },
+ "currently-unhandled": {
+ "version": "0.4.1",
+ "resolved": "https://registry.npmjs.org/currently-unhandled/-/currently-unhandled-0.4.1.tgz",
+ "integrity": "sha1-mI3zP+qxke95mmE2nddsF635V+o=",
+ "dev": true
+ },
+ "d": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/d/-/d-1.0.0.tgz",
+ "integrity": "sha1-dUu1v+VUUdpppYuU1F9MWwRi1Y8=",
+ "dev": true
+ },
+ "dateformat": {
+ "version": "1.0.12",
+ "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-1.0.12.tgz",
+ "integrity": "sha1-nxJLZ1lMk3/3BpMuSmQsyo27/uk=",
+ "dev": true
+ },
+ "debug": {
+ "version": "2.6.8",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz",
+ "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=",
+ "dev": true
+ },
+ "decamelize": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
+ "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA="
+ },
+ "deep-extend": {
+ "version": "0.4.2",
+ "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.4.2.tgz",
+ "integrity": "sha1-SLaZwn4zS/ifEIkr5DL25MfTSn8="
+ },
+ "deep-is": {
+ "version": "0.1.3",
+ "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz",
+ "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=",
+ "dev": true
+ },
+ "del": {
+ "version": "2.2.2",
+ "resolved": "https://registry.npmjs.org/del/-/del-2.2.2.tgz",
+ "integrity": "sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag=",
+ "dev": true
+ },
+ "delegates": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz",
+ "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o="
+ },
+ "diff": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/diff/-/diff-3.2.0.tgz",
+ "integrity": "sha1-yc45Okt8vQsFinJck98pkCeGj/k=",
+ "dev": true
+ },
+ "doctrine": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.0.0.tgz",
+ "integrity": "sha1-xz2NKQnSIpHhoAejlYBNqLZl/mM=",
+ "dev": true
+ },
+ "dotenv": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-4.0.0.tgz",
+ "integrity": "sha1-hk7xN5rO1Vzm+V3r7NzhefegzR0=",
+ "dev": true
+ },
+ "end-of-stream": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.0.tgz",
+ "integrity": "sha1-epDYM+/abPpurA9JSduw+tOmMgY="
+ },
+ "error-ex": {
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.1.tgz",
+ "integrity": "sha1-+FWobOYa3E6GIcPNoh56dhLDqNw="
+ },
+ "es5-ext": {
+ "version": "0.10.21",
+ "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.21.tgz",
+ "integrity": "sha1-Gacl+eUdAwC7wejoIRCf2dr1WSU=",
+ "dev": true
+ },
+ "es6-iterator": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.1.tgz",
+ "integrity": "sha1-jjGcnwRTv1ddN0lAplWSDlnKVRI=",
+ "dev": true
+ },
+ "es6-map": {
+ "version": "0.1.5",
+ "resolved": "https://registry.npmjs.org/es6-map/-/es6-map-0.1.5.tgz",
+ "integrity": "sha1-kTbgUD3MBqMBaQ8LsU/042TpSfA=",
+ "dev": true
+ },
+ "es6-set": {
+ "version": "0.1.5",
+ "resolved": "https://registry.npmjs.org/es6-set/-/es6-set-0.1.5.tgz",
+ "integrity": "sha1-0rPsXU2ADO2BjbU40ol02wpzzLE=",
+ "dev": true
+ },
+ "es6-symbol": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.1.tgz",
+ "integrity": "sha1-vwDvT9q2uhtG7Le2KbTH7VcVzHc=",
+ "dev": true
+ },
+ "es6-weak-map": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.2.tgz",
+ "integrity": "sha1-XjqzIlH/0VOKH45f+hNXdy+S2W8=",
+ "dev": true
+ },
+ "escape-string-regexp": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
+ "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ="
+ },
+ "escope": {
+ "version": "3.6.0",
+ "resolved": "https://registry.npmjs.org/escope/-/escope-3.6.0.tgz",
+ "integrity": "sha1-4Bl16BJ4GhY6ba392AOY3GTIicM=",
+ "dev": true
+ },
+ "eslint": {
+ "version": "3.19.0",
+ "resolved": "https://registry.npmjs.org/eslint/-/eslint-3.19.0.tgz",
+ "integrity": "sha1-yPxiAcf0DdCJQbh8CFdnOGpnmsw=",
+ "dev": true
+ },
+ "eslint-config-airbnb-base": {
+ "version": "11.2.0",
+ "resolved": "https://registry.npmjs.org/eslint-config-airbnb-base/-/eslint-config-airbnb-base-11.2.0.tgz",
+ "integrity": "sha1-GancRIGib3CQRUXsBAEWh2AY+FM=",
+ "dev": true
+ },
+ "eslint-import-resolver-node": {
+ "version": "0.2.3",
+ "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.2.3.tgz",
+ "integrity": "sha1-Wt2BBujJKNssuiMrzZ76hG49oWw=",
+ "dev": true
+ },
+ "eslint-module-utils": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.0.0.tgz",
+ "integrity": "sha1-pvjCHZATWHWc3DXbrBmCrh7li84=",
+ "dev": true,
+ "dependencies": {
+ "debug": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.2.0.tgz",
+ "integrity": "sha1-+HBX6ZWxofauaklgZkE3vFbwOdo=",
+ "dev": true
+ },
+ "ms": {
+ "version": "0.7.1",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.1.tgz",
+ "integrity": "sha1-nNE8A62/8ltl7/3nzoZO6VIBcJg=",
+ "dev": true
+ }
+ }
+ },
+ "eslint-plugin-import": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.3.0.tgz",
+ "integrity": "sha1-N8gB4K2g4pbL3yDD85OstbUq82s=",
+ "dev": true,
+ "dependencies": {
+ "doctrine": {
+ "version": "1.5.0",
+ "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz",
+ "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=",
+ "dev": true
+ }
+ }
+ },
+ "espree": {
+ "version": "3.4.3",
+ "resolved": "https://registry.npmjs.org/espree/-/espree-3.4.3.tgz",
+ "integrity": "sha1-KRC1zNSc6JPC//+qtP2LOjG4I3Q=",
+ "dev": true
+ },
+ "esprima": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/esprima/-/esprima-3.1.3.tgz",
+ "integrity": "sha1-/cpRzuYTOJXjyI1TXOSdv/YqRjM=",
+ "dev": true
+ },
+ "esquery": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.0.tgz",
+ "integrity": "sha1-z7qLV9f7qT8XKYqKAGoEzaE9gPo=",
+ "dev": true
+ },
+ "esrecurse": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.1.0.tgz",
+ "integrity": "sha1-RxO2U2rffyrE8yfVWed1a/9kgiA=",
+ "dev": true,
+ "dependencies": {
+ "estraverse": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.1.1.tgz",
+ "integrity": "sha1-9srKcokzqFDvkGYdDheYK6RxEaI=",
+ "dev": true
+ }
+ }
+ },
+ "estraverse": {
+ "version": "4.2.0",
+ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz",
+ "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=",
+ "dev": true
+ },
+ "esutils": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz",
+ "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=",
+ "dev": true
+ },
+ "event-emitter": {
+ "version": "0.3.5",
+ "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz",
+ "integrity": "sha1-34xp7vFkeSPHFXuc6DhAYQsCzDk=",
+ "dev": true
+ },
+ "eventemitter2": {
+ "version": "0.4.14",
+ "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-0.4.14.tgz",
+ "integrity": "sha1-j2G3XN4BKy6esoTUVFWDtWQ7Yas=",
+ "dev": true
+ },
+ "execa": {
+ "version": "0.5.1",
+ "resolved": "https://registry.npmjs.org/execa/-/execa-0.5.1.tgz",
+ "integrity": "sha1-3j+4XLjW6RyFvLzrFkWBeFy1ezY="
+ },
+ "exit": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz",
+ "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=",
+ "dev": true
+ },
+ "exit-hook": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/exit-hook/-/exit-hook-1.1.1.tgz",
+ "integrity": "sha1-8FyiM7SMBdVP/wd2XfhQfpXAL/g=",
+ "dev": true
+ },
+ "expand-template": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-1.0.3.tgz",
+ "integrity": "sha1-bDAzIxd6YrGyLAcCefeGEoe2mxo="
+ },
+ "fast-levenshtein": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
+ "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=",
+ "dev": true
+ },
+ "figures": {
+ "version": "1.7.0",
+ "resolved": "https://registry.npmjs.org/figures/-/figures-1.7.0.tgz",
+ "integrity": "sha1-y+Hjr/zxzUS4DK3+0o3Hk6lwHS4=",
+ "dev": true
+ },
+ "file-entry-cache": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz",
+ "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=",
+ "dev": true
+ },
+ "find-up": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz",
+ "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c="
+ },
+ "findup-sync": {
+ "version": "0.3.0",
+ "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-0.3.0.tgz",
+ "integrity": "sha1-N5MKpdgWt3fANEXhlmzGeQpMCxY=",
+ "dev": true,
+ "dependencies": {
+ "glob": {
+ "version": "5.0.15",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-5.0.15.tgz",
+ "integrity": "sha1-G8k2ueAvSmA/zCIuz3Yz0wuLk7E=",
+ "dev": true
+ }
+ }
+ },
+ "flat-cache": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.2.2.tgz",
+ "integrity": "sha1-+oZxTnLCHbiGAXYezy9VXRq8a5Y=",
+ "dev": true
+ },
+ "fs.realpath": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
+ "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=",
+ "dev": true
+ },
+ "function-bind": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.0.tgz",
+ "integrity": "sha1-FhdnFMgBeY5Ojyz391KUZ7tKV3E=",
+ "dev": true
+ },
+ "gauge": {
+ "version": "2.7.4",
+ "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz",
+ "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c="
+ },
+ "generate-function": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.0.0.tgz",
+ "integrity": "sha1-aFj+fAlpt9TpCTM3ZHrHn2DfvnQ=",
+ "dev": true
+ },
+ "generate-object-property": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/generate-object-property/-/generate-object-property-1.2.0.tgz",
+ "integrity": "sha1-nA4cQDCM6AT0eDYYuTf6iPmdUNA=",
+ "dev": true
+ },
+ "get-caller-file": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.2.tgz",
+ "integrity": "sha1-9wLmMSfn4jHBYKgMFVSstw1QR+U="
+ },
+ "get-stdin": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-4.0.1.tgz",
+ "integrity": "sha1-uWjGsKBDhDJJAui/Gl3zJXmkUP4=",
+ "dev": true
+ },
+ "get-stream": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-2.3.1.tgz",
+ "integrity": "sha1-Xzj5PzRgCWZu4BUKBUFn+Rvdld4="
+ },
+ "getobject": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/getobject/-/getobject-0.1.0.tgz",
+ "integrity": "sha1-BHpEl4n6Fg0Bj1SG7ZEyC27HiFw=",
+ "dev": true
+ },
+ "github-from-package": {
+ "version": "0.0.0",
+ "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz",
+ "integrity": "sha1-l/tdlr/eiXMxPyDoKI75oWf6ZM4="
+ },
+ "glob": {
+ "version": "7.1.2",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz",
+ "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==",
+ "dev": true
+ },
+ "globals": {
+ "version": "9.17.0",
+ "resolved": "https://registry.npmjs.org/globals/-/globals-9.17.0.tgz",
+ "integrity": "sha1-DAymltm5u2lNLlRwvTd3fKrVAoY=",
+ "dev": true
+ },
+ "globby": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz",
+ "integrity": "sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0=",
+ "dev": true
+ },
+ "graceful-fs": {
+ "version": "4.1.11",
+ "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz",
+ "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg="
+ },
+ "graceful-readlink": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/graceful-readlink/-/graceful-readlink-1.0.1.tgz",
+ "integrity": "sha1-TK+tdrxi8C+gObL5Tpo906ORpyU=",
+ "dev": true
+ },
+ "growl": {
+ "version": "1.9.2",
+ "resolved": "https://registry.npmjs.org/growl/-/growl-1.9.2.tgz",
+ "integrity": "sha1-Dqd0NxXbjY3ixe3hd14bRayFwC8=",
+ "dev": true
+ },
+ "grunt": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/grunt/-/grunt-1.0.1.tgz",
+ "integrity": "sha1-6HeHZOlEsY8yuw8QuQeEdcnftWs=",
+ "dev": true,
+ "dependencies": {
+ "esprima": {
+ "version": "2.7.3",
+ "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz",
+ "integrity": "sha1-luO3DVd59q1JzQMmc9HDEnZ7pYE=",
+ "dev": true
+ },
+ "glob": {
+ "version": "7.0.6",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-7.0.6.tgz",
+ "integrity": "sha1-IRuvr0nlJbjNkyYNFKsTYVKz9Xo=",
+ "dev": true
+ },
+ "grunt-cli": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/grunt-cli/-/grunt-cli-1.2.0.tgz",
+ "integrity": "sha1-VisRnrsGndtGSs4oRVAb6Xs1tqg=",
+ "dev": true
+ },
+ "js-yaml": {
+ "version": "3.5.5",
+ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.5.5.tgz",
+ "integrity": "sha1-A3fDgBfKvHMisNH7zSWkkWQfL74=",
+ "dev": true
+ },
+ "resolve": {
+ "version": "1.1.7",
+ "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.1.7.tgz",
+ "integrity": "sha1-IDEU2CrSxe2ejgQRs5ModeiJ6Xs=",
+ "dev": true
+ },
+ "rimraf": {
+ "version": "2.2.8",
+ "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.2.8.tgz",
+ "integrity": "sha1-5Dm+Kq7jJzIZUnMPmaiSnk/FBYI=",
+ "dev": true
+ }
+ }
+ },
+ "grunt-eslint": {
+ "version": "19.0.0",
+ "resolved": "https://registry.npmjs.org/grunt-eslint/-/grunt-eslint-19.0.0.tgz",
+ "integrity": "sha1-u3TDeQYVmc7B9mFp3vKonYYthhs=",
+ "dev": true
+ },
+ "grunt-known-options": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/grunt-known-options/-/grunt-known-options-1.1.0.tgz",
+ "integrity": "sha1-pCdO6zL6dl2lp6OxcSYXzjsUQUk=",
+ "dev": true
+ },
+ "grunt-legacy-log": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/grunt-legacy-log/-/grunt-legacy-log-1.0.0.tgz",
+ "integrity": "sha1-+4bxgJhHvAfcR4Q/ns1srLYt8tU=",
+ "dev": true,
+ "dependencies": {
+ "lodash": {
+ "version": "3.10.1",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-3.10.1.tgz",
+ "integrity": "sha1-W/Rejkm6QYnhfUgnid/RW9FAt7Y=",
+ "dev": true
+ }
+ }
+ },
+ "grunt-legacy-log-utils": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/grunt-legacy-log-utils/-/grunt-legacy-log-utils-1.0.0.tgz",
+ "integrity": "sha1-p7ji0Ps1taUPSvmG/BEnSevJbz0=",
+ "dev": true,
+ "dependencies": {
+ "lodash": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.3.0.tgz",
+ "integrity": "sha1-79nEpuxT87BUEkKZFcPkgk5NJaQ=",
+ "dev": true
+ }
+ }
+ },
+ "grunt-legacy-util": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/grunt-legacy-util/-/grunt-legacy-util-1.0.0.tgz",
+ "integrity": "sha1-OGqnjcbtUJhsKxiVcmWxtIq7m4Y=",
+ "dev": true,
+ "dependencies": {
+ "lodash": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.3.0.tgz",
+ "integrity": "sha1-79nEpuxT87BUEkKZFcPkgk5NJaQ=",
+ "dev": true
+ }
+ }
+ },
+ "has": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/has/-/has-1.0.1.tgz",
+ "integrity": "sha1-hGFzP1OLCDfJNh45qauelwTcLyg=",
+ "dev": true
+ },
+ "has-ansi": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz",
+ "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE="
+ },
+ "has-flag": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz",
+ "integrity": "sha1-nZ55MWXOAXoA8AQYxD+UKnsdEfo=",
+ "dev": true
+ },
+ "has-unicode": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz",
+ "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk="
+ },
+ "hooker": {
+ "version": "0.2.3",
+ "resolved": "https://registry.npmjs.org/hooker/-/hooker-0.2.3.tgz",
+ "integrity": "sha1-uDT3I8xKJCqmWWNFnfbZhMXT2Vk=",
+ "dev": true
+ },
+ "hosted-git-info": {
+ "version": "2.4.2",
+ "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.4.2.tgz",
+ "integrity": "sha1-AHa59GonBQbduq6lZJaJdGBhKmc="
+ },
+ "iconv-lite": {
+ "version": "0.4.17",
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.17.tgz",
+ "integrity": "sha1-T9qjs4rLwsAxsEXQ7c3+HsqxjI0=",
+ "dev": true
+ },
+ "ignore": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.3.tgz",
+ "integrity": "sha1-QyNS5XrM2HqzEQ6C0/6g5HgSFW0=",
+ "dev": true
+ },
+ "imurmurhash": {
+ "version": "0.1.4",
+ "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
+ "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
+ "dev": true
+ },
+ "indent-string": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-2.1.0.tgz",
+ "integrity": "sha1-ji1INIdCEhtKghi3oTfppSBJ3IA=",
+ "dev": true
+ },
+ "inflight": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+ "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
+ "dev": true
+ },
+ "inherits": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
+ "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
+ },
+ "ini": {
+ "version": "1.3.4",
+ "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.4.tgz",
+ "integrity": "sha1-BTfLedr1m1mhpRff9wbIbsA5Fi4="
+ },
+ "inquirer": {
+ "version": "0.12.0",
+ "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-0.12.0.tgz",
+ "integrity": "sha1-HvK/1jUE3wvHV4X/+MLEHfEvB34=",
+ "dev": true
+ },
+ "interpret": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.0.3.tgz",
+ "integrity": "sha1-y8NcYu7uc/Gat7EKgBURQBr8D5A=",
+ "dev": true
+ },
+ "invert-kv": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz",
+ "integrity": "sha1-EEqOSqym09jNFXqO+L+rLXo//bY="
+ },
+ "is-arrayish": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz",
+ "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0="
+ },
+ "is-builtin-module": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-builtin-module/-/is-builtin-module-1.0.0.tgz",
+ "integrity": "sha1-VAVy0096wxGfj3bDDLwbHgN6/74="
+ },
+ "is-finite": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/is-finite/-/is-finite-1.0.2.tgz",
+ "integrity": "sha1-zGZ3aVYCvlUO8R6LSqYwU0K20Ko=",
+ "dev": true
+ },
+ "is-fullwidth-code-point": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+ "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs="
+ },
+ "is-my-json-valid": {
+ "version": "2.16.0",
+ "resolved": "https://registry.npmjs.org/is-my-json-valid/-/is-my-json-valid-2.16.0.tgz",
+ "integrity": "sha1-8Hndm/2uZe4gOKrorLyGqxCeNpM=",
+ "dev": true
+ },
+ "is-path-cwd": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz",
+ "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=",
+ "dev": true
+ },
+ "is-path-in-cwd": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.0.tgz",
+ "integrity": "sha1-ZHdYK4IU1gI0YJRWcAO+ip6sBNw=",
+ "dev": true
+ },
+ "is-path-inside": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.0.tgz",
+ "integrity": "sha1-/AbloWg/vaE95mev9xe7wQpI838=",
+ "dev": true
+ },
+ "is-property": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz",
+ "integrity": "sha1-V/4cTkhHTt1lsJkR8msc1Ald2oQ=",
+ "dev": true
+ },
+ "is-resolvable": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.0.0.tgz",
+ "integrity": "sha1-jfV8YeouPFAUCNEA+wE8+NbgzGI=",
+ "dev": true
+ },
+ "is-stream": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz",
+ "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ="
+ },
+ "is-utf8": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz",
+ "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=",
+ "dev": true
+ },
+ "isarray": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
+ "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
+ },
+ "isexe": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+ "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA="
+ },
+ "js-tokens": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.1.tgz",
+ "integrity": "sha1-COnxMkhKLEWjCQfp3E1VZ7fxFNc=",
+ "dev": true
+ },
+ "js-yaml": {
+ "version": "3.8.4",
+ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.8.4.tgz",
+ "integrity": "sha1-UgtFZPhlc7qWZir4Woyvp7S1pvY=",
+ "dev": true
+ },
+ "json-stable-stringify": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz",
+ "integrity": "sha1-mnWdOcXy/1A/1TAGRu1EX4jE+a8=",
+ "dev": true
+ },
+ "json3": {
+ "version": "3.3.2",
+ "resolved": "https://registry.npmjs.org/json3/-/json3-3.3.2.tgz",
+ "integrity": "sha1-PAQ0dD35Pi9cQq7nsZvLSDV19OE=",
+ "dev": true
+ },
+ "jsonify": {
+ "version": "0.0.0",
+ "resolved": "https://registry.npmjs.org/jsonify/-/jsonify-0.0.0.tgz",
+ "integrity": "sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM=",
+ "dev": true
+ },
+ "jsonpointer": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/jsonpointer/-/jsonpointer-4.0.1.tgz",
+ "integrity": "sha1-T9kss04OnbPInIYi7PUfm5eMbLk=",
+ "dev": true
+ },
+ "lcid": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz",
+ "integrity": "sha1-MIrMr6C8SDo4Z7S28rlQYlHRuDU="
+ },
+ "levn": {
+ "version": "0.3.0",
+ "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz",
+ "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=",
+ "dev": true
+ },
+ "load-json-file": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz",
+ "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg="
+ },
+ "locate-path": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz",
+ "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4="
+ },
+ "lodash": {
+ "version": "4.17.4",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz",
+ "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=",
+ "dev": true
+ },
+ "lodash._baseassign": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/lodash._baseassign/-/lodash._baseassign-3.2.0.tgz",
+ "integrity": "sha1-jDigmVAPIVrQnlnxci/QxSv+Ck4=",
+ "dev": true
+ },
+ "lodash._basecopy": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/lodash._basecopy/-/lodash._basecopy-3.0.1.tgz",
+ "integrity": "sha1-jaDmqHbPNEwK2KVIghEd08XHyjY=",
+ "dev": true
+ },
+ "lodash._basecreate": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/lodash._basecreate/-/lodash._basecreate-3.0.3.tgz",
+ "integrity": "sha1-G8ZhYU2qf8MRt9A78WgGoCE8+CE=",
+ "dev": true
+ },
+ "lodash._getnative": {
+ "version": "3.9.1",
+ "resolved": "https://registry.npmjs.org/lodash._getnative/-/lodash._getnative-3.9.1.tgz",
+ "integrity": "sha1-VwvH3t5G1hzc3mh9ZdPuy6o6r/U=",
+ "dev": true
+ },
+ "lodash._isiterateecall": {
+ "version": "3.0.9",
+ "resolved": "https://registry.npmjs.org/lodash._isiterateecall/-/lodash._isiterateecall-3.0.9.tgz",
+ "integrity": "sha1-UgOte6Ql+uhCRg5pbbnPPmqsBXw=",
+ "dev": true
+ },
+ "lodash.cond": {
+ "version": "4.5.2",
+ "resolved": "https://registry.npmjs.org/lodash.cond/-/lodash.cond-4.5.2.tgz",
+ "integrity": "sha1-9HGh2khr5g9quVXRcRVSPdHSVdU=",
+ "dev": true
+ },
+ "lodash.create": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/lodash.create/-/lodash.create-3.1.1.tgz",
+ "integrity": "sha1-1/KEnw29p+BGgruM1yqwIkYd6+c=",
+ "dev": true
+ },
+ "lodash.isarguments": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz",
+ "integrity": "sha1-L1c9hcaiQon/AGY7SRwdM4/zRYo=",
+ "dev": true
+ },
+ "lodash.isarray": {
+ "version": "3.0.4",
+ "resolved": "https://registry.npmjs.org/lodash.isarray/-/lodash.isarray-3.0.4.tgz",
+ "integrity": "sha1-eeTriMNqgSKvhvhEqpvNhRtfu1U=",
+ "dev": true
+ },
+ "lodash.keys": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/lodash.keys/-/lodash.keys-3.1.2.tgz",
+ "integrity": "sha1-TbwEcrFWvlCgsoaFXRvQsMZWCYo=",
+ "dev": true
+ },
+ "loud-rejection": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/loud-rejection/-/loud-rejection-1.6.0.tgz",
+ "integrity": "sha1-W0b4AUft7leIcPCG0Eghz5mOVR8=",
+ "dev": true
+ },
+ "lru-cache": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.0.2.tgz",
+ "integrity": "sha1-HRdnnAac2l0ECZGgnbwsDbN35V4="
+ },
+ "map-obj": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-1.0.1.tgz",
+ "integrity": "sha1-2TPOuSBdgr3PSIb2dCvcK03qFG0=",
+ "dev": true
+ },
+ "mem": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/mem/-/mem-1.1.0.tgz",
+ "integrity": "sha1-Xt1StIXKHZAP5kiVUFOZoN+kX3Y="
+ },
+ "meow": {
+ "version": "3.7.0",
+ "resolved": "https://registry.npmjs.org/meow/-/meow-3.7.0.tgz",
+ "integrity": "sha1-cstmi0JSKCkKu/qFaJJYcwioAfs=",
+ "dev": true,
+ "dependencies": {
+ "find-up": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz",
+ "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=",
+ "dev": true
+ },
+ "load-json-file": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz",
+ "integrity": "sha1-lWkFcI1YtLq0wiYbBPWfMcmTdMA=",
+ "dev": true
+ },
+ "path-exists": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz",
+ "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=",
+ "dev": true
+ },
+ "path-type": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz",
+ "integrity": "sha1-WcRPfuSR2nBNpBXaWkBwuk+P5EE=",
+ "dev": true
+ },
+ "read-pkg": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz",
+ "integrity": "sha1-9f+qXs0pyzHAR0vKfXVra7KePyg=",
+ "dev": true
+ },
+ "read-pkg-up": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz",
+ "integrity": "sha1-nWPBMnbAZZGNV/ACpX9AobZD+wI=",
+ "dev": true
+ },
+ "strip-bom": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz",
+ "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=",
+ "dev": true
+ }
+ }
+ },
+ "mimic-fn": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.1.0.tgz",
+ "integrity": "sha1-5md4PZLonb00KBi1IwudYqZyrRg="
+ },
+ "minimatch": {
+ "version": "3.0.4",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
+ "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
+ "dev": true
+ },
+ "minimist": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
+ "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ="
+ },
+ "mkdirp": {
+ "version": "0.5.1",
+ "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
+ "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=",
+ "dependencies": {
+ "minimist": {
+ "version": "0.0.8",
+ "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
+ "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0="
+ }
+ }
+ },
+ "mocha": {
+ "version": "3.4.2",
+ "resolved": "https://registry.npmjs.org/mocha/-/mocha-3.4.2.tgz",
+ "integrity": "sha1-0O9NMyEm2/GNDWQMmzgt1IvpdZQ=",
+ "dev": true,
+ "dependencies": {
+ "debug": {
+ "version": "2.6.0",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.0.tgz",
+ "integrity": "sha1-vFlryr52F/Edn6FTYe3tVgi4SZs=",
+ "dev": true
+ },
+ "glob": {
+ "version": "7.1.1",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.1.tgz",
+ "integrity": "sha1-gFIR3wT6rxxjo2ADBs31reULLsg=",
+ "dev": true
+ },
+ "ms": {
+ "version": "0.7.2",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-0.7.2.tgz",
+ "integrity": "sha1-riXPJRKziFodldfwN4aNhDESR2U=",
+ "dev": true
+ },
+ "supports-color": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.1.2.tgz",
+ "integrity": "sha1-cqJiiU2dQIuVbKBf83su2KbiotU=",
+ "dev": true
+ }
+ }
+ },
+ "moment": {
+ "version": "2.18.1",
+ "resolved": "https://registry.npmjs.org/moment/-/moment-2.18.1.tgz",
+ "integrity": "sha1-w2GT3Tzhwu7SrbfIAtu8d6gbHA8="
+ },
+ "ms": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+ "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=",
+ "dev": true
+ },
+ "mute-stream": {
+ "version": "0.0.5",
+ "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.5.tgz",
+ "integrity": "sha1-j7+rsKmKJT0xhDMfno3rc3L6xsA=",
+ "dev": true
+ },
+ "nan": {
+ "version": "2.5.1",
+ "resolved": "https://registry.npmjs.org/nan/-/nan-2.5.1.tgz",
+ "integrity": "sha1-1bAWkSUzJql6K77p5hxV2NYDUeI="
+ },
+ "natural-compare": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
+ "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=",
+ "dev": true
+ },
+ "node-abi": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-2.0.2.tgz",
+ "integrity": "sha1-APPgpYEA60gBM7SMmaMswfnmyT4="
+ },
+ "noop-logger": {
+ "version": "0.1.1",
+ "resolved": "https://registry.npmjs.org/noop-logger/-/noop-logger-0.1.1.tgz",
+ "integrity": "sha1-lKKxYzxPExdVMAfYlm/Q6EG2pMI="
+ },
+ "nopt": {
+ "version": "3.0.6",
+ "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz",
+ "integrity": "sha1-xkZdvwirzU2zWTF/eaxopkayj/k=",
+ "dev": true
+ },
+ "normalize-package-data": {
+ "version": "2.3.8",
+ "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.3.8.tgz",
+ "integrity": "sha1-2Bntoqne29H/pWPqQHHZNngilbs="
+ },
+ "npm-run-path": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz",
+ "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8="
+ },
+ "npmlog": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.0.tgz",
+ "integrity": "sha512-ocolIkZYZt8UveuiDS0yAkkIjid1o7lPG8cYm05yNYzBn8ykQtaiPMEGp8fY9tKdDgm8okpdKzkvu1y9hUYugA=="
+ },
+ "number-is-nan": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
+ "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0="
+ },
+ "object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
+ },
+ "once": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+ "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E="
+ },
+ "onetime": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/onetime/-/onetime-1.1.0.tgz",
+ "integrity": "sha1-ofeDj4MUxRbwXs78vEzP4EtO14k=",
+ "dev": true
+ },
+ "optionator": {
+ "version": "0.8.2",
+ "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz",
+ "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=",
+ "dev": true
+ },
+ "os-homedir": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz",
+ "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M="
+ },
+ "os-locale": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-2.0.0.tgz",
+ "integrity": "sha1-FZGN7VEFIrge565aMJ1U9jn8OaQ="
+ },
+ "p-finally": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz",
+ "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4="
+ },
+ "p-limit": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.1.0.tgz",
+ "integrity": "sha1-sH/y2aXYi+yAYDWJWiurZqJ5iLw="
+ },
+ "p-locate": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz",
+ "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM="
+ },
+ "parse-json": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz",
+ "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck="
+ },
+ "path-exists": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
+ "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU="
+ },
+ "path-is-absolute": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+ "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
+ "dev": true
+ },
+ "path-is-inside": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz",
+ "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=",
+ "dev": true
+ },
+ "path-key": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz",
+ "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A="
+ },
+ "path-parse": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz",
+ "integrity": "sha1-PBrfhx6pzWyUMbbqK9dKD/BVxME=",
+ "dev": true
+ },
+ "path-type": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz",
+ "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM="
+ },
+ "pify": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
+ "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw="
+ },
+ "pinkie": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz",
+ "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA="
+ },
+ "pinkie-promise": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz",
+ "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o="
+ },
+ "pkg-dir": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-1.0.0.tgz",
+ "integrity": "sha1-ektQio1bstYp1EcFb/TpyTFM89Q=",
+ "dev": true,
+ "dependencies": {
+ "find-up": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz",
+ "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=",
+ "dev": true
+ },
+ "path-exists": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz",
+ "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=",
+ "dev": true
+ }
+ }
+ },
+ "pluralize": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-1.2.1.tgz",
+ "integrity": "sha1-0aIUg/0iu0HlihL6NCGCMUCJfEU=",
+ "dev": true
+ },
+ "prebuild-install": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-2.1.2.tgz",
+ "integrity": "sha1-2a4MqFMw4Dli2TKS+VqLRMLr9QU="
+ },
+ "prelude-ls": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz",
+ "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=",
+ "dev": true
+ },
+ "process-nextick-args": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-1.0.7.tgz",
+ "integrity": "sha1-FQ4gt1ZZCtP5EJPyWk8q2L/zC6M="
+ },
+ "progress": {
+ "version": "1.1.8",
+ "resolved": "https://registry.npmjs.org/progress/-/progress-1.1.8.tgz",
+ "integrity": "sha1-4mDHj2Fhzdmw5WzD4Khd4Xx6V74=",
+ "dev": true
+ },
+ "pseudomap": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz",
+ "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM="
+ },
+ "pump": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/pump/-/pump-1.0.2.tgz",
+ "integrity": "sha1-Oz7mUS+U8OV1U4wXmV+fFpkKXVE="
+ },
+ "rc": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.1.tgz",
+ "integrity": "sha1-LgPo5C7kULjLPc5lvhv4l04d/ZU="
+ },
+ "read-pkg": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz",
+ "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg="
+ },
+ "read-pkg-up": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz",
+ "integrity": "sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4="
+ },
+ "readable-stream": {
+ "version": "2.2.9",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.2.9.tgz",
+ "integrity": "sha1-z3jsb0ptHrQ9JkiMrJfwQudLf8g="
+ },
+ "readline2": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/readline2/-/readline2-1.0.1.tgz",
+ "integrity": "sha1-QQWWCP/BVHV7cV2ZidGZ/783LjU=",
+ "dev": true
+ },
+ "rechoir": {
+ "version": "0.6.2",
+ "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz",
+ "integrity": "sha1-hSBLVNuoLVdC4oyWdW70OvUOM4Q=",
+ "dev": true
+ },
+ "redent": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/redent/-/redent-1.0.0.tgz",
+ "integrity": "sha1-z5Fqsf1fHxbfsggi3W7H9zDCr94=",
+ "dev": true
+ },
+ "repeating": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/repeating/-/repeating-2.0.1.tgz",
+ "integrity": "sha1-UhTFOpJtNVJwdSf7q0FdvAjQbdo=",
+ "dev": true
+ },
+ "require-directory": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
+ "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I="
+ },
+ "require-main-filename": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz",
+ "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE="
+ },
+ "require-uncached": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz",
+ "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=",
+ "dev": true
+ },
+ "resolve": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.3.3.tgz",
+ "integrity": "sha1-ZVkHw0aahoDcLeOidaj91paR8OU=",
+ "dev": true
+ },
+ "resolve-from": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz",
+ "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=",
+ "dev": true
+ },
+ "restore-cursor": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-1.0.1.tgz",
+ "integrity": "sha1-NGYfRohjJ/7SmRR5FSJS35LapUE=",
+ "dev": true
+ },
+ "rimraf": {
+ "version": "2.6.1",
+ "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.1.tgz",
+ "integrity": "sha1-wjOOxkPfeht/5cVPqG9XQopV8z0=",
+ "dev": true
+ },
+ "run-async": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/run-async/-/run-async-0.1.0.tgz",
+ "integrity": "sha1-yK1KXhEGYeQCp9IbUw4AnyX444k=",
+ "dev": true
+ },
+ "rx-lite": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-3.1.2.tgz",
+ "integrity": "sha1-Gc5QLKVyZl87ZHsQk5+X/RYV8QI=",
+ "dev": true
+ },
+ "safe-buffer": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.0.1.tgz",
+ "integrity": "sha1-0mPKVGls2KMGtcplUekt5XkY++c="
+ },
+ "semver": {
+ "version": "5.3.0",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-5.3.0.tgz",
+ "integrity": "sha1-myzl094C0XxgEq0yaqa00M9U+U8="
+ },
+ "set-blocking": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
+ "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc="
+ },
+ "shelljs": {
+ "version": "0.7.7",
+ "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.7.7.tgz",
+ "integrity": "sha1-svXHfvlxSPS09uImguELuoZnz/E=",
+ "dev": true
+ },
+ "should": {
+ "version": "11.2.1",
+ "resolved": "https://registry.npmjs.org/should/-/should-11.2.1.tgz",
+ "integrity": "sha1-kPVRRVUtAc/CAGZuToGKHJZw7aI=",
+ "dev": true
+ },
+ "should-equal": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/should-equal/-/should-equal-1.0.1.tgz",
+ "integrity": "sha1-C26VFvJgGp+wuy3MNpr6HH4gCvc=",
+ "dev": true
+ },
+ "should-format": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/should-format/-/should-format-3.0.3.tgz",
+ "integrity": "sha1-m/yPdPo5IFxT04w01xcwPidxJPE=",
+ "dev": true
+ },
+ "should-type": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/should-type/-/should-type-1.4.0.tgz",
+ "integrity": "sha1-B1bYzoRt/QmEOmlHcZ36DUz/XPM=",
+ "dev": true
+ },
+ "should-type-adaptors": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/should-type-adaptors/-/should-type-adaptors-1.0.1.tgz",
+ "integrity": "sha1-7+VVPN9oz/ZuXF9RtxLcNRx3vqo=",
+ "dev": true
+ },
+ "should-util": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/should-util/-/should-util-1.0.0.tgz",
+ "integrity": "sha1-yYzaN0qmsZDfi6h8mInCtNtiAGM=",
+ "dev": true
+ },
+ "signal-exit": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz",
+ "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0="
+ },
+ "simple-get": {
+ "version": "1.4.3",
+ "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-1.4.3.tgz",
+ "integrity": "sha1-6XVe2kB+ltpAxeUVjJ6jezO+y+s="
+ },
+ "slice-ansi": {
+ "version": "0.0.4",
+ "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-0.0.4.tgz",
+ "integrity": "sha1-7b+JA/ZvfOL46v1s7tZeJkyDGzU=",
+ "dev": true
+ },
+ "spdx-correct": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-1.0.2.tgz",
+ "integrity": "sha1-SzBz2TP/UfORLwOsVRlJikFQ20A="
+ },
+ "spdx-expression-parse": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-1.0.4.tgz",
+ "integrity": "sha1-m98vIOH0DtRH++JzJmGR/O1RYmw="
+ },
+ "spdx-license-ids": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-1.2.2.tgz",
+ "integrity": "sha1-yd96NCRZSt5r0RkA1ZZpbcBrrFc="
+ },
+ "sprintf-js": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
+ "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
+ "dev": true
+ },
+ "string_decoder": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.1.tgz",
+ "integrity": "sha1-YuIA8DmVWmgQ2N8KM//A8BNmLZg="
+ },
+ "string-width": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+ "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M="
+ },
+ "strip-ansi": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+ "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8="
+ },
+ "strip-bom": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz",
+ "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM="
+ },
+ "strip-eof": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz",
+ "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8="
+ },
+ "strip-indent": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-1.0.1.tgz",
+ "integrity": "sha1-DHlipq3vp7vUrDZkYKY4VSrhoKI=",
+ "dev": true
+ },
+ "strip-json-comments": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
+ "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo="
+ },
+ "supports-color": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz",
+ "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc="
+ },
+ "table": {
+ "version": "3.8.3",
+ "resolved": "https://registry.npmjs.org/table/-/table-3.8.3.tgz",
+ "integrity": "sha1-K7xULw/amGGnVdOUf+/Ys/UThV8=",
+ "dev": true,
+ "dependencies": {
+ "is-fullwidth-code-point": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
+ "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
+ "dev": true
+ },
+ "string-width": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.0.0.tgz",
+ "integrity": "sha1-Y1xUNsxypuDDh87KJ41OLuxSaH4=",
+ "dev": true
+ }
+ }
+ },
+ "tar-fs": {
+ "version": "1.15.2",
+ "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-1.15.2.tgz",
+ "integrity": "sha1-dh9bMpMsezlGGmDVN/rqDYCEgww="
+ },
+ "tar-stream": {
+ "version": "1.5.4",
+ "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-1.5.4.tgz",
+ "integrity": "sha1-NlSc8E7RrumyowwBQyUiONr5QBY="
+ },
+ "text-table": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
+ "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=",
+ "dev": true
+ },
+ "through": {
+ "version": "2.3.8",
+ "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
+ "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
+ "dev": true
+ },
+ "trim-newlines": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/trim-newlines/-/trim-newlines-1.0.0.tgz",
+ "integrity": "sha1-WIeWa7WCpFA6QetST301ARgVphM=",
+ "dev": true
+ },
+ "tryit": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/tryit/-/tryit-1.0.3.tgz",
+ "integrity": "sha1-OTvnMKlEb9Hq1tpZoBQwjzbCics=",
+ "dev": true
+ },
+ "tunnel-agent": {
+ "version": "0.4.3",
+ "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.4.3.tgz",
+ "integrity": "sha1-Y3PbdpCf5XDgjXNYM2Xtgop07us="
+ },
+ "type-check": {
+ "version": "0.3.2",
+ "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz",
+ "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=",
+ "dev": true
+ },
+ "typedarray": {
+ "version": "0.0.6",
+ "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
+ "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=",
+ "dev": true
+ },
+ "ultron": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.0.tgz",
+ "integrity": "sha1-sHoualQagV/Go0zNRTO67DB8qGQ="
+ },
+ "underscore.string": {
+ "version": "3.2.3",
+ "resolved": "https://registry.npmjs.org/underscore.string/-/underscore.string-3.2.3.tgz",
+ "integrity": "sha1-gGmSYzZl1eX8tNsfs6hi62jp5to=",
+ "dev": true
+ },
+ "unzip-response": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-1.0.2.tgz",
+ "integrity": "sha1-uYTwh3/AqJwsdzzB73tbIytbBv4="
+ },
+ "user-home": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/user-home/-/user-home-2.0.0.tgz",
+ "integrity": "sha1-nHC/2Babwdy/SGBODwS4tJzenp8=",
+ "dev": true
+ },
+ "utf-8-validate": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/utf-8-validate/-/utf-8-validate-3.0.1.tgz",
+ "integrity": "sha1-XSuGVrTdz97Uche2R6mJQbY88hM="
+ },
+ "util-deprecate": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+ "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
+ },
+ "validate-npm-package-license": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.1.tgz",
+ "integrity": "sha1-KAS6vnEq0zeUWaz74kdGqywwP7w="
+ },
+ "which": {
+ "version": "1.2.14",
+ "resolved": "https://registry.npmjs.org/which/-/which-1.2.14.tgz",
+ "integrity": "sha1-mofEN48D6CfOyvGs31bHNsAcFOU="
+ },
+ "which-module": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz",
+ "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho="
+ },
+ "wide-align": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.2.tgz",
+ "integrity": "sha512-ijDLlyQ7s6x1JgCLur53osjm/UXUYD9+0PbYKrBsYisYXzCxN+HC3mYDNy/dWdmf3AwqwU3CXwDCvsNgGK1S0w=="
+ },
+ "wordwrap": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz",
+ "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=",
+ "dev": true
+ },
+ "wrap-ansi": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz",
+ "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU="
+ },
+ "wrappy": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+ "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8="
+ },
+ "write": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz",
+ "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=",
+ "dev": true
+ },
+ "ws": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/ws/-/ws-3.0.0.tgz",
+ "integrity": "sha1-mN2wAFbIOQy3Ued4h4hJf5kQO2w="
+ },
+ "xtend": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz",
+ "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68="
+ },
+ "y18n": {
+ "version": "3.2.1",
+ "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz",
+ "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE="
+ },
+ "yallist": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz",
+ "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI="
+ },
+ "yargs": {
+ "version": "8.0.1",
+ "resolved": "https://registry.npmjs.org/yargs/-/yargs-8.0.1.tgz",
+ "integrity": "sha1-Qg73XoQMFFeoCtzKm8b6OEneUao=",
+ "dependencies": {
+ "is-fullwidth-code-point": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
+ "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8="
+ },
+ "string-width": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.0.0.tgz",
+ "integrity": "sha1-Y1xUNsxypuDDh87KJ41OLuxSaH4="
+ }
+ }
+ },
+ "yargs-parser": {
+ "version": "7.0.0",
+ "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-7.0.0.tgz",
+ "integrity": "sha1-jQrELxbqVd69MyyvTEA4s+P139k="
+ }
+ }
+}
diff --git a/package.json b/package.json
index 81106ec..30f679d 100644
--- a/package.json
+++ b/package.json
@@ -1,7 +1,7 @@
{
"name": "whir.io",
- "version": "1.2.0",
- "description": "The whir.io chat client.",
+ "version": "1.3.0",
+ "description": "The whir.io chat client. [alpha]",
"author": {
"name": "Stefan Aichholzer",
"email": "play@analogbird.com",
@@ -24,26 +24,26 @@
},
"license": "MIT",
"engines": {
- "node": "6.4.x"
+ "node": ">=7.10"
},
"dependencies": {
"blessed": "^0.1.81",
- "bufferutil": "^1.2.1",
+ "bufferutil": "^3.0.0",
"chalk": "^1.1.3",
- "utf-8-validate": "^1.2.1",
- "ws": "^1.1.1",
- "yargs": "^6.3.0"
+ "moment": "^2.17.1",
+ "utf-8-validate": "^3.0.1",
+ "ws": "^3.0.0",
+ "yargs": "^8.0.1"
},
"devDependencies": {
- "dotenv": "^2.0.0",
+ "dotenv": "^4.0.0",
+ "eslint": "^3.19.0",
+ "eslint-config-airbnb-base": "^11.2.0",
+ "eslint-plugin-import": "^2.3.0",
"grunt": "^1.0.1",
- "grunt-contrib-jshint": "^1.0.0",
- "grunt-jscs": "^3.0.1",
- "grunt-nodemon": "^0.4.2",
- "jscs": "^3.0.7",
- "jshint": "^2.9.3",
- "mocha": "^3.0.2",
- "should": "^11.1.0"
+ "grunt-eslint": "^19.0.0",
+ "mocha": "^3.2.0",
+ "should": "^11.2.0"
},
"main": "./app/index.js",
"directories": {
@@ -55,7 +55,7 @@
"bugs": {
"url": "https://github.com/WhirIO/Client/issues"
},
- "_from": "whir.io@^1.1.0",
+ "_from": "whir.io@^1.2.0",
"_npmUser": {
"name": "aichholzer",
"email": "theaichholzer@gmail.com"
diff --git a/history/.history b/store/.store
similarity index 100%
rename from history/.history
rename to store/.store
diff --git a/w.sh b/w.sh
new file mode 100755
index 0000000..f77b4e7
--- /dev/null
+++ b/w.sh
@@ -0,0 +1,32 @@
+#!/bin/bash
+
+###
+# This is for development & testing purposes only.
+# The channel defaults to "whir"
+# Runs against localhost:9000, provided this host and port are up.
+# Install the server locally: https://github.com/WhirIO/Server
+###
+
+USER=$1
+CHANNEL='-c whir'
+UNSECURE='UNSECURE_SOCKET=true'
+
+if [ "$USER" == "" ]; then
+ echo
+ echo " 💥 You need a user."
+ exit 1
+fi
+
+if [ "$2" != "" ]; then
+ CHANNEL="-c $2"
+fi
+if [ "$2" == "rand" ]; then
+ CHANNEL=""
+fi
+
+if [ "$3" == "false" ]; then
+ UNSECURE=''
+fi
+
+clear
+eval "$UNSECURE node app/ -h localhost:9000 -u $USER $CHANNEL"