From f4dc9a0be89b78edcd6fe58467f86264657c1bc7 Mon Sep 17 00:00:00 2001 From: spurreiter Date: Mon, 20 Apr 2020 17:19:00 +0200 Subject: [PATCH] chore: replace jshint by eslint standard --- .eslintrc.js | 20 + .jscsrc | 46 - .jshintrc | 89 -- lib/ocsp.js | 18 +- lib/ocsp/agent.js | 191 ++- lib/ocsp/api.js | 20 +- lib/ocsp/cache.js | 138 +- lib/ocsp/check.js | 52 +- lib/ocsp/request.js | 56 +- lib/ocsp/server.js | 185 ++- lib/ocsp/utils.js | 241 ++-- lib/ocsp/verify.js | 115 +- package-lock.json | 2538 ++++++++++++++++++++++++++++++++++++ package.json | 12 +- test/.eslintrc.js | 5 + test/agent-test.js | 39 +- test/api-test.js | 89 +- test/cache-test.js | 87 +- test/fixtures/gen-certs.js | 36 +- test/fixtures/index.js | 131 +- test/server-test.js | 53 +- 21 files changed, 3261 insertions(+), 900 deletions(-) create mode 100644 .eslintrc.js delete mode 100644 .jscsrc delete mode 100644 .jshintrc create mode 100644 package-lock.json create mode 100644 test/.eslintrc.js diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 0000000..aae0ced --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,20 @@ +module.exports = { + "env": { + "commonjs": true, + "es6": true, + "node": true + }, + "extends": [ + "standard" + ], + "globals": { + "Atomics": "readonly", + "SharedArrayBuffer": "readonly" + }, + "parserOptions": { + "ecmaVersion": 2018 + }, + "rules": { + "node/no-deprecated-api": "warn" + } +}; diff --git a/.jscsrc b/.jscsrc deleted file mode 100644 index dbaae20..0000000 --- a/.jscsrc +++ /dev/null @@ -1,46 +0,0 @@ -{ - "disallowKeywordsOnNewLine": [ "else" ], - "disallowMixedSpacesAndTabs": true, - "disallowMultipleLineStrings": true, - "disallowMultipleVarDecl": true, - "disallowNewlineBeforeBlockStatements": true, - "disallowQuotedKeysInObjects": true, - "disallowSpaceAfterObjectKeys": true, - "disallowSpaceAfterPrefixUnaryOperators": true, - "disallowSpaceBeforePostfixUnaryOperators": true, - "disallowSpacesInCallExpression": true, - "disallowTrailingComma": true, - "disallowTrailingWhitespace": true, - "disallowYodaConditions": true, - - "requireCommaBeforeLineBreak": true, - "requireOperatorBeforeLineBreak": true, - "requireSpaceAfterBinaryOperators": true, - "requireSpaceAfterKeywords": [ "if", "for", "while", "else", "try", "catch" ], - "requireSpaceAfterLineComment": true, - "requireSpaceBeforeBinaryOperators": true, - "requireSpaceBeforeBlockStatements": true, - "requireSpaceBeforeKeywords": [ "else", "catch" ], - "requireSpaceBeforeObjectValues": true, - "requireSpaceBetweenArguments": true, - "requireSpacesInAnonymousFunctionExpression": { - "beforeOpeningCurlyBrace": true - }, - "requireSpacesInFunctionDeclaration": { - "beforeOpeningCurlyBrace": true - }, - "requireSpacesInFunctionExpression": { - "beforeOpeningCurlyBrace": true - }, - "requireSpacesInConditionalExpression": true, - "requireSpacesInForStatement": true, - "requireSpacesInsideArrayBrackets": "all", - "requireSpacesInsideObjectBrackets": "all", - "requireDotNotation": true, - - "maximumLineLength": 80, - "validateIndentation": 2, - "validateLineBreaks": "LF", - "validateParameterSeparator": ", ", - "validateQuoteMarks": "'" -} diff --git a/.jshintrc b/.jshintrc deleted file mode 100644 index 7e97390..0000000 --- a/.jshintrc +++ /dev/null @@ -1,89 +0,0 @@ -{ - // JSHint Default Configuration File (as on JSHint website) - // See http://jshint.com/docs/ for more details - - "maxerr" : 50, // {int} Maximum error before stopping - - // Enforcing - "bitwise" : false, // true: Prohibit bitwise operators (&, |, ^, etc.) - "camelcase" : false, // true: Identifiers must be in camelCase - "curly" : false, // true: Require {} for every new block or scope - "eqeqeq" : true, // true: Require triple equals (===) for comparison - "forin" : true, // true: Require filtering for..in loops with obj.hasOwnProperty() - "freeze" : true, // true: prohibits overwriting prototypes of native objects such as Array, Date etc. - "immed" : false, // true: Require immediate invocations to be wrapped in parens e.g. `(function () { } ());` - "indent" : 2, // {int} Number of spaces to use for indentation - "latedef" : true, // true: Require variables/functions to be defined before being used - "newcap" : true, // true: Require capitalization of all constructor functions e.g. `new F()` - "noarg" : true, // true: Prohibit use of `arguments.caller` and `arguments.callee` - "noempty" : false, // true: Prohibit use of empty blocks - "nonbsp" : true, // true: Prohibit "non-breaking whitespace" characters. - "nonew" : false, // true: Prohibit use of constructors for side-effects (without assignment) - "plusplus" : false, // true: Prohibit use of `++` & `--` - "quotmark" : "single", // Quotation mark consistency: - // false : do nothing (default) - // true : ensure whatever is used is consistent - // "single" : require single quotes - // "double" : require double quotes - "undef" : true, // true: Require all non-global variables to be declared (prevents global leaks) - "unused" : true, // true: Require all defined variables be used - "strict" : true, // true: Requires all functions run in ES5 Strict Mode - "maxparams" : false, // {int} Max number of formal params allowed per function - "maxdepth" : 3, // {int} Max depth of nested blocks (within functions) - "maxstatements" : false, // {int} Max number statements per function - "maxcomplexity" : false, // {int} Max cyclomatic complexity per function - "maxlen" : false, // {int} Max number of characters per line - - // Relaxing - "asi" : false, // true: Tolerate Automatic Semicolon Insertion (no semicolons) - "boss" : false, // true: Tolerate assignments where comparisons would be expected - "debug" : false, // true: Allow debugger statements e.g. browser breakpoints. - "eqnull" : false, // true: Tolerate use of `== null` - "es5" : false, // true: Allow ES5 syntax (ex: getters and setters) - "esnext" : false, // true: Allow ES.next (ES6) syntax (ex: `const`) - "moz" : false, // true: Allow Mozilla specific syntax (extends and overrides esnext features) - // (ex: `for each`, multiple try/catch, function expression…) - "evil" : false, // true: Tolerate use of `eval` and `new Function()` - "expr" : false, // true: Tolerate `ExpressionStatement` as Programs - "funcscope" : false, // true: Tolerate defining variables inside control statements - "globalstrict" : false, // true: Allow global "use strict" (also enables 'strict') - "iterator" : false, // true: Tolerate using the `__iterator__` property - "lastsemic" : false, // true: Tolerate omitting a semicolon for the last statement of a 1-line block - "laxbreak" : false, // true: Tolerate possibly unsafe line breakings - "laxcomma" : false, // true: Tolerate comma-first style coding - "loopfunc" : false, // true: Tolerate functions being defined in loops - "multistr" : false, // true: Tolerate multi-line strings - "noyield" : false, // true: Tolerate generator functions with no yield statement in them. - "notypeof" : false, // true: Tolerate invalid typeof operator values - "proto" : false, // true: Tolerate using the `__proto__` property - "scripturl" : false, // true: Tolerate script-targeted URLs - "shadow" : true, // true: Allows re-define variables later in code e.g. `var x=1; x=2;` - "sub" : false, // true: Tolerate using `[]` notation when it can still be expressed in dot notation - "supernew" : false, // true: Tolerate `new function () { ... };` and `new Object;` - "validthis" : false, // true: Tolerate using this in a non-constructor function - - // Environments - "browser" : true, // Web Browser (window, document, etc) - "browserify" : true, // Browserify (node.js code in the browser) - "couch" : false, // CouchDB - "devel" : true, // Development/debugging (alert, confirm, etc) - "dojo" : false, // Dojo Toolkit - "jasmine" : false, // Jasmine - "jquery" : false, // jQuery - "mocha" : true, // Mocha - "mootools" : false, // MooTools - "node" : true, // Node.js - "nonstandard" : false, // Widely adopted globals (escape, unescape, etc) - "prototypejs" : false, // Prototype and Scriptaculous - "qunit" : false, // QUnit - "rhino" : false, // Rhino - "shelljs" : false, // ShellJS - "worker" : false, // Web Workers - "wsh" : false, // Windows Scripting Host - "yui" : false, // Yahoo User Interface - - // Custom Globals - "globals" : { - "module": true - } // additional predefined global variables -} diff --git a/lib/ocsp.js b/lib/ocsp.js index bab4b2c..f10a0ff 100644 --- a/lib/ocsp.js +++ b/lib/ocsp.js @@ -1,12 +1,12 @@ -'use strict'; +'use strict' -exports.Cache = require('./ocsp/cache'); -exports.Agent = require('./ocsp/agent'); -exports.Server = require('./ocsp/server'); +exports.Cache = require('./ocsp/cache') +exports.Agent = require('./ocsp/agent') +exports.Server = require('./ocsp/server') -exports.getOCSPURI = require('./ocsp/api').getOCSPURI; +exports.getOCSPURI = require('./ocsp/api').getOCSPURI -exports.request = require('./ocsp/request'); -exports.check = require('./ocsp/check'); -exports.verify = require('./ocsp/verify'); -exports.utils = require('./ocsp/utils'); +exports.request = require('./ocsp/request') +exports.check = require('./ocsp/check') +exports.verify = require('./ocsp/verify') +exports.utils = require('./ocsp/utils') diff --git a/lib/ocsp/agent.js b/lib/ocsp/agent.js index b9fb640..6d223d9 100644 --- a/lib/ocsp/agent.js +++ b/lib/ocsp/agent.js @@ -1,160 +1,145 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); +var ocsp = require('../ocsp') -var util = require('util'); -var http = require('http'); -var https = require('https'); -var rfc5280 = require('asn1.js-rfc5280'); -var SimpleCache = require('simple-lru-cache'); +var util = require('util') +var http = require('http') +var https = require('https') +var rfc5280 = require('asn1.js-rfc5280') +var SimpleCache = require('simple-lru-cache') -function Agent(options) { - if (!options) - options = {}; +function Agent (options) { + if (!options) { options = {} } - https.Agent.call(this, options); + https.Agent.call(this, options) - this.caCache = new SimpleCache({ maxSize: options.CACacheSize || 1024 }); + this.caCache = new SimpleCache({ maxSize: options.CACacheSize || 1024 }) } -module.exports = Agent; -util.inherits(Agent, https.Agent); - -Agent.prototype.createConnection = function createConnection(port, - host, - options) { +module.exports = Agent +util.inherits(Agent, https.Agent) +Agent.prototype.createConnection = function createConnection (port, + host, + options) { if (port !== null && typeof port === 'object') { - options = port; - port = null; + options = port + port = null } else if (host !== null && typeof host === 'object') { - options = host; - host = null; - } else if (options === null || typeof options !== 'object') - options = {}; + options = host + host = null + } else if (options === null || typeof options !== 'object') { options = {} } - if (typeof port === 'number') - options.port = port; + if (typeof port === 'number') { options.port = port } - if (typeof host === 'string') - options.host = host; + if (typeof host === 'string') { options.host = host } - var ocspOptions = util._extend({ requestOCSP: true }, options); + var ocspOptions = Object.assign({ requestOCSP: true }, options) var socket = https.Agent.prototype.createConnection.call( - this, port, host, ocspOptions); + this, port, host, ocspOptions) - var self = this; - var stapling = null; - socket.on('OCSPResponse', function(data) { - stapling = data; - }); + var self = this + var stapling = null + socket.on('OCSPResponse', function (data) { + stapling = data + }) - socket.on('secure', function() { - return self.handleOCSPResponse(socket, stapling, function(err) { - if (err) - return socket.destroy(err); + socket.on('secure', function () { + return self.handleOCSPResponse(socket, stapling, function (err) { + if (err) { return socket.destroy(err) } // Time to allow all writes! - socket.uncork(); - }); - }); + socket.uncork() + }) + }) // Do not let any writes come through until we will verify OCSP - socket.cork(); + socket.cork() - return socket; -}; + return socket +} -Agent.prototype.handleOCSPResponse = function handleOCSPResponse(socket, - stapling, - cb) { - var cert = socket.ssl.getPeerCertificate(true); - var issuer = cert.issuerCertificate; +Agent.prototype.handleOCSPResponse = function handleOCSPResponse (socket, + stapling, + cb) { + var cert = socket.ssl.getPeerCertificate(true) + var issuer = cert.issuerCertificate - cert = cert.raw; + cert = cert.raw try { - cert = rfc5280.Certificate.decode(cert, 'der'); + cert = rfc5280.Certificate.decode(cert, 'der') if (issuer) { - issuer = issuer.raw; - issuer = rfc5280.Certificate.decode(issuer, 'der'); + issuer = issuer.raw + issuer = rfc5280.Certificate.decode(issuer, 'der') } } catch (e) { - return cb(e); + return cb(e) } - function onIssuer(err, x509) { - if (err) - return cb(err); + function onIssuer (err, x509) { + if (err) { return cb(err) } - issuer = x509; + issuer = x509 if (stapling) { - var req = ocsp.request.generate(cert, issuer); + var req = ocsp.request.generate(cert, issuer) ocsp.verify({ request: req, response: stapling - }, cb); + }, cb) } else { - return ocsp.check({ cert: cert, issuer: issuer }, cb); + return ocsp.check({ cert: cert, issuer: issuer }, cb) } } - if (issuer) - return onIssuer(null, issuer); - else - return this.fetchIssuer(cert, stapling, onIssuer); -}; + if (issuer) { return onIssuer(null, issuer) } else { return this.fetchIssuer(cert, stapling, onIssuer) } +} -Agent.prototype.fetchIssuer = function fetchIssuer(cert, stapling, cb) { - var issuers = ocsp.utils['id-ad-caIssuers'].join('.'); - var self = this; +Agent.prototype.fetchIssuer = function fetchIssuer (cert, stapling, cb) { + var issuers = ocsp.utils['id-ad-caIssuers'].join('.') + var self = this // TODO(indutny): use info from stapling response - ocsp.utils.getAuthorityInfo(cert, issuers, function(err, uri) { - if (err) - return cb(err); + ocsp.utils.getAuthorityInfo(cert, issuers, function (err, uri) { + if (err) { return cb(err) } - var ca = self.caCache.get(uri); - if (ca) - return cb(null, ca); + var ca = self.caCache.get(uri) + if (ca) { return cb(null, ca) } - var once = false; - function done(err, data) { - if (once) - return; + var once = false + function done (err, data) { + if (once) { return } - once = true; - cb(err, data); + once = true + cb(err, data) } - function onResponse(res) { - if (res.statusCode < 200 || res.statusCode >= 400) - return done(new Error('Failed to fetch CA: ' + res.statusCode)); + function onResponse (res) { + if (res.statusCode < 200 || res.statusCode >= 400) { return done(new Error('Failed to fetch CA: ' + res.statusCode)) } - var chunks = []; - res.on('readable', function() { - var chunk = res.read(); - if (!chunk) - return; - chunks.push(chunk); - }); + var chunks = [] + res.on('readable', function () { + var chunk = res.read() + if (!chunk) { return } + chunks.push(chunk) + }) - res.on('end', function() { - var cert = Buffer.concat(chunks); + res.on('end', function () { + var cert = Buffer.concat(chunks) try { - cert = rfc5280.Certificate.decode(cert, 'der'); + cert = rfc5280.Certificate.decode(cert, 'der') } catch (e) { - return done(e); + return done(e) } - self.caCache.set(uri, cert); - done(null, cert); - }); + self.caCache.set(uri, cert) + done(null, cert) + }) } http.get(uri) - .on('error', done) - .on('response', onResponse); - }); -}; + .on('error', done) + .on('response', onResponse) + }) +} diff --git a/lib/ocsp/api.js b/lib/ocsp/api.js index a112edf..39a6083 100644 --- a/lib/ocsp/api.js +++ b/lib/ocsp/api.js @@ -1,14 +1,14 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); -var rfc2560 = require('asn1.js-rfc2560'); -var rfc5280 = require('asn1.js-rfc5280'); +var ocsp = require('../ocsp') +var rfc2560 = require('asn1.js-rfc2560') +var rfc5280 = require('asn1.js-rfc5280') -exports.getOCSPURI = function getOCSPURI(rawCert, cb) { - var ocspMethod = rfc2560['id-pkix-ocsp'].join('.'); +exports.getOCSPURI = function getOCSPURI (rawCert, cb) { + var ocspMethod = rfc2560['id-pkix-ocsp'].join('.') - var cert = ocsp.utils.toDER(rawCert, 'CERTIFICATE'); - cert = rfc5280.Certificate.decode(cert, 'der'); + var cert = ocsp.utils.toDER(rawCert, 'CERTIFICATE') + cert = rfc5280.Certificate.decode(cert, 'der') - ocsp.utils.getAuthorityInfo(cert, ocspMethod, cb); -}; + ocsp.utils.getAuthorityInfo(cert, ocspMethod, cb) +} diff --git a/lib/ocsp/cache.js b/lib/ocsp/cache.js index ed38bb7..2ec3424 100644 --- a/lib/ocsp/cache.js +++ b/lib/ocsp/cache.js @@ -1,117 +1,101 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); +var ocsp = require('../ocsp') -function Cache(options) { - this.options = options || {}; - this.cache = {}; +function Cache (options) { + this.options = options || {} + this.cache = {} // Override methods - if (this.options.probe) - this.probe = this.options.probe; - if (this.options.store) - this.store = this.options.store; - if (this.options.filter) - this.filter = this.options.filter; + if (this.options.probe) { this.probe = this.options.probe } + if (this.options.store) { this.store = this.options.store } + if (this.options.filter) { this.filter = this.options.filter } } -module.exports = Cache; - -Cache.prototype.filter = function filter(url, callback) { - callback(null); -}; - -Cache.prototype.probe = function probe(id, callback) { - if (this.cache.hasOwnProperty(id)) - callback(null, this.cache[id]); - else - callback(null, false); -}; - -Cache.prototype.store = function store(id, response, maxTime, callback) { - if (this.cache.hasOwnProperty(id)) - clearTimeout(this.cache[id].timer); - var self = this; +module.exports = Cache + +Cache.prototype.filter = function filter (url, callback) { + callback(null) +} + +Cache.prototype.probe = function probe (id, callback) { + if (Object.prototype.hasOwnProperty.call(this.cache, id)) { callback(null, this.cache[id]) } else { callback(null, false) } +} + +Cache.prototype.store = function store (id, response, maxTime, callback) { + if (Object.prototype.hasOwnProperty.call(this.cache, id)) { clearTimeout(this.cache[id].timer) } + var self = this this.cache[id] = { response: response, - timer: setTimeout(function() { - delete self.cache[id]; + timer: setTimeout(function () { + delete self.cache[id] }, maxTime) - }; + } - callback(null, null); -}; + callback(null, null) +} -Cache.prototype.request = function request(id, data, callback) { - var self = this; +Cache.prototype.request = function request (id, data, callback) { + var self = this - function done(err, response) { - if (callback) - callback(err, response); - callback = null; + function done (err, response) { + if (callback) { callback(err, response) } + callback = null } - function onResponse(err, ocsp) { - if (err) - return done(err); + function onResponse (err, ocsp) { + if (err) { return done(err) } // Respond early - done(null, ocsp); + done(null, ocsp) // Try parsing and caching response - self.getMaxStoreTime(ocsp, function(err, maxTime) { - if (err) - return; + self.getMaxStoreTime(ocsp, function (err, maxTime) { + if (err) { return } - self.store(id, ocsp, maxTime, function() { + self.store(id, ocsp, maxTime, function () { // No-op - }); - }); + }) + }) } // Check that url isn't blacklisted - this.filter(data.url, function(err) { - if (err) - return done(err, null); - - ocsp.utils.getResponse(data.url, data.ocsp, onResponse); - }); + this.filter(data.url, function (err) { + if (err) { return done(err, null) } -}; + ocsp.utils.getResponse(data.url, data.ocsp, onResponse) + }) +} -Cache.prototype.getMaxStoreTime = function getMaxStoreTime(response, callback) { - var basic; +Cache.prototype.getMaxStoreTime = function getMaxStoreTime (response, callback) { + var basic try { - basic = ocsp.utils.parseResponse(response).value; + basic = ocsp.utils.parseResponse(response).value } catch (e) { - return callback(e); + return callback(e) } // Not enough responses - if (basic.tbsResponseData.responses.length === 0) - return callback(new Error('No OCSP responses')); + if (basic.tbsResponseData.responses.length === 0) { return callback(new Error('No OCSP responses')) } - var responses = basic.tbsResponseData.responses; + var responses = basic.tbsResponseData.responses // Every response should be positive - var good = responses.every(function(response) { - return response.certStatus.type === 'good'; - }); + var good = responses.every(function (response) { + return response.certStatus.type === 'good' + }) // No good - no cache - if (!good) - return callback(new Error('Some OCSP responses are not good')); + if (!good) { return callback(new Error('Some OCSP responses are not good')) } // Find minimum nextUpdate time - var nextUpdate = 0; + var nextUpdate = 0 for (var i = 0; i < responses.length; i++) { - var response = responses[i]; - var responseNext = response.nextUpdate; - if (!responseNext) - continue; + const response = responses[i] + var responseNext = response.nextUpdate + if (!responseNext) { continue } - if (nextUpdate === 0 || nextUpdate > responseNext) - nextUpdate = responseNext; + if (nextUpdate === 0 || nextUpdate > responseNext) { nextUpdate = responseNext } } - return callback(null, Math.max(0, nextUpdate - new Date())); -}; + return callback(null, Math.max(0, nextUpdate - new Date())) +} diff --git a/lib/ocsp/check.js b/lib/ocsp/check.js index 5b3e4a8..5813eb7 100644 --- a/lib/ocsp/check.js +++ b/lib/ocsp/check.js @@ -1,46 +1,44 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); +var ocsp = require('../ocsp') -var rfc2560 = require('asn1.js-rfc2560'); +var rfc2560 = require('asn1.js-rfc2560') -module.exports = function check(options, cb) { - var sync = true; - var req; +module.exports = function check (options, cb) { + var sync = true + var req - function done(err, data) { + function done (err, data) { if (sync) { - sync = false; - process.nextTick(function() { - cb(err, data); - }); - return; + sync = false + process.nextTick(function () { + cb(err, data) + }) + return } - cb(err, data); + cb(err, data) } try { - req = ocsp.request.generate(options.cert, options.issuer); + req = ocsp.request.generate(options.cert, options.issuer) } catch (e) { - return done(e); + return done(e) } - var ocspMethod = rfc2560['id-pkix-ocsp'].join('.'); - ocsp.utils.getAuthorityInfo(req.cert, ocspMethod, function(err, uri) { - if (err) - return done(err); + var ocspMethod = rfc2560['id-pkix-ocsp'].join('.') + ocsp.utils.getAuthorityInfo(req.cert, ocspMethod, function (err, uri) { + if (err) { return done(err) } - ocsp.utils.getResponse(uri, req.data, function(err, raw) { - if (err) - return done(err); + ocsp.utils.getResponse(uri, req.data, function (err, raw) { + if (err) { return done(err) } ocsp.verify({ request: req, response: raw - }, done); - }); - }); + }, done) + }) + }) - sync = false; -}; + sync = false +} diff --git a/lib/ocsp/request.js b/lib/ocsp/request.js index 53473e7..bb1b78d 100644 --- a/lib/ocsp/request.js +++ b/lib/ocsp/request.js @@ -1,61 +1,61 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); -var crypto = require('crypto'); -var rfc2560 = require('asn1.js-rfc2560'); -var rfc5280 = require('asn1.js-rfc5280'); +var ocsp = require('../ocsp') +var crypto = require('crypto') +var rfc2560 = require('asn1.js-rfc2560') +var rfc5280 = require('asn1.js-rfc5280') -function sha1(data) { - return crypto.createHash('sha1').update(data).digest(); +function sha1 (data) { + return crypto.createHash('sha1').update(data).digest() } -exports.generate = function generate(rawCert, rawIssuer) { - var cert; +exports.generate = function generate (rawCert, rawIssuer) { + var cert if (rawCert.tbsCertificate) { - cert = rawCert; + cert = rawCert } else { cert = rfc5280.Certificate.decode( - ocsp.utils.toDER(rawCert, 'CERTIFICATE'), - 'der'); + ocsp.utils.toDER(rawCert, 'CERTIFICATE'), + 'der') } - var issuer; + var issuer if (rawIssuer.tbsCertificate) { - issuer = rawIssuer; + issuer = rawIssuer } else { issuer = rfc5280.Certificate.decode( - ocsp.utils.toDER(rawIssuer, 'CERTIFICATE'), - 'der'); + ocsp.utils.toDER(rawIssuer, 'CERTIFICATE'), + 'der') } - var tbsCert = cert.tbsCertificate; - var tbsIssuer = issuer.tbsCertificate; + var tbsCert = cert.tbsCertificate + var tbsIssuer = issuer.tbsCertificate var certID = { hashAlgorithm: { // algorithm: [ 2, 16, 840, 1, 101, 3, 4, 2, 1 ] // sha256 - algorithm: [ 1, 3, 14, 3, 2, 26 ] // sha1 + algorithm: [1, 3, 14, 3, 2, 26] // sha1 }, issuerNameHash: sha1(rfc5280.Name.encode(tbsCert.issuer, 'der')), issuerKeyHash: sha1( tbsIssuer.subjectPublicKeyInfo.subjectPublicKey.data), serialNumber: tbsCert.serialNumber - }; + } var tbs = { version: 'v1', - requestList: [ { + requestList: [{ reqCert: certID - } ], - requestExtensions: [ { + }], + requestExtensions: [{ extnID: rfc2560['id-pkix-ocsp-nonce'], critical: false, extnValue: rfc2560.Nonce.encode(crypto.randomBytes(16), 'der') - } ] - }; + }] + } var req = { tbsRequest: tbs - }; + } return { id: sha1(rfc2560.CertID.encode(certID, 'der')), @@ -65,5 +65,5 @@ exports.generate = function generate(rawCert, rawIssuer) { // Just to avoid re-parsing DER cert: cert, issuer: issuer - }; -}; + } +} diff --git a/lib/ocsp/server.js b/lib/ocsp/server.js index 81900cc..dd0b2f6 100644 --- a/lib/ocsp/server.js +++ b/lib/ocsp/server.js @@ -1,118 +1,113 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); +var ocsp = require('../ocsp') -var http = require('http'); -var util = require('util'); -var crypto = require('crypto'); +var http = require('http') +var util = require('util') +var crypto = require('crypto') -var async = require('async'); -var rfc2560 = require('asn1.js-rfc2560'); -var rfc5280 = require('asn1.js-rfc5280'); +var async = require('async') +var rfc2560 = require('asn1.js-rfc2560') +var rfc5280 = require('asn1.js-rfc5280') -function Server(options) { - http.Server.call(this, this.handler); +function Server (options) { + http.Server.call(this, this.handler) - this.options = util._extend({ + this.options = Object.assign({ nextUpdate: 24 * 3600 * 1e3 - }, options); + }, options) - this.key = this.options.key; + this.key = this.options.key this.cert = rfc5280.Certificate.decode( - ocsp.utils.toDER(options.cert, 'CERTIFICATE'), - 'der'); - this.cert = this.cert.tbsCertificate; + ocsp.utils.toDER(options.cert, 'CERTIFICATE'), + 'der') + this.cert = this.cert.tbsCertificate - var issuerName = rfc5280.Name.encode(this.cert.subject, 'der'); - var issuerKey = this.cert.subjectPublicKeyInfo.subjectPublicKey.data; + var issuerName = rfc5280.Name.encode(this.cert.subject, 'der') + var issuerKey = this.cert.subjectPublicKeyInfo.subjectPublicKey.data - this.certID = {}; - Object.keys(ocsp.utils.digestRev).forEach(function(digest) { + this.certID = {} + Object.keys(ocsp.utils.digestRev).forEach(function (digest) { this.certID[digest] = { issuerNameHash: crypto.createHash(digest).update(issuerName).digest(), issuerKeyHash: crypto.createHash(digest).update(issuerKey).digest() - }; - }, this); + } + }, this) - this.certs = {}; + this.certs = {} } -util.inherits(Server, http.Server); -module.exports = Server; +util.inherits(Server, http.Server) +module.exports = Server -Server.create = function create(options) { - return new Server(options); -}; +Server.create = function create (options) { + return new Server(options) +} -Server.prototype.addCert = function addCert(serial, status, info) { +Server.prototype.addCert = function addCert (serial, status, info) { this.certs[serial.toString(16)] = { type: status, value: info - }; -}; + } +} -Server.prototype.handler = function handler(req, res) { - if (req.method !== 'POST') - return res.writeHead(400); +Server.prototype.handler = function handler (req, res) { + if (req.method !== 'POST') { return res.writeHead(400) } - if (req.headers['content-type'] !== 'application/ocsp-request') - return res.writeHead(400); + if (req.headers['content-type'] !== 'application/ocsp-request') { return res.writeHead(400) } - var chunks = []; - req.on('readable', function() { - var chunk = req.read(); - if (chunk) - chunks.push(chunk); - }); + var chunks = [] + req.on('readable', function () { + var chunk = req.read() + if (chunk) { chunks.push(chunk) } + }) - function errRes(status) { + function errRes (status) { return rfc2560.OCSPResponse.encode({ responseStatus: status - }, 'der'); + }, 'der') } - function done(out) { + function done (out) { res.writeHead(200, { 'Content-Type': 'application/ocsp-response', 'Content-Length': out.length - }); - res.end(out); + }) + res.end(out) } - var self = this; - req.on('end', function() { - var body = Buffer.concat(chunks); - var ocspReq; + var self = this + req.on('end', function () { + var body = Buffer.concat(chunks) + var ocspReq try { - ocspReq = rfc2560.OCSPRequest.decode(body, 'der'); + ocspReq = rfc2560.OCSPRequest.decode(body, 'der') } catch (e) { - return done(errRes('malformed_request')); + return done(errRes('malformed_request')) } - self.getResponses(ocspReq, function(err, responses) { + self.getResponses(ocspReq, function (err, responses) { // Assume not found if (err) { - res.writeHead(404); - res.end(); - return; + res.writeHead(404) + res.end() + return } - return done(responses); - }); - }); - -}; + return done(responses) + }) + }) +} -Server.prototype.getResponses = function getResponses(req, cb) { - var self = this; +Server.prototype.getResponses = function getResponses (req, cb) { + var self = this - var reqList = req.tbsRequest.requestList; + var reqList = req.tbsRequest.requestList // TODO(indutny): support signed requests - async.map(reqList, function(req, cb) { - self.getResponse(req, cb); - }, function(err, responses) { - if (err) - return cb(err); + async.map(reqList, function (req, cb) { + self.getResponse(req, cb) + }, function (err, responses) { + if (err) { return cb(err) } // TODO(indutny): send extensions var basic = { @@ -132,14 +127,14 @@ Server.prototype.getResponses = function getResponses(req, cb) { signature: null // TODO(indutny): send certs? - }; + } - var sign = crypto.createSign('sha512WithRSAEncryption'); - sign.update(rfc2560.ResponseData.encode(basic.tbsResponseData, 'der')); + var sign = crypto.createSign('sha512WithRSAEncryption') + sign.update(rfc2560.ResponseData.encode(basic.tbsResponseData, 'der')) basic.signature = { unused: 0, data: sign.sign(self.key) - }; + } var res = { responseStatus: 'successful', @@ -147,51 +142,49 @@ Server.prototype.getResponses = function getResponses(req, cb) { responseType: 'id-pkix-ocsp-basic', response: rfc2560.BasicOCSPResponse.encode(basic, 'der') } - }; + } - cb(null, rfc2560.OCSPResponse.encode(res, 'der')); - }); -}; + cb(null, rfc2560.OCSPResponse.encode(res, 'der')) + }) +} -Server.prototype.getResponse = function getResponse(req, cb) { - var certID = req.reqCert; +Server.prototype.getResponse = function getResponse (req, cb) { + var certID = req.reqCert - var digestId = certID.hashAlgorithm.algorithm.join('.'); - var digest = ocsp.utils.digest[digestId]; - if (!digest) - return cb(new Error('Unknown digest: ' + digestId)); + var digestId = certID.hashAlgorithm.algorithm.join('.') + var digest = ocsp.utils.digest[digestId] + if (!digest) { return cb(new Error('Unknown digest: ' + digestId)) } - var expectedID = this.certID[digest]; - if (!expectedID) - return cb(new Error('No pre-generated CertID for digest: ' + digest)); + var expectedID = this.certID[digest] + if (!expectedID) { return cb(new Error('No pre-generated CertID for digest: ' + digest)) } if (expectedID.issuerNameHash.toString('hex') !== certID.issuerNameHash.toString('hex')) { - return cb(new Error('Issuer name mismatch')); + return cb(new Error('Issuer name mismatch')) } if (expectedID.issuerKeyHash.toString('hex') !== certID.issuerKeyHash.toString('hex')) { - return cb(new Error('Issuer key mismatch')); + return cb(new Error('Issuer key mismatch')) } - var serial = certID.serialNumber.toString(16); - var cert = this.certs[serial]; + var serial = certID.serialNumber.toString(16) + var cert = this.certs[serial] var response = { certId: certID, certStatus: null, thisUpdate: new Date(), nextUpdate: new Date(+new Date() + this.options.nextUpdate) - }; + } if (cert) { - response.certStatus = cert; + response.certStatus = cert } else { response.certStatus = { type: 'unknown', value: null - }; + } } - cb(null, response); -}; + cb(null, response) +} diff --git a/lib/ocsp/utils.js b/lib/ocsp/utils.js index ad9af75..bfbbd4c 100644 --- a/lib/ocsp/utils.js +++ b/lib/ocsp/utils.js @@ -1,106 +1,98 @@ -'use strict'; +'use strict' -var http = require('http'); -var util = require('util'); -var url = require('url'); -var asn1 = require('asn1.js'); +var http = require('http') +var url = require('url') +var asn1 = require('asn1.js') -var rfc2560 = require('asn1.js-rfc2560'); +var rfc2560 = require('asn1.js-rfc2560') -exports['id-ad-caIssuers'] = [ 1, 3, 6, 1, 5, 5, 7, 48, 2 ]; -exports['id-kp-OCSPSigning'] = [ 1, 3, 6, 1, 5, 5, 7, 3, 9 ]; +exports['id-ad-caIssuers'] = [1, 3, 6, 1, 5, 5, 7, 48, 2] +exports['id-kp-OCSPSigning'] = [1, 3, 6, 1, 5, 5, 7, 3, 9] -exports.getResponse = function getResponse(uri, req, cb) { - uri = url.parse(uri); +exports.getResponse = function getResponse (uri, req, cb) { + uri = url.parse(uri) - var options = util._extend({ + var options = Object.assign({ method: 'POST', headers: { 'Content-Type': 'application/ocsp-request', 'Content-Length': req.length } - }, uri); + }, uri) - function done(err, response) { - if (cb) - cb(err, response); - cb = null; + function done (err, response) { + if (cb) { cb(err, response) } + cb = null } - function onResponse(response) { + function onResponse (response) { if (response.statusCode < 200 || response.statusCode >= 400) { return done( - new Error('Failed to obtain OCSP response: ' + response.statusCode)); + new Error('Failed to obtain OCSP response: ' + response.statusCode)) } - var chunks = []; - response.on('readable', function() { - var chunk = response.read(); - if (!chunk) - return; - chunks.push(chunk); - }); - response.on('end', function() { - var ocsp = Buffer.concat(chunks); - - done(null, ocsp); - }); + var chunks = [] + response.on('readable', function () { + var chunk = response.read() + if (!chunk) { return } + chunks.push(chunk) + }) + response.on('end', function () { + var ocsp = Buffer.concat(chunks) + + done(null, ocsp) + }) } http.request(options, onResponse) - .on('error', done) - .end(req); -}; + .on('error', done) + .end(req) +} -exports.parseResponse = function parseResponse(raw) { - var body = { start: 0, end: raw.length }; +exports.parseResponse = function parseResponse (raw) { + var body = { start: 0, end: raw.length } var response = rfc2560.OCSPResponse.decode(raw, 'der', { - track: function(key, start, end, type) { - if (type !== 'content' || key !== 'responseBytes/response') - return; - body.start = start; - body.end = end; + track: function (key, start, end, type) { + if (type !== 'content' || key !== 'responseBytes/response') { return } + body.start = start + body.end = end } - }); + }) - var status = response.responseStatus; - if (status !== 'successful') - throw new Error('Bad OCSP response status: ' + status); + var status = response.responseStatus + if (status !== 'successful') { throw new Error('Bad OCSP response status: ' + status) } // Unknown response type - var responseType = response.responseBytes.responseType; - if (responseType !== 'id-pkix-ocsp-basic') - throw new Error('Unknown OCSP response type: ' + responseType); + var responseType = response.responseBytes.responseType + if (responseType !== 'id-pkix-ocsp-basic') { throw new Error('Unknown OCSP response type: ' + responseType) } - var bytes = response.responseBytes.response; + var bytes = response.responseBytes.response - var tbs = { start: body.start, end: body.end }; - var certsTbs = []; + var tbs = { start: body.start, end: body.end } + var certsTbs = [] var basic = rfc2560.BasicOCSPResponse.decode(bytes, 'der', { - track: function(key, start, end, type) { - if (type !== 'tagged') - return; + track: function (key, start, end, type) { + if (type !== 'tagged') { return } if (key === 'tbsResponseData') { - tbs.start = body.start + start; - tbs.end = body.start + end; + tbs.start = body.start + start + tbs.end = body.start + end } else if (key === 'certs/tbsCertificate') { - certsTbs.push({ start: body.start + start, end: body.start + end }); + certsTbs.push({ start: body.start + start, end: body.start + end }) } } - }); + }) - var OCSPSigning = exports['id-kp-OCSPSigning'].join('.'); - var certs = (basic.certs || []).filter(function(cert) { - return cert.tbsCertificate.extensions.some(function(ext) { - if (ext.extnID !== 'extendedKeyUsage') - return false; + var OCSPSigning = exports['id-kp-OCSPSigning'].join('.') + var certs = (basic.certs || []).filter(function (cert) { + return cert.tbsCertificate.extensions.some(function (ext) { + if (ext.extnID !== 'extendedKeyUsage') { return false } - return ext.extnValue.some(function(value) { - return value.join('.') === OCSPSigning; - }); - }); - }); + return ext.extnValue.some(function (value) { + return value.join('.') === OCSPSigning + }) + }) + }) return { start: tbs.start, @@ -108,91 +100,86 @@ exports.parseResponse = function parseResponse(raw) { value: basic, certs: certs, certsTbs: certsTbs - }; -}; + } +} exports.digest = { '1.3.14.3.2.26': 'sha1', '2.16.840.1.101.3.4.2.1': 'sha256' -}; +} exports.digestRev = { sha1: '1.3.14.3.2.26', sha256: '2.16.840.1.101.3.4.2.1' -}; +} exports.sign = { '1.2.840.113549.1.1.5': 'sha1WithRSAEncryption', '1.2.840.113549.1.1.11': 'sha256WithRSAEncryption', '1.2.840.113549.1.1.12': 'sha384WithRSAEncryption', '1.2.840.113549.1.1.13': 'sha512WithRSAEncryption' -}; +} exports.signRev = { - sha1WithRSAEncryption: [ 1, 2, 840, 113549, 1, 1, 5 ], - sha256WithRSAEncryption: [ 1, 2, 840, 113549, 1, 1, 11 ], - sha384WithRSAEncryption: [ 1, 2, 840, 113549, 1, 1, 12 ], - sha512WithRSAEncryption: [ 1, 2, 840, 113549, 1, 1, 13 ] -}; - -exports.toPEM = function toPEM(buf, label) { - var p = buf.toString('base64'); - var out = [ '-----BEGIN ' + label + '-----' ]; - for (var i = 0; i < p.length; i += 64) - out.push(p.slice(i, i + 64)); - out.push('-----END ' + label + '-----'); - return out.join('\n'); -}; - -exports.toDER = function toDER(raw, what) { + sha1WithRSAEncryption: [1, 2, 840, 113549, 1, 1, 5], + sha256WithRSAEncryption: [1, 2, 840, 113549, 1, 1, 11], + sha384WithRSAEncryption: [1, 2, 840, 113549, 1, 1, 12], + sha512WithRSAEncryption: [1, 2, 840, 113549, 1, 1, 13] +} + +exports.toPEM = function toPEM (buf, label) { + var p = buf.toString('base64') + var out = ['-----BEGIN ' + label + '-----'] + for (var i = 0; i < p.length; i += 64) { out.push(p.slice(i, i + 64)) } + out.push('-----END ' + label + '-----') + return out.join('\n') +} + +exports.toDER = function toDER (raw, what) { var der = raw.toString().match(new RegExp( - '-----BEGIN ' + what + '-----([^-]*)-----END ' + what + '-----')); - if (der) - der = new Buffer(der[1].replace(/[\r\n]/g, ''), 'base64'); - else if (typeof raw === 'string') - der = new Buffer(raw); - else - der = raw; - return der; -}; + '-----BEGIN ' + what + '-----([^-]*)-----END ' + what + '-----')) + if (der) { + der = Buffer.from(der[1].replace(/[\r\n]/g, ''), 'base64') + } else if (typeof raw === 'string') { + der = Buffer.from(raw) + } else { + der = raw + } + return der +} -exports.getAuthorityInfo = function getAuthorityInfo(cert, key, done) { - var exts = cert.tbsCertificate.extensions; - if (!exts) - exts = []; +exports.getAuthorityInfo = function getAuthorityInfo (cert, key, done) { + var exts = cert.tbsCertificate.extensions + if (!exts) { exts = [] } - var infoAccess = exts.filter(function(ext) { - return ext.extnID === 'authorityInformationAccess'; - }); + var infoAccess = exts.filter(function (ext) { + return ext.extnID === 'authorityInformationAccess' + }) - if (infoAccess.length === 0) - return done(new Error('AuthorityInfoAccess not found in extensions')); + if (infoAccess.length === 0) { return done(new Error('AuthorityInfoAccess not found in extensions')) } - var res = null; - var found = infoAccess.some(function(info) { - var ext = info.extnValue; + var res = null + var found = infoAccess.some(function (info) { + var ext = info.extnValue - return ext.some(function(ad) { - if (ad.accessMethod.join('.') !== key) - return false; + return ext.some(function (ad) { + if (ad.accessMethod.join('.') !== key) { return false } - var loc = ad.accessLocation; - if (loc.type !== 'uniformResourceIdentifier') - return false; + var loc = ad.accessLocation + if (loc.type !== 'uniformResourceIdentifier') { return false } - res = loc.value + ''; + res = loc.value + '' - return true; - }); - }); + return true + }) + }) - if (!found) - return done(new Error(key + ' not found in AuthorityInfoAccess')); + if (!found) { return done(new Error(key + ' not found in AuthorityInfoAccess')) } - return done(null, res); -}; + return done(null, res) +} -var RSAPrivateKey = asn1.define('RSAPrivateKey', function() { +var RSAPrivateKey = asn1.define('RSAPrivateKey', function () { this.seq().obj( this.key('version').int(), this.key('modulus').int(), @@ -203,6 +190,6 @@ var RSAPrivateKey = asn1.define('RSAPrivateKey', function() { this.key('exponent1').int(), this.key('exponent2').int(), this.key('coefficient').int() - ); -}); -exports.RSAPrivateKey = RSAPrivateKey; + ) +}) +exports.RSAPrivateKey = RSAPrivateKey diff --git a/lib/ocsp/verify.js b/lib/ocsp/verify.js index 37da8e2..8a14101 100644 --- a/lib/ocsp/verify.js +++ b/lib/ocsp/verify.js @@ -1,118 +1,113 @@ -'use strict'; +'use strict' -var ocsp = require('../ocsp'); -var rfc5280 = require('asn1.js-rfc5280'); -var crypto = require('crypto'); +var ocsp = require('../ocsp') +var rfc5280 = require('asn1.js-rfc5280') +var crypto = require('crypto') // TODO(indutny): verify issuer, etc... -function findResponder(issuer, certs, raws) { - var issuerKey = issuer.tbsCertificate.subjectPublicKeyInfo; +function findResponder (issuer, certs, raws) { + var issuerKey = issuer.tbsCertificate.subjectPublicKeyInfo issuerKey = ocsp.utils.toPEM( - rfc5280.SubjectPublicKeyInfo.encode(issuerKey, 'der'), 'PUBLIC KEY'); + rfc5280.SubjectPublicKeyInfo.encode(issuerKey, 'der'), 'PUBLIC KEY') for (var i = 0; i < certs.length; i++) { - var cert = certs[i]; - var signAlg = ocsp.utils.sign[cert.signatureAlgorithm.algorithm.join('.')]; + var cert = certs[i] + var signAlg = ocsp.utils.sign[cert.signatureAlgorithm.algorithm.join('.')] if (!signAlg) { throw new Error('Unknown signature algorithm ' + - cert.signatureAlgorithm.algorithm); + cert.signatureAlgorithm.algorithm) } - var verify = crypto.createVerify(signAlg); + var verify = crypto.createVerify(signAlg) - verify.update(raws[i]); - if (!verify.verify(issuerKey, cert.signature.data)) - throw new Error('Invalid signature'); + verify.update(raws[i]) + if (!verify.verify(issuerKey, cert.signature.data)) { throw new Error('Invalid signature') } - var certKey = cert.tbsCertificate.subjectPublicKeyInfo; + var certKey = cert.tbsCertificate.subjectPublicKeyInfo certKey = ocsp.utils.toPEM( - rfc5280.SubjectPublicKeyInfo.encode(certKey, 'der'), 'PUBLIC KEY'); - return certKey; + rfc5280.SubjectPublicKeyInfo.encode(certKey, 'der'), 'PUBLIC KEY') + return certKey } - return issuerKey; + return issuerKey } -module.exports = function verify(options, cb) { - var req = options.request; - var issuer; - var res; +module.exports = function verify (options, cb) { + var req = options.request + var issuer + var res - function done(err) { - process.nextTick(function() { - cb(err, res && res.certStatus); - }); + function done (err) { + process.nextTick(function () { + cb(err, res && res.certStatus) + }) } try { issuer = req.issuer || rfc5280.Certificate.decode( - ocsp.utils.toDER(options.issuer, 'CERTIFICATE'), 'der'); + ocsp.utils.toDER(options.issuer, 'CERTIFICATE'), 'der') - res = ocsp.utils.parseResponse(options.response); + res = ocsp.utils.parseResponse(options.response) } catch (e) { - return done(e); + return done(e) } - var rawTBS = options.response.slice(res.start, res.end); - var certs = res.certs; - var raws = res.certsTbs.map(function(tbs) { - return options.response.slice(tbs.start, tbs.end); - }); - res = res.value; + var rawTBS = options.response.slice(res.start, res.end) + var certs = res.certs + var raws = res.certsTbs.map(function (tbs) { + return options.response.slice(tbs.start, tbs.end) + }) + res = res.value // Verify signature using CAs Public Key - var signAlg = ocsp.utils.sign[res.signatureAlgorithm.algorithm.join('.')]; + var signAlg = ocsp.utils.sign[res.signatureAlgorithm.algorithm.join('.')] if (!signAlg) { done(new Error('Unknown signature algorithm ' + - res.signatureAlgorithm.algorithm)); - return; + res.signatureAlgorithm.algorithm)) + return } - var responderKey = findResponder(issuer, certs, raws); + var responderKey = findResponder(issuer, certs, raws) - var verify = crypto.createVerify(signAlg); - var tbs = res.tbsResponseData; + var verify = crypto.createVerify(signAlg) + var tbs = res.tbsResponseData - var signature = res.signature.data; + var signature = res.signature.data - verify.update(rawTBS); - if (!verify.verify(responderKey, signature)) - return done(new Error('Invalid signature')); + verify.update(rawTBS) + if (!verify.verify(responderKey, signature)) { return done(new Error('Invalid signature')) } - if (tbs.responses.length < 1) - return done(new Error('Expected at least one response')); + if (tbs.responses.length < 1) { return done(new Error('Expected at least one response')) } - var res = tbs.responses[0]; + res = tbs.responses[0] // Verify CertID // XXX(indutny): verify parameters if (res.certId.hashAlgorithm.algorithm.join('.') !== req.certID.hashAlgorithm.algorithm.join('.')) { - return done(new Error('Hash algorithm mismatch')); + return done(new Error('Hash algorithm mismatch')) } if (res.certId.issuerNameHash.toString('hex') !== req.certID.issuerNameHash.toString('hex')) { - return done(new Error('Issuer name hash mismatch')); + return done(new Error('Issuer name hash mismatch')) } if (res.certId.issuerKeyHash.toString('hex') !== req.certID.issuerKeyHash.toString('hex')) { - return done(new Error('Issuer key hash mismatch')); + return done(new Error('Issuer key hash mismatch')) } - if (res.certId.serialNumber.cmp(req.certID.serialNumber) !== 0) - return done(new Error('Serial number mismatch')); + if (res.certId.serialNumber.cmp(req.certID.serialNumber) !== 0) { return done(new Error('Serial number mismatch')) } if (res.certStatus.type !== 'good') { - return done(new Error('OCSP Status: ' + res.certStatus.type)); + return done(new Error('OCSP Status: ' + res.certStatus.type)) } - var now = +new Date(); - var nudge = options.nudge || 60000; - if (res.thisUpdate - nudge > now || res.nextUpdate + nudge < now) - return done(new Error('OCSP Response expired')); + var now = +new Date() + var nudge = options.nudge || 60000 + if (res.thisUpdate - nudge > now || res.nextUpdate + nudge < now) { return done(new Error('OCSP Response expired')) } - return done(null); -}; + return done(null) +} diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..f73dba3 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,2538 @@ +{ + "name": "ocsp", + "version": "1.2.0", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "@babel/code-frame": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.8.3.tgz", + "integrity": "sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==", + "dev": true, + "requires": { + "@babel/highlight": "^7.8.3" + } + }, + "@babel/helper-validator-identifier": { + "version": "7.9.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.9.5.tgz", + "integrity": "sha512-/8arLKUFq882w4tWGj9JYzRpAlZgiWUJ+dtteNTDqrRBz9Iguck9Rn3ykuBDoUwh2TO4tSAJlrxDUOXWklJe4g==", + "dev": true + }, + "@babel/highlight": { + "version": "7.9.0", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.9.0.tgz", + "integrity": "sha512-lJZPilxX7Op3Nv/2cvFdnlepPXDxi29wxteT57Q965oc5R9v86ztx0jfxVrTcBk8C2kcPkkDa2Z4T3ZsPPVWsQ==", + "dev": true, + "requires": { + "@babel/helper-validator-identifier": "^7.9.0", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "@types/color-name": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", + "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==", + "dev": true + }, + "acorn": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.1.1.tgz", + "integrity": "sha512-add7dgA5ppRPxCFJoAGfMDi7PIBXq1RtGo7BhbLaxwrXPOmw8gq48Y9ozT01hUKy9byMjlR20EJhu5zlkErEkg==", + "dev": true + }, + "acorn-jsx": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.2.0.tgz", + "integrity": "sha512-HiUX/+K2YpkpJ+SzBffkM/AQ2YE03S0U1kjTLVpoJdhZMOWy8qvXVN9JdLqv2QsaQ6MPYQIuNmwD8zOiYUofLQ==", + "dev": true + }, + "ajv": { + "version": "6.12.2", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.2.tgz", + "integrity": "sha512-k+V+hzjm5q/Mr8ef/1Y9goCmlsK4I6Sm74teeyGvFk1XrOsbsKLjEdrvny42CZ+a8sXbk8KWpY/bDwS+FLL2UQ==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ansi-colors": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.3.tgz", + "integrity": "sha512-LEHHyuhlPY3TmuUYMh2oz89lTShfvgbmzaBcxve9t/9Wuy7Dwf4yoAKcND7KFT1HAQfqZ12qtc+DUrBMeKF9nw==", + "dev": true + }, + "ansi-escapes": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.1.tgz", + "integrity": "sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==", + "dev": true, + "requires": { + "type-fest": "^0.11.0" + }, + "dependencies": { + "type-fest": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", + "integrity": "sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ==", + "dev": true + } + } + }, + "anymatch": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", + "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", + "dev": true, + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "requires": { + "sprintf-js": "~1.0.2" + } + }, + "array-includes": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.1.tgz", + "integrity": "sha512-c2VXaCHl7zPsvpkFsw4nxvFie4fh1ur9bpcgsVkIjqn0H/Xwdg+7fv3n2r/isyS8EBj5b06M9kHyZuIr4El6WQ==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0", + "is-string": "^1.0.5" + } + }, + "array.prototype.flat": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.3.tgz", + "integrity": "sha512-gBlRZV0VSmfPIeWfuuy56XZMvbVfbEUnOXUvt3F/eUUUSyzlgLxhEX4YAEpxNAogRGehPSnfXyPtYyKAhkzQhQ==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0-next.1" + } + }, + "asn1.js": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-5.3.0.tgz", + "integrity": "sha512-WHnQJFcOrIWT1RLOkFFBQkFVvyt9BPOOrH+Dp152Zk4R993rSzXUGPmkybIcUFhHE2d/iHH+nCaOWVCDbO8fgA==", + "requires": { + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0", + "safer-buffer": "^2.1.0" + } + }, + "asn1.js-rfc2560": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/asn1.js-rfc2560/-/asn1.js-rfc2560-5.0.1.tgz", + "integrity": "sha512-1PrVg6kuBziDN3PGFmRk3QrjpKvP9h/Hv5yMrFZvC1kpzP6dQRzf5BpKstANqHBkaOUmTpakJWhicTATOA/SbA==", + "requires": { + "asn1.js-rfc5280": "^3.0.0" + } + }, + "asn1.js-rfc5280": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/asn1.js-rfc5280/-/asn1.js-rfc5280-3.0.0.tgz", + "integrity": "sha512-Y2LZPOWeZ6qehv698ZgOGGCZXBQShObWnGthTrIFlIQjuV1gg2B8QOhWFRExq/MR1VnPpIIe7P9vX2vElxv+Pg==", + "requires": { + "asn1.js": "^5.0.0" + } + }, + "astral-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", + "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "dev": true + }, + "async": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.0.tgz", + "integrity": "sha512-TR2mEZFVOj2pLStYxLht7TyfuRzaydfpxr3k9RpHIzMgw7A64dzsdqCxH1WJyQdoe8T10nDXd9wnEigmiuHIZw==" + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", + "dev": true + }, + "binary-extensions": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", + "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==", + "dev": true + }, + "bn.js": { + "version": "4.11.8", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", + "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==" + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=", + "dev": true + }, + "browser-stdout": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", + "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", + "dev": true + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true + }, + "chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "chokidar": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.0.tgz", + "integrity": "sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==", + "dev": true, + "requires": { + "anymatch": "~3.1.1", + "braces": "~3.0.2", + "fsevents": "~2.1.1", + "glob-parent": "~5.1.0", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.2.0" + } + }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", + "dev": true + }, + "cliui": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", + "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "dev": true, + "requires": { + "string-width": "^3.1.0", + "strip-ansi": "^5.2.0", + "wrap-ansi": "^5.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "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=", + "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 + }, + "contains-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", + "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", + "dev": true + }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", + "dev": true + }, + "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 + }, + "define-properties": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", + "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "dev": true, + "requires": { + "object-keys": "^1.0.12" + } + }, + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", + "dev": true + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dev": true, + "requires": { + "is-arrayish": "^0.2.1" + } + }, + "es-abstract": { + "version": "1.17.5", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.5.tgz", + "integrity": "sha512-BR9auzDbySxOcfog0tLECW8l28eRGpDpU3Dm3Hp4q/N+VtLTmyj4EUN088XZWQDW/hzj6sYRDXeOFsaAODKvpg==", + "dev": true, + "requires": { + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1", + "is-callable": "^1.1.5", + "is-regex": "^1.0.5", + "object-inspect": "^1.7.0", + "object-keys": "^1.1.1", + "object.assign": "^4.1.0", + "string.prototype.trimleft": "^2.1.1", + "string.prototype.trimright": "^2.1.1" + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "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=", + "dev": true + }, + "eslint": { + "version": "6.8.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", + "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "ajv": "^6.10.0", + "chalk": "^2.1.0", + "cross-spawn": "^6.0.5", + "debug": "^4.0.1", + "doctrine": "^3.0.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^1.4.3", + "eslint-visitor-keys": "^1.1.0", + "espree": "^6.1.2", + "esquery": "^1.0.1", + "esutils": "^2.0.2", + "file-entry-cache": "^5.0.1", + "functional-red-black-tree": "^1.0.1", + "glob-parent": "^5.0.0", + "globals": "^12.1.0", + "ignore": "^4.0.6", + "import-fresh": "^3.0.0", + "imurmurhash": "^0.1.4", + "inquirer": "^7.0.0", + "is-glob": "^4.0.0", + "js-yaml": "^3.13.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.14", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.3", + "progress": "^2.0.0", + "regexpp": "^2.0.1", + "semver": "^6.1.2", + "strip-ansi": "^5.2.0", + "strip-json-comments": "^3.0.1", + "table": "^5.2.3", + "text-table": "^0.2.0", + "v8-compile-cache": "^2.0.3" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "debug": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", + "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "js-yaml": { + "version": "3.13.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", + "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", + "dev": true + }, + "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 + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "strip-json-comments": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.0.tgz", + "integrity": "sha512-e6/d0eBu7gHtdCqFt0xJr642LdToM5/cN4Qb9DbHjVx1CP5RyeM+zH7pbecEmDv/lBqb0QH+6Uqq75rxFPkM0w==", + "dev": true + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "eslint-config-standard": { + "version": "14.1.1", + "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-14.1.1.tgz", + "integrity": "sha512-Z9B+VR+JIXRxz21udPTL9HpFMyoMUEeX1G251EQ6e05WD9aPVtVBn09XUmZ259wCMlCDmYDSZG62Hhm+ZTJcUg==", + "dev": true + }, + "eslint-import-resolver-node": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.3.tgz", + "integrity": "sha512-b8crLDo0M5RSe5YG8Pu2DYBj71tSB6OvXkfzwbJU2w7y8P4/yo0MyF8jU26IEuEuHF2K5/gcAJE3LhQGqBBbVg==", + "dev": true, + "requires": { + "debug": "^2.6.9", + "resolve": "^1.13.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "eslint-module-utils": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.0.tgz", + "integrity": "sha512-6j9xxegbqe8/kZY8cYpcp0xhbK0EgJlg3g9mib3/miLaExuuwc3n5UEfSnU6hWMbT0FAYVvDbL9RrRgpUeQIvA==", + "dev": true, + "requires": { + "debug": "^2.6.9", + "pkg-dir": "^2.0.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "eslint-plugin-es": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-es/-/eslint-plugin-es-3.0.0.tgz", + "integrity": "sha512-6/Jb/J/ZvSebydwbBJO1R9E5ky7YeElfK56Veh7e4QGFHCXoIXGH9HhVz+ibJLM3XJ1XjP+T7rKBLUa/Y7eIng==", + "dev": true, + "requires": { + "eslint-utils": "^2.0.0", + "regexpp": "^3.0.0" + }, + "dependencies": { + "eslint-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.0.0.tgz", + "integrity": "sha512-0HCPuJv+7Wv1bACm8y5/ECVfYdfsAm9xmVb7saeFlxjPYALefjhbYoCkBjPdPzGH8wWyTpAez82Fh3VKYEZ8OA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "regexpp": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", + "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", + "dev": true + } + } + }, + "eslint-plugin-import": { + "version": "2.20.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.20.2.tgz", + "integrity": "sha512-FObidqpXrR8OnCh4iNsxy+WACztJLXAHBO5hK79T1Hc77PgQZkyDGA5Ag9xAvRpglvLNxhH/zSmZ70/pZ31dHg==", + "dev": true, + "requires": { + "array-includes": "^3.0.3", + "array.prototype.flat": "^1.2.1", + "contains-path": "^0.1.0", + "debug": "^2.6.9", + "doctrine": "1.5.0", + "eslint-import-resolver-node": "^0.3.2", + "eslint-module-utils": "^2.4.1", + "has": "^1.0.3", + "minimatch": "^3.0.4", + "object.values": "^1.1.0", + "read-pkg-up": "^2.0.0", + "resolve": "^1.12.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "doctrine": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", + "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", + "dev": true, + "requires": { + "esutils": "^2.0.2", + "isarray": "^1.0.0" + } + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "eslint-plugin-node": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-node/-/eslint-plugin-node-11.1.0.tgz", + "integrity": "sha512-oUwtPJ1W0SKD0Tr+wqu92c5xuCeQqB3hSCHasn/ZgjFdA9iDGNkNf2Zi9ztY7X+hNuMib23LNGRm6+uN+KLE3g==", + "dev": true, + "requires": { + "eslint-plugin-es": "^3.0.0", + "eslint-utils": "^2.0.0", + "ignore": "^5.1.1", + "minimatch": "^3.0.4", + "resolve": "^1.10.1", + "semver": "^6.1.0" + }, + "dependencies": { + "eslint-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.0.0.tgz", + "integrity": "sha512-0HCPuJv+7Wv1bACm8y5/ECVfYdfsAm9xmVb7saeFlxjPYALefjhbYoCkBjPdPzGH8wWyTpAez82Fh3VKYEZ8OA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "ignore": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.4.tgz", + "integrity": "sha512-MzbUSahkTW1u7JpKKjY7LCARd1fU5W2rLdxlM4kdkayuCwZImjkpluF9CM1aLewYJguPDqewLam18Y6AU69A8A==", + "dev": true + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "eslint-plugin-promise": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-4.2.1.tgz", + "integrity": "sha512-VoM09vT7bfA7D+upt+FjeBO5eHIJQBUWki1aPvB+vbNiHS3+oGIJGIeyBtKQTME6UPXXy3vV07OL1tHd3ANuDw==", + "dev": true + }, + "eslint-plugin-standard": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-standard/-/eslint-plugin-standard-4.0.1.tgz", + "integrity": "sha512-v/KBnfyaOMPmZc/dmc6ozOdWqekGp7bBGq4jLAecEfPGmfKiWS4sA8sC0LqiV9w5qmXAtXVn4M3p1jSyhY85SQ==", + "dev": true + }, + "eslint-scope": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.0.0.tgz", + "integrity": "sha512-oYrhJW7S0bxAFDvWqzvMPRm6pcgcnWc4QnofCAqRTRfQC0JcwenzGglTtsLyIuuWFfkqDG9vz67cnttSd53djw==", + "dev": true, + "requires": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + } + }, + "eslint-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", + "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "eslint-visitor-keys": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.1.0.tgz", + "integrity": "sha512-8y9YjtM1JBJU/A9Kc+SbaOV4y29sSWckBwMHa+FGtVj5gN/sbnKDf6xJUl+8g7FAij9LVaP8C24DUiH/f/2Z9A==", + "dev": true + }, + "espree": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + } + }, + "esquery": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.3.1.tgz", + "integrity": "sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + }, + "dependencies": { + "estraverse": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.1.0.tgz", + "integrity": "sha512-FyohXK+R0vE+y1nHLoBM7ZTyqRpqAlhdZHCWIWEviFLiGB8b04H6bQs8G+XTthacvT8VuwvteiP7RJSxMs8UEw==", + "dev": true + } + } + }, + "esrecurse": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz", + "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==", + "dev": true, + "requires": { + "estraverse": "^4.1.0" + } + }, + "estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "requires": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + } + }, + "fast-deep-equal": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", + "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "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": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "file-entry-cache": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", + "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "dev": true, + "requires": { + "flat-cache": "^2.0.1" + } + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "flat": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz", + "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==", + "dev": true, + "requires": { + "is-buffer": "~2.0.3" + } + }, + "flat-cache": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", + "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "dev": true, + "requires": { + "flatted": "^2.0.0", + "rimraf": "2.6.3", + "write": "1.0.3" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "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 + }, + "fsevents": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.2.tgz", + "integrity": "sha512-R4wDiBwZ0KzpgOWetKDug1FZcYhqYnUYKtfZYt4mD5SBz76q0KR4Q9o7GIPamsVPGmW3EYPPJ0dOOjvx32ldZA==", + "dev": true, + "optional": true + }, + "function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true + }, + "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=", + "dev": true + }, + "get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true + }, + "glob-parent": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", + "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", + "dev": true, + "requires": { + "is-glob": "^4.0.1" + } + }, + "globals": { + "version": "12.4.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", + "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "dev": true, + "requires": { + "type-fest": "^0.8.1" + } + }, + "graceful-fs": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", + "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==", + "dev": true + }, + "growl": { + "version": "1.10.5", + "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", + "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", + "dev": true + }, + "has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, + "requires": { + "function-bind": "^1.1.1" + } + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "has-symbols": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", + "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", + "dev": true + }, + "hash.js": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", + "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, + "he": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "dev": true + }, + "hmac-drbg": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", + "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=", + "dev": true, + "requires": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "hosted-git-info": { + "version": "2.8.8", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", + "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", + "dev": true + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "import-fresh": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.1.tgz", + "integrity": "sha512-6e1q1cnWP2RXD9/keSkxHScg508CdXqXWgWBaETNhyuBFz+kUZlKboh+ISK+bU++DmbHimVBrOz/zzPe0sZ3sQ==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "inquirer": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.1.0.tgz", + "integrity": "sha512-5fJMWEmikSYu0nv/flMc475MhGbB7TSPd/2IpFV4I4rMklboCH2rQjYY5kKiYGHqUF9gvaambupcJFFG9dvReg==", + "dev": true, + "requires": { + "ansi-escapes": "^4.2.1", + "chalk": "^3.0.0", + "cli-cursor": "^3.1.0", + "cli-width": "^2.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.15", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.5.3", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "dependencies": { + "ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true + }, + "ansi-styles": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", + "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", + "dev": true, + "requires": { + "@types/color-name": "^1.1.1", + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true + }, + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", + "dev": true + }, + "string-width": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", + "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", + "dev": true, + "requires": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + } + }, + "strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.0" + } + }, + "supports-color": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", + "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", + "dev": true + }, + "is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "requires": { + "binary-extensions": "^2.0.0" + } + }, + "is-buffer": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.4.tgz", + "integrity": "sha512-Kq1rokWXOPXWuaMAqZiJW4XxsmD9zGx9q4aePabbn3qCRGedtH7Cm+zV8WETitMfu1wdh+Rvd6w5egwSngUX2A==", + "dev": true + }, + "is-callable": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz", + "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==", + "dev": true + }, + "is-date-object": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", + "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "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 + }, + "is-glob": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", + "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "is-promise": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", + "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=", + "dev": true + }, + "is-regex": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz", + "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==", + "dev": true, + "requires": { + "has": "^1.0.3" + } + }, + "is-string": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.5.tgz", + "integrity": "sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ==", + "dev": true + }, + "is-symbol": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", + "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", + "dev": true, + "requires": { + "has-symbols": "^1.0.1" + } + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true + }, + "json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "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=", + "dev": true + }, + "levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", + "dev": true, + "requires": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + } + }, + "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=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "strip-bom": "^3.0.0" + }, + "dependencies": { + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + } + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "log-symbols": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz", + "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==", + "dev": true, + "requires": { + "chalk": "^2.4.2" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "miller-rabin": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/miller-rabin/-/miller-rabin-4.0.1.tgz", + "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", + "dev": true, + "requires": { + "bn.js": "^4.0.0", + "brorand": "^1.0.1" + } + }, + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true + }, + "minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==" + }, + "minimalistic-crypto-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", + "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=", + "dev": true + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", + "dev": true + }, + "mkdirp": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", + "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "dev": true, + "requires": { + "minimist": "^1.2.5" + } + }, + "mocha": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-7.1.1.tgz", + "integrity": "sha512-3qQsu3ijNS3GkWcccT5Zw0hf/rWvu1fTN9sPvEd81hlwsr30GX2GcDSSoBxo24IR8FelmrAydGC6/1J5QQP4WA==", + "dev": true, + "requires": { + "ansi-colors": "3.2.3", + "browser-stdout": "1.3.1", + "chokidar": "3.3.0", + "debug": "3.2.6", + "diff": "3.5.0", + "escape-string-regexp": "1.0.5", + "find-up": "3.0.0", + "glob": "7.1.3", + "growl": "1.10.5", + "he": "1.2.0", + "js-yaml": "3.13.1", + "log-symbols": "3.0.0", + "minimatch": "3.0.4", + "mkdirp": "0.5.3", + "ms": "2.1.1", + "node-environment-flags": "1.0.6", + "object.assign": "4.1.0", + "strip-json-comments": "2.0.1", + "supports-color": "6.0.0", + "which": "1.3.1", + "wide-align": "1.1.3", + "yargs": "13.3.2", + "yargs-parser": "13.1.2", + "yargs-unparser": "1.6.0" + }, + "dependencies": { + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "glob": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "js-yaml": { + "version": "3.13.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", + "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "mkdirp": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.3.tgz", + "integrity": "sha512-P+2gwrFqx8lhew375MQHHeTlY8AuOJSrGf0R5ddkEndUkmwpgUob/vQuBD1V22/Cw1/lJr4x+EjllSezBThzBg==", + "dev": true, + "requires": { + "minimist": "^1.2.5" + } + }, + "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=", + "dev": true + }, + "supports-color": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.0.0.tgz", + "integrity": "sha512-on9Kwidc1IUQo+bQdhi8+Tijpo0e1SS6RoGo2guUwn5vdaxw8RXOF9Vb2ws+ihWOmh4JnCJOvaziZWP1VABaLg==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true + }, + "mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, + "nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", + "dev": true + }, + "node-environment-flags": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.6.tgz", + "integrity": "sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==", + "dev": true, + "requires": { + "object.getownpropertydescriptors": "^2.0.3", + "semver": "^5.7.0" + } + }, + "normalize-package-data": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", + "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "dev": true, + "requires": { + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + } + }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true + }, + "object-inspect": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", + "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", + "dev": true + }, + "object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true + }, + "object.assign": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", + "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", + "dev": true, + "requires": { + "define-properties": "^1.1.2", + "function-bind": "^1.1.1", + "has-symbols": "^1.0.0", + "object-keys": "^1.0.11" + } + }, + "object.getownpropertydescriptors": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", + "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0-next.1" + } + }, + "object.values": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.1.tgz", + "integrity": "sha512-WTa54g2K8iu0kmS/us18jEmdv1a4Wi//BZ/DTVYEcH0XhLM5NYdpDHja3gt57VrZLcNAO2WGA+KpWsDBaHt6eA==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0-next.1", + "function-bind": "^1.1.1", + "has": "^1.0.3" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "onetime": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.0.tgz", + "integrity": "sha512-5NcSkPHhwTVFIQN+TUqXoS5+dlElHXdpAWu9I0HP20YOtIi+aZ0Ct82jdlILDxjLEAWwvm+qj1m6aEtsDVmm6Q==", + "dev": true, + "requires": { + "mimic-fn": "^2.1.0" + } + }, + "optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "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", + "word-wrap": "~1.2.3" + } + }, + "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=", + "dev": true + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true + }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, + "parse-json": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", + "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "dev": true, + "requires": { + "error-ex": "^1.2.0" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "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-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "dev": true + }, + "path-parse": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", + "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", + "dev": true + }, + "path-type": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz", + "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM=", + "dev": true, + "requires": { + "pify": "^2.0.0" + } + }, + "picomatch": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", + "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", + "dev": true + }, + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + }, + "pkg-dir": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-2.0.0.tgz", + "integrity": "sha1-9tXREJ4Z1j7fQo4L1X4Sd3YVM0s=", + "dev": true, + "requires": { + "find-up": "^2.1.0" + }, + "dependencies": { + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + } + }, + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "requires": { + "p-try": "^1.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true + } + } + }, + "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 + }, + "primal": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/primal/-/primal-1.1.1.tgz", + "integrity": "sha1-odhfVM6kTul22sZ0MgkR/+WuUic=", + "dev": true, + "requires": { + "bn.js": "^4.0.0", + "miller-rabin": "^4.0.0" + } + }, + "progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true + }, + "punycode": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", + "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", + "dev": true + }, + "read-pkg": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz", + "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg=", + "dev": true, + "requires": { + "load-json-file": "^2.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^2.0.0" + } + }, + "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=", + "dev": true, + "requires": { + "find-up": "^2.0.0", + "read-pkg": "^2.0.0" + }, + "dependencies": { + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + } + }, + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "requires": { + "p-try": "^1.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true + } + } + }, + "readdirp": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.2.0.tgz", + "integrity": "sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==", + "dev": true, + "requires": { + "picomatch": "^2.0.4" + } + }, + "regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "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=", + "dev": true + }, + "require-main-filename": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", + "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", + "dev": true + }, + "resolve": { + "version": "1.16.1", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.16.1.tgz", + "integrity": "sha512-rmAglCSqWWMrrBv/XM6sW0NuRFiKViw/W4d9EbC4pt+49H8JwHy+mcGmALTEg504AUDcLTvb1T2q3E9AnmY+ig==", + "dev": true, + "requires": { + "path-parse": "^1.0.6" + } + }, + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + }, + "restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "requires": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + } + }, + "run-async": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.0.tgz", + "integrity": "sha512-xJTbh/d7Lm7SBhc1tNvTpeCHaEzoyxPrqNlvSdMfBTYwaY++UJFyXUOxAtsRUXjlqOfj8luNaR9vjCh4KeV+pg==", + "dev": true, + "requires": { + "is-promise": "^2.1.0" + } + }, + "rxjs": { + "version": "6.5.5", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.5.tgz", + "integrity": "sha512-WfQI+1gohdf0Dai/Bbmk5L5ItH5tYqm3ki2c5GdWhKjalzjg93N3avFjVStyZZz+A2Em+ZxKH5bNghw9UeylGQ==", + "dev": true, + "requires": { + "tslib": "^1.9.0" + } + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "selfsigned.js": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/selfsigned.js/-/selfsigned.js-3.0.2.tgz", + "integrity": "sha1-gFm/LMqp85m3WmRTe/kzA6SKzo4=", + "dev": true, + "requires": { + "asn1.js": "^4.9.1", + "asn1.js-rfc5280": "^1.2.2", + "bn.js": "^4.0.0", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.0", + "primal": "^1.1.0" + }, + "dependencies": { + "asn1.js": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", + "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", + "dev": true, + "requires": { + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" + } + }, + "asn1.js-rfc5280": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/asn1.js-rfc5280/-/asn1.js-rfc5280-1.2.2.tgz", + "integrity": "sha1-U6DbzyvMyeWkaP9lE/JOzvY5r8I=", + "dev": true, + "requires": { + "asn1.js": "^4.5.0" + } + } + } + }, + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + }, + "set-blocking": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", + "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", + "dev": true + }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "dev": true, + "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=", + "dev": true + }, + "signal-exit": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", + "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", + "dev": true + }, + "simple-lru-cache": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/simple-lru-cache/-/simple-lru-cache-0.0.2.tgz", + "integrity": "sha1-1ZzDoZPBpdAyD4Tucy9uRxPlEd0=" + }, + "slice-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", + "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.0", + "astral-regex": "^1.0.0", + "is-fullwidth-code-point": "^2.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + } + } + }, + "spdx-correct": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.0.tgz", + "integrity": "sha512-lr2EZCctC2BNR7j7WzJ2FpDznxky1sjfxvvYEyzxNyb6lZXHODmEoJeFu4JupYlkfha1KZpJyoqiJ7pgA1qq8Q==", + "dev": true, + "requires": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-exceptions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.2.0.tgz", + "integrity": "sha512-2XQACfElKi9SlVb1CYadKDXvoajPgBVPn/gOQLrTvHdElaVhr7ZEbqJaRnJLVNeaI4cMEAgVCeBMKF6MWRDCRA==", + "dev": true + }, + "spdx-expression-parse": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz", + "integrity": "sha512-Yg6D3XpRD4kkOmTpdgbUiEJFKghJH03fiC1OPll5h/0sO6neh2jqRDVHOQ4o/LMea0tgCkbMgea5ip/e+MkWyg==", + "dev": true, + "requires": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-license-ids": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.5.tgz", + "integrity": "sha512-J+FWzZoynJEXGphVIS+XEh3kFSjZX/1i9gFBaWQcB+/tmpe2qUsSBABpcxqxnAxFdiUFEgAX1bjYGQvIZmoz9Q==", + "dev": true + }, + "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-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.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=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "string.prototype.trimend": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", + "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.5" + } + }, + "string.prototype.trimleft": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.2.tgz", + "integrity": "sha512-gCA0tza1JBvqr3bfAIFJGqfdRTyPae82+KTnm3coDXkZN9wnuW3HjGgN386D7hfv5CHQYCI022/rJPVlqXyHSw==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.5", + "string.prototype.trimstart": "^1.0.0" + } + }, + "string.prototype.trimright": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.2.tgz", + "integrity": "sha512-ZNRQ7sY3KroTaYjRS6EbNiiHrOkjihL9aQE/8gfQ4DtAC/aEBRHFJa44OmoWxGGqXuJlfKkZW4WcXErGr+9ZFg==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.5", + "string.prototype.trimend": "^1.0.0" + } + }, + "string.prototype.trimstart": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", + "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.5" + } + }, + "table": { + "version": "5.4.6", + "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", + "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", + "dev": true, + "requires": { + "ajv": "^6.10.2", + "lodash": "^4.17.14", + "slice-ansi": "^2.1.0", + "string-width": "^3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "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 + }, + "tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "requires": { + "os-tmpdir": "~1.0.2" + } + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "requires": { + "is-number": "^7.0.0" + } + }, + "tslib": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz", + "integrity": "sha512-aZW88SY8kQbU7gpV19lN24LtXh/yD4ZZg6qieAJDDg+YBsJcSmLGK9QpnUjAKVG/xefmvJGd1WUmfpT/g6AJGA==", + "dev": true + }, + "type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", + "dev": true, + "requires": { + "prelude-ls": "~1.1.2" + } + }, + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + }, + "uri-js": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", + "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", + "dev": true, + "requires": { + "punycode": "^2.1.0" + } + }, + "v8-compile-cache": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.0.tgz", + "integrity": "sha512-usZBT3PW+LOjM25wbqIlZwPeJV+3OSz3M1k1Ws8snlW39dZyYL9lOGC5FgPVHfk0jKmjiDV8Z0mIbVQPiwFs7g==", + "dev": true + }, + "validate-npm-package-license": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "dev": true, + "requires": { + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" + } + }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "which-module": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", + "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", + "dev": true + }, + "wide-align": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", + "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", + "dev": true, + "requires": { + "string-width": "^1.0.2 || 2" + } + }, + "word-wrap": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", + "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", + "dev": true + }, + "wrap-ansi": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", + "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", + "dev": true + }, + "write": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", + "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", + "dev": true, + "requires": { + "mkdirp": "^0.5.1" + } + }, + "y18n": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", + "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", + "dev": true + }, + "yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "requires": { + "cliui": "^5.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.1.2" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "requires": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + }, + "yargs-unparser": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.6.0.tgz", + "integrity": "sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==", + "dev": true, + "requires": { + "flat": "^4.1.0", + "lodash": "^4.17.15", + "yargs": "^13.3.0" + }, + "dependencies": { + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", + "dev": true + } + } + } + } +} diff --git a/package.json b/package.json index dc807a7..2d7e266 100644 --- a/package.json +++ b/package.json @@ -15,8 +15,8 @@ "author": "Fedor Indutny ", "main": "lib/ocsp.js", "scripts": { - "lint": "jshint lib/*.js lib/**/*.js && jscs lib/*.js lib/**/*.js test/*.js test/**/*.js", - "test": "mocha test/*-test.js" + "lint": "eslint lib test", + "test": "mocha" }, "dependencies": { "asn1.js": "^5.3.0", @@ -26,8 +26,12 @@ "simple-lru-cache": "0.0.2" }, "devDependencies": { - "jscs": "^3.0.7", - "jshint": "^2.11.0", + "eslint": "^6.8.0", + "eslint-config-standard": "^14.1.1", + "eslint-plugin-import": "^2.20.2", + "eslint-plugin-node": "^11.1.0", + "eslint-plugin-promise": "^4.2.1", + "eslint-plugin-standard": "^4.0.1", "mocha": "^7.1.1", "selfsigned.js": "^3.0.2" } diff --git a/test/.eslintrc.js b/test/.eslintrc.js new file mode 100644 index 0000000..8f4efbd --- /dev/null +++ b/test/.eslintrc.js @@ -0,0 +1,5 @@ +module.exports = { + env: { + mocha: true + } +} diff --git a/test/agent-test.js b/test/agent-test.js index b3922e0..e17252a 100644 --- a/test/agent-test.js +++ b/test/agent-test.js @@ -1,14 +1,11 @@ -var ocsp = require('../'); -var fixtures = require('./fixtures'); +var ocsp = require('../') +var https = require('https') -var assert = require('assert'); -var https = require('https'); - -describe('OCSP Agent', function() { - var a; - beforeEach(function() { - a = new ocsp.Agent(); - }); +describe('OCSP Agent', function () { + var a + beforeEach(function () { + a = new ocsp.Agent() + }) var websites = [ 'www.google.com', @@ -17,18 +14,18 @@ describe('OCSP Agent', function() { 'yahoo.com', 'nytimes.com', 'microsoft.com' - ]; + ] - websites.forEach(function(host) { - it('should connect and validate ' + host, function(cb) { - var req = https.get({ + websites.forEach(function (host) { + it('should connect and validate ' + host, function (cb) { + https.get({ host: host, port: 443, agent: a - }, function(res) { - res.resume(); - cb(); - }); - }); - }); -}); + }, function (res) { + res.resume() + cb() + }) + }) + }) +}) diff --git a/test/api-test.js b/test/api-test.js index 6ad4b82..700cbe0 100644 --- a/test/api-test.js +++ b/test/api-test.js @@ -1,66 +1,65 @@ -var ocsp = require('../'); -var fixtures = require('./fixtures'); +var ocsp = require('../') +var fixtures = require('./fixtures') -var assert = require('assert'); -var https = require('https'); +var assert = require('assert') +var https = require('https') -describe('OCSP Stapling Provider', function() { - describe('.check()', function() { - it('should validate google.com', function(cb) { +describe('OCSP Stapling Provider', function () { + describe('.check()', function () { + it('should validate google.com', function (cb) { ocsp.check({ cert: fixtures.google, issuer: fixtures.googleIssuer - }, function(err, res) { - if (err) - throw err; + }, function (err, res) { + if (err) { throw err } - assert.equal(res.type, 'good'); - cb(); - }); - }); - }); + assert.equal(res.type, 'good') + cb() + }) + }) + }) - describe('.verify()', function() { - it('should verify reddit.com\'s stapling', function(cb) { + describe('.verify()', function () { + it('should verify reddit.com\'s stapling', function (cb) { var req = https.request({ host: 'reddit.com', port: 443, requestOCSP: true - }, function(res) { + }, function (res) { // Should not be called - assert(false); - }); + assert(false) + }) - req.on('socket', function(socket) { - socket.on('OCSPResponse', function(stapling) { - onOCSPResponse(socket, stapling); - }); - }); + req.on('socket', function (socket) { + socket.on('OCSPResponse', function (stapling) { + onOCSPResponse(socket, stapling) + }) + }) - function onOCSPResponse(socket, stapling) { - var cert = socket.getPeerCertificate(true); + function onOCSPResponse (socket, stapling) { + var cert = socket.getPeerCertificate(true) - var req = ocsp.request.generate(cert.raw, cert.issuerCertificate.raw); + var req = ocsp.request.generate(cert.raw, cert.issuerCertificate.raw) ocsp.verify({ request: req, response: stapling - }, function(err, res) { - assert(!err); + }, function (err, res) { + assert(!err) - assert.equal(res.type, 'good'); - socket.destroy(); - cb(); - }); + assert.equal(res.type, 'good') + socket.destroy() + cb() + }) } - }); - }); + }) + }) - describe('.getOCSPURI()', function() { - it('should work on cert without extensions', function(cb) { - ocsp.getOCSPURI(fixtures.noExts, function(err) { - assert(err); - cb(); - }); - }); - }); -}); + describe('.getOCSPURI()', function () { + it('should work on cert without extensions', function (cb) { + ocsp.getOCSPURI(fixtures.noExts, function (err) { + assert(err) + cb() + }) + }) + }) +}) diff --git a/test/cache-test.js b/test/cache-test.js index ec48bbd..72bcc9d 100644 --- a/test/cache-test.js +++ b/test/cache-test.js @@ -1,76 +1,75 @@ -var ocsp = require('../'); -var fixtures = require('./fixtures'); +var ocsp = require('../') +var fixtures = require('./fixtures') -var assert = require('assert'); -var https = require('https'); +var https = require('https') -describe('OCSP Cache', function() { - var issuer = fixtures.certs.issuer; - var good = fixtures.certs.good; - var revoked = fixtures.certs.revoked; +describe('OCSP Cache', function () { + var issuer = fixtures.certs.issuer + var good = fixtures.certs.good - var server; - var agent; - beforeEach(function(cb) { + var server + var agent + var cache + + beforeEach(function (cb) { server = ocsp.Server.create({ cert: issuer.cert, key: issuer.key - }); + }) - server.addCert(43, 'good'); + server.addCert(43, 'good') server.addCert(44, 'revoked', { revocationTime: new Date(), revocationReason: 'CACompromise' - }); + }) - server.listen(8000, function() { - cb(); - }); + server.listen(8000, function () { + cb() + }) - agent = new ocsp.Agent(); + agent = new ocsp.Agent() - cache = new ocsp.Cache(); - }); + cache = new ocsp.Cache() + }) - afterEach(function(cb) { - server.close(cb); - agent = null; - }); + afterEach(function (cb) { + server.close(cb) + agent = null + }) - it('should cache ocsp response', function(cb) { + it('should cache ocsp response', function (cb) { var httpServer = https.createServer({ cert: good.cert + '\n' + good.issuer, key: good.key - }, function(req, res) { - res.end('hello world'); - }); + }, function (req, res) { + res.end('hello world') + }) - httpServer.on('OCSPRequest', function(cert, issuer, cb) { - ocsp.getOCSPURI(cert, function(err, uri) { - if (err) - return cb(err); + httpServer.on('OCSPRequest', function (cert, issuer, cb) { + ocsp.getOCSPURI(cert, function (err, uri) { + if (err) { return cb(err) } var req = ocsp.request.generate(cert, - issuer || fixtures.certs.issuer.cert); + issuer || fixtures.certs.issuer.cert) var options = { url: uri, ocsp: req.data - }; + } - cache.request(req.id, options, cb); - }); - }); + cache.request(req.id, options, cb) + }) + }) - httpServer.listen(8001, function() { + httpServer.listen(8001, function () { https.get({ agent: agent, ca: issuer.cert, rejectUnauthorized: !/^v0.12/.test(process.version), servername: 'local.host', port: 8001 - }, function(res) { - cb(); - }); - }); - }); -}); + }, function (res) { + cb() + }) + }) + }) +}) diff --git a/test/fixtures/gen-certs.js b/test/fixtures/gen-certs.js index a0ce534..e64e860 100755 --- a/test/fixtures/gen-certs.js +++ b/test/fixtures/gen-certs.js @@ -1,33 +1,35 @@ #!/usr/bin/env node -var fs = require('fs'); -var fixtures = require('../fixtures'); +var fs = require('fs') +var { resolve } = require('path') +var fixtures = require('../fixtures') var options = { serial: 42, commonName: 'mega.ca', size: 2048 -}; +} -fixtures.getOCSPCert(options, function(cert, key) { - fs.writeFileSync(__dirname + '/issuer-cert.pem', cert); - fs.writeFileSync(__dirname + '/issuer-key.pem', key); +fixtures.getOCSPCert(options, function (cert, key) { + fs.writeFileSync(resolve(__dirname, '/issuer-cert.pem', cert)) + fs.writeFileSync(resolve(__dirname, '/issuer-key.pem', key)) var options = { issuer: cert, issuerKey: key, serial: 43, size: 2048 - }; + } - fixtures.getOCSPCert(options, function(cert, key) { - fs.writeFileSync(__dirname + '/good-cert.pem', cert); - fs.writeFileSync(__dirname + '/good-key.pem', key); + fixtures.getOCSPCert(options, function (cert, key) { + fs.writeFileSync(resolve(__dirname, '/good-cert.pem', cert)) + fs.writeFileSync(resolve(__dirname, '/good-key.pem', key)) - options.serial++; - fixtures.getOCSPCert(options, function(cert, key) { - fs.writeFileSync(__dirname + '/revoked-cert.pem', cert); - fs.writeFileSync(__dirname + '/revoked-key.pem', key); - }); - }); -}); + options.serial++ + + fixtures.getOCSPCert(options, function (cert, key) { + fs.writeFileSync(resolve(__dirname, '/revoked-cert.pem', cert)) + fs.writeFileSync(resolve(__dirname, '/revoked-key.pem', key)) + }) + }) +}) diff --git a/test/fixtures/index.js b/test/fixtures/index.js index c3ed295..8ee08d7 100644 --- a/test/fixtures/index.js +++ b/test/fixtures/index.js @@ -1,9 +1,10 @@ -var ocsp = require('../../'); +var ocsp = require('../../') -var fs = require('fs'); -var rfc2560 = require('asn1.js-rfc2560'); -var rfc5280 = require('asn1.js-rfc5280'); -var keyGen = require('selfsigned.js').create(); +var fs = require('fs') +var { resolve } = require('path') +var rfc2560 = require('asn1.js-rfc2560') +var rfc5280 = require('asn1.js-rfc5280') +var keyGen = require('selfsigned.js').create() /* AuthorityInfoAccessSyntax ::= @@ -14,97 +15,87 @@ var keyGen = require('selfsigned.js').create(); accessLocation GeneralName } */ -exports.google = fs.readFileSync(__dirname + '/google-cert.pem'); -exports.googleIssuer = fs.readFileSync(__dirname + '/google-issuer.pem'); -exports.noExts = fs.readFileSync(__dirname + '/no-exts-cert.pem'); +exports.google = fs.readFileSync(resolve(__dirname, './google-cert.pem')) +exports.googleIssuer = fs.readFileSync(resolve(__dirname, './google-issuer.pem')) +exports.noExts = fs.readFileSync(resolve(__dirname, './no-exts-cert.pem')) exports.certs = {}; -[ 'issuer', 'good', 'revoked' ].forEach(function(name) { +['issuer', 'good', 'revoked'].forEach(function (name) { exports.certs[name] = { - cert: fs.readFileSync(__dirname + '/' + name + '-cert.pem'), - key: fs.readFileSync(__dirname + '/' + name + '-key.pem') - }; -}); - -exports.getOCSPCert = function getOCSPCert(options, cb) { - if (!options) - options = {}; - - var size = options.size || 256; - var commonName = options.commonName || 'local.host'; - var OCSPEndPoint = options.OCSPEndPoint || 'http://127.0.0.1:8000/ocsp'; - - var issuer = options.issuer; - if (issuer) - issuer = ocsp.utils.toDER(issuer, 'CERTIFICATE'); - if (issuer) - issuer = rfc5280.Certificate.decode(issuer, 'der'); - - var issuerKeyData = options.issuerKey; - - if (issuerKeyData) - issuerKeyData = ocsp.utils.toDER(options.issuerKey, 'RSA PRIVATE KEY'); - - if (issuerKeyData) - issuerKeyData = ocsp.utils.RSAPrivateKey.decode(issuerKeyData, 'der'); - else - issuerKeyData = options.issuerKeyData; - - function getPrime(cb) { - keyGen.getPrime(size >> 1, function(err, prime) { - if (err) - return getPrime(cb); - - cb(prime); - }); + cert: fs.readFileSync(resolve(__dirname, './' + name + '-cert.pem')), + key: fs.readFileSync(resolve(__dirname, './' + name + '-key.pem')) } +}) - function getTwoPrimes(cb) { - var primes = []; - getPrime(done); - getPrime(done); +exports.getOCSPCert = function getOCSPCert (options, cb) { + if (!options) { options = {} } - function done(prime) { - primes.push(prime); - if (primes.length === 2) - return cb(primes[0], primes[1]); + var size = options.size || 256 + var commonName = options.commonName || 'local.host' + var OCSPEndPoint = options.OCSPEndPoint || 'http://127.0.0.1:8000/ocsp' + + var issuer = options.issuer + if (issuer) { issuer = ocsp.utils.toDER(issuer, 'CERTIFICATE') } + if (issuer) { issuer = rfc5280.Certificate.decode(issuer, 'der') } + + var issuerKeyData = options.issuerKey + + if (issuerKeyData) { issuerKeyData = ocsp.utils.toDER(options.issuerKey, 'RSA PRIVATE KEY') } + + if (issuerKeyData) { issuerKeyData = ocsp.utils.RSAPrivateKey.decode(issuerKeyData, 'der') } else { issuerKeyData = options.issuerKeyData } + + function getPrime (cb) { + keyGen.getPrime(size >> 1, function (err, prime) { + if (err) { return getPrime(cb) } + + cb(prime) + }) + } + + function getTwoPrimes (cb) { + var primes = [] + getPrime(done) + getPrime(done) + + function done (prime) { + primes.push(prime) + if (primes.length === 2) { return cb(primes[0], primes[1]) } } } - function getKeyData(cb) { - getTwoPrimes(function(p, q) { - var keyData = keyGen.getKeyData(p, q); - if (!keyData) - return getKeyData(cb); + function getKeyData (cb) { + getTwoPrimes(function (p, q) { + var keyData = keyGen.getKeyData(p, q) + if (!keyData) { return getKeyData(cb) } - cb(keyData); - }); + cb(keyData) + }) } - var ext = rfc5280.AuthorityInfoAccessSyntax.encode([ { + var ext = rfc5280.AuthorityInfoAccessSyntax.encode([{ accessMethod: rfc2560['id-pkix-ocsp'], accessLocation: { type: 'uniformResourceIdentifier', value: OCSPEndPoint } - } ], 'der'); + }], 'der') - getKeyData(function(keyData) { + getKeyData(function (keyData) { var certData = keyGen.getCertData({ serial: options.serial, keyData: keyData, commonName: commonName, issuer: issuer, issuerKeyData: issuerKeyData, - extensions: [ { + extensions: [{ extnID: rfc5280['id-pe-authorityInfoAccess'], critical: false, extnValue: ext - } ] - }); + }] + }) - var pem = keyGen.getCert(certData, 'pem'); - return cb(pem, keyGen.getPrivate(keyData, 'pem')); - }); -}; + var pem = keyGen.getCert(certData, 'pem') + return cb(pem, keyGen.getPrivate(keyData, 'pem')) + }) +} diff --git a/test/server-test.js b/test/server-test.js index 13cf10f..c4b185e 100644 --- a/test/server-test.js +++ b/test/server-test.js @@ -1,48 +1,47 @@ -var ocsp = require('../'); -var fixtures = require('./fixtures'); +var ocsp = require('../') +var fixtures = require('./fixtures') -var assert = require('assert'); +var assert = require('assert') -describe('OCSP Server', function() { - var issuer = fixtures.certs.issuer; - var good = fixtures.certs.good; - var revoked = fixtures.certs.revoked; +describe('OCSP Server', function () { + var issuer = fixtures.certs.issuer + var good = fixtures.certs.good + var revoked = fixtures.certs.revoked - it('should provide ocsp response to the client', function(cb) { + it('should provide ocsp response to the client', function (cb) { var server = ocsp.Server.create({ cert: issuer.cert, key: issuer.key - }); + }) - server.addCert(43, 'good'); + server.addCert(43, 'good') server.addCert(44, 'revoked', { revocationTime: new Date(), revocationReason: 'cACompromise' - }); + }) - server.listen(8000, function() { + server.listen(8000, function () { ocsp.check({ cert: good.cert, issuer: issuer.cert - }, function(err, res) { - if (err) - throw err; + }, function (err, res) { + if (err) { throw err } - assert.equal(res.type, 'good'); + assert.equal(res.type, 'good') - next(); - }); - }); + next() + }) + }) - function next() { + function next () { ocsp.check({ cert: revoked.cert, issuer: issuer.cert - }, function(err, res) { - assert(err); - assert.equal(res.type, 'revoked'); - cb(); - }); + }, function (err, res) { + assert(err) + assert.equal(res.type, 'revoked') + cb() + }) } - }); -}); + }) +})