diff --git a/.eslintrc b/.eslintrc new file mode 100644 index 0000000..d99f427 --- /dev/null +++ b/.eslintrc @@ -0,0 +1,35 @@ +{ + "env": { + "browser": true, + "node": true, + "commonjs": true, + "es6": true + }, + "extends": "eslint:recommended", + "parserOptions": { + "sourceType": "module" + }, + "rules": { + "indent": [ + "warn", + 4, + { + "SwitchCase": 1 + } + ], + "quotes": [ + "warn", + "double" + ], + "semi": [ + "error", + "always" + ], + "no-undef": "error", + "no-unused-vars": "warn", + "eqeqeq": [ + "error", + "always" + ] + } +} \ No newline at end of file diff --git a/discord-bot-core/Client.js b/discord-bot-core/Client.js deleted file mode 100644 index 573e1d6..0000000 --- a/discord-bot-core/Client.js +++ /dev/null @@ -1,102 +0,0 @@ -const CoreUtil = require("./Util.js"); -const Camo = require("camo"); -const CronJob = require("cron").CronJob; -const Discord = require("discord.js"); -const HandleGuildMessage = require("./HandleGuildMessage"); -const InternalConfig = require("./internal-config.json"); -const RequireAll = require("require-all"); -const Util = require("./Util.js"); - -let neDB; - -module.exports = class Client extends Discord.Client { - /** - * Construct a new Discord.Client with some added functionality - * @param {string} token bot token - * @param {string} commandsDir location of dir containing commands .js files - * @param {*} guildDataModel GuildData model to be used for app; must extend BaseGuildData - */ - constructor(token, commandsDir, guildDataModel) { - super({ - messageCacheMaxSize: 16, - disabledEvents: InternalConfig.disabledEvents - }); - - this._token = token; - this.commandsDir = commandsDir; - this.guildDataModel = guildDataModel; - - this.commands = RequireAll(this.commandsDir); - - this.on("ready", this._onReady); - this.on("message", this._onMessage); - this.on("debug", this._onDebug); - this.on("guildCreate", this._onGuildCreate); - this.on("guildDelete", this._onGuildDelete); - process.on("uncaughtException", err => this._onUnhandledException(this, err)); - } - - _onReady() { - this.user.setGame(InternalConfig.website.replace(/^https?:\/\//, "")); - CoreUtil.dateLog(`Registered bot ${this.user.username}`); - - this.removeDeletedGuilds(); - } - - _onMessage(message) { - if (message.channel.type === "text" && message.member) - HandleGuildMessage(this, message, this.commands); - } - - _onDebug(info) { - info = info.replace(/Authenticated using token [^ ]+/, "Authenticated using token [redacted]"); - if (!InternalConfig.debugIgnores.some(x => info.startsWith(x))) - CoreUtil.dateDebug(info); - } - - _onGuildCreate(guild) { - CoreUtil.dateLog(`Added to guild ${guild.name}`); - } - - _onGuildDelete(guild) { - this.guildDataModel.findOneAndDelete({ guildID: guild.id }); - - CoreUtil.dateLog(`Removed from guild ${guild.name}, removing data for this guild`); - } - - _onUnhandledException(client, err) { - CoreUtil.dateError("Unhandled exception!\n", err); - CoreUtil.dateLog("Destroying existing client..."); - client.destroy().then(() => { - CoreUtil.dateLog("Client destroyed, recreating..."); - setTimeout(() => client.login(client._token), InternalConfig.reconnectTimeout); - }); - } - - bootstrap() { - Camo.connect("nedb://guilds-data").then(db => { - neDB = db; - new CronJob(InternalConfig.dbCompactionSchedule, compactCollections, null, true); - - this.emit("beforeLogin"); - this.login(this._token); - }); - } - - removeDeletedGuilds() { - this.guildDataModel.find().then(guildDatas => { - for (let guildData of guildDatas) - if (!this.guilds.get(guildData.guildID)) - guildData.delete(); - }); - } -}; - -function compactCollections() { - /*I realise it is a bit of a cheat to just access _collections in this manner, but in the absence of - camo actually having any kind of solution for this it's the easiest method I could come up with. - Maybe at some point in future I should fork camo and add this feature. The compaction function is NeDB only - and camo is designed to work with both NeDB and MongoDB, which is presumably why it doesn't alraedy exist */ - for (let collectionName of Object.keys(neDB._collections)) - neDB._collections[collectionName].persistence.compactDatafile(); -} \ No newline at end of file diff --git a/discord-bot-core/Command.js b/discord-bot-core/Command.js deleted file mode 100644 index 612a284..0000000 --- a/discord-bot-core/Command.js +++ /dev/null @@ -1,15 +0,0 @@ -module.exports = class Command { - constructor({ name, description, syntax, admin, invoke }) { - this.name = name; - this.description = description; - this.syntax = syntax; - this.admin = admin; - this.invoke = invoke; - - const params = this.syntax.split(/ +/); - const optionalParams = params.filter(x => x.match(/^\[.+\]$/)); - - this.maxParamCount = params.length - 1; - this.expectedParamCount = this.maxParamCount - optionalParams.length; - } -}; \ No newline at end of file diff --git a/discord-bot-core/HandleGuildMessage.js b/discord-bot-core/HandleGuildMessage.js deleted file mode 100644 index 5f6eaf7..0000000 --- a/discord-bot-core/HandleGuildMessage.js +++ /dev/null @@ -1,53 +0,0 @@ -const RequireAll = require("require-all"); - -const internalCommands = RequireAll(__dirname + "/core-commands"); - -function handleGuildMessage(client, message, commands) { - if (isCommand(message)) - client.guildDataModel.findOne({ guildID: message.guild.id }) - .then(guildData => - handleGuildCommand( - client, - message, - Object.assign({}, internalCommands, commands), - guildData || client.guildDataModel.create({ guildID: message.guild.id }) - )); -} - -function handleGuildCommand(client, message, commands, guildData) { - const { botName, isMemberAdmin, params, command } = parseDetails(message, commands); - - if (!command) - return; - - if (params.length < command.expectedParamCount) - message.reply(`Incorrect syntax!\n**Expected:** *${botName} ${command.syntax}*\n**Need help?** *${botName} help*`); - - else if (isMemberAdmin || !command.admin) - command.invoke({ message, params, guildData, client, commands, isMemberAdmin }) - .then(response => { - guildData.save(); - if (response) message.reply(response); - }) - .catch(err => err && message.reply(err)); -} - -function parseDetails(message, commands) { - const split = message.content.split(/ +/); - const commandName = Object.keys(commands).find(x => - /**/ commands[x].name.toLowerCase() === (split[1] || "").toLowerCase()); - - return { - botName: "@" + (message.guild.me.nickname || message.guild.me.user.username), - isMemberAdmin: message.member.permissions.has("ADMINISTRATOR"), - params: split.slice(2, split.length), - command: commands[commandName] - }; -} - -function isCommand(message) { - //criteria for a command is bot being mentioned - return new RegExp(`^<@!?${/[0-9]{18}/.exec(message.guild.me.toString())[0]}>`).exec(message.content); -} - -module.exports = handleGuildMessage; \ No newline at end of file diff --git a/discord-bot-core/Util.js b/discord-bot-core/Util.js deleted file mode 100644 index 3f97b8d..0000000 --- a/discord-bot-core/Util.js +++ /dev/null @@ -1,74 +0,0 @@ -// @ts-ignore -const InternalConfig = require("./internal-config.json"); -const Console = require("console"); -const SimpleFileWriter = require("simple-file-writer"); - -const logWriter = new SimpleFileWriter("./console.log"); -const debugLogWriter = new SimpleFileWriter("./debug.log"); - -function ask(client, textChannel, member, question) { - //return a promise which will resolve once the user next sends a message in this textChannel - return new Promise((resolve, reject) => { - const cancelAsk = () => { - client.removeListener("message", handler); - textChannel.send("Response to question timed out"); - }; - - const askTimeout = setTimeout(cancelAsk, InternalConfig.askTimeout); - - const handler = responseMessage => { - if (responseMessage.channel.id === textChannel.id && responseMessage.member && responseMessage.member.id === member.id) { - clearTimeout(askTimeout); - resolve(responseMessage); - } - }; - - client.on("message", handler); - - textChannel.send(member.toString() + " " + question).catch(reject); - }); -} - -function dateLog(...args) { - doDateLog(Console.log, logWriter, args, "INFO"); -} - -function dateError(...args) { - doDateLog(Console.error, logWriter, args, "ERROR"); -} - -function dateDebugError(...args) { - doDateLog(null, null, args, "DEBUG ERROR"); -} - -function dateDebug(...args) { - doDateLog(null, null, args, "DEBUG"); -} - -function doDateLog(consoleMethod, fileWriter, args, prefix = "") { - args = formatArgs([`[${prefix}]`].concat(args)); - - if (consoleMethod !== null) - consoleMethod.apply(this, args); - - if (fileWriter !== null) - fileWriter.write(formatArgsForFile(args)); - - debugLogWriter.write(formatArgsForFile(args)); -} - -function formatArgs(args) { - return [`[${new Date().toUTCString()}]`].concat(args); -} - -function formatArgsForFile(args) { - return args.join(" ") + "\n"; -} - -module.exports = { - dateError, - dateLog, - dateDebug, - dateDebugError, - ask -}; \ No newline at end of file diff --git a/discord-bot-core/BaseEmbeddedData.js b/discord-bot-core/base-embedded-data.js similarity index 72% rename from discord-bot-core/BaseEmbeddedData.js rename to discord-bot-core/base-embedded-data.js index 43d4aa7..02a4cd0 100644 --- a/discord-bot-core/BaseEmbeddedData.js +++ b/discord-bot-core/base-embedded-data.js @@ -2,7 +2,7 @@ const Camo = require("camo"); // @ts-ignore module.exports = class BaseEmbeddedData extends Camo.EmbeddedDocument { - constructor() { - super(); - } + constructor() { + super(); + } }; \ No newline at end of file diff --git a/discord-bot-core/BaseGuildData.js b/discord-bot-core/base-guild-data.js similarity index 55% rename from discord-bot-core/BaseGuildData.js rename to discord-bot-core/base-guild-data.js index b6e8268..547fadf 100644 --- a/discord-bot-core/BaseGuildData.js +++ b/discord-bot-core/base-guild-data.js @@ -1,9 +1,9 @@ const Camo = require("camo"); module.exports = class BaseGuildData extends Camo.Document { - constructor() { - super(); + constructor() { + super(); - this.guildID = String; - } + this.guildID = String; + } }; \ No newline at end of file diff --git a/discord-bot-core/client.js b/discord-bot-core/client.js new file mode 100644 index 0000000..db4f99d --- /dev/null +++ b/discord-bot-core/client.js @@ -0,0 +1,102 @@ +const CoreUtil = require("./util.js"); +const Camo = require("camo"); +const CronJob = require("cron").CronJob; +const Discord = require("discord.js"); +const HandleGuildMessage = require("./handle-guild-message.js"); +// @ts-ignore +const InternalConfig = require("./internal-config.json"); +const RequireAll = require("require-all"); + +let neDB; + +module.exports = class Client extends Discord.Client { + /** + * Construct a new Discord.Client with some added functionality + * @param {string} token bot token + * @param {string} commandsDir location of dir containing commands .js files + * @param {*} guildDataModel GuildData model to be used for app; must extend BaseGuildData + */ + constructor(token, commandsDir, guildDataModel) { + super({ + messageCacheMaxSize: 16, + disabledEvents: InternalConfig.disabledEvents + }); + + this._token = token; + this.commandsDir = commandsDir; + this.guildDataModel = guildDataModel; + + this.commands = RequireAll(this.commandsDir); + + this.on("ready", this._onReady); + this.on("message", this._onMessage); + this.on("debug", this._onDebug); + this.on("guildCreate", this._onGuildCreate); + this.on("guildDelete", this._onGuildDelete); + process.on("uncaughtException", err => this._onUnhandledException(this, err)); + } + + _onReady() { + this.user.setGame(InternalConfig.website.replace(/^https?:\/\//, "")); + CoreUtil.dateLog(`Registered bot ${this.user.username}`); + + this.removeDeletedGuilds(); + } + + _onMessage(message) { + if (message.channel.type === "text" && message.member) + HandleGuildMessage(this, message, this.commands); + } + + _onDebug(info) { + info = info.replace(/Authenticated using token [^ ]+/, "Authenticated using token [redacted]"); + if (!InternalConfig.debugIgnores.some(x => info.startsWith(x))) + CoreUtil.dateDebug(info); + } + + _onGuildCreate(guild) { + CoreUtil.dateLog(`Added to guild ${guild.name}`); + } + + _onGuildDelete(guild) { + this.guildDataModel.findOneAndDelete({ guildID: guild.id }); + + CoreUtil.dateLog(`Removed from guild ${guild.name}, removing data for this guild`); + } + + _onUnhandledException(client, err) { + CoreUtil.dateError("Unhandled exception!\n", err); + CoreUtil.dateLog("Destroying existing client..."); + client.destroy().then(() => { + CoreUtil.dateLog("Client destroyed, recreating..."); + setTimeout(() => client.login(client._token), InternalConfig.reconnectTimeout); + }); + } + + bootstrap() { + Camo.connect("nedb://guilds-data").then(db => { + neDB = db; + new CronJob(InternalConfig.dbCompactionSchedule, compactCollections, null, true); + + this.emit("beforeLogin"); + this.login(this._token); + }); + } + + removeDeletedGuilds() { + this.guildDataModel.find().then(guildDatas => { + for (let guildData of guildDatas) + if (!this.guilds.get(guildData.guildID)) + guildData.delete(); + }); + } +}; + +function compactCollections() { + /*I realise it is a bit of a cheat to just access _collections in this manner, but in the absence of + camo actually having any kind of solution for this it's the easiest method I could come up with. + Maybe at some point in future I should fork camo and add this feature. The compaction function is NeDB only + and camo is designed to work with both NeDB and MongoDB, which is presumably why it doesn't alraedy exist */ + for (let collectionName of Object.keys(neDB._collections)) + neDB._collections[collectionName].persistence.compactDatafile(); +} \ No newline at end of file diff --git a/discord-bot-core/command.js b/discord-bot-core/command.js new file mode 100644 index 0000000..396592c --- /dev/null +++ b/discord-bot-core/command.js @@ -0,0 +1,15 @@ +module.exports = class Command { + constructor({ name, description, syntax, admin, invoke }) { + this.name = name; + this.description = description; + this.syntax = syntax; + this.admin = admin; + this.invoke = invoke; + + const params = this.syntax.split(/ +/); + const optionalParams = params.filter(x => x.match(/^\[.+\]$/)); + + this.maxParamCount = params.length - 1; + this.expectedParamCount = this.maxParamCount - optionalParams.length; + } +}; \ No newline at end of file diff --git a/discord-bot-core/handle-guild-message.js b/discord-bot-core/handle-guild-message.js new file mode 100644 index 0000000..aadee27 --- /dev/null +++ b/discord-bot-core/handle-guild-message.js @@ -0,0 +1,53 @@ +const RequireAll = require("require-all"); + +const internalCommands = RequireAll(__dirname + "/core-commands"); + +function handleGuildMessage(client, message, commands) { + if (isCommand(message)) + client.guildDataModel.findOne({ guildID: message.guild.id }) + .then(guildData => + handleGuildCommand( + client, + message, + Object.assign({}, internalCommands, commands), + guildData || client.guildDataModel.create({ guildID: message.guild.id }) + )); +} + +function handleGuildCommand(client, message, commands, guildData) { + const { botName, isMemberAdmin, params, command } = parseDetails(message, commands); + + if (!command) + return; + + if (params.length < command.expectedParamCount) + message.reply(`Incorrect syntax!\n**Expected:** *${botName} ${command.syntax}*\n**Need help?** *${botName} help*`); + + else if (isMemberAdmin || !command.admin) + command.invoke({ message, params, guildData, client, commands, isMemberAdmin }) + .then(response => { + guildData.save(); + if (response) message.reply(response); + }) + .catch(err => err && message.reply(err)); +} + +function parseDetails(message, commands) { + const split = message.content.split(/ +/); + const commandName = Object.keys(commands).find(x => + /**/ commands[x].name.toLowerCase() === (split[1] || "").toLowerCase()); + + return { + botName: "@" + (message.guild.me.nickname || message.guild.me.user.username), + isMemberAdmin: message.member.permissions.has("ADMINISTRATOR"), + params: split.slice(2, split.length), + command: commands[commandName] + }; +} + +function isCommand(message) { + //criteria for a command is bot being mentioned + return new RegExp(`^<@!?${/[0-9]{18}/.exec(message.guild.me.toString())[0]}>`).exec(message.content); +} + +module.exports = handleGuildMessage; \ No newline at end of file diff --git a/discord-bot-core/index.js b/discord-bot-core/index.js index 532a802..c7d3869 100644 --- a/discord-bot-core/index.js +++ b/discord-bot-core/index.js @@ -2,13 +2,13 @@ const InternalConfig = require("./internal-config.json"); module.exports = { - Client: require("./Client.js"), - BaseGuildData: require("./BaseGuildData.js"), - BaseEmbeddedData: require("./BaseEmbeddedData.js"), - Command: require("./Command.js"), - util: require("./Util.js"), - details: { - website: InternalConfig.website, - discordInvite: InternalConfig.discordInvite - } + Client: require("./client.js"), + BaseGuildData: require("./base-guild-data.js"), + BaseEmbeddedData: require("./base-embedded-data.js"), + Command: require("./command.js"), + util: require("./util.js"), + details: { + website: InternalConfig.website, + discordInvite: InternalConfig.discordInvite + } }; diff --git a/discord-bot-core/internal-config.json b/discord-bot-core/internal-config.json index 4f0f239..7bdfd6c 100644 --- a/discord-bot-core/internal-config.json +++ b/discord-bot-core/internal-config.json @@ -1,21 +1,21 @@ { - "dbCompactionSchedule": "0 * * * * *", - "restartSchedule": "0 0 0 * * *", - "restartTimeout": 5000, - "website": "https://benji7425.github.io", - "discordInvite": "https://discord.gg/SSkbwSJ", - "debugIgnores": [ - "[ws] [connection] Sending a heartbeat", - "[ws] [connection] Heartbeat acknowledged" - ], - "disabledEvents": [ - "CHANNEL_PINS_UPDATE", - "GUILD_BAN_ADD", - "GUILD_BAN_REMOVE", - "PRESENCE_UPDATE", - "TYPING_START", - "USER_NOTE_UPDATE", - "USER_SETTINGS_UPDATE" - ], - "askTimeout": 60000 + "dbCompactionSchedule": "0 * * * * *", + "restartSchedule": "0 0 0 * * *", + "restartTimeout": 5000, + "website": "https://benji7425.github.io", + "discordInvite": "https://discord.gg/SSkbwSJ", + "debugIgnores": [ + "[ws] [connection] Sending a heartbeat", + "[ws] [connection] Heartbeat acknowledged" + ], + "disabledEvents": [ + "CHANNEL_PINS_UPDATE", + "GUILD_BAN_ADD", + "GUILD_BAN_REMOVE", + "PRESENCE_UPDATE", + "TYPING_START", + "USER_NOTE_UPDATE", + "USER_SETTINGS_UPDATE" + ], + "askTimeout": 60000 } \ No newline at end of file diff --git a/discord-bot-core/monitor.js b/discord-bot-core/monitor.js index a761777..44a7e54 100644 --- a/discord-bot-core/monitor.js +++ b/discord-bot-core/monitor.js @@ -11,23 +11,23 @@ restart(); new CronJob(InternalConfig.restartSchedule, restart, null, true); function restart() { - ensureKilledInstance() - .then(bootstrapNewInstance) - .catch(DiscordUtil.dateError); + ensureKilledInstance() + .then(bootstrapNewInstance) + .catch(DiscordUtil.dateError); } function bootstrapNewInstance() { - instance = fork(process.argv[2]); + instance = fork(process.argv[2]); } function ensureKilledInstance() { - return new Promise((resolve, reject) => { - if (instance) { - instance.kill(); - DiscordUtil.dateLog(`Killed existing instance for scheduled restart in ${InternalConfig.restartTimeout / 1000} sec`); - setTimeout(resolve, InternalConfig.restartTimeout); - } - else - resolve(); - }); + return new Promise((resolve, reject) => { + if (instance) { + instance.kill(); + DiscordUtil.dateLog(`Killed existing instance for scheduled restart in ${InternalConfig.restartTimeout / 1000} sec`); + setTimeout(resolve, InternalConfig.restartTimeout); + } + else + resolve(); + }); } \ No newline at end of file diff --git a/discord-bot-core/util.js b/discord-bot-core/util.js new file mode 100644 index 0000000..f6d4176 --- /dev/null +++ b/discord-bot-core/util.js @@ -0,0 +1,74 @@ +// @ts-ignore +const InternalConfig = require("./internal-config.json"); +const Console = require("console"); +const SimpleFileWriter = require("simple-file-writer"); + +const logWriter = new SimpleFileWriter("./console.log"); +const debugLogWriter = new SimpleFileWriter("./debug.log"); + +function ask(client, textChannel, member, question) { + //return a promise which will resolve once the user next sends a message in this textChannel + return new Promise((resolve, reject) => { + const cancelAsk = () => { + client.removeListener("message", handler); + textChannel.send("Response to question timed out"); + }; + + const askTimeout = setTimeout(cancelAsk, InternalConfig.askTimeout); + + const handler = responseMessage => { + if (responseMessage.channel.id === textChannel.id && responseMessage.member && responseMessage.member.id === member.id) { + clearTimeout(askTimeout); + resolve(responseMessage); + } + }; + + client.on("message", handler); + + textChannel.send(member.toString() + " " + question).catch(reject); + }); +} + +function dateLog(...args) { + doDateLog(Console.log, logWriter, args, "INFO"); +} + +function dateError(...args) { + doDateLog(Console.error, logWriter, args, "ERROR"); +} + +function dateDebugError(...args) { + doDateLog(null, null, args, "DEBUG ERROR"); +} + +function dateDebug(...args) { + doDateLog(null, null, args, "DEBUG"); +} + +function doDateLog(consoleMethod, fileWriter, args, prefix = "") { + args = formatArgs([`[${prefix}]`].concat(args)); + + if (consoleMethod !== null) + consoleMethod.apply(this, args); + + if (fileWriter !== null) + fileWriter.write(formatArgsForFile(args)); + + debugLogWriter.write(formatArgsForFile(args)); +} + +function formatArgs(args) { + return [`[${new Date().toUTCString()}]`].concat(args); +} + +function formatArgsForFile(args) { + return args.join(" ") + "\n"; +} + +module.exports = { + dateError, + dateLog, + dateDebug, + dateDebugError, + ask +}; \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 747d877..847afd2 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "discord-bot-rss-feed", - "version": "3.3.0", + "version": "3.4.0", "lockfileVersion": 1, "requires": true, "dependencies": { @@ -15,12 +15,84 @@ "integrity": "sha1-yM4n3grMdtiW0rH6099YjZ6C8BQ=", "optional": true }, + "acorn-jsx": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz", + "integrity": "sha1-r9+UiPsezvyDSPb7IvRk4ypYs2s=", + "requires": { + "acorn": "3.3.0" + }, + "dependencies": { + "acorn": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz", + "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo=" + } + } + }, + "ajv": { + "version": "5.5.2", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", + "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=", + "requires": { + "co": "4.6.0", + "fast-deep-equal": "1.0.0", + "fast-json-stable-stringify": "2.0.0", + "json-schema-traverse": "0.3.1" + } + }, + "ajv-keywords": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-2.1.1.tgz", + "integrity": "sha1-YXmX/F9gV2iUxDX5QNgZ4TW4B2I=" + }, "amdefine": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", "integrity": "sha1-SlKCrBZHKek2Gbz9OtFR+BfOkfU=", "optional": true }, + "ansi-escapes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.0.0.tgz", + "integrity": "sha512-O/klc27mWNUigtv0F8NJWbLF00OcegQalkqKURWdosW08YZKi4m6CnSUSvIZG1otNJbTWhN01Hhz389DW7mvDQ==" + }, + "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=" + }, + "argparse": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.9.tgz", + "integrity": "sha1-c9g7wmP4bpf4zE9rrhsOkKfSLIY=", + "requires": { + "sprintf-js": "1.0.3" + } + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "requires": { + "array-uniq": "1.0.3" + } + }, + "array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=" + }, + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=" + }, "ast-types": { "version": "0.8.15", "resolved": "https://registry.npmjs.org/ast-types/-/ast-types-0.8.15.tgz", @@ -37,6 +109,43 @@ "version": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz", "integrity": "sha1-ePrtjD0HSrgfIrTphdeehzj3IPg=" }, + "babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", + "requires": { + "chalk": "1.1.3", + "esutils": "2.0.2", + "js-tokens": "3.0.2" + }, + "dependencies": { + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "requires": { + "ansi-styles": "2.2.1", + "escape-string-regexp": "1.0.5", + "has-ansi": "2.0.0", + "strip-ansi": "3.0.1", + "supports-color": "2.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "requires": { + "ansi-regex": "2.1.1" + } + } + } + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" + }, "base62": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/base62/-/base62-0.1.1.tgz", @@ -60,12 +169,34 @@ } } }, + "brace-expansion": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.8.tgz", + "integrity": "sha1-wHshHHyVLsH479Uad+8NHTmQopI=", + "requires": { + "balanced-match": "1.0.0", + "concat-map": "0.0.1" + } + }, "bson": { "version": "0.4.23", "resolved": "https://registry.npmjs.org/bson/-/bson-0.4.23.tgz", "integrity": "sha1-5louPHUH/63kEJvHV1p25Q+NqRU=", "optional": true }, + "caller-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", + "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", + "requires": { + "callsites": "0.2.0" + } + }, + "callsites": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", + "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=" + }, "camo": { "version": "git+https://github.com/scottwrobinson/camo.git#3acde553b37ec7d2161a20424fa1975243f96179", "requires": { @@ -75,10 +206,160 @@ "nedb": "1.8.0" } }, + "chalk": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.0.tgz", + "integrity": "sha512-Az5zJR2CBujap2rqXGaJKaPHyJ0IrUimvYNX+ncCy8PJP4ltOGTrHUIo097ZaL2zMeKYpiCdqDvS6zdrTFok3Q==", + "requires": { + "ansi-styles": "3.2.0", + "escape-string-regexp": "1.0.5", + "supports-color": "4.5.0" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.0.tgz", + "integrity": "sha512-NnSOmMEYtVR2JVMIGTzynRkkaxtiq1xnFBcdQD/DnNCYPoEPsVJhM98BDyaoNOQIi7p4okdi3E27eN7GQbsUug==", + "requires": { + "color-convert": "1.9.1" + } + }, + "supports-color": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-4.5.0.tgz", + "integrity": "sha1-vnoN5ITexcXN34s9WRJQRJEvY1s=", + "requires": { + "has-flag": "2.0.0" + } + } + } + }, + "chardet": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz", + "integrity": "sha1-tUc7M9yXxCTl2Y3IfVXU2KKci/I=" + }, + "circular-json": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", + "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==" + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "requires": { + "restore-cursor": "2.0.0" + } + }, + "cli-width": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz", + "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=" + }, + "co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=" + }, + "color-convert": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.1.tgz", + "integrity": "sha512-mjGanIiwQJskCC18rPR6OmrZ6fm2Lc7PeGFYwCmy5J34wC6F1PzdGL6xeMfmgicfYcNLGuVFA3WzXtIDCQSZxQ==", + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "concat-stream": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.0.tgz", + "integrity": "sha1-CqxmL9Ur54lk1VMvaUeE5wEQrPc=", + "requires": { + "inherits": "2.0.3", + "readable-stream": "2.3.3", + "typedarray": "0.0.6" + }, + "dependencies": { + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" + }, + "readable-stream": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz", + "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==", + "requires": { + "core-util-is": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", + "inherits": "2.0.3", + "isarray": "1.0.0", + "process-nextick-args": "1.0.7", + "safe-buffer": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", + "string_decoder": "1.0.3", + "util-deprecate": "1.0.2" + } + }, + "string_decoder": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", + "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", + "requires": { + "safe-buffer": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz" + } + } + } + }, "core-util-is": { "version": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" }, + "cross-spawn": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", + "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", + "requires": { + "lru-cache": "4.1.1", + "shebang-command": "1.2.0", + "which": "1.3.0" + } + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "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=" + }, + "del": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/del/-/del-2.2.2.tgz", + "integrity": "sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag=", + "requires": { + "globby": "5.0.0", + "is-path-cwd": "1.0.0", + "is-path-in-cwd": "1.0.0", + "object-assign": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "pify": "2.3.0", + "pinkie-promise": "2.0.1", + "rimraf": "2.6.2" + } + }, "depd": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.0.tgz", @@ -95,6 +376,14 @@ "ws": "https://registry.npmjs.org/ws/-/ws-3.2.0.tgz" } }, + "doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "requires": { + "esutils": "2.0.2" + } + }, "entities": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/entities/-/entities-1.1.1.tgz", @@ -117,17 +406,140 @@ "integrity": "sha1-A+jzxyl5KOVHjWqx0GQyUVB73t0=", "optional": 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=" + }, + "eslint": { + "version": "4.16.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-4.16.0.tgz", + "integrity": "sha512-YVXV4bDhNoHHcv0qzU4Meof7/P26B4EuaktMi5L1Tnt52Aov85KmYA8c5D+xyZr/BkhvwUqr011jDSD/QTULxg==", + "requires": { + "ajv": "5.5.2", + "babel-code-frame": "6.26.0", + "chalk": "2.3.0", + "concat-stream": "1.6.0", + "cross-spawn": "5.1.0", + "debug": "3.1.0", + "doctrine": "2.1.0", + "eslint-scope": "3.7.1", + "eslint-visitor-keys": "1.0.0", + "espree": "3.5.2", + "esquery": "1.0.0", + "esutils": "2.0.2", + "file-entry-cache": "2.0.0", + "functional-red-black-tree": "1.0.1", + "glob": "7.1.2", + "globals": "11.1.0", + "ignore": "3.3.7", + "imurmurhash": "0.1.4", + "inquirer": "3.3.0", + "is-resolvable": "1.1.0", + "js-yaml": "3.10.0", + "json-stable-stringify-without-jsonify": "1.0.1", + "levn": "0.3.0", + "lodash": "4.17.4", + "minimatch": "3.0.4", + "mkdirp": "0.5.1", + "natural-compare": "1.4.0", + "optionator": "0.8.2", + "path-is-inside": "1.0.2", + "pluralize": "7.0.0", + "progress": "2.0.0", + "require-uncached": "1.0.3", + "semver": "5.5.0", + "strip-ansi": "4.0.0", + "strip-json-comments": "2.0.1", + "table": "4.0.2", + "text-table": "0.2.0" + }, + "dependencies": { + "lodash": { + "version": "4.17.4", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", + "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" + } + } + }, + "eslint-scope": { + "version": "3.7.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-3.7.1.tgz", + "integrity": "sha1-PWPD7f2gLgbgGkUq2IyqzHzctug=", + "requires": { + "esrecurse": "4.2.0", + "estraverse": "4.2.0" + } + }, + "eslint-visitor-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", + "integrity": "sha512-qzm/XxIbxm/FHyH341ZrbnMUpe+5Bocte9xkmFMzPMjRaZMcXww+MpBptFvtU+79L362nqiLhekCxCxDPaUMBQ==" + }, "esmangle-evaluator": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/esmangle-evaluator/-/esmangle-evaluator-1.0.1.tgz", "integrity": "sha1-Yg2GbvSGGzMR91dm1SqFcrs8YzY=", "optional": true }, + "espree": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/espree/-/espree-3.5.2.tgz", + "integrity": "sha512-sadKeYwaR/aJ3stC2CdvgXu1T16TdYN+qwCpcWbMnGJ8s0zNWemzrvb2GbD4OhmJ/fwpJjudThAlLobGbWZbCQ==", + "requires": { + "acorn": "5.3.0", + "acorn-jsx": "3.0.1" + }, + "dependencies": { + "acorn": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.3.0.tgz", + "integrity": "sha512-Yej+zOJ1Dm/IMZzzj78OntP/r3zHEaKcyNoU2lAaxPtrseM6rF0xwqoz5Q5ysAiED9hTjI2hgtvLXitlCN1/Ug==" + } + } + }, "esprima-fb": { "version": "3001.1.0-dev-harmony-fb", "resolved": "https://registry.npmjs.org/esprima-fb/-/esprima-fb-3001.0001.0000-dev-harmony-fb.tgz", "integrity": "sha1-t303q8046gt3Qmu4vCkizmtCZBE=" }, + "esquery": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.0.tgz", + "integrity": "sha1-z7qLV9f7qT8XKYqKAGoEzaE9gPo=", + "requires": { + "estraverse": "4.2.0" + } + }, + "esrecurse": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.0.tgz", + "integrity": "sha1-+pVo2Y04I/mkHZHpAtyrnqblsWM=", + "requires": { + "estraverse": "4.2.0", + "object-assign": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz" + } + }, + "estraverse": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz", + "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=" + }, + "esutils": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", + "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=" + }, + "external-editor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-2.1.0.tgz", + "integrity": "sha512-E44iT5QVOUJBKij4IIV3uvxuNlbKS38Tw1HiupxEIHPv9qtC2PrDYohbXV5U+1jnfIXttny8gUhj+oZvflFlzA==", + "requires": { + "chardet": "0.4.2", + "iconv-lite": "0.4.19", + "tmp": "0.0.33" + } + }, "falafel": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/falafel/-/falafel-1.2.0.tgz", @@ -140,12 +552,65 @@ "object-keys": "1.0.11" } }, + "fast-deep-equal": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.0.0.tgz", + "integrity": "sha1-liVqO8l1WV6zbYLpkp0GDYk0Of8=" + }, + "fast-json-stable-stringify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", + "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=" + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=" + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "requires": { + "escape-string-regexp": "1.0.5" + } + }, + "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=", + "requires": { + "flat-cache": "1.3.0", + "object-assign": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz" + } + }, + "flat-cache": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.0.tgz", + "integrity": "sha1-0wMLMrOBVPTjt+nHCfSQ9++XxIE=", + "requires": { + "circular-json": "0.3.3", + "del": "2.2.2", + "graceful-fs": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", + "write": "0.2.1" + } + }, "foreach": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/foreach/-/foreach-2.0.5.tgz", "integrity": "sha1-C+4AUBiusmDQo6865ljdATbsG5k=", "optional": true }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" + }, + "functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=" + }, "get-urls": { "version": "https://registry.npmjs.org/get-urls/-/get-urls-7.0.0.tgz", "integrity": "sha1-xICtx9TGzFy7ZLUxgj3GO5nsHlo=", @@ -154,10 +619,63 @@ "url-regex": "https://registry.npmjs.org/url-regex/-/url-regex-4.1.1.tgz" } }, + "glob": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", + "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", + "requires": { + "fs.realpath": "1.0.0", + "inflight": "1.0.6", + "inherits": "2.0.3", + "minimatch": "3.0.4", + "once": "1.4.0", + "path-is-absolute": "1.0.1" + } + }, + "globals": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.1.0.tgz", + "integrity": "sha512-uEuWt9mqTlPDwSqi+sHjD4nWU/1N+q0fiWI9T1mZpD2UENqX20CFD5T/ziLZvztPaBKl7ZylUi1q6Qfm7E2CiQ==" + }, + "globby": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz", + "integrity": "sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0=", + "requires": { + "array-union": "1.0.2", + "arrify": "1.0.1", + "glob": "7.1.2", + "object-assign": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "pify": "2.3.0", + "pinkie-promise": "2.0.1" + } + }, "graceful-fs": { "version": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", - "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", - "optional": true + "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=" + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "requires": { + "ansi-regex": "2.1.1" + } + }, + "has-flag": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz", + "integrity": "sha1-6CB68cx7MNRGzHC3NLXovhj4jVE=" + }, + "iconv-lite": { + "version": "0.4.19", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz", + "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ==" + }, + "ignore": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.7.tgz", + "integrity": "sha512-YGG3ejvBNHRqu0559EOxxNFihD0AjpvHlC/pdGKd3X3ofe+CoJkYazwNJYTNebqpPKN+VVQbh4ZFn1DivMNuHA==" }, "immediate": { "version": "3.0.6", @@ -165,6 +683,20 @@ "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=", "optional": true }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=" + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "requires": { + "once": "1.4.0", + "wrappy": "1.0.2" + } + }, "inherits": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", @@ -180,19 +712,119 @@ "through2": "0.6.5" } }, + "inquirer": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-3.3.0.tgz", + "integrity": "sha512-h+xtnyk4EwKvFWHrUYsWErEVR+igKtLdchu+o0Z1RL7VU/jVMFbYir2bp6bAj8efFNxWqHX0dIss6fJQ+/+qeQ==", + "requires": { + "ansi-escapes": "3.0.0", + "chalk": "2.3.0", + "cli-cursor": "2.1.0", + "cli-width": "2.2.0", + "external-editor": "2.1.0", + "figures": "2.0.0", + "lodash": "4.17.4", + "mute-stream": "0.0.7", + "run-async": "2.3.0", + "rx-lite": "4.0.8", + "rx-lite-aggregates": "4.0.8", + "string-width": "2.1.1", + "strip-ansi": "4.0.0", + "through": "2.3.8" + }, + "dependencies": { + "lodash": { + "version": "4.17.4", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", + "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" + } + } + }, "ip-regex": { "version": "https://registry.npmjs.org/ip-regex/-/ip-regex-1.0.3.tgz", "integrity": "sha1-3FiQdvZZ9BnCIgOaMzFvHHOH7/0=" }, + "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=" + }, + "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=" + }, + "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=", + "requires": { + "is-path-inside": "1.0.1" + } + }, + "is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", + "requires": { + "path-is-inside": "1.0.2" + } + }, "is-plain-obj": { "version": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=" }, + "is-promise": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", + "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=" + }, + "is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==" + }, "isarray": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" }, + "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.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=" + }, + "js-yaml": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.10.0.tgz", + "integrity": "sha512-O2v52ffjLa9VeM43J4XocZE//WT9N0IiwDa3KSHH7Tu8CtH+1qM8SIZvnsTh6v+4yFy5KUY3BHUVwjpfAWsjIA==", + "requires": { + "argparse": "1.0.9", + "esprima": "4.0.0" + }, + "dependencies": { + "esprima": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.0.tgz", + "integrity": "sha512-oftTcaMu/EGrEIu904mWteKIv8vMuOgGYo7EhVJJN00R/EED9DCua/xxHRdYnKtcECzVg7xOWhflvJMnqcFZjw==" + } + } + }, + "json-schema-traverse": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", + "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=" + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=" + }, "jsonfile": { "version": "https://registry.npmjs.org/jsonfile/-/jsonfile-3.0.1.tgz", "integrity": "sha1-pezG9l9T9mLEQVx2daAzHQmS7GY=", @@ -220,6 +852,15 @@ "nan": "2.5.1" } }, + "levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", + "requires": { + "prelude-ls": "1.1.2", + "type-check": "0.3.2" + } + }, "lie": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/lie/-/lie-3.0.2.tgz", @@ -250,17 +891,37 @@ "version": "https://registry.npmjs.org/long/-/long-3.2.0.tgz", "integrity": "sha1-2CG3E4yhy1gcFymQ7xTbIAtcR0s=" }, + "lru-cache": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.1.tgz", + "integrity": "sha512-q4spe4KTfsAS1SUHLO0wz8Qiyf1+vMIAgpRYioFYDMNqKfHQbg+AVDH3i4fvpl71/P1L0dBl+fQi+P37UYf0ew==", + "requires": { + "pseudomap": "1.0.2", + "yallist": "2.1.2" + } + }, + "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==", + "requires": { + "brace-expansion": "1.1.8" + } + }, "minimist": { "version": "0.0.8", "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", - "optional": true + "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" }, "mkdirp": { "version": "0.5.1", "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", - "optional": true, "requires": { "minimist": "0.0.8" } @@ -286,12 +947,27 @@ "kerberos": "0.0.23" } }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=" + }, "nan": { "version": "2.5.1", "resolved": "https://registry.npmjs.org/nan/-/nan-2.5.1.tgz", "integrity": "sha1-1bAWkSUzJql6K77p5hxV2NYDUeI=", "optional": true }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=" + }, "nedb": { "version": "1.8.0", "resolved": "https://registry.npmjs.org/nedb/-/nedb-1.8.0.tgz", @@ -333,6 +1009,78 @@ "integrity": "sha1-xUYBd4rVYPEULODgG8yotW0TQm0=", "optional": true }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "requires": { + "wrappy": "1.0.2" + } + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "requires": { + "mimic-fn": "1.1.0" + } + }, + "optionator": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz", + "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=", + "requires": { + "deep-is": "0.1.3", + "fast-levenshtein": "2.0.6", + "levn": "0.3.0", + "prelude-ls": "1.1.2", + "type-check": "0.3.2", + "wordwrap": "1.0.0" + } + }, + "os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=" + }, + "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=" + }, + "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=" + }, + "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=", + "requires": { + "pinkie": "2.0.4" + } + }, + "pluralize": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz", + "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==" + }, + "prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=" + }, "prepend-http": { "version": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=" @@ -347,6 +1095,21 @@ "integrity": "sha512-VvivMrbvd2nKkiG38qjULzlc+4Vx4wm/whI9pQD35YrARNnhxeiRktSOhSukRLFNlzg6Br/cJPet5J/u19r/mg==", "optional": 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": "2.0.0", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.0.tgz", + "integrity": "sha1-ihvjZr+Pwj2yvSPxDG/pILQ4nR8=" + }, + "pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=" + }, "query-string": { "version": "https://registry.npmjs.org/query-string/-/query-string-4.3.4.tgz", "integrity": "sha1-u7aTucqRXCMlFbIosaArYJBD2+s=", @@ -393,6 +1156,37 @@ } } }, + "require-uncached": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", + "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=", + "requires": { + "caller-path": "0.1.0", + "resolve-from": "1.0.1" + } + }, + "resolve-from": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", + "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=" + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "requires": { + "onetime": "2.0.1", + "signal-exit": "3.0.2" + } + }, + "rimraf": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.2.tgz", + "integrity": "sha512-lreewLK/BlghmxtfH36YYVg1i8IAce4TI7oao75I1g245+6BctqTVQiBP3YUJ9C6DQOXJmkYR9X9fCLtCOJc5w==", + "requires": { + "glob": "7.1.2" + } + }, "rss-parser": { "version": "2.12.0", "resolved": "https://registry.npmjs.org/rss-parser/-/rss-parser-2.12.0.tgz", @@ -402,14 +1196,66 @@ "xml2js": "0.4.19" } }, + "run-async": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz", + "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=", + "requires": { + "is-promise": "2.1.0" + } + }, + "rx-lite": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-4.0.8.tgz", + "integrity": "sha1-Cx4Rr4vESDbwSmQH6S2kJGe3lEQ=" + }, + "rx-lite-aggregates": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz", + "integrity": "sha1-dTuHqJoRyVRnxKwWJsTvxOBcZ74=", + "requires": { + "rx-lite": "4.0.8" + } + }, "safe-buffer": { "version": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", "integrity": "sha1-iTMSr2myEj3vcfV4iQAWce6yyFM=" }, + "semver": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.5.0.tgz", + "integrity": "sha512-4SJ3dm0WAwWy/NVeioZh5AntkdJoWKxHxcmyP622fOkgHa4z3R0TdBJICINyaSDE6uNwVc8gZr+ZinwZAH4xIA==" + }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "requires": { + "shebang-regex": "1.0.0" + } + }, + "shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=" + }, "shortid": { "version": "https://registry.npmjs.org/shortid/-/shortid-2.2.8.tgz", "integrity": "sha1-AzsRfWoul1gE9vCWnb59PQs1UTE=" }, + "signal-exit": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", + "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=" + }, + "slice-ansi": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-1.0.0.tgz", + "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==", + "requires": { + "is-fullwidth-code-point": "2.0.0" + } + }, "snekfetch": { "version": "https://registry.npmjs.org/snekfetch/-/snekfetch-3.3.0.tgz", "integrity": "sha1-l6oNedWSDxjCbUyKf92+kSBa0Po=" @@ -430,20 +1276,83 @@ "amdefine": "1.0.1" } }, + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" + }, "strict-uri-encode": { "version": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=" }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "requires": { + "is-fullwidth-code-point": "2.0.0", + "strip-ansi": "4.0.0" + } + }, "string_decoder": { "version": "0.10.31", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "requires": { + "ansi-regex": "3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=" + } + } + }, + "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": "4.0.2", + "resolved": "https://registry.npmjs.org/table/-/table-4.0.2.tgz", + "integrity": "sha512-UUkEAPdSGxtRpiV9ozJ5cMTtYiqz7Ni1OGqLXRCynrvzdtR1p+cfOWe2RJLwvUG8hNanaSRjecIqwOjqeatDsA==", + "requires": { + "ajv": "5.5.2", + "ajv-keywords": "2.1.1", + "chalk": "2.3.0", + "lodash": "4.17.4", + "slice-ansi": "1.0.0", + "string-width": "2.1.1" + }, + "dependencies": { + "lodash": { + "version": "4.17.4", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", + "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" + } + } + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=" + }, "through": { "version": "2.3.8", "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", - "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", - "optional": true + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=" }, "through2": { "version": "0.6.5", @@ -471,10 +1380,31 @@ "version": "https://registry.npmjs.org/tlds/-/tlds-1.196.0.tgz", "integrity": "sha1-SddN29H53zAjizv+9N+Chitbu0g=" }, + "tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "requires": { + "os-tmpdir": "1.0.2" + } + }, "tweetnacl": { "version": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-1.0.0.tgz", "integrity": "sha1-cT2LgY2kIGh0C/aDhtBHnmb8ins=" }, + "type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", + "requires": { + "prelude-ls": "1.1.2" + } + }, + "typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" + }, "ultron": { "version": "https://registry.npmjs.org/ultron/-/ultron-1.1.0.tgz", "integrity": "sha1-sHoualQagV/Go0zNRTO67DB8qGQ=" @@ -498,6 +1428,37 @@ "tlds": "https://registry.npmjs.org/tlds/-/tlds-1.196.0.tgz" } }, + "util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" + }, + "which": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.0.tgz", + "integrity": "sha512-xcJpopdamTuY5duC/KnTTNBraPK54YwpenP4lzxU8H91GudWpFv38u0CKjclE1Wi2EH2EDz5LRcHcKbCIzqGyg==", + "requires": { + "isexe": "2.0.0" + } + }, + "wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=" + }, + "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=", + "requires": { + "mkdirp": "0.5.1" + } + }, "ws": { "version": "https://registry.npmjs.org/ws/-/ws-3.2.0.tgz", "integrity": "sha1-1dPWsRr/cec/gI9AzGnVK7bUoYU=", @@ -532,6 +1493,11 @@ "version": "4.0.1", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=" + }, + "yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=" } } } diff --git a/package.json b/package.json index ba483fb..3474223 100644 --- a/package.json +++ b/package.json @@ -9,6 +9,7 @@ "@types/node": "9.3.0", "camo": "git+https://github.com/scottwrobinson/camo.git", "discord.js": "11.2.0", + "eslint": "4.16.0", "get-urls": "7.0.0", "jsonfile": "3.0.1", "rss-parser": "2.12.0",