diff --git a/.gitignore b/.gitignore index f0c275a..8a5ef0d 100644 --- a/.gitignore +++ b/.gitignore @@ -2,3 +2,4 @@ coverage .nyc_output node_modules +.vscode/ diff --git a/accumulative.js b/accumulative.js index 42fd28f..3fc0537 100644 --- a/accumulative.js +++ b/accumulative.js @@ -1,38 +1,48 @@ var utils = require('./utils') +var ext = require('./bn-extensions') // add inputs until we reach or surpass the target value (or deplete) // worst-case: O(n) module.exports = function accumulative (utxos, outputs, feeRate) { - if (!isFinite(utils.uintOrNaN(feeRate))) return {} + if (!isFinite(utils.bnOrNaN(feeRate))) return {} var bytesAccum = utils.transactionBytes([], outputs) - var inAccum = 0 + var inAccum = ext.BN_ZERO var inputs = [] var outAccum = utils.sumOrNaN(outputs) for (var i = 0; i < utxos.length; ++i) { var utxo = utxos[i] var utxoBytes = utils.inputBytes(utxo) - var utxoFee = feeRate * utxoBytes - var utxoValue = utils.uintOrNaN(utxo.value) + var utxoFee = ext.mul(feeRate, utxoBytes) + var utxoValue = utils.bnOrNaN(utxo.value) // skip detrimental input - if (utxoFee > utxo.value) { - if (i === utxos.length - 1) return { fee: feeRate * (bytesAccum + utxoBytes) } + var feeIsMoreThanValue = ext.gt(utxoFee, utxoValue) + // utxoFee > utxoValue + if (feeIsMoreThanValue) { + if (i === utxos.length - 1) { + var bytesSum = ext.add(bytesAccum, utxoBytes) + return { + fee: ext.mul(feeRate, bytesSum) + } + } continue } - bytesAccum += utxoBytes - inAccum += utxoValue + bytesAccum = ext.add(bytesAccum, utxoBytes) + inAccum = ext.add(inAccum, utxoValue) inputs.push(utxo) - var fee = feeRate * bytesAccum + var fee = ext.mul(feeRate, bytesAccum) // go again? - if (inAccum < outAccum + fee) continue + if (ext.lt(inAccum, ext.add(outAccum, fee))) continue return utils.finalize(inputs, outputs, feeRate) } - return { fee: feeRate * bytesAccum } + return { + fee: feeRate.mul(bytesAccum) + } } diff --git a/blackjack.js b/blackjack.js index 316568b..60fdca2 100644 --- a/blackjack.js +++ b/blackjack.js @@ -1,13 +1,14 @@ var utils = require('./utils') +var ext = require('./bn-extensions') // only add inputs if they don't bust the target value (aka, exact match) // worst-case: O(n) module.exports = function blackjack (utxos, outputs, feeRate) { - if (!isFinite(utils.uintOrNaN(feeRate))) return {} + if (!isFinite(utils.bnOrNaN(feeRate))) return {} var bytesAccum = utils.transactionBytes([], outputs) - var inAccum = 0 + var inAccum = ext.BN_ZERO var inputs = [] var outAccum = utils.sumOrNaN(outputs) var threshold = utils.dustThreshold({}, feeRate) @@ -15,21 +16,27 @@ module.exports = function blackjack (utxos, outputs, feeRate) { for (var i = 0; i < utxos.length; ++i) { var input = utxos[i] var inputBytes = utils.inputBytes(input) - var fee = feeRate * (bytesAccum + inputBytes) - var inputValue = utils.uintOrNaN(input.value) + var fee = ext.mul(feeRate, ext.add(bytesAccum, inputBytes)) + var inputValue = utils.bnOrNaN(input.value) // would it waste value? - if ((inAccum + inputValue) > (outAccum + fee + threshold)) continue + var totalInputs = ext.add(inAccum, inputValue) + var outputsAndFee = ext.add(outAccum, fee) + var totalOutputs = ext.add(outputsAndFee, threshold) + var inputsAreGreaterThanOutputs = ext.gt(totalInputs, totalOutputs) - bytesAccum += inputBytes - inAccum += inputValue + if (inputsAreGreaterThanOutputs) continue + + bytesAccum = ext.add(bytesAccum, inputBytes) + inAccum = ext.add(inAccum, inputValue) inputs.push(input) // go again? - if (inAccum < outAccum + fee) continue + if (ext.lt(inAccum, outputsAndFee)) continue return utils.finalize(inputs, outputs, feeRate) } - - return { fee: feeRate * bytesAccum } + return { + fee: feeRate.mul(bytesAccum) + } } diff --git a/bn-extensions.js b/bn-extensions.js new file mode 100644 index 0000000..83fb4d7 --- /dev/null +++ b/bn-extensions.js @@ -0,0 +1,74 @@ +var BN = require('bn.js') + +var BN_ZERO = new BN(0) +var BN_ONE = new BN(1) + +function mul (multiplicand, multiplier) { + if (!BN.isBN(multiplicand) || !BN.isBN(multiplier)) return NaN + + return multiplicand.mul(multiplier) +} + +function div (dividend, divisor) { + if (!BN.isBN(dividend) || !BN.isBN(divisor)) return NaN + if (divisor.cmp(BN_ZERO) === 0) return Infinity + + return dividend.div(divisor) +} + +function add (arg1, arg2, arg3) { + // Add two items + if (!BN.isBN(arg1) || !BN.isBN(arg2)) return NaN + if (typeof arg3 === 'undefined') return arg1.add(arg2) + + // Add three items + if (!BN.isBN(arg3)) return NaN + return arg1.add(arg2).add(arg3) +} + +function sub (arg1, arg2, arg3) { + // Subtract two items + if (!BN.isBN(arg1) || !BN.isBN(arg2)) return NaN + if (typeof arg3 === 'undefined') return arg1.sub(arg2) + + // Subtract three items + if (!BN.isBN(arg3)) return NaN + return arg1.sub(arg2).sub(arg3) +} + +function shrn (argument, shiftBy) { + if (!BN.isBN(argument)) return NaN + if (BN.isBN(shiftBy)) shiftBy = shiftBy.toNumber() + if (typeof shiftBy !== 'number') return NaN + + return argument.shrn(shiftBy) +} + +function isZero (argument) { + if (!BN.isBN(argument)) return false + if (argument.cmp(BN_ZERO) === 0) return true + return false +} + +function lt (subject, argument) { + if (!BN.isBN(argument) || !BN.isBN(subject)) return false + return subject.lt(argument) +} + +function gt (subject, argument) { + if (!BN.isBN(argument) || !BN.isBN(subject)) return false + return subject.gt(argument) +} + +module.exports = { + mul: mul, + div: div, + add: add, + sub: sub, + shrn: shrn, + isZero: isZero, + lt: lt, + gt: gt, + BN_ZERO: BN_ZERO, + BN_ONE: BN_ONE +} diff --git a/break.js b/break.js index 91dd72c..f910bf9 100644 --- a/break.js +++ b/break.js @@ -1,32 +1,39 @@ var utils = require('./utils') +var ext = require('./bn-extensions') // break utxos into the maximum number of 'output' possible module.exports = function broken (utxos, output, feeRate) { - if (!isFinite(utils.uintOrNaN(feeRate))) return {} + if (!isFinite(utils.bnOrNaN(feeRate))) return {} var bytesAccum = utils.transactionBytes(utxos, []) - var value = utils.uintOrNaN(output.value) + var value = utils.bnOrNaN(output.value) var inAccum = utils.sumOrNaN(utxos) + if (!isFinite(value) || - !isFinite(inAccum)) return { fee: feeRate * bytesAccum } + !isFinite(inAccum)) return { fee: feeRate.mul(bytesAccum) } var outputBytes = utils.outputBytes(output) - var outAccum = 0 + var outAccum = ext.BN_ZERO var outputs = [] while (true) { - var fee = feeRate * (bytesAccum + outputBytes) + // feeRate * (bytesAccum + outputBytes) + var fee = ext.mul(feeRate, ext.add(bytesAccum, outputBytes)) // did we bust? - if (inAccum < (outAccum + fee + value)) { + if (ext.lt(inAccum, ext.add(outAccum, fee, value))) { + var isZero = ext.isZero(outAccum) // premature? - if (outAccum === 0) return { fee: fee } - + if (isZero) { + return { + fee: fee + } + } break } - bytesAccum += outputBytes - outAccum += value + bytesAccum = ext.add(bytesAccum, outputBytes) + outAccum = ext.add(outAccum, value) outputs.push(output) } diff --git a/index.js b/index.js index 19aa484..dd096dd 100644 --- a/index.js +++ b/index.js @@ -4,12 +4,12 @@ var utils = require('./utils') // order by descending value, minus the inputs approximate fee function utxoScore (x, feeRate) { - return x.value - (feeRate * utils.inputBytes(x)) + return x.value.sub((feeRate.mul(utils.inputBytes(x)))) } module.exports = function coinSelect (utxos, outputs, feeRate) { utxos = utxos.concat().sort(function (a, b) { - return utxoScore(b, feeRate) - utxoScore(a, feeRate) + return utxoScore(b, feeRate).sub(utxoScore(a, feeRate)) }) // attempt to use the blackjack strategy first (no change output) diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..a009382 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,3108 @@ +{ + "name": "coinselect", + "version": "3.1.11", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "acorn": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.1.1.tgz", + "integrity": "sha512-jPTiwtOxaHNaAPg/dmrJ/beuzLRnXtB0kQPQ8JpotKJgTB6rX6c8mlf315941pyjBSaPg8NHXS9fhP4u17DpGA==", + "dev": true + }, + "acorn-jsx": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.0.1.tgz", + "integrity": "sha512-HJ7CfNHrfJLlNTzIEUTj43LNWGkqpRLxm3YjAlcD0ACydk9XynzYsCBHxut+iqt+1aBXkx9UP/w/ZqMr13XIzg==", + "dev": true + }, + "ajv": { + "version": "6.10.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.0.tgz", + "integrity": "sha512-nffhOpkymDECQyR0mnsUtoCE8RlX38G0rYP+wgLWFyZuUyuuojSSvi/+euOiQBIn63whYwYVIIH1TvE3tu4OEg==", + "dev": true, + "requires": { + "fast-deep-equal": "^2.0.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ajv-keywords": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.4.0.tgz", + "integrity": "sha512-aUjdRFISbuFOl0EIZc+9e4FfZp0bDZgAdOOf30bJmw8VM9v84SHyVyxDfbWxpGYbdZD/9XoKxfHVNmxPkhwyGw==", + "dev": true + }, + "ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true + }, + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "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.0.3", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.0.3.tgz", + "integrity": "sha1-GEtI9i2S10UrsxsyMWXH+L0CJm0=", + "dev": true, + "requires": { + "define-properties": "^1.1.2", + "es-abstract": "^1.7.0" + } + }, + "babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", + "dev": true, + "requires": { + "chalk": "^1.1.3", + "esutils": "^2.0.2", + "js-tokens": "^3.0.2" + }, + "dependencies": { + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", + "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" + } + }, + "caller-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", + "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", + "dev": true, + "requires": { + "callsites": "^0.2.0" + } + }, + "callsites": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", + "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=", + "dev": true + }, + "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" + }, + "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" + } + }, + "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" + } + } + } + }, + "chardet": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz", + "integrity": "sha1-tUc7M9yXxCTl2Y3IfVXU2KKci/I=", + "dev": true + }, + "circular-json": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", + "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", + "dev": true + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "cli-width": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz", + "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=", + "dev": true + }, + "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" + } + }, + "debug-log": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/debug-log/-/debug-log-1.0.1.tgz", + "integrity": "sha1-IwdjLUwEOCuN+KMvcLiVBG1SdF8=", + "dev": true + }, + "deep-equal": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-1.0.1.tgz", + "integrity": "sha1-9dJgKStmDghO/0zbyfCK0yR0SLU=", + "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" + } + }, + "defined": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/defined/-/defined-1.0.0.tgz", + "integrity": "sha1-yY2bzvdWdBiOEQlpFRGZ45sfppM=", + "dev": true + }, + "deglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/deglob/-/deglob-2.1.1.tgz", + "integrity": "sha512-2kjwuGGonL7gWE1XU4Fv79+vVzpoQCl0V+boMwWtOQJV2AGDabCwez++nB1Nli/8BabAfZQ/UuHPlp6AymKdWw==", + "dev": true, + "requires": { + "find-root": "^1.0.0", + "glob": "^7.0.5", + "ignore": "^3.0.9", + "pkg-config": "^1.1.0", + "run-parallel": "^1.1.2", + "uniq": "^1.0.1" + }, + "dependencies": { + "ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + } + } + }, + "doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "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.13.0", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.13.0.tgz", + "integrity": "sha512-vDZfg/ykNxQVwup/8E1BZhVzFfBxs9NqMzGcvIJrqg5k2/5Za2bWo40dK2J1pgLngZ7c+Shh8lwYtLGyrwPutg==", + "dev": true, + "requires": { + "es-to-primitive": "^1.2.0", + "function-bind": "^1.1.1", + "has": "^1.0.3", + "is-callable": "^1.1.4", + "is-regex": "^1.0.4", + "object-keys": "^1.0.12" + } + }, + "es-to-primitive": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.0.tgz", + "integrity": "sha512-qZryBOJjV//LaxLTV6UC//WewneB3LcXOL9NP++ozKVXsIIIpm/2c13UDiD9Jp2eThsecw9m3jPqDwTyobcdbg==", + "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": "5.4.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-5.4.0.tgz", + "integrity": "sha512-UIpL91XGex3qtL6qwyCQJar2j3osKxK9e3ano3OcGEIRM4oWIpCkDg9x95AXEC2wMs7PnxzOkPZ2gq+tsMS9yg==", + "dev": true, + "requires": { + "ajv": "^6.5.0", + "babel-code-frame": "^6.26.0", + "chalk": "^2.1.0", + "cross-spawn": "^6.0.5", + "debug": "^3.1.0", + "doctrine": "^2.1.0", + "eslint-scope": "^4.0.0", + "eslint-utils": "^1.3.1", + "eslint-visitor-keys": "^1.0.0", + "espree": "^4.0.0", + "esquery": "^1.0.1", + "esutils": "^2.0.2", + "file-entry-cache": "^2.0.0", + "functional-red-black-tree": "^1.0.1", + "glob": "^7.1.2", + "globals": "^11.7.0", + "ignore": "^4.0.2", + "imurmurhash": "^0.1.4", + "inquirer": "^5.2.0", + "is-resolvable": "^1.1.0", + "js-yaml": "^3.11.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.5", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.2", + "path-is-inside": "^1.0.2", + "pluralize": "^7.0.0", + "progress": "^2.0.0", + "regexpp": "^2.0.0", + "require-uncached": "^1.0.3", + "semver": "^5.5.0", + "strip-ansi": "^4.0.0", + "strip-json-comments": "^2.0.1", + "table": "^4.0.3", + "text-table": "^0.2.0" + } + }, + "eslint-config-standard": { + "version": "12.0.0", + "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-12.0.0.tgz", + "integrity": "sha512-COUz8FnXhqFitYj4DTqHzidjIL/t4mumGZto5c7DrBpvWoie+Sn3P4sLEzUGeYhRElWuFEf8K1S1EfvD1vixCQ==", + "dev": true + }, + "eslint-config-standard-jsx": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/eslint-config-standard-jsx/-/eslint-config-standard-jsx-6.0.2.tgz", + "integrity": "sha512-D+YWAoXw+2GIdbMBRAzWwr1ZtvnSf4n4yL0gKGg7ShUOGXkSOLerI17K4F6LdQMJPNMoWYqepzQD/fKY+tXNSg==", + "dev": true + }, + "eslint-import-resolver-node": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.2.tgz", + "integrity": "sha512-sfmTqJfPSizWu4aymbPr4Iidp5yKm8yDkHp+Ir3YiTHiiDfxh69mOUsmiqW6RZ9zRXFaF64GtYmN7e+8GHBv6Q==", + "dev": true, + "requires": { + "debug": "^2.6.9", + "resolve": "^1.5.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-module-utils": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.3.0.tgz", + "integrity": "sha512-lmDJgeOOjk8hObTysjqH7wyMi+nsHwwvfBykwfhjR1LNdd7C2uFJBvx4OpWYpXOw4df1yE1cDEVd1yLHitk34w==", + "dev": true, + "requires": { + "debug": "^2.6.8", + "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": "1.4.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-es/-/eslint-plugin-es-1.4.0.tgz", + "integrity": "sha512-XfFmgFdIUDgvaRAlaXUkxrRg5JSADoRC8IkKLc/cISeR3yHVMefFHQZpcyXXEUUPHfy5DwviBcrfqlyqEwlQVw==", + "dev": true, + "requires": { + "eslint-utils": "^1.3.0", + "regexpp": "^2.0.1" + } + }, + "eslint-plugin-import": { + "version": "2.14.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.14.0.tgz", + "integrity": "sha512-FpuRtniD/AY6sXByma2Wr0TXvXJ4nA/2/04VPlfpmUDPOpOY264x+ILiwnrk/k4RINgDAyFZByxqPUbSQ5YE7g==", + "dev": true, + "requires": { + "contains-path": "^0.1.0", + "debug": "^2.6.8", + "doctrine": "1.5.0", + "eslint-import-resolver-node": "^0.3.1", + "eslint-module-utils": "^2.2.0", + "has": "^1.0.1", + "lodash": "^4.17.4", + "minimatch": "^3.0.3", + "read-pkg-up": "^2.0.0", + "resolve": "^1.6.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" + } + }, + "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": "7.0.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-node/-/eslint-plugin-node-7.0.1.tgz", + "integrity": "sha512-lfVw3TEqThwq0j2Ba/Ckn2ABdwmL5dkOgAux1rvOk6CO7A6yGyPI2+zIxN6FyNkp1X1X/BSvKOceD6mBWSj4Yw==", + "dev": true, + "requires": { + "eslint-plugin-es": "^1.3.1", + "eslint-utils": "^1.3.1", + "ignore": "^4.0.2", + "minimatch": "^3.0.4", + "resolve": "^1.8.1", + "semver": "^5.5.0" + } + }, + "eslint-plugin-promise": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-4.0.1.tgz", + "integrity": "sha512-Si16O0+Hqz1gDHsys6RtFRrW7cCTB6P7p3OJmKp3Y3dxpQE2qwOA7d3xnV+0mBmrPoi0RBnxlCKvqu70te6wjg==", + "dev": true + }, + "eslint-plugin-react": { + "version": "7.11.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.11.1.tgz", + "integrity": "sha512-cVVyMadRyW7qsIUh3FHp3u6QHNhOgVrLQYdQEB1bPWBsgbNCHdFAeNMquBMCcZJu59eNthX053L70l7gRt4SCw==", + "dev": true, + "requires": { + "array-includes": "^3.0.3", + "doctrine": "^2.1.0", + "has": "^1.0.3", + "jsx-ast-utils": "^2.0.1", + "prop-types": "^15.6.2" + } + }, + "eslint-plugin-standard": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-standard/-/eslint-plugin-standard-4.0.0.tgz", + "integrity": "sha512-OwxJkR6TQiYMmt1EsNRMe5qG3GsbjlcOhbGUBY4LtavF9DsLaTcoR+j2Tdjqi23oUwKNUqX7qcn5fPStafMdlA==", + "dev": true + }, + "eslint-scope": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.3.tgz", + "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", + "dev": true, + "requires": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + } + }, + "eslint-utils": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.3.1.tgz", + "integrity": "sha512-Z7YjnIldX+2XMcjr7ZkgEsOj/bREONV60qYeB/bjMAqqqZ4zxKyWX+BOUkdmRmA9riiIPVvo5x86m5elviOk0Q==", + "dev": true + }, + "eslint-visitor-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", + "integrity": "sha512-qzm/XxIbxm/FHyH341ZrbnMUpe+5Bocte9xkmFMzPMjRaZMcXww+MpBptFvtU+79L362nqiLhekCxCxDPaUMBQ==", + "dev": true + }, + "espree": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-4.1.0.tgz", + "integrity": "sha512-I5BycZW6FCVIub93TeVY1s7vjhP9CY6cXCznIRfiig7nRviKZYdRnj/sHEWC6A7WE9RDWOFq9+7OsWSYz8qv2w==", + "dev": true, + "requires": { + "acorn": "^6.0.2", + "acorn-jsx": "^5.0.0", + "eslint-visitor-keys": "^1.0.0" + } + }, + "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 + }, + "esquery": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.1.tgz", + "integrity": "sha512-SmiyZ5zIWH9VM+SRUReLS5Q8a7GxtRdxEBVZpm98rJM7Sb+A9DVCndXfkeFUd3byderg+EbDkfnevfCwynWaNA==", + "dev": true, + "requires": { + "estraverse": "^4.0.0" + } + }, + "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.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz", + "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=", + "dev": true + }, + "esutils": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", + "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", + "dev": true + }, + "external-editor": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-2.2.0.tgz", + "integrity": "sha512-bSn6gvGxKt+b7+6TKEv1ZycHleA7aHhRHyAqJyp5pbUFuYYNIzpZnQDk7AsYckyWdEnTeAnay0aCy2aV6iTk9A==", + "dev": true, + "requires": { + "chardet": "^0.4.0", + "iconv-lite": "^0.4.17", + "tmp": "^0.0.33" + } + }, + "fast-deep-equal": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", + "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", + "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=", + "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": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "file-entry-cache": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", + "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=", + "dev": true, + "requires": { + "flat-cache": "^1.2.1", + "object-assign": "^4.0.1" + } + }, + "find-root": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/find-root/-/find-root-1.1.0.tgz", + "integrity": "sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==", + "dev": true + }, + "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" + } + }, + "flat-cache": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.4.tgz", + "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", + "dev": true, + "requires": { + "circular-json": "^0.3.1", + "graceful-fs": "^4.1.2", + "rimraf": "~2.6.2", + "write": "^0.2.1" + } + }, + "for-each": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", + "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", + "dev": true, + "requires": { + "is-callable": "^1.1.3" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "function-bind": { + "version": "1.1.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-stdin": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-6.0.0.tgz", + "integrity": "sha512-jp4tHawyV7+fkkSKyvjuLZswblUtz+SQKzSWnBbii16BuZksJlU1wuBYXY75r+duh/llF1ur6oNwi+2ZzjKZ7g==", + "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" + } + }, + "globals": { + "version": "11.11.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.11.0.tgz", + "integrity": "sha512-WHq43gS+6ufNOEqlrDBxVEbb8ntfXrfAUU2ZOpCxrBdGKW3gyv8mCxAfIBD0DroPKGrJ2eSsXsLtY9MPntsyTw==", + "dev": true + }, + "graceful-fs": { + "version": "4.1.15", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.15.tgz", + "integrity": "sha512-6uHUhOPEBgQ24HM+r6b/QwWfZq+yiFcipKFrOFiBEnWdy5sdzYoi+pJeQaPI5qOLRFqWmAXUPQNsielzdLoecA==", + "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-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.0.tgz", + "integrity": "sha1-uhqPGvKg/DllD1yFA2dwQSIGO0Q=", + "dev": true + }, + "hosted-git-info": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.7.1.tgz", + "integrity": "sha512-7T/BxH19zbcCTa8XkMlbK5lTo1WtgkFi3GvdWEyNuc4Vex7/9Dqbnpsf4JMydcfj9HCg4zUWFTL3Za6lapg5/w==", + "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 + }, + "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.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "inquirer": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-5.2.0.tgz", + "integrity": "sha512-E9BmnJbAKLPGonz0HeWHtbKf+EeSP93paWO3ZYoUpq/aowXvYGjjCSuashhXPpzbArIjBbji39THkxTz9ZeEUQ==", + "dev": true, + "requires": { + "ansi-escapes": "^3.0.0", + "chalk": "^2.0.0", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^2.1.0", + "figures": "^2.0.0", + "lodash": "^4.3.0", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rxjs": "^5.5.2", + "string-width": "^2.1.0", + "strip-ansi": "^4.0.0", + "through": "^2.3.6" + } + }, + "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-callable": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.4.tgz", + "integrity": "sha512-r5p9sxJjYnArLjObpjA4xu5EKI3CuKHkJXMhT7kwbpUyIFD1n5PMAsoPvWnvtZiNz7LjkYDRZhd7FlI0eMijEA==", + "dev": true + }, + "is-date-object": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.1.tgz", + "integrity": "sha1-mqIOtq7rv/d/vTPnTKAbM1gdOhY=", + "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-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.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.4.tgz", + "integrity": "sha1-VRdIm1RwkbCTDglWVM7SXul+lJE=", + "dev": true, + "requires": { + "has": "^1.0.1" + } + }, + "is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", + "dev": true + }, + "is-symbol": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.2.tgz", + "integrity": "sha512-HS8bZ9ox60yCJLH9snBpIwv9pYUAkcuLhSA1oero1UB5y9aiQpRA8y2ex945AOtCZL1lJDeIk3G5LthswI46Lw==", + "dev": true, + "requires": { + "has-symbols": "^1.0.0" + } + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, + "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": "3.0.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", + "dev": true + }, + "js-yaml": { + "version": "3.12.2", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.2.tgz", + "integrity": "sha512-QHn/Lh/7HhZ/Twc7vJYQTkjuCa0kaCcDcjK5Zlk2rvnUpy7DxMJ23+Jc2dcyvltwQVg1nygAVlB2oRDFHoRS5Q==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", + "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 + }, + "jsx-ast-utils": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-2.0.1.tgz", + "integrity": "sha1-6AGxs5mF4g//yHtA43SAgOLcrH8=", + "dev": true, + "requires": { + "array-includes": "^3.0.3" + } + }, + "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" + } + }, + "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" + } + }, + "lodash": { + "version": "4.17.11", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", + "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==", + "dev": true + }, + "loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "dev": true, + "requires": { + "js-tokens": "^3.0.0 || ^4.0.0" + } + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "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": "0.0.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", + "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", + "dev": true + }, + "mkdirp": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", + "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", + "dev": true, + "requires": { + "minimist": "0.0.8" + } + }, + "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.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", + "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 + }, + "nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", + "dev": true + }, + "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" + } + }, + "nyc": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/nyc/-/nyc-8.4.0.tgz", + "integrity": "sha1-ZgNxyAfK7wQn+5sJSPdBgGJOpuQ=", + "dev": true, + "requires": { + "archy": "^1.0.0", + "arrify": "^1.0.1", + "caching-transform": "^1.0.0", + "convert-source-map": "^1.3.0", + "default-require-extensions": "^1.0.0", + "find-cache-dir": "^0.1.1", + "find-up": "^1.1.2", + "foreground-child": "^1.5.3", + "glob": "^7.0.6", + "istanbul-lib-coverage": "^1.0.0", + "istanbul-lib-hook": "^1.0.0-alpha.4", + "istanbul-lib-instrument": "^1.2.0", + "istanbul-lib-report": "^1.0.0-alpha.3", + "istanbul-lib-source-maps": "^1.0.2", + "istanbul-reports": "^1.0.0", + "md5-hex": "^1.2.0", + "micromatch": "^2.3.11", + "mkdirp": "^0.5.0", + "resolve-from": "^2.0.0", + "rimraf": "^2.5.4", + "signal-exit": "^3.0.1", + "spawn-wrap": "^1.2.4", + "test-exclude": "^2.1.3", + "yargs": "^6.0.0", + "yargs-parser": "^4.0.2" + }, + "dependencies": { + "align-text": { + "version": "0.1.4", + "bundled": true, + "dev": true, + "requires": { + "kind-of": "^3.0.2", + "longest": "^1.0.1", + "repeat-string": "^1.5.2" + } + }, + "amdefine": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "ansi-regex": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "bundled": true, + "dev": true + }, + "append-transform": { + "version": "0.3.0", + "bundled": true, + "dev": true + }, + "archy": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "arr-diff": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "requires": { + "arr-flatten": "^1.0.1" + } + }, + "arr-flatten": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "array-unique": { + "version": "0.2.1", + "bundled": true, + "dev": true + }, + "arrify": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "async": { + "version": "1.5.2", + "bundled": true, + "dev": true + }, + "babel-code-frame": { + "version": "6.16.0", + "bundled": true, + "dev": true, + "requires": { + "chalk": "^1.1.0", + "esutils": "^2.0.2", + "js-tokens": "^2.0.0" + } + }, + "babel-generator": { + "version": "6.18.0", + "bundled": true, + "dev": true, + "requires": { + "babel-messages": "^6.8.0", + "babel-runtime": "^6.9.0", + "babel-types": "^6.18.0", + "detect-indent": "^4.0.0", + "jsesc": "^1.3.0", + "lodash": "^4.2.0", + "source-map": "^0.5.0" + } + }, + "babel-messages": { + "version": "6.8.0", + "bundled": true, + "dev": true, + "requires": { + "babel-runtime": "^6.0.0" + } + }, + "babel-runtime": { + "version": "6.18.0", + "bundled": true, + "dev": true, + "requires": { + "core-js": "^2.4.0", + "regenerator-runtime": "^0.9.5" + } + }, + "babel-template": { + "version": "6.16.0", + "bundled": true, + "dev": true, + "requires": { + "babel-runtime": "^6.9.0", + "babel-traverse": "^6.16.0", + "babel-types": "^6.16.0", + "babylon": "^6.11.0", + "lodash": "^4.2.0" + } + }, + "babel-traverse": { + "version": "6.18.0", + "bundled": true, + "dev": true, + "requires": { + "babel-code-frame": "^6.16.0", + "babel-messages": "^6.8.0", + "babel-runtime": "^6.9.0", + "babel-types": "^6.18.0", + "babylon": "^6.11.0", + "debug": "^2.2.0", + "globals": "^9.0.0", + "invariant": "^2.2.0", + "lodash": "^4.2.0" + } + }, + "babel-types": { + "version": "6.18.0", + "bundled": true, + "dev": true, + "requires": { + "babel-runtime": "^6.9.1", + "esutils": "^2.0.2", + "lodash": "^4.2.0", + "to-fast-properties": "^1.0.1" + } + }, + "babylon": { + "version": "6.13.1", + "bundled": true, + "dev": true + }, + "balanced-match": { + "version": "0.4.2", + "bundled": true, + "dev": true + }, + "brace-expansion": { + "version": "1.1.6", + "bundled": true, + "dev": true, + "requires": { + "balanced-match": "^0.4.1", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "1.8.5", + "bundled": true, + "dev": true, + "requires": { + "expand-range": "^1.8.1", + "preserve": "^0.2.0", + "repeat-element": "^1.1.2" + } + }, + "builtin-modules": { + "version": "1.1.1", + "bundled": true, + "dev": true + }, + "caching-transform": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "requires": { + "md5-hex": "^1.2.0", + "mkdirp": "^0.5.1", + "write-file-atomic": "^1.1.4" + } + }, + "camelcase": { + "version": "1.2.1", + "bundled": true, + "dev": true, + "optional": true + }, + "center-align": { + "version": "0.1.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "align-text": "^0.1.3", + "lazy-cache": "^1.0.3" + } + }, + "chalk": { + "version": "1.1.3", + "bundled": true, + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + }, + "cliui": { + "version": "2.1.0", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "center-align": "^0.1.1", + "right-align": "^0.1.1", + "wordwrap": "0.0.2" + }, + "dependencies": { + "wordwrap": { + "version": "0.0.2", + "bundled": true, + "dev": true, + "optional": true + } + } + }, + "code-point-at": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "commondir": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "bundled": true, + "dev": true + }, + "convert-source-map": { + "version": "1.3.0", + "bundled": true, + "dev": true + }, + "core-js": { + "version": "2.4.1", + "bundled": true, + "dev": true + }, + "cross-spawn": { + "version": "4.0.2", + "bundled": true, + "dev": true, + "requires": { + "lru-cache": "^4.0.1", + "which": "^1.2.9" + } + }, + "debug": { + "version": "2.2.0", + "bundled": true, + "dev": true, + "requires": { + "ms": "0.7.1" + } + }, + "decamelize": { + "version": "1.2.0", + "bundled": true, + "dev": true + }, + "default-require-extensions": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "strip-bom": "^2.0.0" + } + }, + "detect-indent": { + "version": "4.0.0", + "bundled": true, + "dev": true, + "requires": { + "repeating": "^2.0.0" + } + }, + "error-ex": { + "version": "1.3.0", + "bundled": true, + "dev": true, + "requires": { + "is-arrayish": "^0.2.1" + } + }, + "escape-string-regexp": { + "version": "1.0.5", + "bundled": true, + "dev": true + }, + "esutils": { + "version": "2.0.2", + "bundled": true, + "dev": true + }, + "expand-brackets": { + "version": "0.1.5", + "bundled": true, + "dev": true, + "requires": { + "is-posix-bracket": "^0.1.0" + } + }, + "expand-range": { + "version": "1.8.2", + "bundled": true, + "dev": true, + "requires": { + "fill-range": "^2.1.0" + } + }, + "extglob": { + "version": "0.3.2", + "bundled": true, + "dev": true, + "requires": { + "is-extglob": "^1.0.0" + } + }, + "filename-regex": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "fill-range": { + "version": "2.2.3", + "bundled": true, + "dev": true, + "requires": { + "is-number": "^2.1.0", + "isobject": "^2.0.0", + "randomatic": "^1.1.3", + "repeat-element": "^1.1.2", + "repeat-string": "^1.5.2" + } + }, + "find-cache-dir": { + "version": "0.1.1", + "bundled": true, + "dev": true, + "requires": { + "commondir": "^1.0.1", + "mkdirp": "^0.5.1", + "pkg-dir": "^1.0.0" + } + }, + "find-up": { + "version": "1.1.2", + "bundled": true, + "dev": true, + "requires": { + "path-exists": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "for-in": { + "version": "0.1.6", + "bundled": true, + "dev": true + }, + "for-own": { + "version": "0.1.4", + "bundled": true, + "dev": true, + "requires": { + "for-in": "^0.1.5" + } + }, + "foreground-child": { + "version": "1.5.3", + "bundled": true, + "dev": true, + "requires": { + "cross-spawn": "^4", + "signal-exit": "^3.0.0" + } + }, + "fs.realpath": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "get-caller-file": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "glob": { + "version": "7.1.1", + "bundled": true, + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.2", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-base": { + "version": "0.3.0", + "bundled": true, + "dev": true, + "requires": { + "glob-parent": "^2.0.0", + "is-glob": "^2.0.0" + } + }, + "glob-parent": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "requires": { + "is-glob": "^2.0.0" + } + }, + "globals": { + "version": "9.12.0", + "bundled": true, + "dev": true + }, + "graceful-fs": { + "version": "4.1.9", + "bundled": true, + "dev": true + }, + "handlebars": { + "version": "4.0.5", + "bundled": true, + "dev": true, + "requires": { + "async": "^1.4.0", + "optimist": "^0.6.1", + "source-map": "^0.4.4", + "uglify-js": "^2.6" + }, + "dependencies": { + "source-map": { + "version": "0.4.4", + "bundled": true, + "dev": true, + "requires": { + "amdefine": ">=0.0.4" + } + } + } + }, + "has-ansi": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "has-flag": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "hosted-git-info": { + "version": "2.1.5", + "bundled": true, + "dev": true + }, + "imurmurhash": { + "version": "0.1.4", + "bundled": true, + "dev": true + }, + "inflight": { + "version": "1.0.6", + "bundled": true, + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.3", + "bundled": true, + "dev": true + }, + "invariant": { + "version": "2.2.1", + "bundled": true, + "dev": true, + "requires": { + "loose-envify": "^1.0.0" + } + }, + "invert-kv": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "is-arrayish": { + "version": "0.2.1", + "bundled": true, + "dev": true + }, + "is-buffer": { + "version": "1.1.4", + "bundled": true, + "dev": true + }, + "is-builtin-module": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "builtin-modules": "^1.0.0" + } + }, + "is-dotfile": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "is-equal-shallow": { + "version": "0.1.3", + "bundled": true, + "dev": true, + "requires": { + "is-primitive": "^2.0.0" + } + }, + "is-extendable": { + "version": "0.1.1", + "bundled": true, + "dev": true + }, + "is-extglob": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "is-finite": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "is-glob": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "requires": { + "is-extglob": "^1.0.0" + } + }, + "is-number": { + "version": "2.1.0", + "bundled": true, + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "is-posix-bracket": { + "version": "0.1.1", + "bundled": true, + "dev": true + }, + "is-primitive": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "is-utf8": { + "version": "0.2.1", + "bundled": true, + "dev": true + }, + "isarray": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "isexe": { + "version": "1.1.2", + "bundled": true, + "dev": true + }, + "isobject": { + "version": "2.1.0", + "bundled": true, + "dev": true, + "requires": { + "isarray": "1.0.0" + } + }, + "istanbul-lib-coverage": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "istanbul-lib-hook": { + "version": "1.0.0-alpha.4", + "bundled": true, + "dev": true, + "requires": { + "append-transform": "^0.3.0" + } + }, + "istanbul-lib-instrument": { + "version": "1.2.0", + "bundled": true, + "dev": true, + "requires": { + "babel-generator": "^6.18.0", + "babel-template": "^6.16.0", + "babel-traverse": "^6.18.0", + "babel-types": "^6.18.0", + "babylon": "^6.13.0", + "istanbul-lib-coverage": "^1.0.0", + "semver": "^5.3.0" + } + }, + "istanbul-lib-report": { + "version": "1.0.0-alpha.3", + "bundled": true, + "dev": true, + "requires": { + "async": "^1.4.2", + "istanbul-lib-coverage": "^1.0.0-alpha", + "mkdirp": "^0.5.1", + "path-parse": "^1.0.5", + "rimraf": "^2.4.3", + "supports-color": "^3.1.2" + }, + "dependencies": { + "supports-color": { + "version": "3.1.2", + "bundled": true, + "dev": true, + "requires": { + "has-flag": "^1.0.0" + } + } + } + }, + "istanbul-lib-source-maps": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "istanbul-lib-coverage": "^1.0.0-alpha.0", + "mkdirp": "^0.5.1", + "rimraf": "^2.4.4", + "source-map": "^0.5.3" + } + }, + "istanbul-reports": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "handlebars": "^4.0.3" + } + }, + "js-tokens": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "jsesc": { + "version": "1.3.0", + "bundled": true, + "dev": true + }, + "kind-of": { + "version": "3.0.4", + "bundled": true, + "dev": true, + "requires": { + "is-buffer": "^1.0.2" + } + }, + "lazy-cache": { + "version": "1.0.4", + "bundled": true, + "dev": true, + "optional": true + }, + "lcid": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "invert-kv": "^1.0.0" + } + }, + "load-json-file": { + "version": "1.1.0", + "bundled": true, + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0", + "strip-bom": "^2.0.0" + } + }, + "lodash": { + "version": "4.16.6", + "bundled": true, + "dev": true + }, + "longest": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "loose-envify": { + "version": "1.3.0", + "bundled": true, + "dev": true, + "requires": { + "js-tokens": "^2.0.0" + } + }, + "lru-cache": { + "version": "4.0.1", + "bundled": true, + "dev": true, + "requires": { + "pseudomap": "^1.0.1", + "yallist": "^2.0.0" + } + }, + "md5-hex": { + "version": "1.3.0", + "bundled": true, + "dev": true, + "requires": { + "md5-o-matic": "^0.1.1" + } + }, + "md5-o-matic": { + "version": "0.1.1", + "bundled": true, + "dev": true + }, + "micromatch": { + "version": "2.3.11", + "bundled": true, + "dev": true, + "requires": { + "arr-diff": "^2.0.0", + "array-unique": "^0.2.1", + "braces": "^1.8.2", + "expand-brackets": "^0.1.4", + "extglob": "^0.3.1", + "filename-regex": "^2.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.1", + "kind-of": "^3.0.2", + "normalize-path": "^2.0.1", + "object.omit": "^2.0.0", + "parse-glob": "^3.0.4", + "regex-cache": "^0.4.2" + } + }, + "minimatch": { + "version": "3.0.3", + "bundled": true, + "dev": true, + "requires": { + "brace-expansion": "^1.0.0" + } + }, + "minimist": { + "version": "0.0.8", + "bundled": true, + "dev": true + }, + "mkdirp": { + "version": "0.5.1", + "bundled": true, + "dev": true, + "requires": { + "minimist": "0.0.8" + } + }, + "ms": { + "version": "0.7.1", + "bundled": true, + "dev": true + }, + "normalize-package-data": { + "version": "2.3.5", + "bundled": true, + "dev": true, + "requires": { + "hosted-git-info": "^2.1.4", + "is-builtin-module": "^1.0.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + } + }, + "normalize-path": { + "version": "2.0.1", + "bundled": true, + "dev": true + }, + "number-is-nan": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "object-assign": { + "version": "4.1.0", + "bundled": true, + "dev": true + }, + "object.omit": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "requires": { + "for-own": "^0.1.4", + "is-extendable": "^0.1.1" + } + }, + "once": { + "version": "1.4.0", + "bundled": true, + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "optimist": { + "version": "0.6.1", + "bundled": true, + "dev": true, + "requires": { + "minimist": "~0.0.1", + "wordwrap": "~0.0.2" + } + }, + "os-homedir": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "os-locale": { + "version": "1.4.0", + "bundled": true, + "dev": true, + "requires": { + "lcid": "^1.0.0" + } + }, + "parse-glob": { + "version": "3.0.4", + "bundled": true, + "dev": true, + "requires": { + "glob-base": "^0.3.0", + "is-dotfile": "^1.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.0" + } + }, + "parse-json": { + "version": "2.2.0", + "bundled": true, + "dev": true, + "requires": { + "error-ex": "^1.2.0" + } + }, + "path-exists": { + "version": "2.1.0", + "bundled": true, + "dev": true, + "requires": { + "pinkie-promise": "^2.0.0" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "path-parse": { + "version": "1.0.5", + "bundled": true, + "dev": true + }, + "path-type": { + "version": "1.1.0", + "bundled": true, + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "pify": { + "version": "2.3.0", + "bundled": true, + "dev": true + }, + "pinkie": { + "version": "2.0.4", + "bundled": true, + "dev": true + }, + "pinkie-promise": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "requires": { + "pinkie": "^2.0.0" + } + }, + "pkg-dir": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "find-up": "^1.0.0" + } + }, + "preserve": { + "version": "0.2.0", + "bundled": true, + "dev": true + }, + "pseudomap": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "randomatic": { + "version": "1.1.5", + "bundled": true, + "dev": true, + "requires": { + "is-number": "^2.0.2", + "kind-of": "^3.0.2" + } + }, + "read-pkg": { + "version": "1.1.0", + "bundled": true, + "dev": true, + "requires": { + "load-json-file": "^1.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^1.0.0" + } + }, + "read-pkg-up": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "requires": { + "find-up": "^1.0.0", + "read-pkg": "^1.0.0" + } + }, + "regenerator-runtime": { + "version": "0.9.5", + "bundled": true, + "dev": true + }, + "regex-cache": { + "version": "0.4.3", + "bundled": true, + "dev": true, + "requires": { + "is-equal-shallow": "^0.1.3", + "is-primitive": "^2.0.0" + } + }, + "repeat-element": { + "version": "1.1.2", + "bundled": true, + "dev": true + }, + "repeat-string": { + "version": "1.6.1", + "bundled": true, + "dev": true + }, + "repeating": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "requires": { + "is-finite": "^1.0.0" + } + }, + "require-directory": { + "version": "2.1.1", + "bundled": true, + "dev": true + }, + "require-main-filename": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "resolve-from": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "right-align": { + "version": "0.1.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "align-text": "^0.1.1" + } + }, + "rimraf": { + "version": "2.5.4", + "bundled": true, + "dev": true, + "requires": { + "glob": "^7.0.5" + } + }, + "semver": { + "version": "5.3.0", + "bundled": true, + "dev": true + }, + "set-blocking": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "signal-exit": { + "version": "3.0.1", + "bundled": true, + "dev": true + }, + "slide": { + "version": "1.1.6", + "bundled": true, + "dev": true + }, + "source-map": { + "version": "0.5.6", + "bundled": true, + "dev": true + }, + "spawn-wrap": { + "version": "1.2.4", + "bundled": true, + "dev": true, + "requires": { + "foreground-child": "^1.3.3", + "mkdirp": "^0.5.0", + "os-homedir": "^1.0.1", + "rimraf": "^2.3.3", + "signal-exit": "^2.0.0", + "which": "^1.2.4" + }, + "dependencies": { + "signal-exit": { + "version": "2.1.2", + "bundled": true, + "dev": true + } + } + }, + "spdx-correct": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "spdx-license-ids": "^1.0.2" + } + }, + "spdx-expression-parse": { + "version": "1.0.4", + "bundled": true, + "dev": true + }, + "spdx-license-ids": { + "version": "1.2.2", + "bundled": true, + "dev": true + }, + "string-width": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "bundled": true, + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "strip-bom": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "requires": { + "is-utf8": "^0.2.0" + } + }, + "supports-color": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "test-exclude": { + "version": "2.1.3", + "bundled": true, + "dev": true, + "requires": { + "arrify": "^1.0.1", + "micromatch": "^2.3.11", + "object-assign": "^4.1.0", + "read-pkg-up": "^1.0.1", + "require-main-filename": "^1.0.1" + } + }, + "to-fast-properties": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "uglify-js": { + "version": "2.7.4", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "async": "~0.2.6", + "source-map": "~0.5.1", + "uglify-to-browserify": "~1.0.0", + "yargs": "~3.10.0" + }, + "dependencies": { + "async": { + "version": "0.2.10", + "bundled": true, + "dev": true, + "optional": true + }, + "yargs": { + "version": "3.10.0", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "camelcase": "^1.0.2", + "cliui": "^2.1.0", + "decamelize": "^1.0.0", + "window-size": "0.1.0" + } + } + } + }, + "uglify-to-browserify": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "validate-npm-package-license": { + "version": "3.0.1", + "bundled": true, + "dev": true, + "requires": { + "spdx-correct": "~1.0.0", + "spdx-expression-parse": "~1.0.0" + } + }, + "which": { + "version": "1.2.11", + "bundled": true, + "dev": true, + "requires": { + "isexe": "^1.1.1" + } + }, + "which-module": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "window-size": { + "version": "0.1.0", + "bundled": true, + "dev": true, + "optional": true + }, + "wordwrap": { + "version": "0.0.3", + "bundled": true, + "dev": true + }, + "wrap-ansi": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "requires": { + "string-width": "^1.0.1" + } + }, + "wrappy": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "write-file-atomic": { + "version": "1.2.0", + "bundled": true, + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "imurmurhash": "^0.1.4", + "slide": "^1.1.5" + } + }, + "y18n": { + "version": "3.2.1", + "bundled": true, + "dev": true + }, + "yallist": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "yargs": { + "version": "6.3.0", + "bundled": true, + "dev": true, + "requires": { + "camelcase": "^3.0.0", + "cliui": "^3.2.0", + "decamelize": "^1.1.1", + "get-caller-file": "^1.0.1", + "os-locale": "^1.4.0", + "read-pkg-up": "^1.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^1.0.2", + "which-module": "^1.0.0", + "window-size": "^0.2.0", + "y18n": "^3.2.1", + "yargs-parser": "^4.0.2" + }, + "dependencies": { + "camelcase": { + "version": "3.0.0", + "bundled": true, + "dev": true + }, + "cliui": { + "version": "3.2.0", + "bundled": true, + "dev": true, + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wrap-ansi": "^2.0.0" + } + }, + "window-size": { + "version": "0.2.0", + "bundled": true, + "dev": true + } + } + }, + "yargs-parser": { + "version": "4.0.2", + "bundled": true, + "dev": true, + "requires": { + "camelcase": "^3.0.0" + }, + "dependencies": { + "camelcase": { + "version": "3.0.0", + "bundled": true, + "dev": true + } + } + } + } + }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", + "dev": true + }, + "object-inspect": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.6.0.tgz", + "integrity": "sha512-GJzfBZ6DgDAmnuaM3104jR4s1Myxr3Y3zfIyN4z3UdqN69oSRacNK8UhnobDdC+7J2AHCjGwxQubNJfE70SXXQ==", + "dev": true + }, + "object-keys": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.0.tgz", + "integrity": "sha512-6OO5X1+2tYkNyNEx6TsCxEqFfRWaqx6EtMiSbGrw8Ob8v9Ne+Hl8rBAgLBZn5wjEz3s/s6U1WXFUFOcxxAwUpg==", + "dev": true + }, + "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": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, + "optionator": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz", + "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=", + "dev": true, + "requires": { + "deep-is": "~0.1.3", + "fast-levenshtein": "~2.0.4", + "levn": "~0.3.0", + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2", + "wordwrap": "~1.0.0" + } + }, + "os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", + "dev": true + }, + "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 + }, + "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-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", + "dev": true + }, + "path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "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" + } + }, + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + }, + "pkg-conf": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/pkg-conf/-/pkg-conf-2.1.0.tgz", + "integrity": "sha1-ISZRTKbyq/69FoWW3xi6V4Z/AFg=", + "dev": true, + "requires": { + "find-up": "^2.0.0", + "load-json-file": "^4.0.0" + }, + "dependencies": { + "load-json-file": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-4.0.0.tgz", + "integrity": "sha1-L19Fq5HjMhYjT9U62rZo607AmTs=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^4.0.0", + "pify": "^3.0.0", + "strip-bom": "^3.0.0" + } + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + } + } + }, + "pkg-config": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/pkg-config/-/pkg-config-1.1.1.tgz", + "integrity": "sha1-VX7yLXPaPIg3EHdmxS6tq94pj+Q=", + "dev": true, + "requires": { + "debug-log": "^1.0.0", + "find-root": "^1.0.0", + "xtend": "^4.0.1" + } + }, + "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" + } + }, + "pluralize": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz", + "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==", + "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 + }, + "progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true + }, + "prop-types": { + "version": "15.7.2", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.7.2.tgz", + "integrity": "sha512-8QQikdH7//R2vurIJSutZ1smHYTcLpRWEOlHnzcWHmBYrOGUysKwSsrC89BCiFj3CbrfJ/nXFdJepOVrY1GCHQ==", + "dev": true, + "requires": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.8.1" + } + }, + "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 + }, + "react-is": { + "version": "16.8.4", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.8.4.tgz", + "integrity": "sha512-PVadd+WaUDOAciICm/J1waJaSvgq+4rHE/K70j0PFqKhkTBsPv/82UGQJNXAngz1fOQLLxI6z1sEDmJDQhCTAA==", + "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" + } + }, + "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-uncached": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", + "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=", + "dev": true, + "requires": { + "caller-path": "^0.1.0", + "resolve-from": "^1.0.0" + } + }, + "resolve": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.10.0.tgz", + "integrity": "sha512-3sUr9aq5OfSg2S9pNtPA9hL1FVEAjvfOC4leW0SNf/mpnaakz2a9femSd6LqAww2RaFctwyf1lCqnTHuF1rxDg==", + "dev": true, + "requires": { + "path-parse": "^1.0.6" + } + }, + "resolve-from": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", + "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=", + "dev": true + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + }, + "resumer": { + "version": "0.0.0", + "resolved": "https://registry.npmjs.org/resumer/-/resumer-0.0.0.tgz", + "integrity": "sha1-8ej0YeQGS6Oegq883CqMiT0HZ1k=", + "dev": true, + "requires": { + "through": "~2.3.4" + } + }, + "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" + } + }, + "run-async": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz", + "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=", + "dev": true, + "requires": { + "is-promise": "^2.1.0" + } + }, + "run-parallel": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.1.9.tgz", + "integrity": "sha512-DEqnSRTDw/Tc3FXf49zedI638Z9onwUotBMiUFKmrO2sdFKIbXamXGQ3Axd4qgphxKB4kw/qP1w5kTxnfU1B9Q==", + "dev": true + }, + "rxjs": { + "version": "5.5.12", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-5.5.12.tgz", + "integrity": "sha512-xx2itnL5sBbqeeiVgNPVuQQ1nC8Jp2WfNJhXWHmElW9YmrpS9UVnNzhP3EH3HFqexO5Tlp8GhYY+WEcqcVMvGw==", + "dev": true, + "requires": { + "symbol-observable": "1.0.1" + } + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true + }, + "semver": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.6.0.tgz", + "integrity": "sha512-RS9R6R35NYgQn++fkDWaOmqGoj4Ek9gGs+DPxNUZKuwE183xjJroKvyo1IzVFeXvUrvmALy6FWD5xrdJT25gMg==", + "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.2", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", + "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", + "dev": true + }, + "slice-ansi": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-1.0.0.tgz", + "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.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.3", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.3.tgz", + "integrity": "sha512-uBIcIl3Ih6Phe3XHK1NqboJLdGfwr1UN3k6wSD1dZpmPsIkb8AGNbZYJ1fOBk834+Gxy8rpfDxrS6XLEMZMY2g==", + "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 + }, + "standard": { + "version": "12.0.1", + "resolved": "https://registry.npmjs.org/standard/-/standard-12.0.1.tgz", + "integrity": "sha512-UqdHjh87OG2gUrNCSM4QRLF5n9h3TFPwrCNyVlkqu31Hej0L/rc8hzKqVvkb2W3x0WMq7PzZdkLfEcBhVOR6lg==", + "dev": true, + "requires": { + "eslint": "~5.4.0", + "eslint-config-standard": "12.0.0", + "eslint-config-standard-jsx": "6.0.2", + "eslint-plugin-import": "~2.14.0", + "eslint-plugin-node": "~7.0.1", + "eslint-plugin-promise": "~4.0.0", + "eslint-plugin-react": "~7.11.1", + "eslint-plugin-standard": "~4.0.0", + "standard-engine": "~9.0.0" + } + }, + "standard-engine": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/standard-engine/-/standard-engine-9.0.0.tgz", + "integrity": "sha512-ZfNfCWZ2Xq67VNvKMPiVMKHnMdvxYzvZkf1AH8/cw2NLDBm5LRsxMqvEJpsjLI/dUosZ3Z1d6JlHDp5rAvvk2w==", + "dev": true, + "requires": { + "deglob": "^2.1.0", + "get-stdin": "^6.0.0", + "minimist": "^1.1.0", + "pkg-conf": "^2.0.0" + }, + "dependencies": { + "minimist": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", + "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", + "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" + } + }, + "string.prototype.trim": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.1.2.tgz", + "integrity": "sha1-0E3iyJ4Tf019IG8Ia17S+ua+jOo=", + "dev": true, + "requires": { + "define-properties": "^1.1.2", + "es-abstract": "^1.5.0", + "function-bind": "^1.0.2" + } + }, + "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" + }, + "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-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + }, + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + }, + "symbol-observable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.0.1.tgz", + "integrity": "sha1-g0D8RwLDEi310iKI+IKD9RPT/dQ=", + "dev": true + }, + "table": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/table/-/table-4.0.3.tgz", + "integrity": "sha512-S7rnFITmBH1EnyKcvxBh1LjYeQMmnZtCXSEbHcH6S0NoKit24ZuFO/T1vDcLdYsLQkM188PVVhQmzKIuThNkKg==", + "dev": true, + "requires": { + "ajv": "^6.0.1", + "ajv-keywords": "^3.0.0", + "chalk": "^2.1.0", + "lodash": "^4.17.4", + "slice-ansi": "1.0.0", + "string-width": "^2.1.1" + } + }, + "tape": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/tape/-/tape-4.10.1.tgz", + "integrity": "sha512-G0DywYV1jQeY3axeYnXUOt6ktnxS9OPJh97FGR3nrua8lhWi1zPflLxcAHavZ7Jf3qUfY7cxcVIVFa4mY2IY1w==", + "dev": true, + "requires": { + "deep-equal": "~1.0.1", + "defined": "~1.0.0", + "for-each": "~0.3.3", + "function-bind": "~1.1.1", + "glob": "~7.1.3", + "has": "~1.0.3", + "inherits": "~2.0.3", + "minimist": "~1.2.0", + "object-inspect": "~1.6.0", + "resolve": "~1.10.0", + "resumer": "~0.0.0", + "string.prototype.trim": "~1.1.2", + "through": "~2.3.8" + }, + "dependencies": { + "minimist": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", + "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", + "dev": true + } + } + }, + "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" + } + }, + "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" + } + }, + "uniq": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", + "integrity": "sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8=", + "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" + } + }, + "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" + } + }, + "wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=", + "dev": true + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", + "dev": true + }, + "write": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", + "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=", + "dev": true, + "requires": { + "mkdirp": "^0.5.1" + } + }, + "xtend": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", + "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=", + "dev": true + } + } +} diff --git a/package.json b/package.json index e8504a7..e220e60 100644 --- a/package.json +++ b/package.json @@ -46,5 +46,7 @@ "standard": "*", "tape": "^4.5.1" }, - "dependencies": {} + "dependencies": { + "bn.js": "^4.11.8" + } } diff --git a/split.js b/split.js index 4d3ce5d..f2bd436 100644 --- a/split.js +++ b/split.js @@ -1,33 +1,42 @@ var utils = require('./utils') +var BN = require('bn.js') +var ext = require('./bn-extensions') // split utxos between each output, ignores outputs with .value defined module.exports = function split (utxos, outputs, feeRate) { - if (!isFinite(utils.uintOrNaN(feeRate))) return {} + if (!isFinite(utils.bnOrNaN(feeRate))) return {} var bytesAccum = utils.transactionBytes(utxos, outputs) - var fee = feeRate * bytesAccum + var fee = ext.mul(feeRate, bytesAccum) if (outputs.length === 0) return { fee: fee } var inAccum = utils.sumOrNaN(utxos) var outAccum = utils.sumForgiving(outputs) - var remaining = inAccum - outAccum - fee + var remaining = ext.sub(inAccum, outAccum, fee) if (!isFinite(remaining) || remaining < 0) return { fee: fee } var unspecified = outputs.reduce(function (a, x) { return a + !isFinite(x.value) }, 0) - if (remaining === 0 && unspecified === 0) return utils.finalize(utxos, outputs, feeRate) + if (ext.isZero(remaining) && unspecified === 0) return utils.finalize(utxos, outputs, feeRate) - var splitOutputsCount = outputs.reduce(function (a, x) { + // Counts the number of split outputs left + var splitOutputsCount = new BN(outputs.reduce(function (a, x) { return a + !x.value - }, 0) - var splitValue = (remaining / splitOutputsCount) >>> 0 + }, 0)) + + // any number / 0 = infinity (shift right = 0) + var splitValue = ext.shrn(ext.div(remaining, splitOutputsCount), 0) // ensure every output is either user defined, or over the threshold if (!outputs.every(function (x) { - return x.value !== undefined || (splitValue > utils.dustThreshold(x, feeRate)) - })) return { fee: fee } + return x.value !== undefined || ext.gt(splitValue, utils.dustThreshold(x, feeRate)) + })) { + return { + fee: fee + } + } // assign splitValue to outputs not user defined outputs = outputs.map(function (x) { diff --git a/stats/index.js b/stats/index.js index ea99dfc..4a17411 100644 --- a/stats/index.js +++ b/stats/index.js @@ -41,7 +41,7 @@ let selectedData = walletType === 'p2pkh' ? pubkeyhashScriptLengthData : scripth let inLengthProbs = selectedData.inLengthPercs let outLengthProbs = {}; -[scripthashScriptLengthData, pubkeyhashScriptLengthData].forEach(({prob, outLength}) => { +[scripthashScriptLengthData, pubkeyhashScriptLengthData].forEach(({ prob, outLength }) => { outLengthProbs[outLength] = prob }) diff --git a/test/_utils.js b/test/_utils.js index 4c3739d..9375893 100644 --- a/test/_utils.js +++ b/test/_utils.js @@ -1,16 +1,26 @@ +var BN = require('bn.js') + function expand (values, indices) { if (indices) { return values.map(function (x, i) { - if (typeof x === 'number') return { i: i, value: x } - - var y = { i: i } + if (BN.isBN(x)) { + return { + i: i, + value: x + } + } + var y = { + i: i + } for (var k in x) y[k] = x[k] return y }) } return values.map(function (x, i) { - return typeof x === 'object' ? x : { value: x } + return (typeof x === 'object' && !BN.isBN(x)) ? x : { + value: x + } }) } diff --git a/test/accumulative.js b/test/accumulative.js index 6f6edd7..5279295 100644 --- a/test/accumulative.js +++ b/test/accumulative.js @@ -9,10 +9,17 @@ fixtures.forEach(function (f) { var outputs = utils.expand(f.outputs) var actual = coinAccum(inputs, outputs, f.feeRate) - t.same(actual, f.expected) + t.same(actual.inputs, f.expected.inputs) + t.same(actual.outputs, f.expected.outputs) + if (f.expected.fee) t.ok(actual.fee.cmp(f.expected.fee) === 0) + else t.ok(actual.fee === f.expected.fee) + if (actual.inputs) { var feedback = coinAccum(actual.inputs, actual.outputs, f.feeRate) - t.same(feedback, f.expected) + t.same(feedback.inputs, f.expected.inputs) + t.same(feedback.outputs, f.expected.outputs) + if (f.expected.fee) t.ok(feedback.fee.cmp(f.expected.fee) === 0) + else t.ok(actual.fee === f.expected.fee) } t.end() diff --git a/test/fixtures/accumulative.js b/test/fixtures/accumulative.js new file mode 100644 index 0000000..02090ed --- /dev/null +++ b/test/fixtures/accumulative.js @@ -0,0 +1,589 @@ +var BN = require('bn.js') + +module.exports = [{ + 'description': '1 output, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('102001') + ], + 'outputs': [ + new BN('100000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('102001') + }], + 'outputs': [{ + 'value': new BN('100000') + }], + 'fee': new BN('2001') + } +}, +{ + 'description': '1 output, change expected', + 'feeRate': new BN('5'), + 'inputs': [ + new BN('106001') + ], + 'outputs': [ + new BN('100000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('106001') + }], + 'outputs': [{ + 'value': new BN('100000') + }, + { + 'value': new BN('4871') + } + ], + 'fee': new BN('1130') + } +}, +{ + 'description': '1 output, change expected, value > 2^32', + 'feeRate': new BN('5'), + 'inputs': [ + new BN('5000000000') + ], + 'outputs': [ + new BN('1') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('5000000000') + }], + 'outputs': [{ + 'value': new BN('1') + }, + { + 'value': new BN('4999998869') + } + ], + 'fee': new BN('1130') + } +}, +{ + 'description': '1 output, sub-optimal inputs (if re-ordered), direct possible', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('40000'), + new BN('40000') + ], + 'outputs': [ + new BN('7700') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('7700') + }], + 'fee': new BN('2300') + } +}, +{ + 'description': '1 output, sub-optimal inputs (if re-ordered), direct possible, but slightly higher fee', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('40000'), + new BN('40000') + ], + 'outputs': [ + new BN('6800') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('6800') + }], + 'fee': new BN('3200') + } +}, +{ + 'description': '1 output, sub-optimal inputs (if re-ordered, no direct possible), change expected', + 'feeRate': new BN('5'), + 'inputs': [ + new BN('10000'), + new BN('40000'), + new BN('40000') + ], + 'outputs': [ + new BN('4700') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('4700') + }, + { + 'value': new BN('4170') + } + ], + 'fee': new BN('1130') + } +}, +{ + 'description': '1 output, fails, skips (and finishes on) detrimental input', + 'feeRate': new BN('55'), + 'inputs': [{ + 'value': new BN('44000') + }, + { + 'value': new BN('800') + } + ], + 'outputs': [ + new BN('38000') + ], + 'expected': { + 'fee': new BN('18700') + } +}, +{ + 'description': '1 output, optimal inputs, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000') + ], + 'outputs': [ + new BN('7700') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('7700') + }], + 'fee': new BN('2300') + } +}, +{ + 'description': '1 output, no fee, change expected', + 'feeRate': new BN('0'), + 'inputs': [ + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000') + ], + 'outputs': [ + new BN('28000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('5000') + }, + { + 'i': 1, + 'value': new BN('5000') + }, + { + 'i': 2, + 'value': new BN('5000') + }, + { + 'i': 3, + 'value': new BN('5000') + }, + { + 'i': 4, + 'value': new BN('5000') + }, + { + 'i': 5, + 'value': new BN('5000') + } + ], + 'outputs': [{ + 'value': new BN('28000') + }, + { + 'value': new BN('2000') + } + ], + 'fee': new BN('0') + } +}, +{ + 'description': '1 output, 2 inputs (related), no change', + 'feeRate': new BN('10'), + 'inputs': [{ + 'address': 'a', + 'value': new BN('100000') + }, + { + 'address': 'a', + 'value': new BN('2000') + } + ], + 'outputs': [ + new BN('98000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'address': 'a', + 'value': new BN('100000') + }], + 'outputs': [{ + 'value': new BN('98000') + }], + 'fee': new BN('2000') + } +}, +{ + 'description': 'many outputs, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('30000'), + new BN('12220'), + new BN('10001') + ], + 'outputs': [ + new BN('35000'), + new BN('5000'), + new BN('5000'), + new BN('1000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('30000') + }, + { + 'i': 1, + 'value': new BN('12220') + }, + { + 'i': 2, + 'value': new BN('10001') + } + ], + 'outputs': [{ + 'value': new BN('35000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('1000') + } + ], + 'fee': new BN('6221') + } +}, +{ + 'description': 'many outputs, change expected', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('30000'), + new BN('14220'), + new BN('10001') + ], + 'outputs': [ + new BN('35000'), + new BN('5000'), + new BN('5000'), + new BN('1000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('30000') + }, + { + 'i': 1, + 'value': new BN('14220') + }, + { + 'i': 2, + 'value': new BN('10001') + } + ], + 'outputs': [{ + 'value': new BN('35000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('1000') + }, + { + 'value': new BN('1981') + } + ], + 'fee': new BN('6240') + } +}, +{ + 'description': 'many outputs, no fee, change expected', + 'feeRate': new BN('0'), + 'inputs': [ + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000') + ], + 'outputs': [ + new BN('28000'), + new BN('1000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('5000') + }, + { + 'i': 1, + 'value': new BN('5000') + }, + { + 'i': 2, + 'value': new BN('5000') + }, + { + 'i': 3, + 'value': new BN('5000') + }, + { + 'i': 4, + 'value': new BN('5000') + }, + { + 'i': 5, + 'value': new BN('5000') + } + ], + 'outputs': [{ + 'value': new BN('28000') + }, + { + 'value': new BN('1000') + }, + { + 'value': new BN('1000') + } + ], + 'fee': new BN('0') + } +}, +{ + 'description': 'no outputs, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('1900') + ], + 'outputs': [], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('1900') + }], + 'outputs': [], + 'fee': new BN('1900') + } +}, +{ + 'description': 'no outputs, change expected', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('20000') + }], + 'outputs': [{ + 'value': new BN('18080') + }], + 'fee': new BN('1920') + } +}, +{ + 'description': 'not enough funds, empty result', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('40000') + ], + 'expected': { + 'fee': new BN('1920') + } +}, +{ + 'description': 'not enough funds (w/ fee), empty result', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('40000') + ], + 'outputs': [ + new BN('40000') + ], + 'expected': { + 'fee': new BN('1920') + } +}, +{ + 'description': 'not enough funds (no inputs), empty result', + 'feeRate': new BN('10'), + 'inputs': [], + 'outputs': [], + 'expected': { + 'fee': new BN('100') + } +}, +{ + 'description': 'not enough funds (no inputs), empty result (>1KiB)', + 'feeRate': new BN('10'), + 'inputs': [], + 'outputs': [ + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1') + ], + 'expected': { + 'fee': new BN('9960') + } +}, +{ + 'description': '2 outputs, some with missing value (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('1000'), + {} + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': 'input with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + 20000.5 + ], + 'outputs': [ + new BN('10000'), + new BN('1200') + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': '2 outputs, with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + 10000.25, + 1200.5 + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': '2 outputs, string values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [{ + 'value': '100' + }, + { + 'value': '204' + } + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': '1', + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('10000') + ], + 'expected': {} +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': 1.5, + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('10000') + ], + 'expected': {} +} +] diff --git a/test/fixtures/accumulative.json b/test/fixtures/accumulative.json deleted file mode 100644 index b14a989..0000000 --- a/test/fixtures/accumulative.json +++ /dev/null @@ -1,634 +0,0 @@ -[ - { - "description": "1 output, no change", - "feeRate": 10, - "inputs": [ - 102001 - ], - "outputs": [ - 100000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 102001 - } - ], - "outputs": [ - { - "value": 100000 - } - ], - "fee": 2001 - } - }, - { - "description": "1 output, change expected", - "feeRate": 5, - "inputs": [ - 106001 - ], - "outputs": [ - 100000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 106001 - } - ], - "outputs": [ - { - "value": 100000 - }, - { - "value": 4871 - } - ], - "fee": 1130 - } - }, - { - "description": "1 output, change expected, value > 2^32", - "feeRate": 5, - "inputs": [ - 5000000000 - ], - "outputs": [ - 1 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 5000000000 - } - ], - "outputs": [ - { - "value": 1 - }, - { - "value": 4999998869 - } - ], - "fee": 1130 - } - }, - { - "description": "1 output, sub-optimal inputs (if re-ordered), direct possible", - "feeRate": 10, - "inputs": [ - 10000, - 40000, - 40000 - ], - "outputs": [ - 7700 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 7700 - } - ], - "fee": 2300 - } - }, - { - "description": "1 output, sub-optimal inputs (if re-ordered), direct possible, but slightly higher fee", - "feeRate": 10, - "inputs": [ - 10000, - 40000, - 40000 - ], - "outputs": [ - 6800 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 6800 - } - ], - "fee": 3200 - } - }, - { - "description": "1 output, sub-optimal inputs (if re-ordered, no direct possible), change expected", - "feeRate": 5, - "inputs": [ - 10000, - 40000, - 40000 - ], - "outputs": [ - 4700 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 4700 - }, - { - "value": 4170 - } - ], - "fee": 1130 - } - }, - { - "description": "1 output, fails, skips (and finishes on) detrimental input", - "feeRate": 55, - "inputs": [ - { - "value": 44000 - }, - { - "value": 800 - } - ], - "outputs": [ - 38000 - ], - "expected": { - "fee": 18700 - } - }, - { - "description": "1 output, optimal inputs, no change", - "feeRate": 10, - "inputs": [ - 10000 - ], - "outputs": [ - 7700 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 7700 - } - ], - "fee": 2300 - } - }, - { - "description": "1 output, no fee, change expected", - "feeRate": 0, - "inputs": [ - 5000, - 5000, - 5000, - 5000, - 5000, - 5000 - ], - "outputs": [ - 28000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 5000 - }, - { - "i": 1, - "value": 5000 - }, - { - "i": 2, - "value": 5000 - }, - { - "i": 3, - "value": 5000 - }, - { - "i": 4, - "value": 5000 - }, - { - "i": 5, - "value": 5000 - } - ], - "outputs": [ - { - "value": 28000 - }, - { - "value": 2000 - } - ], - "fee": 0 - } - }, - { - "description": "1 output, 2 inputs (related), no change", - "feeRate": 10, - "inputs": [ - { - "address": "a", - "value": 100000 - }, - { - "address": "a", - "value": 2000 - } - ], - "outputs": [ - 98000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "address": "a", - "value": 100000 - } - ], - "outputs": [ - { - "value": 98000 - } - ], - "fee": 2000 - } - }, - { - "description": "many outputs, no change", - "feeRate": 10, - "inputs": [ - 30000, - 12220, - 10001 - ], - "outputs": [ - 35000, - 5000, - 5000, - 1000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 30000 - }, - { - "i": 1, - "value": 12220 - }, - { - "i": 2, - "value": 10001 - } - ], - "outputs": [ - { - "value": 35000 - }, - { - "value": 5000 - }, - { - "value": 5000 - }, - { - "value": 1000 - } - ], - "fee": 6221 - } - }, - { - "description": "many outputs, change expected", - "feeRate": 10, - "inputs": [ - 30000, - 14220, - 10001 - ], - "outputs": [ - 35000, - 5000, - 5000, - 1000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 30000 - }, - { - "i": 1, - "value": 14220 - }, - { - "i": 2, - "value": 10001 - } - ], - "outputs": [ - { - "value": 35000 - }, - { - "value": 5000 - }, - { - "value": 5000 - }, - { - "value": 1000 - }, - { - "value": 1981 - } - ], - "fee": 6240 - } - }, - { - "description": "many outputs, no fee, change expected", - "feeRate": 0, - "inputs": [ - 5000, - 5000, - 5000, - 5000, - 5000, - 5000 - ], - "outputs": [ - 28000, - 1000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 5000 - }, - { - "i": 1, - "value": 5000 - }, - { - "i": 2, - "value": 5000 - }, - { - "i": 3, - "value": 5000 - }, - { - "i": 4, - "value": 5000 - }, - { - "i": 5, - "value": 5000 - } - ], - "outputs": [ - { - "value": 28000 - }, - { - "value": 1000 - }, - { - "value": 1000 - } - ], - "fee": 0 - } - }, - { - "description": "no outputs, no change", - "feeRate": 10, - "inputs": [ - 1900 - ], - "outputs": [], - "expected": { - "inputs": [ - { - "i": 0, - "value": 1900 - } - ], - "outputs": [], - "fee": 1900 - } - }, - { - "description": "no outputs, change expected", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [], - "expected": { - "inputs": [ - { - "i": 0, - "value": 20000 - } - ], - "outputs": [ - { - "value": 18080 - } - ], - "fee": 1920 - } - }, - { - "description": "not enough funds, empty result", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - 40000 - ], - "expected": { - "fee": 1920 - } - }, - { - "description": "not enough funds (w/ fee), empty result", - "feeRate": 10, - "inputs": [ - 40000 - ], - "outputs": [ - 40000 - ], - "expected": { - "fee": 1920 - } - }, - { - "description": "not enough funds (no inputs), empty result", - "feeRate": 10, - "inputs": [], - "outputs": [], - "expected": { - "fee": 100 - } - }, - { - "description": "not enough funds (no inputs), empty result (>1KiB)", - "feeRate": 10, - "inputs": [], - "outputs": [ - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1 - ], - "expected": { - "fee": 9960 - } - }, - { - "description": "2 outputs, some with missing value (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - 1000, - {} - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "input with float values (NaN)", - "feeRate": 10, - "inputs": [ - 20000.5 - ], - "outputs": [ - 10000, - 1200 - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "2 outputs, with float values (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - 10000.25, - 1200.5 - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "2 outputs, string values (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - { - "value": "100" - }, - { - "value": "204" - } - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": "1", - "inputs": [ - 20000 - ], - "outputs": [ - 10000 - ], - "expected": {} - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": 1.5, - "inputs": [ - 20000 - ], - "outputs": [ - 10000 - ], - "expected": {} - } -] diff --git a/test/fixtures/break.js b/test/fixtures/break.js new file mode 100644 index 0000000..cdbc998 --- /dev/null +++ b/test/fixtures/break.js @@ -0,0 +1,274 @@ +var BN = require('bn.js') + +module.exports = [{ + 'description': '1:1, no remainder', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('11920') + ], + 'output': new BN('10000'), + 'expected': { + 'inputs': [{ + 'value': new BN('11920') + }], + 'outputs': [{ + 'value': new BN('10000') + }], + 'fee': new BN('1920') + } +}, +{ + 'description': '1:1', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('12000') + ], + 'output': { + 'address': 'woop', + 'value': new BN('10000') + }, + 'expected': { + 'fee': new BN('2000'), + 'inputs': [{ + 'value': new BN('12000') + }], + 'outputs': [{ + 'address': 'woop', + 'value': new BN('10000') + }] + } +}, +{ + 'description': '1:1, w/ change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('12000') + ], + 'output': new BN('8000'), + 'expected': { + 'inputs': [{ + 'value': new BN('12000') + }], + 'outputs': [{ + 'value': new BN('8000') + }, + { + 'value': new BN('1740') + } + ], + 'fee': new BN('2260') + } +}, +{ + 'description': '1:4', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('12000') + ], + 'output': new BN('2000'), + 'expected': { + 'inputs': [{ + 'value': new BN('12000') + }], + 'outputs': [{ + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + } + ], + 'fee': new BN('4000') + } +}, +{ + 'description': '2:5', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('3000'), + new BN('12000') + ], + 'output': new BN('2000'), + 'expected': { + 'inputs': [{ + 'value': new BN('3000') + }, + { + 'value': new BN('12000') + } + ], + 'outputs': [{ + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + }, + { + 'value': new BN('2000') + } + ], + 'fee': new BN('5000') + } +}, +{ + 'description': '2:5, no fee', + 'feeRate': new BN('0'), + 'inputs': [ + new BN('5000'), + new BN('10000') + ], + 'output': new BN('3000'), + 'expected': { + 'inputs': [{ + 'value': new BN('5000') + }, + { + 'value': new BN('10000') + } + ], + 'outputs': [{ + 'value': new BN('3000') + }, + { + 'value': new BN('3000') + }, + { + 'value': new BN('3000') + }, + { + 'value': new BN('3000') + }, + { + 'value': new BN('3000') + } + ], + 'fee': new BN('0') + } +}, +{ + 'description': '2:2 (+1), w/ change', + 'feeRate': new BN('7'), + 'inputs': [ + new BN('16000') + ], + 'output': new BN('6000'), + 'expected': { + 'inputs': [{ + 'value': new BN('16000') + }], + 'outputs': [{ + 'value': new BN('6000') + }, + { + 'value': new BN('6000') + }, + { + 'value': new BN('2180') + } + ], + 'fee': new BN('1820') + } +}, +{ + 'description': '2:3 (+1), no fee, w/ change', + 'feeRate': new BN('0'), + 'inputs': [ + new BN('5000'), + new BN('10000') + ], + 'output': new BN('4000'), + 'expected': { + 'inputs': [{ + 'value': new BN('5000') + }, + { + 'value': new BN('10000') + } + ], + 'outputs': [{ + 'value': new BN('4000') + }, + { + 'value': new BN('4000') + }, + { + 'value': new BN('4000') + }, + { + 'value': new BN('3000') + } + ], + 'fee': new BN('0') + } +}, +{ + 'description': 'not enough funds', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('41000'), + new BN('1000') + ], + 'output': new BN('40000'), + 'expected': { + 'fee': new BN('3400') + } +}, +{ + 'description': 'no inputs', + 'feeRate': new BN('10'), + 'inputs': [], + 'output': new BN('2000'), + 'expected': { + 'fee': new BN('440') + } +}, +{ + 'description': 'invalid output (NaN)', + 'feeRate': new BN('10'), + 'inputs': [], + 'output': {}, + 'expected': { + 'fee': new BN('100') + } +}, +{ + 'description': 'input with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + 10000.5 + ], + 'output': new BN('5000'), + 'expected': { + 'fee': new BN('1580') + } +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': '1', + 'inputs': [ + new BN('20000') + ], + 'output': new BN('10000'), + 'expected': {} +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': 1.5, + 'inputs': [ + new BN('20000') + ], + 'output': new BN('10000'), + 'expected': {} +} +] diff --git a/test/fixtures/break.json b/test/fixtures/break.json deleted file mode 100644 index 8977e4d..0000000 --- a/test/fixtures/break.json +++ /dev/null @@ -1,296 +0,0 @@ -[ - { - "description": "1:1, no remainder", - "feeRate": 10, - "inputs": [ - 11920 - ], - "output": 10000, - "expected": { - "inputs": [ - { - "value": 11920 - } - ], - "outputs": [ - { - "value": 10000 - } - ], - "fee": 1920 - } - }, - { - "description": "1:1", - "feeRate": 10, - "inputs": [ - 12000 - ], - "output": { - "address": "woop", - "value": 10000 - }, - "expected": { - "fee": 2000, - "inputs": [ - { - "value": 12000 - } - ], - "outputs": [ - { - "address": "woop", - "value": 10000 - } - ] - } - }, - { - "description": "1:1, w/ change", - "feeRate": 10, - "inputs": [ - 12000 - ], - "output": 8000, - "expected": { - "inputs": [ - { - "value": 12000 - } - ], - "outputs": [ - { - "value": 8000 - }, - { - "value": 1740 - } - ], - "fee": 2260 - } - }, - { - "description": "1:4", - "feeRate": 10, - "inputs": [ - 12000 - ], - "output": 2000, - "expected": { - "inputs": [ - { - "value": 12000 - } - ], - "outputs": [ - { - "value": 2000 - }, - { - "value": 2000 - }, - { - "value": 2000 - }, - { - "value": 2000 - } - ], - "fee": 4000 - } - }, - { - "description": "2:5", - "feeRate": 10, - "inputs": [ - 3000, - 12000 - ], - "output": 2000, - "expected": { - "inputs": [ - { - "value": 3000 - }, - { - "value": 12000 - } - ], - "outputs": [ - { - "value": 2000 - }, - { - "value": 2000 - }, - { - "value": 2000 - }, - { - "value": 2000 - }, - { - "value": 2000 - } - ], - "fee": 5000 - } - }, - { - "description": "2:5, no fee", - "feeRate": 0, - "inputs": [ - 5000, - 10000 - ], - "output": 3000, - "expected": { - "inputs": [ - { - "value": 5000 - }, - { - "value": 10000 - } - ], - "outputs": [ - { - "value": 3000 - }, - { - "value": 3000 - }, - { - "value": 3000 - }, - { - "value": 3000 - }, - { - "value": 3000 - } - ], - "fee": 0 - } - }, - { - "description": "2:2 (+1), w/ change", - "feeRate": 7, - "inputs": [ - 16000 - ], - "output": 6000, - "expected": { - "inputs": [ - { - "value": 16000 - } - ], - "outputs": [ - { - "value": 6000 - }, - { - "value": 6000 - }, - { - "value": 2180 - } - ], - "fee": 1820 - } - }, - { - "description": "2:3 (+1), no fee, w/ change", - "feeRate": 0, - "inputs": [ - 5000, - 10000 - ], - "output": 4000, - "expected": { - "inputs": [ - { - "value": 5000 - }, - { - "value": 10000 - } - ], - "outputs": [ - { - "value": 4000 - }, - { - "value": 4000 - }, - { - "value": 4000 - }, - { - "value": 3000 - } - ], - "fee": 0 - } - }, - { - "description": "not enough funds", - "feeRate": 10, - "inputs": [ - 41000, - 1000 - ], - "output": 40000, - "expected": { - "fee": 3400 - } - }, - { - "description": "no inputs", - "feeRate": 10, - "inputs": [], - "output": 2000, - "expected": { - "fee": 440 - } - }, - { - "description": "invalid output (NaN)", - "feeRate": 10, - "inputs": [], - "output": {}, - "expected": { - "fee": 100 - } - }, - { - "description": "input with float values (NaN)", - "feeRate": 10, - "inputs": [ - 10000.5 - ], - "output": 5000, - "expected": { - "fee": 1580 - } - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": "1", - "inputs": [ - 20000 - ], - "output": 10000, - "expected": {} - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": 1.5, - "inputs": [ - 20000 - ], - "output": 10000, - "expected": {} - } -] diff --git a/test/fixtures/index.js b/test/fixtures/index.js new file mode 100644 index 0000000..60e7ab1 --- /dev/null +++ b/test/fixtures/index.js @@ -0,0 +1,548 @@ +var BN = require('bn.js') + +module.exports = [{ + 'description': '1 output, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('102001') + ], + 'outputs': [ + new BN('100000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('102001') + }], + 'outputs': [{ + 'value': new BN('100000') + }], + 'fee': new BN('2001') + } +}, +{ + 'description': '1 output, change expected', + 'feeRate': new BN('5'), + 'inputs': [ + new BN('106001') + ], + 'outputs': [ + new BN('100000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('106001') + }], + 'outputs': [{ + 'value': new BN('100000') + }, + { + 'value': new BN('4871') + } + ], + 'fee': new BN('1130') + } +}, +{ + 'description': '1 output, sub-optimal inputs (if re-ordered), direct possible', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('40000'), + new BN('40000') + ], + 'outputs': [ + new BN('7700') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('7700') + }], + 'fee': new BN('2300') + } +}, +{ + 'description': '1 output, sub-optimal inputs (if re-ordered), direct possible, but slightly higher fee', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('40000'), + new BN('40000') + ], + 'outputs': [ + new BN('6800') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('6800') + }], + 'fee': new BN('3200') + } +}, +{ + 'description': '1 output, sub-optimal inputs (if re-ordered, no direct possible), change expected', + 'feeRate': new BN('5'), + 'inputs': [ + new BN('10000'), + new BN('40000'), + new BN('40000') + ], + 'outputs': [ + new BN('4700') + ], + 'expected': { + 'inputs': [{ + 'i': 1, + 'value': new BN('40000') + }], + 'outputs': [{ + 'value': new BN('4700') + }, + { + 'value': new BN('34170') + } + ], + 'fee': new BN('1130') + } +}, +{ + 'description': '1 output, optimal inputs, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000') + ], + 'outputs': [ + new BN('7700') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('10000') + }], + 'outputs': [{ + 'value': new BN('7700') + }], + 'fee': new BN('2300') + } +}, +{ + 'description': '1 output, no fee, change expected', + 'feeRate': new BN('0'), + 'inputs': [ + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000') + ], + 'outputs': [ + new BN('28000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('5000') + }, + { + 'i': 1, + 'value': new BN('5000') + }, + { + 'i': 2, + 'value': new BN('5000') + }, + { + 'i': 3, + 'value': new BN('5000') + }, + { + 'i': 4, + 'value': new BN('5000') + }, + { + 'i': 5, + 'value': new BN('5000') + } + ], + 'outputs': [{ + 'value': new BN('28000') + }, + { + 'value': new BN('2000') + } + ], + 'fee': new BN('0') + } +}, +{ + 'description': '1 output, 2 inputs (related), no change', + 'feeRate': new BN('10'), + 'inputs': [{ + 'address': 'a', + 'value': new BN('100000') + }, + { + 'address': 'a', + 'value': new BN('2000') + } + ], + 'outputs': [ + new BN('98000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'address': 'a', + 'value': new BN('100000') + }], + 'outputs': [{ + 'value': new BN('98000') + }], + 'fee': new BN('2000') + } +}, +{ + 'description': 'many outputs, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('30000'), + new BN('12220'), + new BN('10001') + ], + 'outputs': [ + new BN('35000'), + new BN('5000'), + new BN('5000'), + new BN('1000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('30000') + }, + { + 'i': 1, + 'value': new BN('12220') + }, + { + 'i': 2, + 'value': new BN('10001') + } + ], + 'outputs': [{ + 'value': new BN('35000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('1000') + } + ], + 'fee': new BN('6221') + } +}, +{ + 'description': 'many outputs, change expected', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('30000'), + new BN('14220'), + new BN('10001') + ], + 'outputs': [ + new BN('35000'), + new BN('5000'), + new BN('5000'), + new BN('1000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('30000') + }, + { + 'i': 1, + 'value': new BN('14220') + }, + { + 'i': 2, + 'value': new BN('10001') + } + ], + 'outputs': [{ + 'value': new BN('35000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('5000') + }, + { + 'value': new BN('1000') + }, + { + 'value': new BN('1981') + } + ], + 'fee': new BN('6240') + } +}, +{ + 'description': 'many outputs, no fee, change expected', + 'feeRate': new BN('0'), + 'inputs': [ + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000'), + new BN('5000') + ], + 'outputs': [ + new BN('28000'), + new BN('1000') + ], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('5000') + }, + { + 'i': 1, + 'value': new BN('5000') + }, + { + 'i': 2, + 'value': new BN('5000') + }, + { + 'i': 3, + 'value': new BN('5000') + }, + { + 'i': 4, + 'value': new BN('5000') + }, + { + 'i': 5, + 'value': new BN('5000') + } + ], + 'outputs': [{ + 'value': new BN('28000') + }, + { + 'value': new BN('1000') + }, + { + 'value': new BN('1000') + } + ], + 'fee': new BN('0') + } +}, +{ + 'description': 'no outputs, no change', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('1900') + ], + 'outputs': [], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('1900') + }], + 'outputs': [], + 'fee': new BN('1900') + } +}, +{ + 'description': 'no outputs, change expected', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [], + 'expected': { + 'inputs': [{ + 'i': 0, + 'value': new BN('20000') + }], + 'outputs': [{ + 'value': new BN('18080') + }], + 'fee': new BN('1920') + } +}, +{ + 'description': 'not enough funds, empty result', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('40000') + ], + 'expected': { + 'fee': new BN('1920') + } +}, +{ + 'description': 'not enough funds (w/ fee), empty result', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('40000') + ], + 'outputs': [ + new BN('40000') + ], + 'expected': { + 'fee': new BN('1920') + } +}, +{ + 'description': 'not enough funds (no inputs), empty result', + 'feeRate': new BN('10'), + 'inputs': [], + 'outputs': [], + 'expected': { + 'fee': new BN('100') + } +}, +{ + 'description': 'not enough funds (no inputs), empty result (>1KiB)', + 'feeRate': new BN('10'), + 'inputs': [], + 'outputs': [ + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1'), + new BN('1') + ], + 'expected': { + 'fee': new BN('9960') + } +}, +{ + 'description': '2 outputs, some with missing value (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('1000'), + {} + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': 'input with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + 20000.5 + ], + 'outputs': [ + 10000, + 1200 + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': '2 outputs, with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('10000.25'), + new BN('1200.5') + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': '2 outputs, string values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [{ + 'value': '100' + }, + { + 'value': '204' + } + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': '1', + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('10000') + ], + 'expected': {} +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': 1.5, + 'inputs': [ + new BN('20000') + ], + 'outputs': [ + new BN('10000') + ], + 'expected': {} +} +] diff --git a/test/fixtures/index.json b/test/fixtures/index.json deleted file mode 100644 index 9886706..0000000 --- a/test/fixtures/index.json +++ /dev/null @@ -1,590 +0,0 @@ -[ - { - "description": "1 output, no change", - "feeRate": 10, - "inputs": [ - 102001 - ], - "outputs": [ - 100000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 102001 - } - ], - "outputs": [ - { - "value": 100000 - } - ], - "fee": 2001 - } - }, - { - "description": "1 output, change expected", - "feeRate": 5, - "inputs": [ - 106001 - ], - "outputs": [ - 100000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 106001 - } - ], - "outputs": [ - { - "value": 100000 - }, - { - "value": 4871 - } - ], - "fee": 1130 - } - }, - { - "description": "1 output, sub-optimal inputs (if re-ordered), direct possible", - "feeRate": 10, - "inputs": [ - 10000, - 40000, - 40000 - ], - "outputs": [ - 7700 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 7700 - } - ], - "fee": 2300 - } - }, - { - "description": "1 output, sub-optimal inputs (if re-ordered), direct possible, but slightly higher fee", - "feeRate": 10, - "inputs": [ - 10000, - 40000, - 40000 - ], - "outputs": [ - 6800 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 6800 - } - ], - "fee": 3200 - } - }, - { - "description": "1 output, sub-optimal inputs (if re-ordered, no direct possible), change expected", - "feeRate": 5, - "inputs": [ - 10000, - 40000, - 40000 - ], - "outputs": [ - 4700 - ], - "expected": { - "inputs": [ - { - "i": 1, - "value": 40000 - } - ], - "outputs": [ - { - "value": 4700 - }, - { - "value": 34170 - } - ], - "fee": 1130 - } - }, - { - "description": "1 output, optimal inputs, no change", - "feeRate": 10, - "inputs": [ - 10000 - ], - "outputs": [ - 7700 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 10000 - } - ], - "outputs": [ - { - "value": 7700 - } - ], - "fee": 2300 - } - }, - { - "description": "1 output, no fee, change expected", - "feeRate": 0, - "inputs": [ - 5000, - 5000, - 5000, - 5000, - 5000, - 5000 - ], - "outputs": [ - 28000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 5000 - }, - { - "i": 1, - "value": 5000 - }, - { - "i": 2, - "value": 5000 - }, - { - "i": 3, - "value": 5000 - }, - { - "i": 4, - "value": 5000 - }, - { - "i": 5, - "value": 5000 - } - ], - "outputs": [ - { - "value": 28000 - }, - { - "value": 2000 - } - ], - "fee": 0 - } - }, - { - "description": "1 output, 2 inputs (related), no change", - "feeRate": 10, - "inputs": [ - { - "address": "a", - "value": 100000 - }, - { - "address": "a", - "value": 2000 - } - ], - "outputs": [ - 98000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "address": "a", - "value": 100000 - } - ], - "outputs": [ - { - "value": 98000 - } - ], - "fee": 2000 - } - }, - { - "description": "many outputs, no change", - "feeRate": 10, - "inputs": [ - 30000, - 12220, - 10001 - ], - "outputs": [ - 35000, - 5000, - 5000, - 1000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 30000 - }, - { - "i": 1, - "value": 12220 - }, - { - "i": 2, - "value": 10001 - } - ], - "outputs": [ - { - "value": 35000 - }, - { - "value": 5000 - }, - { - "value": 5000 - }, - { - "value": 1000 - } - ], - "fee": 6221 - } - }, - { - "description": "many outputs, change expected", - "feeRate": 10, - "inputs": [ - 30000, - 14220, - 10001 - ], - "outputs": [ - 35000, - 5000, - 5000, - 1000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 30000 - }, - { - "i": 1, - "value": 14220 - }, - { - "i": 2, - "value": 10001 - } - ], - "outputs": [ - { - "value": 35000 - }, - { - "value": 5000 - }, - { - "value": 5000 - }, - { - "value": 1000 - }, - { - "value": 1981 - } - ], - "fee": 6240 - } - }, - { - "description": "many outputs, no fee, change expected", - "feeRate": 0, - "inputs": [ - 5000, - 5000, - 5000, - 5000, - 5000, - 5000 - ], - "outputs": [ - 28000, - 1000 - ], - "expected": { - "inputs": [ - { - "i": 0, - "value": 5000 - }, - { - "i": 1, - "value": 5000 - }, - { - "i": 2, - "value": 5000 - }, - { - "i": 3, - "value": 5000 - }, - { - "i": 4, - "value": 5000 - }, - { - "i": 5, - "value": 5000 - } - ], - "outputs": [ - { - "value": 28000 - }, - { - "value": 1000 - }, - { - "value": 1000 - } - ], - "fee": 0 - } - }, - { - "description": "no outputs, no change", - "feeRate": 10, - "inputs": [ - 1900 - ], - "outputs": [], - "expected": { - "inputs": [ - { - "i": 0, - "value": 1900 - } - ], - "outputs": [], - "fee": 1900 - } - }, - { - "description": "no outputs, change expected", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [], - "expected": { - "inputs": [ - { - "i": 0, - "value": 20000 - } - ], - "outputs": [ - { - "value": 18080 - } - ], - "fee": 1920 - } - }, - { - "description": "not enough funds, empty result", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - 40000 - ], - "expected": { - "fee": 1920 - } - }, - { - "description": "not enough funds (w/ fee), empty result", - "feeRate": 10, - "inputs": [ - 40000 - ], - "outputs": [ - 40000 - ], - "expected": { - "fee": 1920 - } - }, - { - "description": "not enough funds (no inputs), empty result", - "feeRate": 10, - "inputs": [], - "outputs": [], - "expected": { - "fee": 100 - } - }, - { - "description": "not enough funds (no inputs), empty result (>1KiB)", - "feeRate": 10, - "inputs": [], - "outputs": [ - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1, - 1 - ], - "expected": { - "fee": 9960 - } - }, - { - "description": "2 outputs, some with missing value (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - 1000, - {} - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "input with float values (NaN)", - "feeRate": 10, - "inputs": [ - 20000.5 - ], - "outputs": [ - 10000, - 1200 - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "2 outputs, with float values (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - 10000.25, - 1200.5 - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "2 outputs, string values (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - { - "value": "100" - }, - { - "value": "204" - } - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": "1", - "inputs": [ - 20000 - ], - "outputs": [ - 10000 - ], - "expected": {} - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": 1.5, - "inputs": [ - 20000 - ], - "outputs": [ - 10000 - ], - "expected": {} - } -] - diff --git a/test/fixtures/split.js b/test/fixtures/split.js new file mode 100644 index 0000000..6962ed6 --- /dev/null +++ b/test/fixtures/split.js @@ -0,0 +1,264 @@ +var BN = require('bn.js') + +module.exports = [{ + 'description': '1 to 3', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('18000') + ], + 'outputs': [{}, + {}, + {} + ], + 'expected': { + 'inputs': [{ + 'value': new BN('18000') + }], + 'outputs': [{ + 'value': new BN('5133') + }, + { + 'value': new BN('5133') + }, + { + 'value': new BN('5133') + } + ], + 'fee': new BN('2601') + } +}, +{ + 'description': '5 to 2', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('10000'), + new BN('10000'), + new BN('10000'), + new BN('10000') + ], + 'outputs': [{}, + {} + ], + 'expected': { + 'inputs': [{ + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + } + ], + 'outputs': [{ + 'value': new BN('20910') + }, + { + 'value': new BN('20910') + } + ], + 'fee': new BN('8180') + } +}, +{ + 'description': '3 to 1', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('10000'), + new BN('10000') + ], + 'outputs': [{}], + 'expected': { + 'inputs': [{ + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + } + ], + 'outputs': [{ + 'value': new BN('25120') + }], + 'fee': new BN('4880') + } +}, +{ + 'description': '3 to 3 (1 output pre-defined)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('10000'), + new BN('10000') + ], + 'outputs': [{ + 'address': 'foobar', + 'value': new BN('12000') + }, + { + 'address': 'fizzbuzz' + }, + {} + ], + 'expected': { + 'inputs': [{ + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + }, + { + 'value': new BN('10000') + } + ], + 'outputs': [{ + 'address': 'foobar', + 'value': new BN('12000') + }, + { + 'address': 'fizzbuzz', + 'value': new BN('6220') + }, + { + 'value': new BN('6220') + } + ], + 'fee': new BN('5560') + } +}, +{ + 'description': '2 to 0 (no result)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('10000'), + new BN('10000') + ], + 'outputs': [], + 'expected': { + 'fee': new BN('3060') + } +}, +{ + 'description': '0 to 2 (no result)', + 'feeRate': new BN('10'), + 'inputs': [], + 'outputs': [{}, + {} + ], + 'expected': { + 'fee': new BN('780') + } +}, +{ + 'description': '1 to 2, output is dust (no result)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('2000') + ], + 'outputs': [{}], + 'expected': { + 'fee': new BN('1920') + } +}, +{ + 'description': '2 outputs, some with missing value (NaN)', + 'feeRate': new BN('11'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [{ + 'value': new BN('4000') + }, + {} + ], + 'expected': { + 'inputs': [{ + 'value': new BN('20000') + }], + 'outputs': [{ + 'value': new BN('4000') + }, + { + 'value': new BN('13514') + } + ], + 'fee': new BN('2486') + } +}, + +// TODO +{ + 'description': '2 outputs, some with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [{ + 'value': 4000.5 + }, + {} + ], + 'expected': { + 'fee': new BN('2260') + } +}, + +{ + 'description': '2 outputs, string values (NaN)', + 'feeRate': new BN('11'), + 'inputs': [ + new BN('20000') + ], + 'outputs': [{ + 'value': '100' + }, + { + 'value': '204' + } + ], + 'expected': { + 'fee': new BN('2486') + } +}, +{ + 'description': 'input with float values (NaN)', + 'feeRate': new BN('10'), + 'inputs': [ + 20000.5 + ], + 'outputs': [{}, + {} + ], + 'expected': { + 'fee': new BN('2260') + } +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': '1', + 'inputs': [ + new BN('20000') + ], + 'outputs': [{}], + 'expected': {} +}, +{ + 'description': 'inputs and outputs, bad feeRate (NaN)', + 'feeRate': 1.5, + 'inputs': [ + new BN('20000') + ], + 'outputs': [{}], + 'expected': {} +} +] diff --git a/test/fixtures/split.json b/test/fixtures/split.json deleted file mode 100644 index 30c1dd2..0000000 --- a/test/fixtures/split.json +++ /dev/null @@ -1,289 +0,0 @@ -[ - { - "description": "1 to 3", - "feeRate": 10, - "inputs": [ - 18000 - ], - "outputs": [ - {}, - {}, - {} - ], - "expected": { - "inputs": [ - { - "value": 18000 - } - ], - "outputs": [ - { - "value": 5133 - }, - { - "value": 5133 - }, - { - "value": 5133 - } - ], - "fee": 2601 - } - }, - { - "description": "5 to 2", - "feeRate": 10, - "inputs": [ - 10000, - 10000, - 10000, - 10000, - 10000 - ], - "outputs": [ - {}, - {} - ], - "expected": { - "inputs": [ - { - "value": 10000 - }, - { - "value": 10000 - }, - { - "value": 10000 - }, - { - "value": 10000 - }, - { - "value": 10000 - } - ], - "outputs": [ - { - "value": 20910 - }, - { - "value": 20910 - } - ], - "fee": 8180 - } - }, - { - "description": "3 to 1", - "feeRate": 10, - "inputs": [ - 10000, - 10000, - 10000 - ], - "outputs": [ - {} - ], - "expected": { - "inputs": [ - { - "value": 10000 - }, - { - "value": 10000 - }, - { - "value": 10000 - } - ], - "outputs": [ - { - "value": 25120 - } - ], - "fee": 4880 - } - }, - { - "description": "3 to 3 (1 output pre-defined)", - "feeRate": 10, - "inputs": [ - 10000, - 10000, - 10000 - ], - "outputs": [ - { - "address": "foobar", - "value": 12000 - }, - { - "address": "fizzbuzz" - }, - {} - ], - "expected": { - "inputs": [ - { - "value": 10000 - }, - { - "value": 10000 - }, - { - "value": 10000 - } - ], - "outputs": [ - { - "address": "foobar", - "value": 12000 - }, - { - "address": "fizzbuzz", - "value": 6220 - }, - { - "value": 6220 - } - ], - "fee": 5560 - } - }, - { - "description": "2 to 0 (no result)", - "feeRate": 10, - "inputs": [ - 10000, - 10000 - ], - "outputs": [], - "expected": { - "fee": 3060 - } - }, - { - "description": "0 to 2 (no result)", - "feeRate": 10, - "inputs": [], - "outputs": [ - {}, - {} - ], - "expected": { - "fee": 780 - } - }, - { - "description": "1 to 2, output is dust (no result)", - "feeRate": 10, - "inputs": [ - 2000 - ], - "outputs": [ - {} - ], - "expected": { - "fee": 1920 - } - }, - { - "description": "2 outputs, some with missing value (NaN)", - "feeRate": 11, - "inputs": [ - 20000 - ], - "outputs": [ - { - "value": 4000 - }, - {} - ], - "expected": { - "inputs": [ - { - "value": 20000 - } - ], - "outputs": [ - { - "value": 4000 - }, - { - "value": 13514 - } - ], - "fee": 2486 - } - }, - { - "description": "2 outputs, some with float values (NaN)", - "feeRate": 10, - "inputs": [ - 20000 - ], - "outputs": [ - { - "value": 4000.5 - }, - {} - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "2 outputs, string values (NaN)", - "feeRate": 11, - "inputs": [ - 20000 - ], - "outputs": [ - { - "value": "100" - }, - { - "value": "204" - } - ], - "expected": { - "fee": 2486 - } - }, - { - "description": "input with float values (NaN)", - "feeRate": 10, - "inputs": [ - 20000.5 - ], - "outputs": [ - {}, - {} - ], - "expected": { - "fee": 2260 - } - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": "1", - "inputs": [ - 20000 - ], - "outputs": [ - {} - ], - "expected": {} - }, - { - "description": "inputs and outputs, bad feeRate (NaN)", - "feeRate": 1.5, - "inputs": [ - 20000 - ], - "outputs": [ - {} - ], - "expected": {} - } -] diff --git a/test/index.js b/test/index.js index 3554363..9b84948 100644 --- a/test/index.js +++ b/test/index.js @@ -6,7 +6,9 @@ var utils = require('./_utils') fixtures.forEach(function (f) { tape(f.description, function (t) { var inputs = utils.expand(f.inputs, true) + var outputs = utils.expand(f.outputs) + var actual = coinSelect(inputs, outputs, f.feeRate) t.same(actual, f.expected) diff --git a/test/utils.js b/test/utils.js index 57ce34f..5428b33 100644 --- a/test/utils.js +++ b/test/utils.js @@ -1,18 +1,19 @@ var tape = require('tape') var utils = require('../utils') +var BN = require('bn.js') +var ext = require('../bn-extensions') tape('utils', function (t) { - t.test('uintOrNaN', function (t) { + t.test('bnOrNaN', function (t) { t.plan(8) - - t.equal(utils.uintOrNaN(1), 1) - t.equal(isNaN(utils.uintOrNaN('')), true) - t.equal(isNaN(utils.uintOrNaN(Infinity)), true) - t.equal(isNaN(utils.uintOrNaN(NaN)), true) - t.equal(isNaN(utils.uintOrNaN('1')), true) - t.equal(isNaN(utils.uintOrNaN('1.1')), true) - t.equal(isNaN(utils.uintOrNaN(1.1)), true) - t.equal(isNaN(utils.uintOrNaN(-1)), true) + t.ok(utils.bnOrNaN(new BN(1)).cmp(ext.BN_ONE) === 0) + t.equal(isNaN(utils.bnOrNaN('')), true) + t.equal(isNaN(utils.bnOrNaN(Infinity)), true) + t.equal(isNaN(utils.bnOrNaN(NaN)), true) + t.equal(isNaN(utils.bnOrNaN('1')), true) + t.equal(isNaN(utils.bnOrNaN('1.1')), true) + t.equal(isNaN(utils.bnOrNaN(1.1)), true) + t.equal(isNaN(utils.bnOrNaN(-1)), true) }) t.end() diff --git a/utils.js b/utils.js index 4e5ca31..589c0ca 100644 --- a/utils.js +++ b/utils.js @@ -1,16 +1,19 @@ +var BN = require('bn.js') +var ext = require('./bn-extensions') + // baseline estimates, used to improve performance -var TX_BASE_SIZE = 10 +var TX_BASE_SIZE = new BN('10') var TX_INPUT_SIZE = { - LEGACY: 148, - P2SH: 92, - BECH32: 69 + LEGACY: new BN('148'), + P2SH: new BN('92'), + BECH32: new BN('69') } var TX_OUTPUT_SIZE = { - LEGACY: 34, - P2SH: 32, - BECH32: 31 + LEGACY: new BN('34'), + P2SH: new BN('32'), + BECH32: new BN('31') } function inputBytes (input) { @@ -23,45 +26,54 @@ function outputBytes (output) { function dustThreshold (output, feeRate) { /* ... classify the output for input estimate */ - return inputBytes({}) * feeRate + return ext.mul(inputBytes({}), feeRate) } function transactionBytes (inputs, outputs) { - return TX_BASE_SIZE + - inputs.reduce(function (a, x) { return a + inputBytes(x) }, 0) + - outputs.reduce(function (a, x) { return a + outputBytes(x) }, 0) + return TX_BASE_SIZE + .add(inputs.reduce(function (a, x) { + return a.add(inputBytes(x)) + }, ext.BN_ZERO)) + .add(outputs.reduce(function (a, x) { + return a.add(outputBytes(x)) + }, ext.BN_ZERO)) } -function uintOrNaN (v) { - if (typeof v !== 'number') return NaN +function bnOrNaN (v) { if (!isFinite(v)) return NaN - if (Math.floor(v) !== v) return NaN - if (v < 0) return NaN + if (!BN.isBN(v)) return NaN + if (v.isNeg()) return NaN return v } function sumForgiving (range) { - return range.reduce(function (a, x) { return a + (isFinite(x.value) ? x.value : 0) }, 0) + return range.reduce(function (a, x) { + var valueOrZero = BN.isBN(x.value) ? x.value : ext.BN_ZERO + return ext.add(a, valueOrZero) + }, + ext.BN_ZERO) } function sumOrNaN (range) { - return range.reduce(function (a, x) { return a + uintOrNaN(x.value) }, 0) + return range.reduce(function (a, x) { + return ext.add(a, bnOrNaN(x.value)) + }, ext.BN_ZERO) } var BLANK_OUTPUT = outputBytes({}) function finalize (inputs, outputs, feeRate) { var bytesAccum = transactionBytes(inputs, outputs) - var feeAfterExtraOutput = feeRate * (bytesAccum + BLANK_OUTPUT) - var remainderAfterExtraOutput = sumOrNaN(inputs) - (sumOrNaN(outputs) + feeAfterExtraOutput) + var feeAfterExtraOutput = ext.mul(feeRate, ext.add(bytesAccum, BLANK_OUTPUT)) + var remainderAfterExtraOutput = ext.sub(sumOrNaN(inputs), ext.add(sumOrNaN(outputs), feeAfterExtraOutput)) // is it worth a change output? - if (remainderAfterExtraOutput > dustThreshold({}, feeRate)) { + if (ext.gt(remainderAfterExtraOutput, dustThreshold({}, feeRate))) { outputs = outputs.concat({ value: remainderAfterExtraOutput }) } - var fee = sumOrNaN(inputs) - sumOrNaN(outputs) - if (!isFinite(fee)) return { fee: feeRate * bytesAccum } + var fee = ext.sub(sumOrNaN(inputs), sumOrNaN(outputs)) + if (!isFinite(fee)) return { fee: ext.mul(feeRate, bytesAccum) } return { inputs: inputs, @@ -78,5 +90,5 @@ module.exports = { sumOrNaN: sumOrNaN, sumForgiving: sumForgiving, transactionBytes: transactionBytes, - uintOrNaN: uintOrNaN + bnOrNaN: bnOrNaN }