You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
28113 lines
802 KiB
28113 lines
802 KiB
require=
|
|
// modules are defined as an array
|
|
// [ module function, map of requireuires ]
|
|
//
|
|
// map of requireuires is short require name -> numeric require
|
|
//
|
|
// anything defined in a previous bundle is accessed via the
|
|
// orig method which is the requireuire for previous bundles
|
|
(function outer (modules, cache, entry) {
|
|
// Save the require from previous bundle to this closure if any
|
|
var previousRequire = typeof require == "function" && require;
|
|
|
|
function newRequire(name, jumped, inSkipCache){
|
|
|
|
var m, skipCache = inSkipCache;
|
|
if (typeof name === 'string') {
|
|
if (name.charAt(0) === '!' ) {
|
|
name = name.substr(1);
|
|
skipCache=true;
|
|
}
|
|
}
|
|
if(skipCache || !cache[name]) {
|
|
if(!modules[name]) {
|
|
// if we cannot find the the module within our internal map or
|
|
// cache jump to the current global require ie. the last bundle
|
|
// that was added to the page.
|
|
var currentRequire = typeof require == "function" && require;
|
|
if (!jumped && currentRequire) return currentRequire(name, true);
|
|
|
|
// If there are other bundles on this page the require from the
|
|
// previous one is saved to 'previousRequire'. Repeat this as
|
|
// many times as there are bundles until the module is found or
|
|
// we exhaust the require chain.
|
|
if (previousRequire) return previousRequire(name, true);
|
|
throw new Error('Cannot find module \'' + name + '\'');
|
|
}
|
|
|
|
m = {exports:{}};
|
|
var nextSkipCache = inSkipCache ? false : skipCache;
|
|
if (!skipCache) cache[name] = m;
|
|
skipCache = false;
|
|
modules[name][0].call(m.exports, function(x){
|
|
var id = modules[name][1][x];
|
|
return newRequire(id ? id : x, false, nextSkipCache);
|
|
},m,m.exports,outer,modules,cache,entry);
|
|
}
|
|
return m ? m.exports:cache[name].exports;
|
|
}
|
|
for(var i=0;i<entry.length;i++) newRequire(entry[i]);
|
|
|
|
// Override the current require with this new one
|
|
return newRequire;
|
|
})
|
|
({"bitcore":[function(require,module,exports){
|
|
module.exports=require('tmvhGl');
|
|
},{}],"tmvhGl":[function(require,module,exports){
|
|
(function (Buffer){
|
|
/*
|
|
One way to require files is this simple way:
|
|
module.exports.Address = require('./Address');
|
|
|
|
However, that will load all classes in memory even if they are not used.
|
|
Instead, we can set the 'get' property of each class to only require them when
|
|
they are accessed, saving memory if they are not used in a given project.
|
|
*/
|
|
var requireWhenAccessed = function(name, file) {
|
|
Object.defineProperty(module.exports, name, {get: function() {return require(file)}});
|
|
};
|
|
|
|
requireWhenAccessed('Bignum', 'bignum');
|
|
Object.defineProperty(module.exports, 'bignum', {get: function() {
|
|
console.log('bignum (with a lower-case "b") is deprecated. Use bitcore.Bignum (capital "B") instead.');
|
|
return require('bignum');
|
|
}});
|
|
requireWhenAccessed('Base58', './lib/Base58');
|
|
Object.defineProperty(module.exports, 'base58', {get: function() {
|
|
console.log('base58 (with a lower-case "b") is deprecated. Use bitcore.Base58 (capital "B") instead.');
|
|
return require('./lib/Base58');
|
|
}});
|
|
requireWhenAccessed('bufferput', 'bufferput');
|
|
requireWhenAccessed('buffertools', 'buffertools');
|
|
requireWhenAccessed('Buffers.monkey', './patches/Buffers.monkey');
|
|
requireWhenAccessed('config', './config');
|
|
requireWhenAccessed('const', './const');
|
|
requireWhenAccessed('Curve', './lib/Curve');
|
|
requireWhenAccessed('Deserialize', './lib/Deserialize');
|
|
requireWhenAccessed('log', './util/log');
|
|
requireWhenAccessed('networks', './networks');
|
|
requireWhenAccessed('SecureRandom', './lib/SecureRandom');
|
|
requireWhenAccessed('util', './util/util');
|
|
requireWhenAccessed('EncodedData', './util/EncodedData');
|
|
requireWhenAccessed('VersionedData', './util/VersionedData');
|
|
requireWhenAccessed('BinaryParser', './util/BinaryParser');
|
|
requireWhenAccessed('Address', './lib/Address');
|
|
requireWhenAccessed('HierarchicalKey', './lib/HierarchicalKey');
|
|
Object.defineProperty(module.exports, 'BIP32', {get: function() {
|
|
console.log('BIP32 is deprecated. Use bitcore.HierarchicalKey instead.');
|
|
return require('./lib/HierarchicalKey');
|
|
}});
|
|
requireWhenAccessed('Point', './lib/Point');
|
|
requireWhenAccessed('Opcode', './lib/Opcode');
|
|
requireWhenAccessed('Script', './lib/Script');
|
|
requireWhenAccessed('Transaction', './lib/Transaction');
|
|
requireWhenAccessed('TransactionBuilder', './lib/TransactionBuilder');
|
|
requireWhenAccessed('Connection', './lib/Connection');
|
|
requireWhenAccessed('Peer', './lib/Peer');
|
|
requireWhenAccessed('Block', './lib/Block');
|
|
requireWhenAccessed('ScriptInterpreter', './lib/ScriptInterpreter');
|
|
requireWhenAccessed('Bloom', './lib/Bloom');
|
|
requireWhenAccessed('Key', './lib/Key');
|
|
Object.defineProperty(module.exports, 'KeyModule', {get: function() {
|
|
console.log('KeyModule is deprecated.');
|
|
return require('bindings')('KeyModule');
|
|
}});
|
|
requireWhenAccessed('SINKey', './lib/SINKey');
|
|
requireWhenAccessed('SIN', './lib/SIN');
|
|
requireWhenAccessed('PrivateKey', './lib/PrivateKey');
|
|
requireWhenAccessed('RpcClient', './lib/RpcClient');
|
|
requireWhenAccessed('Wallet', './lib/Wallet');
|
|
requireWhenAccessed('WalletKey', './lib/WalletKey');
|
|
requireWhenAccessed('PeerManager', './lib/PeerManager');
|
|
requireWhenAccessed('Message', './lib/Message');
|
|
requireWhenAccessed('Electrum', './lib/Electrum');
|
|
requireWhenAccessed('Armory', './lib/Armory');
|
|
module.exports.Buffer = Buffer;
|
|
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"./lib/Base58":"6VqyzY","./lib/HierarchicalKey":"x1O6JW","bignum":58,"bindings":73,"buffer":80}],3:[function(require,module,exports){
|
|
if ('undefined' === typeof window) window = this;
|
|
Bitcoin = {};
|
|
if (typeof navigator === 'undefined') {
|
|
var navigator = {};
|
|
navigator.appName = 'NodeJS';
|
|
}
|
|
/*!
|
|
* Crypto-JS v2.0.0
|
|
* http://code.google.com/p/crypto-js/
|
|
* Copyright (c) 2009, Jeff Mott. All rights reserved.
|
|
* http://code.google.com/p/crypto-js/wiki/License
|
|
*/
|
|
|
|
var base64map = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
|
|
// Global Crypto object
|
|
var Crypto = window.Crypto = {};
|
|
|
|
// Crypto utilities
|
|
var util = Crypto.util = {
|
|
|
|
// Bit-wise rotate left
|
|
rotl: function (n, b) {
|
|
return (n << b) | (n >>> (32 - b));
|
|
},
|
|
|
|
// Bit-wise rotate right
|
|
rotr: function (n, b) {
|
|
return (n << (32 - b)) | (n >>> b);
|
|
},
|
|
|
|
// Swap big-endian to little-endian and vice versa
|
|
endian: function (n) {
|
|
|
|
// If number given, swap endian
|
|
if (n.constructor == Number) {
|
|
return util.rotl(n, 8) & 0x00FF00FF |
|
|
util.rotl(n, 24) & 0xFF00FF00;
|
|
}
|
|
|
|
// Else, assume array and swap all items
|
|
for (var i = 0; i < n.length; i++)
|
|
n[i] = util.endian(n[i]);
|
|
return n;
|
|
|
|
},
|
|
|
|
// Generate an array of any length of random bytes
|
|
randomBytes: function (n) {
|
|
for (var bytes = []; n > 0; n--)
|
|
bytes.push(Math.floor(Math.random() * 256));
|
|
return bytes;
|
|
},
|
|
|
|
// Convert a byte array to big-endian 32-bit words
|
|
bytesToWords: function (bytes) {
|
|
for (var words = [], i = 0, b = 0; i < bytes.length; i++, b += 8)
|
|
words[b >>> 5] |= bytes[i] << (24 - b % 32);
|
|
return words;
|
|
},
|
|
|
|
// Convert big-endian 32-bit words to a byte array
|
|
wordsToBytes: function (words) {
|
|
for (var bytes = [], b = 0; b < words.length * 32; b += 8)
|
|
bytes.push((words[b >>> 5] >>> (24 - b % 32)) & 0xFF);
|
|
return bytes;
|
|
},
|
|
|
|
// Convert a byte array to a hex string
|
|
bytesToHex: function (bytes) {
|
|
for (var hex = [], i = 0; i < bytes.length; i++) {
|
|
hex.push((bytes[i] >>> 4).toString(16));
|
|
hex.push((bytes[i] & 0xF).toString(16));
|
|
}
|
|
return hex.join("");
|
|
},
|
|
|
|
// Convert a hex string to a byte array
|
|
hexToBytes: function (hex) {
|
|
for (var bytes = [], c = 0; c < hex.length; c += 2)
|
|
bytes.push(parseInt(hex.substr(c, 2), 16));
|
|
return bytes;
|
|
},
|
|
|
|
// Convert a byte array to a base-64 string
|
|
bytesToBase64: function (bytes) {
|
|
|
|
// Use browser-native function if it exists
|
|
if (typeof btoa == "function") return btoa(Binary.bytesToString(bytes));
|
|
|
|
for(var base64 = [], i = 0; i < bytes.length; i += 3) {
|
|
var triplet = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];
|
|
for (var j = 0; j < 4; j++) {
|
|
if (i * 8 + j * 6 <= bytes.length * 8)
|
|
base64.push(base64map.charAt((triplet >>> 6 * (3 - j)) & 0x3F));
|
|
else base64.push("=");
|
|
}
|
|
}
|
|
|
|
return base64.join("");
|
|
|
|
},
|
|
|
|
// Convert a base-64 string to a byte array
|
|
base64ToBytes: function (base64) {
|
|
|
|
// Use browser-native function if it exists
|
|
if (typeof atob == "function") return Binary.stringToBytes(atob(base64));
|
|
|
|
// Remove non-base-64 characters
|
|
base64 = base64.replace(/[^A-Z0-9+\/]/ig, "");
|
|
|
|
for (var bytes = [], i = 0, imod4 = 0; i < base64.length; imod4 = ++i % 4) {
|
|
if (imod4 == 0) continue;
|
|
bytes.push(((base64map.indexOf(base64.charAt(i - 1)) & (Math.pow(2, -2 * imod4 + 8) - 1)) << (imod4 * 2)) |
|
|
(base64map.indexOf(base64.charAt(i)) >>> (6 - imod4 * 2)));
|
|
}
|
|
|
|
return bytes;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
// Crypto mode namespace
|
|
Crypto.mode = {};
|
|
|
|
// Crypto character encodings
|
|
var charenc = Crypto.charenc = {};
|
|
|
|
// UTF-8 encoding
|
|
var UTF8 = charenc.UTF8 = {
|
|
|
|
// Convert a string to a byte array
|
|
stringToBytes: function (str) {
|
|
return Binary.stringToBytes(unescape(encodeURIComponent(str)));
|
|
},
|
|
|
|
// Convert a byte array to a string
|
|
bytesToString: function (bytes) {
|
|
return decodeURIComponent(escape(Binary.bytesToString(bytes)));
|
|
}
|
|
|
|
};
|
|
|
|
// Binary encoding
|
|
var Binary = charenc.Binary = {
|
|
|
|
// Convert a string to a byte array
|
|
stringToBytes: function (str) {
|
|
for (var bytes = [], i = 0; i < str.length; i++)
|
|
bytes.push(str.charCodeAt(i));
|
|
return bytes;
|
|
},
|
|
|
|
// Convert a byte array to a string
|
|
bytesToString: function (bytes) {
|
|
for (var str = [], i = 0; i < bytes.length; i++)
|
|
str.push(String.fromCharCode(bytes[i]));
|
|
return str.join("");
|
|
}
|
|
|
|
};
|
|
|
|
/*
|
|
CryptoJS v3.1.2
|
|
code.google.com/p/crypto-js
|
|
(c) 2009-2013 by Jeff Mott. All rights reserved.
|
|
code.google.com/p/crypto-js/wiki/License
|
|
*/
|
|
/*
|
|
|
|
(c) 2012 by C?dric Mesnil. All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
|
|
|
|
- Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
|
|
- Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
var CryptoJS=CryptoJS||function(j,k){var e={},l=e.lib={},z=function(){},t=l.Base={extend:function(a){z.prototype=this;var c=new z;a&&c.mixIn(a);c.hasOwnProperty("init")||(c.init=function(){c.$super.init.apply(this,arguments)});c.init.prototype=c;c.$super=this;return c},create:function(){var a=this.extend();a.init.apply(a,arguments);return a},init:function(){},mixIn:function(a){for(var c in a)a.hasOwnProperty(c)&&(this[c]=a[c]);a.hasOwnProperty("toString")&&(this.toString=a.toString)},clone:function(){return this.init.prototype.extend(this)}},
|
|
u=l.WordArray=t.extend({init:function(a,c){a=this.words=a||[];this.sigBytes=c!=k?c:4*a.length},toString:function(a){return(a||D).stringify(this)},concat:function(a){var c=this.words,h=a.words,d=this.sigBytes;a=a.sigBytes;this.clamp();if(d%4)for(var b=0;b<a;b++)c[d+b>>>2]|=(h[b>>>2]>>>24-8*(b%4)&255)<<24-8*((d+b)%4);else if(65535<h.length)for(b=0;b<a;b+=4)c[d+b>>>2]=h[b>>>2];else c.push.apply(c,h);this.sigBytes+=a;return this},clamp:function(){var a=this.words,c=this.sigBytes;a[c>>>2]&=4294967295<<
|
|
32-8*(c%4);a.length=j.ceil(c/4)},clone:function(){var a=t.clone.call(this);a.words=this.words.slice(0);return a},random:function(a){for(var c=[],b=0;b<a;b+=4)c.push(4294967296*j.random()|0);return new u.init(c,a)}}),w=e.enc={},D=w.Hex={stringify:function(a){var c=a.words;a=a.sigBytes;for(var b=[],d=0;d<a;d++){var g=c[d>>>2]>>>24-8*(d%4)&255;b.push((g>>>4).toString(16));b.push((g&15).toString(16))}return b.join("")},parse:function(a){for(var c=a.length,b=[],d=0;d<c;d+=2)b[d>>>3]|=parseInt(a.substr(d,
|
|
2),16)<<24-4*(d%8);return new u.init(b,c/2)}},A=w.Latin1={stringify:function(a){var c=a.words;a=a.sigBytes;for(var b=[],d=0;d<a;d++)b.push(String.fromCharCode(c[d>>>2]>>>24-8*(d%4)&255));return b.join("")},parse:function(a){for(var b=a.length,h=[],d=0;d<b;d++)h[d>>>2]|=(a.charCodeAt(d)&255)<<24-8*(d%4);return new u.init(h,b)}},g=w.Utf8={stringify:function(a){try{return decodeURIComponent(escape(A.stringify(a)))}catch(b){throw Error("Malformed UTF-8 data");}},parse:function(a){return A.parse(unescape(encodeURIComponent(a)))}},
|
|
v=l.BufferedBlockAlgorithm=t.extend({reset:function(){this._data=new u.init;this._nDataBytes=0},_append:function(a){"string"==typeof a&&(a=g.parse(a));this._data.concat(a);this._nDataBytes+=a.sigBytes},_process:function(a){var b=this._data,h=b.words,d=b.sigBytes,g=this.blockSize,v=d/(4*g),v=a?j.ceil(v):j.max((v|0)-this._minBufferSize,0);a=v*g;d=j.min(4*a,d);if(a){for(var e=0;e<a;e+=g)this._doProcessBlock(h,e);e=h.splice(0,a);b.sigBytes-=d}return new u.init(e,d)},clone:function(){var a=t.clone.call(this);
|
|
a._data=this._data.clone();return a},_minBufferSize:0});l.Hasher=v.extend({cfg:t.extend(),init:function(a){this.cfg=this.cfg.extend(a);this.reset()},reset:function(){v.reset.call(this);this._doReset()},update:function(a){this._append(a);this._process();return this},finalize:function(a){a&&this._append(a);return this._doFinalize()},blockSize:16,_createHelper:function(a){return function(b,g){return(new a.init(g)).finalize(b)}},_createHmacHelper:function(a){return function(c,g){return(new b.HMAC.init(a,
|
|
g)).finalize(c)}}});var b=e.algo={};return e}(Math);
|
|
(function(){var j=CryptoJS,k=j.lib,e=k.WordArray,l=k.Hasher,k=j.algo,z=e.create([0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,7,4,13,1,10,6,15,3,12,0,9,5,2,14,11,8,3,10,14,4,9,15,8,1,2,7,0,6,13,11,5,12,1,9,11,10,0,8,12,4,13,3,7,15,14,5,6,2,4,0,5,9,7,12,2,10,14,1,3,8,11,6,15,13]),t=e.create([5,14,7,0,9,2,11,4,13,6,15,8,1,10,3,12,6,11,3,7,0,13,5,10,14,15,8,12,4,9,1,2,15,5,1,3,7,14,6,9,11,8,12,2,10,0,4,13,8,6,4,1,3,11,15,0,5,12,2,13,9,7,10,14,12,15,10,4,1,5,8,7,6,2,13,14,0,3,9,11]),u=e.create([11,14,15,12,
|
|
5,8,7,9,11,13,14,15,6,7,9,8,7,6,8,13,11,9,7,15,7,12,15,9,11,7,13,12,11,13,6,7,14,9,13,15,14,8,13,6,5,12,7,5,11,12,14,15,14,15,9,8,9,14,5,6,8,6,5,12,9,15,5,11,6,8,13,12,5,12,13,14,11,8,5,6]),w=e.create([8,9,9,11,13,15,15,5,7,7,8,11,14,14,12,6,9,13,15,7,12,8,9,11,7,7,12,7,6,15,13,11,9,7,15,11,8,6,6,14,12,13,5,14,13,13,7,5,15,5,8,11,14,14,6,14,6,9,12,9,12,5,15,8,8,5,12,9,12,5,14,6,8,13,6,5,15,13,11,11]),D=e.create([0,1518500249,1859775393,2400959708,2840853838]),A=e.create([1352829926,1548603684,1836072691,
|
|
2053994217,0]),k=k.RIPEMD160=l.extend({_doReset:function(){this._hash=e.create([1732584193,4023233417,2562383102,271733878,3285377520])},_doProcessBlock:function(g,e){for(var b=0;16>b;b++){var a=e+b,c=g[a];g[a]=(c<<8|c>>>24)&16711935|(c<<24|c>>>8)&4278255360}var a=this._hash.words,c=D.words,h=A.words,d=z.words,j=t.words,k=u.words,l=w.words,B,m,n,p,x,C,q,r,s,y;C=B=a[0];q=m=a[1];r=n=a[2];s=p=a[3];y=x=a[4];for(var f,b=0;80>b;b+=1)f=B+g[e+d[b]]|0,f=16>b?f+((m^n^p)+c[0]):32>b?f+((m&n|~m&p)+c[1]):48>b?
|
|
f+(((m|~n)^p)+c[2]):64>b?f+((m&p|n&~p)+c[3]):f+((m^(n|~p))+c[4]),f|=0,f=f<<k[b]|f>>>32-k[b],f=f+x|0,B=x,x=p,p=n<<10|n>>>22,n=m,m=f,f=C+g[e+j[b]]|0,f=16>b?f+((q^(r|~s))+h[0]):32>b?f+((q&s|r&~s)+h[1]):48>b?f+(((q|~r)^s)+h[2]):64>b?f+((q&r|~q&s)+h[3]):f+((q^r^s)+h[4]),f|=0,f=f<<l[b]|f>>>32-l[b],f=f+y|0,C=y,y=s,s=r<<10|r>>>22,r=q,q=f;f=a[1]+n+s|0;a[1]=a[2]+p+y|0;a[2]=a[3]+x+C|0;a[3]=a[4]+B+q|0;a[4]=a[0]+m+r|0;a[0]=f},_doFinalize:function(){var g=this._data,e=g.words,b=8*this._nDataBytes,a=8*g.sigBytes;
|
|
e[a>>>5]|=128<<24-a%32;e[(a+64>>>9<<4)+14]=(b<<8|b>>>24)&16711935|(b<<24|b>>>8)&4278255360;g.sigBytes=4*(e.length+1);this._process();g=this._hash;e=g.words;for(b=0;5>b;b++)a=e[b],e[b]=(a<<8|a>>>24)&16711935|(a<<24|a>>>8)&4278255360;return g},clone:function(){var e=l.clone.call(this);e._hash=this._hash.clone();return e}});j.RIPEMD160=l._createHelper(k);j.HmacRIPEMD160=l._createHmacHelper(k)})(Math);
|
|
|
|
module.exports.crypto31 = CryptoJS;
|
|
// Copyright (c) 2005 Tom Wu
|
|
// All Rights Reserved.
|
|
// See "LICENSE" for details.
|
|
|
|
// Basic JavaScript BN library - subset useful for RSA encryption.
|
|
|
|
// Bits per digit
|
|
var dbits;
|
|
|
|
// JavaScript engine analysis
|
|
var canary = 0xdeadbeefcafe;
|
|
var j_lm = ((canary&0xffffff)==0xefcafe);
|
|
|
|
// (public) Constructor
|
|
function BigInteger(a,b,c) {
|
|
if(a != null)
|
|
if("number" == typeof a) this.fromNumber(a,b,c);
|
|
else if(b == null && "string" != typeof a) this.fromString(a,256);
|
|
else this.fromString(a,b);
|
|
}
|
|
|
|
// return new, unset BigInteger
|
|
function nbi() { return new BigInteger(null); }
|
|
|
|
// am: Compute w_j += (x*this_i), propagate carries,
|
|
// c is initial carry, returns final carry.
|
|
// c < 3*dvalue, x < 2*dvalue, this_i < dvalue
|
|
// We need to select the fastest one that works in this environment.
|
|
|
|
// am1: use a single mult and divide to get the high bits,
|
|
// max digit bits should be 26 because
|
|
// max internal value = 2*dvalue^2-2*dvalue (< 2^53)
|
|
function am1(i,x,w,j,c,n) {
|
|
while(--n >= 0) {
|
|
var v = x*this[i++]+w[j]+c;
|
|
c = Math.floor(v/0x4000000);
|
|
w[j++] = v&0x3ffffff;
|
|
}
|
|
return c;
|
|
}
|
|
// am2 avoids a big mult-and-extract completely.
|
|
// Max digit bits should be <= 30 because we do bitwise ops
|
|
// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)
|
|
function am2(i,x,w,j,c,n) {
|
|
var xl = x&0x7fff, xh = x>>15;
|
|
while(--n >= 0) {
|
|
var l = this[i]&0x7fff;
|
|
var h = this[i++]>>15;
|
|
var m = xh*l+h*xl;
|
|
l = xl*l+((m&0x7fff)<<15)+w[j]+(c&0x3fffffff);
|
|
c = (l>>>30)+(m>>>15)+xh*h+(c>>>30);
|
|
w[j++] = l&0x3fffffff;
|
|
}
|
|
return c;
|
|
}
|
|
// Alternately, set max digit bits to 28 since some
|
|
// browsers slow down when dealing with 32-bit numbers.
|
|
function am3(i,x,w,j,c,n) {
|
|
var xl = x&0x3fff, xh = x>>14;
|
|
while(--n >= 0) {
|
|
var l = this[i]&0x3fff;
|
|
var h = this[i++]>>14;
|
|
var m = xh*l+h*xl;
|
|
l = xl*l+((m&0x3fff)<<14)+w[j]+c;
|
|
c = (l>>28)+(m>>14)+xh*h;
|
|
w[j++] = l&0xfffffff;
|
|
}
|
|
return c;
|
|
}
|
|
if(j_lm && (navigator.appName == "Microsoft Internet Explorer")) {
|
|
BigInteger.prototype.am = am2;
|
|
dbits = 30;
|
|
}
|
|
else if(j_lm && (navigator.appName != "Netscape")) {
|
|
BigInteger.prototype.am = am1;
|
|
dbits = 26;
|
|
}
|
|
else { // Mozilla/Netscape seems to prefer am3
|
|
BigInteger.prototype.am = am3;
|
|
dbits = 28;
|
|
}
|
|
|
|
BigInteger.prototype.DB = dbits;
|
|
BigInteger.prototype.DM = ((1<<dbits)-1);
|
|
BigInteger.prototype.DV = (1<<dbits);
|
|
|
|
var BI_FP = 52;
|
|
BigInteger.prototype.FV = Math.pow(2,BI_FP);
|
|
BigInteger.prototype.F1 = BI_FP-dbits;
|
|
BigInteger.prototype.F2 = 2*dbits-BI_FP;
|
|
|
|
// Digit conversions
|
|
var BI_RM = "0123456789abcdefghijklmnopqrstuvwxyz";
|
|
var BI_RC = new Array();
|
|
var rr,vv;
|
|
rr = "0".charCodeAt(0);
|
|
for(vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;
|
|
rr = "a".charCodeAt(0);
|
|
for(vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
|
|
rr = "A".charCodeAt(0);
|
|
for(vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
|
|
|
|
function int2char(n) { return BI_RM.charAt(n); }
|
|
function intAt(s,i) {
|
|
var c = BI_RC[s.charCodeAt(i)];
|
|
return (c==null)?-1:c;
|
|
}
|
|
|
|
// (protected) copy this to r
|
|
function bnpCopyTo(r) {
|
|
for(var i = this.t-1; i >= 0; --i) r[i] = this[i];
|
|
r.t = this.t;
|
|
r.s = this.s;
|
|
}
|
|
|
|
// (protected) set from integer value x, -DV <= x < DV
|
|
function bnpFromInt(x) {
|
|
this.t = 1;
|
|
this.s = (x<0)?-1:0;
|
|
if(x > 0) this[0] = x;
|
|
else if(x < -1) this[0] = x+DV;
|
|
else this.t = 0;
|
|
}
|
|
|
|
// return bigint initialized to value
|
|
function nbv(i) { var r = nbi(); r.fromInt(i); return r; }
|
|
|
|
// (protected) set from string and radix
|
|
function bnpFromString(s,b) {
|
|
var k;
|
|
if(b == 16) k = 4;
|
|
else if(b == 8) k = 3;
|
|
else if(b == 256) k = 8; // byte array
|
|
else if(b == 2) k = 1;
|
|
else if(b == 32) k = 5;
|
|
else if(b == 4) k = 2;
|
|
else { this.fromRadix(s,b); return; }
|
|
this.t = 0;
|
|
this.s = 0;
|
|
var i = s.length, mi = false, sh = 0;
|
|
while(--i >= 0) {
|
|
var x = (k==8)?s[i]&0xff:intAt(s,i);
|
|
if(x < 0) {
|
|
if(s.charAt(i) == "-") mi = true;
|
|
continue;
|
|
}
|
|
mi = false;
|
|
if(sh == 0)
|
|
this[this.t++] = x;
|
|
else if(sh+k > this.DB) {
|
|
this[this.t-1] |= (x&((1<<(this.DB-sh))-1))<<sh;
|
|
this[this.t++] = (x>>(this.DB-sh));
|
|
}
|
|
else
|
|
this[this.t-1] |= x<<sh;
|
|
sh += k;
|
|
if(sh >= this.DB) sh -= this.DB;
|
|
}
|
|
if(k == 8 && (s[0]&0x80) != 0) {
|
|
this.s = -1;
|
|
if(sh > 0) this[this.t-1] |= ((1<<(this.DB-sh))-1)<<sh;
|
|
}
|
|
this.clamp();
|
|
if(mi) BigInteger.ZERO.subTo(this,this);
|
|
}
|
|
|
|
// (protected) clamp off excess high words
|
|
function bnpClamp() {
|
|
var c = this.s&this.DM;
|
|
while(this.t > 0 && this[this.t-1] == c) --this.t;
|
|
}
|
|
|
|
// (public) return string representation in given radix
|
|
function bnToString(b) {
|
|
if(this.s < 0) return "-"+this.negate().toString(b);
|
|
var k;
|
|
if(b == 16) k = 4;
|
|
else if(b == 8) k = 3;
|
|
else if(b == 2) k = 1;
|
|
else if(b == 32) k = 5;
|
|
else if(b == 4) k = 2;
|
|
else return this.toRadix(b);
|
|
var km = (1<<k)-1, d, m = false, r = "", i = this.t;
|
|
var p = this.DB-(i*this.DB)%k;
|
|
if(i-- > 0) {
|
|
if(p < this.DB && (d = this[i]>>p) > 0) { m = true; r = int2char(d); }
|
|
while(i >= 0) {
|
|
if(p < k) {
|
|
d = (this[i]&((1<<p)-1))<<(k-p);
|
|
d |= this[--i]>>(p+=this.DB-k);
|
|
}
|
|
else {
|
|
d = (this[i]>>(p-=k))&km;
|
|
if(p <= 0) { p += this.DB; --i; }
|
|
}
|
|
if(d > 0) m = true;
|
|
if(m) r += int2char(d);
|
|
}
|
|
}
|
|
return m?r:"0";
|
|
}
|
|
|
|
// (public) -this
|
|
function bnNegate() { var r = nbi(); BigInteger.ZERO.subTo(this,r); return r; }
|
|
|
|
// (public) |this|
|
|
function bnAbs() { return (this.s<0)?this.negate():this; }
|
|
|
|
// (public) return + if this > a, - if this < a, 0 if equal
|
|
function bnCompareTo(a) {
|
|
var r = this.s-a.s;
|
|
if(r != 0) return r;
|
|
var i = this.t;
|
|
r = i-a.t;
|
|
if(r != 0) return (this.s<0)?-r:r;
|
|
while(--i >= 0) if((r=this[i]-a[i]) != 0) return r;
|
|
return 0;
|
|
}
|
|
|
|
// returns bit length of the integer x
|
|
function nbits(x) {
|
|
var r = 1, t;
|
|
if((t=x>>>16) != 0) { x = t; r += 16; }
|
|
if((t=x>>8) != 0) { x = t; r += 8; }
|
|
if((t=x>>4) != 0) { x = t; r += 4; }
|
|
if((t=x>>2) != 0) { x = t; r += 2; }
|
|
if((t=x>>1) != 0) { x = t; r += 1; }
|
|
return r;
|
|
}
|
|
|
|
// (public) return the number of bits in "this"
|
|
function bnBitLength() {
|
|
if(this.t <= 0) return 0;
|
|
return this.DB*(this.t-1)+nbits(this[this.t-1]^(this.s&this.DM));
|
|
}
|
|
|
|
// (protected) r = this << n*DB
|
|
function bnpDLShiftTo(n,r) {
|
|
var i;
|
|
for(i = this.t-1; i >= 0; --i) r[i+n] = this[i];
|
|
for(i = n-1; i >= 0; --i) r[i] = 0;
|
|
r.t = this.t+n;
|
|
r.s = this.s;
|
|
}
|
|
|
|
// (protected) r = this >> n*DB
|
|
function bnpDRShiftTo(n,r) {
|
|
for(var i = n; i < this.t; ++i) r[i-n] = this[i];
|
|
r.t = Math.max(this.t-n,0);
|
|
r.s = this.s;
|
|
}
|
|
|
|
// (protected) r = this << n
|
|
function bnpLShiftTo(n,r) {
|
|
var bs = n%this.DB;
|
|
var cbs = this.DB-bs;
|
|
var bm = (1<<cbs)-1;
|
|
var ds = Math.floor(n/this.DB), c = (this.s<<bs)&this.DM, i;
|
|
for(i = this.t-1; i >= 0; --i) {
|
|
r[i+ds+1] = (this[i]>>cbs)|c;
|
|
c = (this[i]&bm)<<bs;
|
|
}
|
|
for(i = ds-1; i >= 0; --i) r[i] = 0;
|
|
r[ds] = c;
|
|
r.t = this.t+ds+1;
|
|
r.s = this.s;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = this >> n
|
|
function bnpRShiftTo(n,r) {
|
|
r.s = this.s;
|
|
var ds = Math.floor(n/this.DB);
|
|
if(ds >= this.t) { r.t = 0; return; }
|
|
var bs = n%this.DB;
|
|
var cbs = this.DB-bs;
|
|
var bm = (1<<bs)-1;
|
|
r[0] = this[ds]>>bs;
|
|
for(var i = ds+1; i < this.t; ++i) {
|
|
r[i-ds-1] |= (this[i]&bm)<<cbs;
|
|
r[i-ds] = this[i]>>bs;
|
|
}
|
|
if(bs > 0) r[this.t-ds-1] |= (this.s&bm)<<cbs;
|
|
r.t = this.t-ds;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = this - a
|
|
function bnpSubTo(a,r) {
|
|
var i = 0, c = 0, m = Math.min(a.t,this.t);
|
|
while(i < m) {
|
|
c += this[i]-a[i];
|
|
r[i++] = c&this.DM;
|
|
c >>= this.DB;
|
|
}
|
|
if(a.t < this.t) {
|
|
c -= a.s;
|
|
while(i < this.t) {
|
|
c += this[i];
|
|
r[i++] = c&this.DM;
|
|
c >>= this.DB;
|
|
}
|
|
c += this.s;
|
|
}
|
|
else {
|
|
c += this.s;
|
|
while(i < a.t) {
|
|
c -= a[i];
|
|
r[i++] = c&this.DM;
|
|
c >>= this.DB;
|
|
}
|
|
c -= a.s;
|
|
}
|
|
r.s = (c<0)?-1:0;
|
|
if(c < -1) r[i++] = this.DV+c;
|
|
else if(c > 0) r[i++] = c;
|
|
r.t = i;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = this * a, r != this,a (HAC 14.12)
|
|
// "this" should be the larger one if appropriate.
|
|
function bnpMultiplyTo(a,r) {
|
|
var x = this.abs(), y = a.abs();
|
|
var i = x.t;
|
|
r.t = i+y.t;
|
|
while(--i >= 0) r[i] = 0;
|
|
for(i = 0; i < y.t; ++i) r[i+x.t] = x.am(0,y[i],r,i,0,x.t);
|
|
r.s = 0;
|
|
r.clamp();
|
|
if(this.s != a.s) BigInteger.ZERO.subTo(r,r);
|
|
}
|
|
|
|
// (protected) r = this^2, r != this (HAC 14.16)
|
|
function bnpSquareTo(r) {
|
|
var x = this.abs();
|
|
var i = r.t = 2*x.t;
|
|
while(--i >= 0) r[i] = 0;
|
|
for(i = 0; i < x.t-1; ++i) {
|
|
var c = x.am(i,x[i],r,2*i,0,1);
|
|
if((r[i+x.t]+=x.am(i+1,2*x[i],r,2*i+1,c,x.t-i-1)) >= x.DV) {
|
|
r[i+x.t] -= x.DV;
|
|
r[i+x.t+1] = 1;
|
|
}
|
|
}
|
|
if(r.t > 0) r[r.t-1] += x.am(i,x[i],r,2*i,0,1);
|
|
r.s = 0;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)
|
|
// r != q, this != m. q or r may be null.
|
|
function bnpDivRemTo(m,q,r) {
|
|
var pm = m.abs();
|
|
if(pm.t <= 0) return;
|
|
var pt = this.abs();
|
|
if(pt.t < pm.t) {
|
|
if(q != null) q.fromInt(0);
|
|
if(r != null) this.copyTo(r);
|
|
return;
|
|
}
|
|
if(r == null) r = nbi();
|
|
var y = nbi(), ts = this.s, ms = m.s;
|
|
var nsh = this.DB-nbits(pm[pm.t-1]); // normalize modulus
|
|
if(nsh > 0) { pm.lShiftTo(nsh,y); pt.lShiftTo(nsh,r); }
|
|
else { pm.copyTo(y); pt.copyTo(r); }
|
|
var ys = y.t;
|
|
var y0 = y[ys-1];
|
|
if(y0 == 0) return;
|
|
var yt = y0*(1<<this.F1)+((ys>1)?y[ys-2]>>this.F2:0);
|
|
var d1 = this.FV/yt, d2 = (1<<this.F1)/yt, e = 1<<this.F2;
|
|
var i = r.t, j = i-ys, t = (q==null)?nbi():q;
|
|
y.dlShiftTo(j,t);
|
|
if(r.compareTo(t) >= 0) {
|
|
r[r.t++] = 1;
|
|
r.subTo(t,r);
|
|
}
|
|
BigInteger.ONE.dlShiftTo(ys,t);
|
|
t.subTo(y,y); // "negative" y so we can replace sub with am later
|
|
while(y.t < ys) y[y.t++] = 0;
|
|
while(--j >= 0) {
|
|
// Estimate quotient digit
|
|
var qd = (r[--i]==y0)?this.DM:Math.floor(r[i]*d1+(r[i-1]+e)*d2);
|
|
if((r[i]+=y.am(0,qd,r,j,0,ys)) < qd) { // Try it out
|
|
y.dlShiftTo(j,t);
|
|
r.subTo(t,r);
|
|
while(r[i] < --qd) r.subTo(t,r);
|
|
}
|
|
}
|
|
if(q != null) {
|
|
r.drShiftTo(ys,q);
|
|
if(ts != ms) BigInteger.ZERO.subTo(q,q);
|
|
}
|
|
r.t = ys;
|
|
r.clamp();
|
|
if(nsh > 0) r.rShiftTo(nsh,r); // Denormalize remainder
|
|
if(ts < 0) BigInteger.ZERO.subTo(r,r);
|
|
}
|
|
|
|
// (public) this mod a
|
|
function bnMod(a) {
|
|
var r = nbi();
|
|
this.abs().divRemTo(a,null,r);
|
|
if(this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r,r);
|
|
return r;
|
|
}
|
|
|
|
// Modular reduction using "classic" algorithm
|
|
function Classic(m) { this.m = m; }
|
|
function cConvert(x) {
|
|
if(x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);
|
|
else return x;
|
|
}
|
|
function cRevert(x) { return x; }
|
|
function cReduce(x) { x.divRemTo(this.m,null,x); }
|
|
function cMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
|
|
function cSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
|
|
|
|
Classic.prototype.convert = cConvert;
|
|
Classic.prototype.revert = cRevert;
|
|
Classic.prototype.reduce = cReduce;
|
|
Classic.prototype.mulTo = cMulTo;
|
|
Classic.prototype.sqrTo = cSqrTo;
|
|
|
|
// (protected) return "-1/this % 2^DB"; useful for Mont. reduction
|
|
// justification:
|
|
// xy == 1 (mod m)
|
|
// xy = 1+km
|
|
// xy(2-xy) = (1+km)(1-km)
|
|
// x[y(2-xy)] = 1-k^2m^2
|
|
// x[y(2-xy)] == 1 (mod m^2)
|
|
// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2
|
|
// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.
|
|
// JS multiply "overflows" differently from C/C++, so care is needed here.
|
|
function bnpInvDigit() {
|
|
if(this.t < 1) return 0;
|
|
var x = this[0];
|
|
if((x&1) == 0) return 0;
|
|
var y = x&3; // y == 1/x mod 2^2
|
|
y = (y*(2-(x&0xf)*y))&0xf; // y == 1/x mod 2^4
|
|
y = (y*(2-(x&0xff)*y))&0xff; // y == 1/x mod 2^8
|
|
y = (y*(2-(((x&0xffff)*y)&0xffff)))&0xffff; // y == 1/x mod 2^16
|
|
// last step - calculate inverse mod DV directly;
|
|
// assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints
|
|
y = (y*(2-x*y%this.DV))%this.DV; // y == 1/x mod 2^dbits
|
|
// we really want the negative inverse, and -DV < y < DV
|
|
return (y>0)?this.DV-y:-y;
|
|
}
|
|
|
|
// Montgomery reduction
|
|
function Montgomery(m) {
|
|
this.m = m;
|
|
this.mp = m.invDigit();
|
|
this.mpl = this.mp&0x7fff;
|
|
this.mph = this.mp>>15;
|
|
this.um = (1<<(m.DB-15))-1;
|
|
this.mt2 = 2*m.t;
|
|
}
|
|
|
|
// xR mod m
|
|
function montConvert(x) {
|
|
var r = nbi();
|
|
x.abs().dlShiftTo(this.m.t,r);
|
|
r.divRemTo(this.m,null,r);
|
|
if(x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r,r);
|
|
return r;
|
|
}
|
|
|
|
// x/R mod m
|
|
function montRevert(x) {
|
|
var r = nbi();
|
|
x.copyTo(r);
|
|
this.reduce(r);
|
|
return r;
|
|
}
|
|
|
|
// x = x/R mod m (HAC 14.32)
|
|
function montReduce(x) {
|
|
while(x.t <= this.mt2) // pad x so am has enough room later
|
|
x[x.t++] = 0;
|
|
for(var i = 0; i < this.m.t; ++i) {
|
|
// faster way of calculating u0 = x[i]*mp mod DV
|
|
var j = x[i]&0x7fff;
|
|
var u0 = (j*this.mpl+(((j*this.mph+(x[i]>>15)*this.mpl)&this.um)<<15))&x.DM;
|
|
// use am to combine the multiply-shift-add into one call
|
|
j = i+this.m.t;
|
|
x[j] += this.m.am(0,u0,x,i,0,this.m.t);
|
|
// propagate carry
|
|
while(x[j] >= x.DV) { x[j] -= x.DV; x[++j]++; }
|
|
}
|
|
x.clamp();
|
|
x.drShiftTo(this.m.t,x);
|
|
if(x.compareTo(this.m) >= 0) x.subTo(this.m,x);
|
|
}
|
|
|
|
// r = "x^2/R mod m"; x != r
|
|
function montSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
|
|
|
|
// r = "xy/R mod m"; x,y != r
|
|
function montMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
|
|
|
|
Montgomery.prototype.convert = montConvert;
|
|
Montgomery.prototype.revert = montRevert;
|
|
Montgomery.prototype.reduce = montReduce;
|
|
Montgomery.prototype.mulTo = montMulTo;
|
|
Montgomery.prototype.sqrTo = montSqrTo;
|
|
|
|
// (protected) true iff this is even
|
|
function bnpIsEven() { return ((this.t>0)?(this[0]&1):this.s) == 0; }
|
|
|
|
// (protected) this^e, e < 2^32, doing sqr and mul with "r" (HAC 14.79)
|
|
function bnpExp(e,z) {
|
|
if(e > 0xffffffff || e < 1) return BigInteger.ONE;
|
|
var r = nbi(), r2 = nbi(), g = z.convert(this), i = nbits(e)-1;
|
|
g.copyTo(r);
|
|
while(--i >= 0) {
|
|
z.sqrTo(r,r2);
|
|
if((e&(1<<i)) > 0) z.mulTo(r2,g,r);
|
|
else { var t = r; r = r2; r2 = t; }
|
|
}
|
|
return z.revert(r);
|
|
}
|
|
|
|
// (public) this^e % m, 0 <= e < 2^32
|
|
function bnModPowInt(e,m) {
|
|
var z;
|
|
if(e < 256 || m.isEven()) z = new Classic(m); else z = new Montgomery(m);
|
|
return this.exp(e,z);
|
|
}
|
|
|
|
// protected
|
|
BigInteger.prototype.copyTo = bnpCopyTo;
|
|
BigInteger.prototype.fromInt = bnpFromInt;
|
|
BigInteger.prototype.fromString = bnpFromString;
|
|
BigInteger.prototype.clamp = bnpClamp;
|
|
BigInteger.prototype.dlShiftTo = bnpDLShiftTo;
|
|
BigInteger.prototype.drShiftTo = bnpDRShiftTo;
|
|
BigInteger.prototype.lShiftTo = bnpLShiftTo;
|
|
BigInteger.prototype.rShiftTo = bnpRShiftTo;
|
|
BigInteger.prototype.subTo = bnpSubTo;
|
|
BigInteger.prototype.multiplyTo = bnpMultiplyTo;
|
|
BigInteger.prototype.squareTo = bnpSquareTo;
|
|
BigInteger.prototype.divRemTo = bnpDivRemTo;
|
|
BigInteger.prototype.invDigit = bnpInvDigit;
|
|
BigInteger.prototype.isEven = bnpIsEven;
|
|
BigInteger.prototype.exp = bnpExp;
|
|
|
|
// public
|
|
BigInteger.prototype.toString = bnToString;
|
|
BigInteger.prototype.negate = bnNegate;
|
|
BigInteger.prototype.abs = bnAbs;
|
|
BigInteger.prototype.compareTo = bnCompareTo;
|
|
BigInteger.prototype.bitLength = bnBitLength;
|
|
BigInteger.prototype.mod = bnMod;
|
|
BigInteger.prototype.modPowInt = bnModPowInt;
|
|
|
|
// "constants"
|
|
BigInteger.ZERO = nbv(0);
|
|
BigInteger.ONE = nbv(1);
|
|
// Copyright (c) 2005-2009 Tom Wu
|
|
// All Rights Reserved.
|
|
// See "LICENSE" for details.
|
|
|
|
// Extended JavaScript BN functions, required for RSA private ops.
|
|
|
|
// Version 1.1: new BigInteger("0", 10) returns "proper" zero
|
|
// Version 1.2: square() API, isProbablePrime fix
|
|
|
|
// (public)
|
|
function bnClone() { var r = nbi(); this.copyTo(r); return r; }
|
|
|
|
// (public) return value as integer
|
|
function bnIntValue() {
|
|
if(this.s < 0) {
|
|
if(this.t == 1) return this[0]-this.DV;
|
|
else if(this.t == 0) return -1;
|
|
}
|
|
else if(this.t == 1) return this[0];
|
|
else if(this.t == 0) return 0;
|
|
// assumes 16 < DB < 32
|
|
return ((this[1]&((1<<(32-this.DB))-1))<<this.DB)|this[0];
|
|
}
|
|
|
|
// (public) return value as byte
|
|
function bnByteValue() { return (this.t==0)?this.s:(this[0]<<24)>>24; }
|
|
|
|
// (public) return value as short (assumes DB>=16)
|
|
function bnShortValue() { return (this.t==0)?this.s:(this[0]<<16)>>16; }
|
|
|
|
// (protected) return x s.t. r^x < DV
|
|
function bnpChunkSize(r) { return Math.floor(Math.LN2*this.DB/Math.log(r)); }
|
|
|
|
// (public) 0 if this == 0, 1 if this > 0
|
|
function bnSigNum() {
|
|
if(this.s < 0) return -1;
|
|
else if(this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;
|
|
else return 1;
|
|
}
|
|
|
|
// (protected) convert to radix string
|
|
function bnpToRadix(b) {
|
|
if(b == null) b = 10;
|
|
if(this.signum() == 0 || b < 2 || b > 36) return "0";
|
|
var cs = this.chunkSize(b);
|
|
var a = Math.pow(b,cs);
|
|
var d = nbv(a), y = nbi(), z = nbi(), r = "";
|
|
this.divRemTo(d,y,z);
|
|
while(y.signum() > 0) {
|
|
r = (a+z.intValue()).toString(b).substr(1) + r;
|
|
y.divRemTo(d,y,z);
|
|
}
|
|
return z.intValue().toString(b) + r;
|
|
}
|
|
|
|
// (protected) convert from radix string
|
|
function bnpFromRadix(s,b) {
|
|
this.fromInt(0);
|
|
if(b == null) b = 10;
|
|
var cs = this.chunkSize(b);
|
|
var d = Math.pow(b,cs), mi = false, j = 0, w = 0;
|
|
for(var i = 0; i < s.length; ++i) {
|
|
var x = intAt(s,i);
|
|
if(x < 0) {
|
|
if(s.charAt(i) == "-" && this.signum() == 0) mi = true;
|
|
continue;
|
|
}
|
|
w = b*w+x;
|
|
if(++j >= cs) {
|
|
this.dMultiply(d);
|
|
this.dAddOffset(w,0);
|
|
j = 0;
|
|
w = 0;
|
|
}
|
|
}
|
|
if(j > 0) {
|
|
this.dMultiply(Math.pow(b,j));
|
|
this.dAddOffset(w,0);
|
|
}
|
|
if(mi) BigInteger.ZERO.subTo(this,this);
|
|
}
|
|
|
|
// (protected) alternate constructor
|
|
function bnpFromNumber(a,b,c) {
|
|
if("number" == typeof b) {
|
|
// new BigInteger(int,int,RNG)
|
|
if(a < 2) this.fromInt(1);
|
|
else {
|
|
this.fromNumber(a,c);
|
|
if(!this.testBit(a-1)) // force MSB set
|
|
this.bitwiseTo(BigInteger.ONE.shiftLeft(a-1),op_or,this);
|
|
if(this.isEven()) this.dAddOffset(1,0); // force odd
|
|
while(!this.isProbablePrime(b)) {
|
|
this.dAddOffset(2,0);
|
|
if(this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a-1),this);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// new BigInteger(int,RNG)
|
|
var x = new Array(), t = a&7;
|
|
x.length = (a>>3)+1;
|
|
b.nextBytes(x);
|
|
if(t > 0) x[0] &= ((1<<t)-1); else x[0] = 0;
|
|
this.fromString(x,256);
|
|
}
|
|
}
|
|
|
|
// (public) convert to bigendian byte array
|
|
function bnToByteArray() {
|
|
var i = this.t, r = new Array();
|
|
r[0] = this.s;
|
|
var p = this.DB-(i*this.DB)%8, d, k = 0;
|
|
if(i-- > 0) {
|
|
if(p < this.DB && (d = this[i]>>p) != (this.s&this.DM)>>p)
|
|
r[k++] = d|(this.s<<(this.DB-p));
|
|
while(i >= 0) {
|
|
if(p < 8) {
|
|
d = (this[i]&((1<<p)-1))<<(8-p);
|
|
d |= this[--i]>>(p+=this.DB-8);
|
|
}
|
|
else {
|
|
d = (this[i]>>(p-=8))&0xff;
|
|
if(p <= 0) { p += this.DB; --i; }
|
|
}
|
|
if((d&0x80) != 0) d |= -256;
|
|
if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;
|
|
if(k > 0 || d != this.s) r[k++] = d;
|
|
}
|
|
}
|
|
return r;
|
|
}
|
|
|
|
function bnEquals(a) { return(this.compareTo(a)==0); }
|
|
function bnMin(a) { return(this.compareTo(a)<0)?this:a; }
|
|
function bnMax(a) { return(this.compareTo(a)>0)?this:a; }
|
|
|
|
// (protected) r = this op a (bitwise)
|
|
function bnpBitwiseTo(a,op,r) {
|
|
var i, f, m = Math.min(a.t,this.t);
|
|
for(i = 0; i < m; ++i) r[i] = op(this[i],a[i]);
|
|
if(a.t < this.t) {
|
|
f = a.s&this.DM;
|
|
for(i = m; i < this.t; ++i) r[i] = op(this[i],f);
|
|
r.t = this.t;
|
|
}
|
|
else {
|
|
f = this.s&this.DM;
|
|
for(i = m; i < a.t; ++i) r[i] = op(f,a[i]);
|
|
r.t = a.t;
|
|
}
|
|
r.s = op(this.s,a.s);
|
|
r.clamp();
|
|
}
|
|
|
|
// (public) this & a
|
|
function op_and(x,y) { return x&y; }
|
|
function bnAnd(a) { var r = nbi(); this.bitwiseTo(a,op_and,r); return r; }
|
|
|
|
// (public) this | a
|
|
function op_or(x,y) { return x|y; }
|
|
function bnOr(a) { var r = nbi(); this.bitwiseTo(a,op_or,r); return r; }
|
|
|
|
// (public) this ^ a
|
|
function op_xor(x,y) { return x^y; }
|
|
function bnXor(a) { var r = nbi(); this.bitwiseTo(a,op_xor,r); return r; }
|
|
|
|
// (public) this & ~a
|
|
function op_andnot(x,y) { return x&~y; }
|
|
function bnAndNot(a) { var r = nbi(); this.bitwiseTo(a,op_andnot,r); return r; }
|
|
|
|
// (public) ~this
|
|
function bnNot() {
|
|
var r = nbi();
|
|
for(var i = 0; i < this.t; ++i) r[i] = this.DM&~this[i];
|
|
r.t = this.t;
|
|
r.s = ~this.s;
|
|
return r;
|
|
}
|
|
|
|
// (public) this << n
|
|
function bnShiftLeft(n) {
|
|
var r = nbi();
|
|
if(n < 0) this.rShiftTo(-n,r); else this.lShiftTo(n,r);
|
|
return r;
|
|
}
|
|
|
|
// (public) this >> n
|
|
function bnShiftRight(n) {
|
|
var r = nbi();
|
|
if(n < 0) this.lShiftTo(-n,r); else this.rShiftTo(n,r);
|
|
return r;
|
|
}
|
|
|
|
// return index of lowest 1-bit in x, x < 2^31
|
|
function lbit(x) {
|
|
if(x == 0) return -1;
|
|
var r = 0;
|
|
if((x&0xffff) == 0) { x >>= 16; r += 16; }
|
|
if((x&0xff) == 0) { x >>= 8; r += 8; }
|
|
if((x&0xf) == 0) { x >>= 4; r += 4; }
|
|
if((x&3) == 0) { x >>= 2; r += 2; }
|
|
if((x&1) == 0) ++r;
|
|
return r;
|
|
}
|
|
|
|
// (public) returns index of lowest 1-bit (or -1 if none)
|
|
function bnGetLowestSetBit() {
|
|
for(var i = 0; i < this.t; ++i)
|
|
if(this[i] != 0) return i*this.DB+lbit(this[i]);
|
|
if(this.s < 0) return this.t*this.DB;
|
|
return -1;
|
|
}
|
|
|
|
// return number of 1 bits in x
|
|
function cbit(x) {
|
|
var r = 0;
|
|
while(x != 0) { x &= x-1; ++r; }
|
|
return r;
|
|
}
|
|
|
|
// (public) return number of set bits
|
|
function bnBitCount() {
|
|
var r = 0, x = this.s&this.DM;
|
|
for(var i = 0; i < this.t; ++i) r += cbit(this[i]^x);
|
|
return r;
|
|
}
|
|
|
|
// (public) true iff nth bit is set
|
|
function bnTestBit(n) {
|
|
var j = Math.floor(n/this.DB);
|
|
if(j >= this.t) return(this.s!=0);
|
|
return((this[j]&(1<<(n%this.DB)))!=0);
|
|
}
|
|
|
|
// (protected) this op (1<<n)
|
|
function bnpChangeBit(n,op) {
|
|
var r = BigInteger.ONE.shiftLeft(n);
|
|
this.bitwiseTo(r,op,r);
|
|
return r;
|
|
}
|
|
|
|
// (public) this | (1<<n)
|
|
function bnSetBit(n) { return this.changeBit(n,op_or); }
|
|
|
|
// (public) this & ~(1<<n)
|
|
function bnClearBit(n) { return this.changeBit(n,op_andnot); }
|
|
|
|
// (public) this ^ (1<<n)
|
|
function bnFlipBit(n) { return this.changeBit(n,op_xor); }
|
|
|
|
// (protected) r = this + a
|
|
function bnpAddTo(a,r) {
|
|
var i = 0, c = 0, m = Math.min(a.t,this.t);
|
|
while(i < m) {
|
|
c += this[i]+a[i];
|
|
r[i++] = c&this.DM;
|
|
c >>= this.DB;
|
|
}
|
|
if(a.t < this.t) {
|
|
c += a.s;
|
|
while(i < this.t) {
|
|
c += this[i];
|
|
r[i++] = c&this.DM;
|
|
c >>= this.DB;
|
|
}
|
|
c += this.s;
|
|
}
|
|
else {
|
|
c += this.s;
|
|
while(i < a.t) {
|
|
c += a[i];
|
|
r[i++] = c&this.DM;
|
|
c >>= this.DB;
|
|
}
|
|
c += a.s;
|
|
}
|
|
r.s = (c<0)?-1:0;
|
|
if(c > 0) r[i++] = c;
|
|
else if(c < -1) r[i++] = this.DV+c;
|
|
r.t = i;
|
|
r.clamp();
|
|
}
|
|
|
|
// (public) this + a
|
|
function bnAdd(a) { var r = nbi(); this.addTo(a,r); return r; }
|
|
|
|
// (public) this - a
|
|
function bnSubtract(a) { var r = nbi(); this.subTo(a,r); return r; }
|
|
|
|
// (public) this * a
|
|
function bnMultiply(a) { var r = nbi(); this.multiplyTo(a,r); return r; }
|
|
|
|
// (public) this^2
|
|
function bnSquare() { var r = nbi(); this.squareTo(r); return r; }
|
|
|
|
// (public) this / a
|
|
function bnDivide(a) { var r = nbi(); this.divRemTo(a,r,null); return r; }
|
|
|
|
// (public) this % a
|
|
function bnRemainder(a) { var r = nbi(); this.divRemTo(a,null,r); return r; }
|
|
|
|
// (public) [this/a,this%a]
|
|
function bnDivideAndRemainder(a) {
|
|
var q = nbi(), r = nbi();
|
|
this.divRemTo(a,q,r);
|
|
return new Array(q,r);
|
|
}
|
|
|
|
// (protected) this *= n, this >= 0, 1 < n < DV
|
|
function bnpDMultiply(n) {
|
|
this[this.t] = this.am(0,n-1,this,0,0,this.t);
|
|
++this.t;
|
|
this.clamp();
|
|
}
|
|
|
|
// (protected) this += n << w words, this >= 0
|
|
function bnpDAddOffset(n,w) {
|
|
if(n == 0) return;
|
|
while(this.t <= w) this[this.t++] = 0;
|
|
this[w] += n;
|
|
while(this[w] >= this.DV) {
|
|
this[w] -= this.DV;
|
|
if(++w >= this.t) this[this.t++] = 0;
|
|
++this[w];
|
|
}
|
|
}
|
|
|
|
// A "null" reducer
|
|
function NullExp() {}
|
|
function nNop(x) { return x; }
|
|
function nMulTo(x,y,r) { x.multiplyTo(y,r); }
|
|
function nSqrTo(x,r) { x.squareTo(r); }
|
|
|
|
NullExp.prototype.convert = nNop;
|
|
NullExp.prototype.revert = nNop;
|
|
NullExp.prototype.mulTo = nMulTo;
|
|
NullExp.prototype.sqrTo = nSqrTo;
|
|
|
|
// (public) this^e
|
|
function bnPow(e) { return this.exp(e,new NullExp()); }
|
|
|
|
// (protected) r = lower n words of "this * a", a.t <= n
|
|
// "this" should be the larger one if appropriate.
|
|
function bnpMultiplyLowerTo(a,n,r) {
|
|
var i = Math.min(this.t+a.t,n);
|
|
r.s = 0; // assumes a,this >= 0
|
|
r.t = i;
|
|
while(i > 0) r[--i] = 0;
|
|
var j;
|
|
for(j = r.t-this.t; i < j; ++i) r[i+this.t] = this.am(0,a[i],r,i,0,this.t);
|
|
for(j = Math.min(a.t,n); i < j; ++i) this.am(0,a[i],r,i,0,n-i);
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = "this * a" without lower n words, n > 0
|
|
// "this" should be the larger one if appropriate.
|
|
function bnpMultiplyUpperTo(a,n,r) {
|
|
--n;
|
|
var i = r.t = this.t+a.t-n;
|
|
r.s = 0; // assumes a,this >= 0
|
|
while(--i >= 0) r[i] = 0;
|
|
for(i = Math.max(n-this.t,0); i < a.t; ++i)
|
|
r[this.t+i-n] = this.am(n-i,a[i],r,0,0,this.t+i-n);
|
|
r.clamp();
|
|
r.drShiftTo(1,r);
|
|
}
|
|
|
|
// Barrett modular reduction
|
|
function Barrett(m) {
|
|
// setup Barrett
|
|
this.r2 = nbi();
|
|
this.q3 = nbi();
|
|
BigInteger.ONE.dlShiftTo(2*m.t,this.r2);
|
|
this.mu = this.r2.divide(m);
|
|
this.m = m;
|
|
}
|
|
|
|
function barrettConvert(x) {
|
|
if(x.s < 0 || x.t > 2*this.m.t) return x.mod(this.m);
|
|
else if(x.compareTo(this.m) < 0) return x;
|
|
else { var r = nbi(); x.copyTo(r); this.reduce(r); return r; }
|
|
}
|
|
|
|
function barrettRevert(x) { return x; }
|
|
|
|
// x = x mod m (HAC 14.42)
|
|
function barrettReduce(x) {
|
|
x.drShiftTo(this.m.t-1,this.r2);
|
|
if(x.t > this.m.t+1) { x.t = this.m.t+1; x.clamp(); }
|
|
this.mu.multiplyUpperTo(this.r2,this.m.t+1,this.q3);
|
|
this.m.multiplyLowerTo(this.q3,this.m.t+1,this.r2);
|
|
while(x.compareTo(this.r2) < 0) x.dAddOffset(1,this.m.t+1);
|
|
x.subTo(this.r2,x);
|
|
while(x.compareTo(this.m) >= 0) x.subTo(this.m,x);
|
|
}
|
|
|
|
// r = x^2 mod m; x != r
|
|
function barrettSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
|
|
|
|
// r = x*y mod m; x,y != r
|
|
function barrettMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
|
|
|
|
Barrett.prototype.convert = barrettConvert;
|
|
Barrett.prototype.revert = barrettRevert;
|
|
Barrett.prototype.reduce = barrettReduce;
|
|
Barrett.prototype.mulTo = barrettMulTo;
|
|
Barrett.prototype.sqrTo = barrettSqrTo;
|
|
|
|
// (public) this^e % m (HAC 14.85)
|
|
function bnModPow(e,m) {
|
|
var i = e.bitLength(), k, r = nbv(1), z;
|
|
if(i <= 0) return r;
|
|
else if(i < 18) k = 1;
|
|
else if(i < 48) k = 3;
|
|
else if(i < 144) k = 4;
|
|
else if(i < 768) k = 5;
|
|
else k = 6;
|
|
if(i < 8)
|
|
z = new Classic(m);
|
|
else if(m.isEven())
|
|
z = new Barrett(m);
|
|
else
|
|
z = new Montgomery(m);
|
|
|
|
// precomputation
|
|
var g = new Array(), n = 3, k1 = k-1, km = (1<<k)-1;
|
|
g[1] = z.convert(this);
|
|
if(k > 1) {
|
|
var g2 = nbi();
|
|
z.sqrTo(g[1],g2);
|
|
while(n <= km) {
|
|
g[n] = nbi();
|
|
z.mulTo(g2,g[n-2],g[n]);
|
|
n += 2;
|
|
}
|
|
}
|
|
|
|
var j = e.t-1, w, is1 = true, r2 = nbi(), t;
|
|
i = nbits(e[j])-1;
|
|
while(j >= 0) {
|
|
if(i >= k1) w = (e[j]>>(i-k1))&km;
|
|
else {
|
|
w = (e[j]&((1<<(i+1))-1))<<(k1-i);
|
|
if(j > 0) w |= e[j-1]>>(this.DB+i-k1);
|
|
}
|
|
|
|
n = k;
|
|
while((w&1) == 0) { w >>= 1; --n; }
|
|
if((i -= n) < 0) { i += this.DB; --j; }
|
|
if(is1) { // ret == 1, don't bother squaring or multiplying it
|
|
g[w].copyTo(r);
|
|
is1 = false;
|
|
}
|
|
else {
|
|
while(n > 1) { z.sqrTo(r,r2); z.sqrTo(r2,r); n -= 2; }
|
|
if(n > 0) z.sqrTo(r,r2); else { t = r; r = r2; r2 = t; }
|
|
z.mulTo(r2,g[w],r);
|
|
}
|
|
|
|
while(j >= 0 && (e[j]&(1<<i)) == 0) {
|
|
z.sqrTo(r,r2); t = r; r = r2; r2 = t;
|
|
if(--i < 0) { i = this.DB-1; --j; }
|
|
}
|
|
}
|
|
return z.revert(r);
|
|
}
|
|
|
|
// (public) gcd(this,a) (HAC 14.54)
|
|
function bnGCD(a) {
|
|
var x = (this.s<0)?this.negate():this.clone();
|
|
var y = (a.s<0)?a.negate():a.clone();
|
|
if(x.compareTo(y) < 0) { var t = x; x = y; y = t; }
|
|
var i = x.getLowestSetBit(), g = y.getLowestSetBit();
|
|
if(g < 0) return x;
|
|
if(i < g) g = i;
|
|
if(g > 0) {
|
|
x.rShiftTo(g,x);
|
|
y.rShiftTo(g,y);
|
|
}
|
|
while(x.signum() > 0) {
|
|
if((i = x.getLowestSetBit()) > 0) x.rShiftTo(i,x);
|
|
if((i = y.getLowestSetBit()) > 0) y.rShiftTo(i,y);
|
|
if(x.compareTo(y) >= 0) {
|
|
x.subTo(y,x);
|
|
x.rShiftTo(1,x);
|
|
}
|
|
else {
|
|
y.subTo(x,y);
|
|
y.rShiftTo(1,y);
|
|
}
|
|
}
|
|
if(g > 0) y.lShiftTo(g,y);
|
|
return y;
|
|
}
|
|
|
|
// (protected) this % n, n < 2^26
|
|
function bnpModInt(n) {
|
|
if(n <= 0) return 0;
|
|
var d = this.DV%n, r = (this.s<0)?n-1:0;
|
|
if(this.t > 0)
|
|
if(d == 0) r = this[0]%n;
|
|
else for(var i = this.t-1; i >= 0; --i) r = (d*r+this[i])%n;
|
|
return r;
|
|
}
|
|
|
|
// (public) 1/this % m (HAC 14.61)
|
|
function bnModInverse(m) {
|
|
var ac = m.isEven();
|
|
if((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;
|
|
var u = m.clone(), v = this.clone();
|
|
var a = nbv(1), b = nbv(0), c = nbv(0), d = nbv(1);
|
|
while(u.signum() != 0) {
|
|
while(u.isEven()) {
|
|
u.rShiftTo(1,u);
|
|
if(ac) {
|
|
if(!a.isEven() || !b.isEven()) { a.addTo(this,a); b.subTo(m,b); }
|
|
a.rShiftTo(1,a);
|
|
}
|
|
else if(!b.isEven()) b.subTo(m,b);
|
|
b.rShiftTo(1,b);
|
|
}
|
|
while(v.isEven()) {
|
|
v.rShiftTo(1,v);
|
|
if(ac) {
|
|
if(!c.isEven() || !d.isEven()) { c.addTo(this,c); d.subTo(m,d); }
|
|
c.rShiftTo(1,c);
|
|
}
|
|
else if(!d.isEven()) d.subTo(m,d);
|
|
d.rShiftTo(1,d);
|
|
}
|
|
if(u.compareTo(v) >= 0) {
|
|
u.subTo(v,u);
|
|
if(ac) a.subTo(c,a);
|
|
b.subTo(d,b);
|
|
}
|
|
else {
|
|
v.subTo(u,v);
|
|
if(ac) c.subTo(a,c);
|
|
d.subTo(b,d);
|
|
}
|
|
}
|
|
if(v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;
|
|
if(d.compareTo(m) >= 0) return d.subtract(m);
|
|
if(d.signum() < 0) d.addTo(m,d); else return d;
|
|
if(d.signum() < 0) return d.add(m); else return d;
|
|
}
|
|
|
|
var lowprimes = [2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71,73,79,83,89,97,101,103,107,109,113,127,131,137,139,149,151,157,163,167,173,179,181,191,193,197,199,211,223,227,229,233,239,241,251,257,263,269,271,277,281,283,293,307,311,313,317,331,337,347,349,353,359,367,373,379,383,389,397,401,409,419,421,431,433,439,443,449,457,461,463,467,479,487,491,499,503,509,521,523,541,547,557,563,569,571,577,587,593,599,601,607,613,617,619,631,641,643,647,653,659,661,673,677,683,691,701,709,719,727,733,739,743,751,757,761,769,773,787,797,809,811,821,823,827,829,839,853,857,859,863,877,881,883,887,907,911,919,929,937,941,947,953,967,971,977,983,991,997];
|
|
var lplim = (1<<26)/lowprimes[lowprimes.length-1];
|
|
|
|
// (public) test primality with certainty >= 1-.5^t
|
|
function bnIsProbablePrime(t) {
|
|
var i, x = this.abs();
|
|
if(x.t == 1 && x[0] <= lowprimes[lowprimes.length-1]) {
|
|
for(i = 0; i < lowprimes.length; ++i)
|
|
if(x[0] == lowprimes[i]) return true;
|
|
return false;
|
|
}
|
|
if(x.isEven()) return false;
|
|
i = 1;
|
|
while(i < lowprimes.length) {
|
|
var m = lowprimes[i], j = i+1;
|
|
while(j < lowprimes.length && m < lplim) m *= lowprimes[j++];
|
|
m = x.modInt(m);
|
|
while(i < j) if(m%lowprimes[i++] == 0) return false;
|
|
}
|
|
return x.millerRabin(t);
|
|
}
|
|
|
|
// (protected) true if probably prime (HAC 4.24, Miller-Rabin)
|
|
function bnpMillerRabin(t) {
|
|
var n1 = this.subtract(BigInteger.ONE);
|
|
var k = n1.getLowestSetBit();
|
|
if(k <= 0) return false;
|
|
var r = n1.shiftRight(k);
|
|
t = (t+1)>>1;
|
|
if(t > lowprimes.length) t = lowprimes.length;
|
|
var a = nbi();
|
|
for(var i = 0; i < t; ++i) {
|
|
//Pick bases at random, instead of starting at 2
|
|
a.fromInt(lowprimes[Math.floor(Math.random()*lowprimes.length)]);
|
|
var y = a.modPow(r,this);
|
|
if(y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {
|
|
var j = 1;
|
|
while(j++ < k && y.compareTo(n1) != 0) {
|
|
y = y.modPowInt(2,this);
|
|
if(y.compareTo(BigInteger.ONE) == 0) return false;
|
|
}
|
|
if(y.compareTo(n1) != 0) return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// protected
|
|
BigInteger.prototype.chunkSize = bnpChunkSize;
|
|
BigInteger.prototype.toRadix = bnpToRadix;
|
|
BigInteger.prototype.fromRadix = bnpFromRadix;
|
|
BigInteger.prototype.fromNumber = bnpFromNumber;
|
|
BigInteger.prototype.bitwiseTo = bnpBitwiseTo;
|
|
BigInteger.prototype.changeBit = bnpChangeBit;
|
|
BigInteger.prototype.addTo = bnpAddTo;
|
|
BigInteger.prototype.dMultiply = bnpDMultiply;
|
|
BigInteger.prototype.dAddOffset = bnpDAddOffset;
|
|
BigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;
|
|
BigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;
|
|
BigInteger.prototype.modInt = bnpModInt;
|
|
BigInteger.prototype.millerRabin = bnpMillerRabin;
|
|
|
|
// public
|
|
BigInteger.prototype.clone = bnClone;
|
|
BigInteger.prototype.intValue = bnIntValue;
|
|
BigInteger.prototype.byteValue = bnByteValue;
|
|
BigInteger.prototype.shortValue = bnShortValue;
|
|
BigInteger.prototype.signum = bnSigNum;
|
|
BigInteger.prototype.toByteArray = bnToByteArray;
|
|
BigInteger.prototype.equals = bnEquals;
|
|
BigInteger.prototype.min = bnMin;
|
|
BigInteger.prototype.max = bnMax;
|
|
BigInteger.prototype.and = bnAnd;
|
|
BigInteger.prototype.or = bnOr;
|
|
BigInteger.prototype.xor = bnXor;
|
|
BigInteger.prototype.andNot = bnAndNot;
|
|
BigInteger.prototype.not = bnNot;
|
|
BigInteger.prototype.shiftLeft = bnShiftLeft;
|
|
BigInteger.prototype.shiftRight = bnShiftRight;
|
|
BigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;
|
|
BigInteger.prototype.bitCount = bnBitCount;
|
|
BigInteger.prototype.testBit = bnTestBit;
|
|
BigInteger.prototype.setBit = bnSetBit;
|
|
BigInteger.prototype.clearBit = bnClearBit;
|
|
BigInteger.prototype.flipBit = bnFlipBit;
|
|
BigInteger.prototype.add = bnAdd;
|
|
BigInteger.prototype.subtract = bnSubtract;
|
|
BigInteger.prototype.multiply = bnMultiply;
|
|
BigInteger.prototype.divide = bnDivide;
|
|
BigInteger.prototype.remainder = bnRemainder;
|
|
BigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;
|
|
BigInteger.prototype.modPow = bnModPow;
|
|
BigInteger.prototype.modInverse = bnModInverse;
|
|
BigInteger.prototype.pow = bnPow;
|
|
BigInteger.prototype.gcd = bnGCD;
|
|
BigInteger.prototype.isProbablePrime = bnIsProbablePrime;
|
|
|
|
// JSBN-specific extension
|
|
BigInteger.prototype.square = bnSquare;
|
|
|
|
// BigInteger interfaces not implemented in jsbn:
|
|
|
|
// BigInteger(int signum, byte[] magnitude)
|
|
// double doubleValue()
|
|
// float floatValue()
|
|
// int hashCode()
|
|
// long longValue()
|
|
// static BigInteger valueOf(long val)
|
|
|
|
module.exports.BigInteger = BigInteger;
|
|
// prng4.js - uses Arcfour as a PRNG
|
|
|
|
function Arcfour() {
|
|
this.i = 0;
|
|
this.j = 0;
|
|
this.S = new Array();
|
|
}
|
|
|
|
// Initialize arcfour context from key, an array of ints, each from [0..255]
|
|
function ARC4init(key) {
|
|
var i, j, t;
|
|
for(i = 0; i < 256; ++i)
|
|
this.S[i] = i;
|
|
j = 0;
|
|
for(i = 0; i < 256; ++i) {
|
|
j = (j + this.S[i] + key[i % key.length]) & 255;
|
|
t = this.S[i];
|
|
this.S[i] = this.S[j];
|
|
this.S[j] = t;
|
|
}
|
|
this.i = 0;
|
|
this.j = 0;
|
|
}
|
|
|
|
function ARC4next() {
|
|
var t;
|
|
this.i = (this.i + 1) & 255;
|
|
this.j = (this.j + this.S[this.i]) & 255;
|
|
t = this.S[this.i];
|
|
this.S[this.i] = this.S[this.j];
|
|
this.S[this.j] = t;
|
|
return this.S[(t + this.S[this.i]) & 255];
|
|
}
|
|
|
|
Arcfour.prototype.init = ARC4init;
|
|
Arcfour.prototype.next = ARC4next;
|
|
|
|
// Plug in your RNG constructor here
|
|
function prng_newstate() {
|
|
return new Arcfour();
|
|
}
|
|
|
|
// Pool size must be a multiple of 4 and greater than 32.
|
|
// An array of bytes the size of the pool will be passed to init()
|
|
var rng_psize = 256;
|
|
// BigInteger monkey patching
|
|
BigInteger.valueOf = nbv;
|
|
|
|
/**
|
|
* Returns a byte array representation of the big integer.
|
|
*
|
|
* This returns the absolute of the contained value in big endian
|
|
* form. A value of zero results in an empty array.
|
|
*/
|
|
BigInteger.prototype.toByteArrayUnsigned = function () {
|
|
var ba = this.abs().toByteArray();
|
|
if (ba.length) {
|
|
if (ba[0] == 0) {
|
|
ba = ba.slice(1);
|
|
}
|
|
return ba.map(function (v) {
|
|
return (v < 0) ? v + 256 : v;
|
|
});
|
|
} else {
|
|
// Empty array, nothing to do
|
|
return ba;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Turns a byte array into a big integer.
|
|
*
|
|
* This function will interpret a byte array as a big integer in big
|
|
* endian notation and ignore leading zeros.
|
|
*/
|
|
BigInteger.fromByteArrayUnsigned = function (ba) {
|
|
if (!ba.length) {
|
|
return ba.valueOf(0);
|
|
} else if (ba[0] & 0x80) {
|
|
// Prepend a zero so the BigInteger class doesn't mistake this
|
|
// for a negative integer.
|
|
return new BigInteger([0].concat(ba));
|
|
} else {
|
|
return new BigInteger(ba);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Converts big integer to signed byte representation.
|
|
*
|
|
* The format for this value uses a the most significant bit as a sign
|
|
* bit. If the most significant bit is already occupied by the
|
|
* absolute value, an extra byte is prepended and the sign bit is set
|
|
* there.
|
|
*
|
|
* Examples:
|
|
*
|
|
* 0 => 0x00
|
|
* 1 => 0x01
|
|
* -1 => 0x81
|
|
* 127 => 0x7f
|
|
* -127 => 0xff
|
|
* 128 => 0x0080
|
|
* -128 => 0x8080
|
|
* 255 => 0x00ff
|
|
* -255 => 0x80ff
|
|
* 16300 => 0x3fac
|
|
* -16300 => 0xbfac
|
|
* 62300 => 0x00f35c
|
|
* -62300 => 0x80f35c
|
|
*/
|
|
BigInteger.prototype.toByteArraySigned = function () {
|
|
var val = this.abs().toByteArrayUnsigned();
|
|
var neg = this.compareTo(BigInteger.ZERO) < 0;
|
|
|
|
if (neg) {
|
|
if (val[0] & 0x80) {
|
|
val.unshift(0x80);
|
|
} else {
|
|
val[0] |= 0x80;
|
|
}
|
|
} else {
|
|
if (val[0] & 0x80) {
|
|
val.unshift(0x00);
|
|
}
|
|
}
|
|
|
|
return val;
|
|
};
|
|
|
|
/**
|
|
* Parse a signed big integer byte representation.
|
|
*
|
|
* For details on the format please see BigInteger.toByteArraySigned.
|
|
*/
|
|
BigInteger.fromByteArraySigned = function (ba) {
|
|
// Check for negative value
|
|
if (ba[0] & 0x80) {
|
|
// Remove sign bit
|
|
ba[0] &= 0x7f;
|
|
|
|
return BigInteger.fromByteArrayUnsigned(ba).negate();
|
|
} else {
|
|
return BigInteger.fromByteArrayUnsigned(ba);
|
|
}
|
|
};
|
|
|
|
// Console ignore
|
|
var names = ["log", "debug", "info", "warn", "error", "assert", "dir",
|
|
"dirxml", "group", "groupEnd", "time", "timeEnd", "count",
|
|
"trace", "profile", "profileEnd"];
|
|
|
|
if ("undefined" == typeof window.console) window.console = {};
|
|
for (var i = 0; i < names.length; ++i)
|
|
if ("undefined" == typeof window.console[names[i]])
|
|
window.console[names[i]] = function() {};
|
|
|
|
// Bitcoin utility functions
|
|
Bitcoin.Util = {
|
|
/**
|
|
* Cross-browser compatibility version of Array.isArray.
|
|
*/
|
|
isArray: Array.isArray || function(o)
|
|
{
|
|
return Object.prototype.toString.call(o) === '[object Array]';
|
|
},
|
|
|
|
/**
|
|
* Create an array of a certain length filled with a specific value.
|
|
*/
|
|
makeFilledArray: function (len, val)
|
|
{
|
|
var array = [];
|
|
var i = 0;
|
|
while (i < len) {
|
|
array[i++] = val;
|
|
}
|
|
return array;
|
|
},
|
|
|
|
/**
|
|
* Turn an integer into a "var_int".
|
|
*
|
|
* "var_int" is a variable length integer used by Bitcoin's binary format.
|
|
*
|
|
* Returns a byte array.
|
|
*/
|
|
numToVarInt: function (i)
|
|
{
|
|
if (i < 0xfd) {
|
|
// unsigned char
|
|
return [i];
|
|
} else if (i < 0x10000) {
|
|
// unsigned short (LE)
|
|
return [0xfd, i & 255 , i >>> 8];
|
|
} else if (i < 0x100000000) {
|
|
// unsigned int (LE)
|
|
return [0xfe].concat(Crypto.util.wordsToBytes([i]).reverse());
|
|
} else {
|
|
throw 'quadword not implemented'
|
|
// unsigned long long (LE)
|
|
//return [0xff].concat(Crypto.util.wordsToBytes([i >>> 32, i]));
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Parse a Bitcoin value byte array, returning a BigInteger.
|
|
*/
|
|
valueToBigInt: function (valueBuffer)
|
|
{
|
|
if (valueBuffer instanceof BigInteger) return valueBuffer;
|
|
|
|
// Prepend zero byte to prevent interpretation as negative integer
|
|
return BigInteger.fromByteArrayUnsigned(valueBuffer);
|
|
},
|
|
|
|
/**
|
|
* Format a Bitcoin value as a string.
|
|
*
|
|
* Takes a BigInteger or byte-array and returns that amount of Bitcoins in a
|
|
* nice standard formatting.
|
|
*
|
|
* Examples:
|
|
* 12.3555
|
|
* 0.1234
|
|
* 900.99998888
|
|
* 34.00
|
|
*/
|
|
formatValue: function (valueBuffer) {
|
|
var value = this.valueToBigInt(valueBuffer).toString();
|
|
var integerPart = value.length > 8 ? value.substr(0, value.length-8) : '0';
|
|
var decimalPart = value.length > 8 ? value.substr(value.length-8) : value;
|
|
while (decimalPart.length < 8) decimalPart = "0"+decimalPart;
|
|
decimalPart = decimalPart.replace(/0*$/, '');
|
|
while (decimalPart.length < 2) decimalPart += "0";
|
|
return integerPart+"."+decimalPart;
|
|
},
|
|
|
|
/**
|
|
* Parse a floating point string as a Bitcoin value.
|
|
*
|
|
* Keep in mind that parsing user input is messy. You should always display
|
|
* the parsed value back to the user to make sure we understood his input
|
|
* correctly.
|
|
*/
|
|
parseValue: function (valueString) {
|
|
// TODO: Detect other number formats (e.g. comma as decimal separator)
|
|
var valueComp = valueString.split('.');
|
|
var integralPart = valueComp[0];
|
|
var fractionalPart = valueComp[1] || "0";
|
|
while (fractionalPart.length < 8) fractionalPart += "0";
|
|
fractionalPart = fractionalPart.replace(/^0+/g, '');
|
|
var value = BigInteger.valueOf(parseInt(integralPart));
|
|
value = value.multiply(BigInteger.valueOf(100000000));
|
|
value = value.add(BigInteger.valueOf(parseInt(fractionalPart)));
|
|
return value;
|
|
},
|
|
|
|
/**
|
|
* Calculate RIPEMD160(SHA256(data)).
|
|
*
|
|
* Takes an arbitrary byte array as inputs and returns the hash as a byte
|
|
* array.
|
|
*/
|
|
sha256ripe160: function (data) {
|
|
return Crypto.RIPEMD160(Crypto.SHA256(data, {asBytes: true}), {asBytes: true});
|
|
}
|
|
};
|
|
|
|
for (var i in Crypto.util) {
|
|
if (Crypto.util.hasOwnProperty(i)) {
|
|
Bitcoin.Util[i] = Crypto.util[i];
|
|
}
|
|
}
|
|
// Random number generator - requires a PRNG backend, e.g. prng4.js
|
|
|
|
// For best results, put code like
|
|
// <body onClick='rng_seed_time();' onKeyPress='rng_seed_time();'>
|
|
// in your main HTML document.
|
|
|
|
var rng_state;
|
|
var rng_pool;
|
|
var rng_pptr;
|
|
|
|
// Mix in a 32-bit integer into the pool
|
|
function rng_seed_int(x) {
|
|
rng_pool[rng_pptr++] ^= x & 255;
|
|
rng_pool[rng_pptr++] ^= (x >> 8) & 255;
|
|
rng_pool[rng_pptr++] ^= (x >> 16) & 255;
|
|
rng_pool[rng_pptr++] ^= (x >> 24) & 255;
|
|
if(rng_pptr >= rng_psize) rng_pptr -= rng_psize;
|
|
}
|
|
|
|
// Mix in the current time (w/milliseconds) into the pool
|
|
function rng_seed_time() {
|
|
rng_seed_int(new Date().getTime());
|
|
}
|
|
|
|
// Initialize the pool with junk if needed.
|
|
if(rng_pool == null) {
|
|
rng_pool = new Array();
|
|
rng_pptr = 0;
|
|
var t;
|
|
if(navigator.appName == "Netscape" && navigator.appVersion < "5" && window.crypto) {
|
|
// Extract entropy (256 bits) from NS4 RNG if available
|
|
var z = window.crypto.random(32);
|
|
for(t = 0; t < z.length; ++t)
|
|
rng_pool[rng_pptr++] = z.charCodeAt(t) & 255;
|
|
}
|
|
while(rng_pptr < rng_psize) { // extract some randomness from Math.random()
|
|
t = Math.floor(65536 * Math.random());
|
|
rng_pool[rng_pptr++] = t >>> 8;
|
|
rng_pool[rng_pptr++] = t & 255;
|
|
}
|
|
rng_pptr = 0;
|
|
rng_seed_time();
|
|
//rng_seed_int(window.screenX);
|
|
//rng_seed_int(window.screenY);
|
|
}
|
|
|
|
function rng_get_byte() {
|
|
if(rng_state == null) {
|
|
rng_seed_time();
|
|
rng_state = prng_newstate();
|
|
rng_state.init(rng_pool);
|
|
for(rng_pptr = 0; rng_pptr < rng_pool.length; ++rng_pptr)
|
|
rng_pool[rng_pptr] = 0;
|
|
rng_pptr = 0;
|
|
//rng_pool = null;
|
|
}
|
|
// TODO: allow reseeding after first request
|
|
return rng_state.next();
|
|
}
|
|
|
|
function rng_get_bytes(ba) {
|
|
var i;
|
|
for(i = 0; i < ba.length; ++i) ba[i] = rng_get_byte();
|
|
}
|
|
|
|
function SecureRandom() {}
|
|
|
|
SecureRandom.prototype.nextBytes = function() {
|
|
throw new Error('Should not use old RNG');
|
|
};
|
|
// Basic Javascript Elliptic Curve implementation
|
|
// Ported loosely from BouncyCastle's Java EC code
|
|
// Only Fp curves implemented for now
|
|
|
|
// Requires jsbn.js and jsbn2.js
|
|
|
|
// ----------------
|
|
// ECFieldElementFp
|
|
|
|
// constructor
|
|
function ECFieldElementFp(q,x) {
|
|
this.x = x;
|
|
// TODO if(x.compareTo(q) >= 0) error
|
|
this.q = q;
|
|
}
|
|
|
|
function feFpEquals(other) {
|
|
if(other == this) return true;
|
|
return (this.q.equals(other.q) && this.x.equals(other.x));
|
|
}
|
|
|
|
function feFpToBigInteger() {
|
|
return this.x;
|
|
}
|
|
|
|
function feFpNegate() {
|
|
return new ECFieldElementFp(this.q, this.x.negate().mod(this.q));
|
|
}
|
|
|
|
function feFpAdd(b) {
|
|
return new ECFieldElementFp(this.q, this.x.add(b.toBigInteger()).mod(this.q));
|
|
}
|
|
|
|
function feFpSubtract(b) {
|
|
return new ECFieldElementFp(this.q, this.x.subtract(b.toBigInteger()).mod(this.q));
|
|
}
|
|
|
|
function feFpMultiply(b) {
|
|
return new ECFieldElementFp(this.q, this.x.multiply(b.toBigInteger()).mod(this.q));
|
|
}
|
|
|
|
function feFpSquare() {
|
|
return new ECFieldElementFp(this.q, this.x.square().mod(this.q));
|
|
}
|
|
|
|
function feFpDivide(b) {
|
|
return new ECFieldElementFp(this.q, this.x.multiply(b.toBigInteger().modInverse(this.q)).mod(this.q));
|
|
}
|
|
|
|
ECFieldElementFp.prototype.equals = feFpEquals;
|
|
ECFieldElementFp.prototype.toBigInteger = feFpToBigInteger;
|
|
ECFieldElementFp.prototype.negate = feFpNegate;
|
|
ECFieldElementFp.prototype.add = feFpAdd;
|
|
ECFieldElementFp.prototype.subtract = feFpSubtract;
|
|
ECFieldElementFp.prototype.multiply = feFpMultiply;
|
|
ECFieldElementFp.prototype.square = feFpSquare;
|
|
ECFieldElementFp.prototype.divide = feFpDivide;
|
|
|
|
// ----------------
|
|
// ECPointFp
|
|
|
|
// constructor
|
|
function ECPointFp(curve,x,y,z) {
|
|
this.curve = curve;
|
|
this.x = x;
|
|
this.y = y;
|
|
// Projective coordinates: either zinv == null or z * zinv == 1
|
|
// z and zinv are just BigIntegers, not fieldElements
|
|
if(z == null) {
|
|
this.z = BigInteger.ONE;
|
|
}
|
|
else {
|
|
this.z = z;
|
|
}
|
|
this.zinv = null;
|
|
//TODO: compression flag
|
|
}
|
|
|
|
function pointFpGetX() {
|
|
if(this.zinv == null) {
|
|
this.zinv = this.z.modInverse(this.curve.q);
|
|
}
|
|
return this.curve.fromBigInteger(this.x.toBigInteger().multiply(this.zinv).mod(this.curve.q));
|
|
}
|
|
|
|
function pointFpGetY() {
|
|
if(this.zinv == null) {
|
|
this.zinv = this.z.modInverse(this.curve.q);
|
|
}
|
|
return this.curve.fromBigInteger(this.y.toBigInteger().multiply(this.zinv).mod(this.curve.q));
|
|
}
|
|
|
|
function pointFpEquals(other) {
|
|
if(other == this) return true;
|
|
if(this.isInfinity()) return other.isInfinity();
|
|
if(other.isInfinity()) return this.isInfinity();
|
|
var u, v;
|
|
// u = Y2 * Z1 - Y1 * Z2
|
|
u = other.y.toBigInteger().multiply(this.z).subtract(this.y.toBigInteger().multiply(other.z)).mod(this.curve.q);
|
|
if(!u.equals(BigInteger.ZERO)) return false;
|
|
// v = X2 * Z1 - X1 * Z2
|
|
v = other.x.toBigInteger().multiply(this.z).subtract(this.x.toBigInteger().multiply(other.z)).mod(this.curve.q);
|
|
return v.equals(BigInteger.ZERO);
|
|
}
|
|
|
|
function pointFpIsInfinity() {
|
|
if((this.x == null) && (this.y == null)) return true;
|
|
return this.z.equals(BigInteger.ZERO) && !this.y.toBigInteger().equals(BigInteger.ZERO);
|
|
}
|
|
|
|
function pointFpNegate() {
|
|
return new ECPointFp(this.curve, this.x, this.y.negate(), this.z);
|
|
}
|
|
|
|
function pointFpAdd(b) {
|
|
if(this.isInfinity()) return b;
|
|
if(b.isInfinity()) return this;
|
|
|
|
// u = Y2 * Z1 - Y1 * Z2
|
|
var u = b.y.toBigInteger().multiply(this.z).subtract(this.y.toBigInteger().multiply(b.z)).mod(this.curve.q);
|
|
// v = X2 * Z1 - X1 * Z2
|
|
var v = b.x.toBigInteger().multiply(this.z).subtract(this.x.toBigInteger().multiply(b.z)).mod(this.curve.q);
|
|
|
|
if(BigInteger.ZERO.equals(v)) {
|
|
if(BigInteger.ZERO.equals(u)) {
|
|
return this.twice(); // this == b, so double
|
|
}
|
|
return this.curve.getInfinity(); // this = -b, so infinity
|
|
}
|
|
|
|
var THREE = new BigInteger("3");
|
|
var x1 = this.x.toBigInteger();
|
|
var y1 = this.y.toBigInteger();
|
|
var x2 = b.x.toBigInteger();
|
|
var y2 = b.y.toBigInteger();
|
|
|
|
var v2 = v.square();
|
|
var v3 = v2.multiply(v);
|
|
var x1v2 = x1.multiply(v2);
|
|
var zu2 = u.square().multiply(this.z);
|
|
|
|
// x3 = v * (z2 * (z1 * u^2 - 2 * x1 * v^2) - v^3)
|
|
var x3 = zu2.subtract(x1v2.shiftLeft(1)).multiply(b.z).subtract(v3).multiply(v).mod(this.curve.q);
|
|
// y3 = z2 * (3 * x1 * u * v^2 - y1 * v^3 - z1 * u^3) + u * v^3
|
|
var y3 = x1v2.multiply(THREE).multiply(u).subtract(y1.multiply(v3)).subtract(zu2.multiply(u)).multiply(b.z).add(u.multiply(v3)).mod(this.curve.q);
|
|
// z3 = v^3 * z1 * z2
|
|
var z3 = v3.multiply(this.z).multiply(b.z).mod(this.curve.q);
|
|
|
|
return new ECPointFp(this.curve, this.curve.fromBigInteger(x3), this.curve.fromBigInteger(y3), z3);
|
|
}
|
|
|
|
function pointFpTwice() {
|
|
if(this.isInfinity()) return this;
|
|
if(this.y.toBigInteger().signum() == 0) return this.curve.getInfinity();
|
|
|
|
// TODO: optimized handling of constants
|
|
var THREE = new BigInteger("3");
|
|
var x1 = this.x.toBigInteger();
|
|
var y1 = this.y.toBigInteger();
|
|
|
|
var y1z1 = y1.multiply(this.z);
|
|
var y1sqz1 = y1z1.multiply(y1).mod(this.curve.q);
|
|
var a = this.curve.a.toBigInteger();
|
|
|
|
// w = 3 * x1^2 + a * z1^2
|
|
var w = x1.square().multiply(THREE);
|
|
if(!BigInteger.ZERO.equals(a)) {
|
|
w = w.add(this.z.square().multiply(a));
|
|
}
|
|
w = w.mod(this.curve.q);
|
|
// x3 = 2 * y1 * z1 * (w^2 - 8 * x1 * y1^2 * z1)
|
|
var x3 = w.square().subtract(x1.shiftLeft(3).multiply(y1sqz1)).shiftLeft(1).multiply(y1z1).mod(this.curve.q);
|
|
// y3 = 4 * y1^2 * z1 * (3 * w * x1 - 2 * y1^2 * z1) - w^3
|
|
var y3 = w.multiply(THREE).multiply(x1).subtract(y1sqz1.shiftLeft(1)).shiftLeft(2).multiply(y1sqz1).subtract(w.square().multiply(w)).mod(this.curve.q);
|
|
// z3 = 8 * (y1 * z1)^3
|
|
var z3 = y1z1.square().multiply(y1z1).shiftLeft(3).mod(this.curve.q);
|
|
|
|
return new ECPointFp(this.curve, this.curve.fromBigInteger(x3), this.curve.fromBigInteger(y3), z3);
|
|
}
|
|
|
|
// Simple NAF (Non-Adjacent Form) multiplication algorithm
|
|
// TODO: modularize the multiplication algorithm
|
|
function pointFpMultiply(k) {
|
|
if(this.isInfinity()) return this;
|
|
if(k.signum() == 0) return this.curve.getInfinity();
|
|
|
|
var e = k;
|
|
var h = e.multiply(new BigInteger("3"));
|
|
|
|
var neg = this.negate();
|
|
var R = this;
|
|
|
|
var i;
|
|
for(i = h.bitLength() - 2; i > 0; --i) {
|
|
R = R.twice();
|
|
|
|
var hBit = h.testBit(i);
|
|
var eBit = e.testBit(i);
|
|
|
|
if (hBit != eBit) {
|
|
R = R.add(hBit ? this : neg);
|
|
}
|
|
}
|
|
|
|
return R;
|
|
}
|
|
|
|
// Compute this*j + x*k (simultaneous multiplication)
|
|
function pointFpMultiplyTwo(j,x,k) {
|
|
var i;
|
|
if(j.bitLength() > k.bitLength())
|
|
i = j.bitLength() - 1;
|
|
else
|
|
i = k.bitLength() - 1;
|
|
|
|
var R = this.curve.getInfinity();
|
|
var both = this.add(x);
|
|
while(i >= 0) {
|
|
R = R.twice();
|
|
if(j.testBit(i)) {
|
|
if(k.testBit(i)) {
|
|
R = R.add(both);
|
|
}
|
|
else {
|
|
R = R.add(this);
|
|
}
|
|
}
|
|
else {
|
|
if(k.testBit(i)) {
|
|
R = R.add(x);
|
|
}
|
|
}
|
|
--i;
|
|
}
|
|
|
|
return R;
|
|
}
|
|
|
|
ECPointFp.prototype.getX = pointFpGetX;
|
|
ECPointFp.prototype.getY = pointFpGetY;
|
|
ECPointFp.prototype.equals = pointFpEquals;
|
|
ECPointFp.prototype.isInfinity = pointFpIsInfinity;
|
|
ECPointFp.prototype.negate = pointFpNegate;
|
|
ECPointFp.prototype.add = pointFpAdd;
|
|
ECPointFp.prototype.twice = pointFpTwice;
|
|
ECPointFp.prototype.multiply = pointFpMultiply;
|
|
ECPointFp.prototype.multiplyTwo = pointFpMultiplyTwo;
|
|
|
|
// ----------------
|
|
// ECCurveFp
|
|
|
|
// constructor
|
|
function ECCurveFp(q,a,b) {
|
|
this.q = q;
|
|
this.a = this.fromBigInteger(a);
|
|
this.b = this.fromBigInteger(b);
|
|
this.infinity = new ECPointFp(this, null, null);
|
|
}
|
|
|
|
function curveFpGetQ() {
|
|
return this.q;
|
|
}
|
|
|
|
function curveFpGetA() {
|
|
return this.a;
|
|
}
|
|
|
|
function curveFpGetB() {
|
|
return this.b;
|
|
}
|
|
|
|
function curveFpEquals(other) {
|
|
if(other == this) return true;
|
|
return(this.q.equals(other.q) && this.a.equals(other.a) && this.b.equals(other.b));
|
|
}
|
|
|
|
function curveFpGetInfinity() {
|
|
return this.infinity;
|
|
}
|
|
|
|
function curveFpFromBigInteger(x) {
|
|
return new ECFieldElementFp(this.q, x);
|
|
}
|
|
|
|
// for now, work with hex strings because they're easier in JS
|
|
function curveFpDecodePointHex(s) {
|
|
switch(parseInt(s.substr(0,2), 16)) { // first byte
|
|
case 0:
|
|
return this.infinity;
|
|
case 2:
|
|
case 3:
|
|
// point compression not supported yet
|
|
return null;
|
|
case 4:
|
|
case 6:
|
|
case 7:
|
|
var len = (s.length - 2) / 2;
|
|
var xHex = s.substr(2, len);
|
|
var yHex = s.substr(len+2, len);
|
|
|
|
return new ECPointFp(this,
|
|
this.fromBigInteger(new BigInteger(xHex, 16)),
|
|
this.fromBigInteger(new BigInteger(yHex, 16)));
|
|
|
|
default: // unsupported
|
|
return null;
|
|
}
|
|
}
|
|
|
|
ECCurveFp.prototype.getQ = curveFpGetQ;
|
|
ECCurveFp.prototype.getA = curveFpGetA;
|
|
ECCurveFp.prototype.getB = curveFpGetB;
|
|
ECCurveFp.prototype.equals = curveFpEquals;
|
|
ECCurveFp.prototype.getInfinity = curveFpGetInfinity;
|
|
ECCurveFp.prototype.fromBigInteger = curveFpFromBigInteger;
|
|
ECCurveFp.prototype.decodePointHex = curveFpDecodePointHex;
|
|
|
|
module.exports.ECPointFp = ECPointFp;
|
|
module.exports.ECFieldElementFp = ECFieldElementFp;
|
|
// Named EC curves
|
|
|
|
// Requires ec.js, jsbn.js, and jsbn2.js
|
|
|
|
// ----------------
|
|
// X9ECParameters
|
|
|
|
// constructor
|
|
function X9ECParameters(curve,g,n,h) {
|
|
this.curve = curve;
|
|
this.g = g;
|
|
this.n = n;
|
|
this.h = h;
|
|
}
|
|
|
|
function x9getCurve() {
|
|
return this.curve;
|
|
}
|
|
|
|
function x9getG() {
|
|
return this.g;
|
|
}
|
|
|
|
function x9getN() {
|
|
return this.n;
|
|
}
|
|
|
|
function x9getH() {
|
|
return this.h;
|
|
}
|
|
|
|
X9ECParameters.prototype.getCurve = x9getCurve;
|
|
X9ECParameters.prototype.getG = x9getG;
|
|
X9ECParameters.prototype.getN = x9getN;
|
|
X9ECParameters.prototype.getH = x9getH;
|
|
|
|
// ----------------
|
|
// SECNamedCurves
|
|
|
|
function fromHex(s) { return new BigInteger(s, 16); }
|
|
|
|
function secp128r1() {
|
|
// p = 2^128 - 2^97 - 1
|
|
var p = fromHex("FFFFFFFDFFFFFFFFFFFFFFFFFFFFFFFF");
|
|
var a = fromHex("FFFFFFFDFFFFFFFFFFFFFFFFFFFFFFFC");
|
|
var b = fromHex("E87579C11079F43DD824993C2CEE5ED3");
|
|
//byte[] S = Hex.decode("000E0D4D696E6768756151750CC03A4473D03679");
|
|
var n = fromHex("FFFFFFFE0000000075A30D1B9038A115");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "161FF7528B899B2D0C28607CA52C5B86"
|
|
+ "CF5AC8395BAFEB13C02DA292DDED7A83");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp160k1() {
|
|
// p = 2^160 - 2^32 - 2^14 - 2^12 - 2^9 - 2^8 - 2^7 - 2^3 - 2^2 - 1
|
|
var p = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFAC73");
|
|
var a = BigInteger.ZERO;
|
|
var b = fromHex("7");
|
|
//byte[] S = null;
|
|
var n = fromHex("0100000000000000000001B8FA16DFAB9ACA16B6B3");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "3B4C382CE37AA192A4019E763036F4F5DD4D7EBB"
|
|
+ "938CF935318FDCED6BC28286531733C3F03C4FEE");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp160r1() {
|
|
// p = 2^160 - 2^31 - 1
|
|
var p = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFF");
|
|
var a = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFC");
|
|
var b = fromHex("1C97BEFC54BD7A8B65ACF89F81D4D4ADC565FA45");
|
|
//byte[] S = Hex.decode("1053CDE42C14D696E67687561517533BF3F83345");
|
|
var n = fromHex("0100000000000000000001F4C8F927AED3CA752257");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "4A96B5688EF573284664698968C38BB913CBFC82"
|
|
+ "23A628553168947D59DCC912042351377AC5FB32");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp192k1() {
|
|
// p = 2^192 - 2^32 - 2^12 - 2^8 - 2^7 - 2^6 - 2^3 - 1
|
|
var p = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFEE37");
|
|
var a = BigInteger.ZERO;
|
|
var b = fromHex("3");
|
|
//byte[] S = null;
|
|
var n = fromHex("FFFFFFFFFFFFFFFFFFFFFFFE26F2FC170F69466A74DEFD8D");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "DB4FF10EC057E9AE26B07D0280B7F4341DA5D1B1EAE06C7D"
|
|
+ "9B2F2F6D9C5628A7844163D015BE86344082AA88D95E2F9D");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp192r1() {
|
|
// p = 2^192 - 2^64 - 1
|
|
var p = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF");
|
|
var a = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFC");
|
|
var b = fromHex("64210519E59C80E70FA7E9AB72243049FEB8DEECC146B9B1");
|
|
//byte[] S = Hex.decode("3045AE6FC8422F64ED579528D38120EAE12196D5");
|
|
var n = fromHex("FFFFFFFFFFFFFFFFFFFFFFFF99DEF836146BC9B1B4D22831");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "188DA80EB03090F67CBF20EB43A18800F4FF0AFD82FF1012"
|
|
+ "07192B95FFC8DA78631011ED6B24CDD573F977A11E794811");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp224r1() {
|
|
// p = 2^224 - 2^96 + 1
|
|
var p = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF000000000000000000000001");
|
|
var a = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFFFFFFFFFE");
|
|
var b = fromHex("B4050A850C04B3ABF54132565044B0B7D7BFD8BA270B39432355FFB4");
|
|
//byte[] S = Hex.decode("BD71344799D5C7FCDC45B59FA3B9AB8F6A948BC5");
|
|
var n = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFF16A2E0B8F03E13DD29455C5C2A3D");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "B70E0CBD6BB4BF7F321390B94A03C1D356C21122343280D6115C1D21"
|
|
+ "BD376388B5F723FB4C22DFE6CD4375A05A07476444D5819985007E34");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp256k1() {
|
|
// p = 2^256 - 2^32 - 2^9 - 2^8 - 2^7 - 2^6 - 2^4 - 1
|
|
var p = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F");
|
|
var a = BigInteger.ZERO;
|
|
var b = fromHex("7");
|
|
//byte[] S = null;
|
|
var n = fromHex("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798"
|
|
+ "483ADA7726A3C4655DA4FBFC0E1108A8FD17B448A68554199C47D08FFB10D4B8");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
function secp256r1() {
|
|
// p = 2^224 (2^32 - 1) + 2^192 + 2^96 - 1
|
|
var p = fromHex("FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFF");
|
|
var a = fromHex("FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFC");
|
|
var b = fromHex("5AC635D8AA3A93E7B3EBBD55769886BC651D06B0CC53B0F63BCE3C3E27D2604B");
|
|
//byte[] S = Hex.decode("C49D360886E704936A6678E1139D26B7819F7E90");
|
|
var n = fromHex("FFFFFFFF00000000FFFFFFFFFFFFFFFFBCE6FAADA7179E84F3B9CAC2FC632551");
|
|
var h = BigInteger.ONE;
|
|
var curve = new ECCurveFp(p, a, b);
|
|
var G = curve.decodePointHex("04"
|
|
+ "6B17D1F2E12C4247F8BCE6E563A440F277037D812DEB33A0F4A13945D898C296"
|
|
+ "4FE342E2FE1A7F9B8EE7EB4A7C0F9E162BCE33576B315ECECBB6406837BF51F5");
|
|
return new X9ECParameters(curve, G, n, h);
|
|
}
|
|
|
|
// TODO: make this into a proper hashtable
|
|
function getSECCurveByName(name) {
|
|
if(name == "secp128r1") return secp128r1();
|
|
if(name == "secp160k1") return secp160k1();
|
|
if(name == "secp160r1") return secp160r1();
|
|
if(name == "secp192k1") return secp192k1();
|
|
if(name == "secp192r1") return secp192r1();
|
|
if(name == "secp224r1") return secp224r1();
|
|
if(name == "secp256k1") return secp256k1();
|
|
if(name == "secp256r1") return secp256r1();
|
|
return null;
|
|
}
|
|
|
|
module.exports.getSECCurveByName = getSECCurveByName;
|
|
function integerToBytes(i, len) {
|
|
var bytes = i.toByteArrayUnsigned();
|
|
|
|
if (len < bytes.length) {
|
|
bytes = bytes.slice(bytes.length-len);
|
|
} else while (len > bytes.length) {
|
|
bytes.unshift(0);
|
|
}
|
|
|
|
return bytes;
|
|
};
|
|
|
|
ECFieldElementFp.prototype.getByteLength = function () {
|
|
return Math.floor((this.toBigInteger().bitLength() + 7) / 8);
|
|
};
|
|
|
|
ECPointFp.prototype.getEncoded = function (compressed) {
|
|
var x = this.getX().toBigInteger();
|
|
var y = this.getY().toBigInteger();
|
|
|
|
// Get value as a 32-byte Buffer
|
|
// Fixed length based on a patch by bitaddress.org and Casascius
|
|
var enc = integerToBytes(x, 32);
|
|
|
|
if (compressed) {
|
|
if (y.isEven()) {
|
|
// Compressed even pubkey
|
|
// M = 02 || X
|
|
enc.unshift(0x02);
|
|
} else {
|
|
// Compressed uneven pubkey
|
|
// M = 03 || X
|
|
enc.unshift(0x03);
|
|
}
|
|
} else {
|
|
// Uncompressed pubkey
|
|
// M = 04 || X || Y
|
|
enc.unshift(0x04);
|
|
enc = enc.concat(integerToBytes(y, 32));
|
|
}
|
|
return enc;
|
|
};
|
|
|
|
ECPointFp.decodeFrom = function (ecparams, enc) {
|
|
var type = enc[0];
|
|
var dataLen = enc.length-1;
|
|
|
|
// Extract x and y as byte arrays
|
|
if (type === 4) {
|
|
var xBa = enc.slice(1, 1 + dataLen/2),
|
|
yBa = enc.slice(1 + dataLen/2, 1 + dataLen),
|
|
x = BigInteger.fromByteArrayUnsigned(xBa),
|
|
y = BigInteger.fromByteArrayUnsigned(yBa);
|
|
}
|
|
else {
|
|
var xBa = enc.slice(1),
|
|
x = BigInteger.fromByteArrayUnsigned(xBa),
|
|
p = ecparams.getQ(),
|
|
xCubedPlus7 = x.multiply(x).multiply(x).add(new BigInteger('7')).mod(p),
|
|
pPlus1Over4 = p.add(new BigInteger('1'))
|
|
.divide(new BigInteger('4')),
|
|
y = xCubedPlus7.modPow(pPlus1Over4,p);
|
|
if (y.mod(new BigInteger('2')).toString() != ''+(type % 2)) {
|
|
y = p.subtract(y)
|
|
}
|
|
}
|
|
|
|
// Return point
|
|
return new ECPointFp(ecparams,
|
|
ecparams.fromBigInteger(x),
|
|
ecparams.fromBigInteger(y));
|
|
};
|
|
|
|
ECPointFp.prototype.add2D = function (b) {
|
|
if(this.isInfinity()) return b;
|
|
if(b.isInfinity()) return this;
|
|
|
|
if (this.x.equals(b.x)) {
|
|
if (this.y.equals(b.y)) {
|
|
// this = b, i.e. this must be doubled
|
|
return this.twice();
|
|
}
|
|
// this = -b, i.e. the result is the point at infinity
|
|
return this.curve.getInfinity();
|
|
}
|
|
|
|
var x_x = b.x.subtract(this.x);
|
|
var y_y = b.y.subtract(this.y);
|
|
var gamma = y_y.divide(x_x);
|
|
|
|
var x3 = gamma.square().subtract(this.x).subtract(b.x);
|
|
var y3 = gamma.multiply(this.x.subtract(x3)).subtract(this.y);
|
|
|
|
return new ECPointFp(this.curve, x3, y3);
|
|
};
|
|
|
|
ECPointFp.prototype.twice2D = function () {
|
|
if (this.isInfinity()) return this;
|
|
if (this.y.toBigInteger().signum() == 0) {
|
|
// if y1 == 0, then (x1, y1) == (x1, -y1)
|
|
// and hence this = -this and thus 2(x1, y1) == infinity
|
|
return this.curve.getInfinity();
|
|
}
|
|
|
|
var TWO = this.curve.fromBigInteger(BigInteger.valueOf(2));
|
|
var THREE = this.curve.fromBigInteger(BigInteger.valueOf(3));
|
|
var gamma = this.x.square().multiply(THREE).add(this.curve.a).divide(this.y.multiply(TWO));
|
|
|
|
var x3 = gamma.square().subtract(this.x.multiply(TWO));
|
|
var y3 = gamma.multiply(this.x.subtract(x3)).subtract(this.y);
|
|
|
|
return new ECPointFp(this.curve, x3, y3);
|
|
};
|
|
|
|
ECPointFp.prototype.multiply2D = function (k) {
|
|
if(this.isInfinity()) return this;
|
|
if(k.signum() == 0) return this.curve.getInfinity();
|
|
|
|
var e = k;
|
|
var h = e.multiply(new BigInteger("3"));
|
|
|
|
var neg = this.negate();
|
|
var R = this;
|
|
|
|
var i;
|
|
for (i = h.bitLength() - 2; i > 0; --i) {
|
|
R = R.twice();
|
|
|
|
var hBit = h.testBit(i);
|
|
var eBit = e.testBit(i);
|
|
|
|
if (hBit != eBit) {
|
|
R = R.add2D(hBit ? this : neg);
|
|
}
|
|
}
|
|
|
|
return R;
|
|
};
|
|
|
|
ECPointFp.prototype.isOnCurve = function () {
|
|
var x = this.getX().toBigInteger();
|
|
var y = this.getY().toBigInteger();
|
|
var a = this.curve.getA().toBigInteger();
|
|
var b = this.curve.getB().toBigInteger();
|
|
var n = this.curve.getQ();
|
|
var lhs = y.multiply(y).mod(n);
|
|
var rhs = x.multiply(x).multiply(x)
|
|
.add(a.multiply(x)).add(b).mod(n);
|
|
return lhs.equals(rhs);
|
|
};
|
|
|
|
ECPointFp.prototype.toString = function () {
|
|
return '('+this.getX().toBigInteger().toString()+','+
|
|
this.getY().toBigInteger().toString()+')';
|
|
};
|
|
|
|
/**
|
|
* Validate an elliptic curve point.
|
|
*
|
|
* See SEC 1, section 3.2.2.1: Elliptic Curve Public Key Validation Primitive
|
|
*/
|
|
ECPointFp.prototype.validate = function () {
|
|
var n = this.curve.getQ();
|
|
|
|
// Check Q != O
|
|
if (this.isInfinity()) {
|
|
throw new Error("Point is at infinity.");
|
|
}
|
|
|
|
// Check coordinate bounds
|
|
var x = this.getX().toBigInteger();
|
|
var y = this.getY().toBigInteger();
|
|
if (x.compareTo(BigInteger.ONE) < 0 ||
|
|
x.compareTo(n.subtract(BigInteger.ONE)) > 0) {
|
|
throw new Error('x coordinate out of bounds');
|
|
}
|
|
if (y.compareTo(BigInteger.ONE) < 0 ||
|
|
y.compareTo(n.subtract(BigInteger.ONE)) > 0) {
|
|
throw new Error('y coordinate out of bounds');
|
|
}
|
|
|
|
// Check y^2 = x^3 + ax + b (mod n)
|
|
if (!this.isOnCurve()) {
|
|
throw new Error("Point is not on the curve.");
|
|
}
|
|
|
|
// Check nQ = 0 (Q is a scalar multiple of G)
|
|
if (this.multiply(n).isInfinity()) {
|
|
// TODO: This check doesn't work - fix.
|
|
throw new Error("Point is not a scalar multiple of G.");
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
function dmp(v) {
|
|
if (!(v instanceof BigInteger)) v = v.toBigInteger();
|
|
return Crypto.util.bytesToHex(v.toByteArrayUnsigned());
|
|
};
|
|
|
|
Bitcoin.ECDSA = (function () {
|
|
var ecparams = getSECCurveByName("secp256k1");
|
|
var rng = new SecureRandom();
|
|
|
|
var P_OVER_FOUR = null;
|
|
|
|
function implShamirsTrick(P, k, Q, l)
|
|
{
|
|
var m = Math.max(k.bitLength(), l.bitLength());
|
|
var Z = P.add2D(Q);
|
|
var R = P.curve.getInfinity();
|
|
|
|
for (var i = m - 1; i >= 0; --i) {
|
|
R = R.twice2D();
|
|
|
|
R.z = BigInteger.ONE;
|
|
|
|
if (k.testBit(i)) {
|
|
if (l.testBit(i)) {
|
|
R = R.add2D(Z);
|
|
} else {
|
|
R = R.add2D(P);
|
|
}
|
|
} else {
|
|
if (l.testBit(i)) {
|
|
R = R.add2D(Q);
|
|
}
|
|
}
|
|
}
|
|
|
|
return R;
|
|
};
|
|
|
|
var ECDSA = {
|
|
getBigRandom: function (limit) {
|
|
return new BigInteger(limit.bitLength(), rng)
|
|
.mod(limit.subtract(BigInteger.ONE))
|
|
.add(BigInteger.ONE)
|
|
;
|
|
},
|
|
sign: function (hash, priv) {
|
|
var d = priv;
|
|
var n = ecparams.getN();
|
|
var e = BigInteger.fromByteArrayUnsigned(hash);
|
|
|
|
do {
|
|
var k = ECDSA.getBigRandom(n);
|
|
var G = ecparams.getG();
|
|
var Q = G.multiply(k);
|
|
var r = Q.getX().toBigInteger().mod(n);
|
|
} while (r.compareTo(BigInteger.ZERO) <= 0);
|
|
|
|
var s = k.modInverse(n).multiply(e.add(d.multiply(r))).mod(n);
|
|
|
|
return ECDSA.serializeSig(r, s);
|
|
},
|
|
|
|
verify: function (hash, sig, pubkey) {
|
|
var r,s;
|
|
if (Bitcoin.Util.isArray(sig)) {
|
|
var obj = ECDSA.parseSig(sig);
|
|
r = obj.r;
|
|
s = obj.s;
|
|
} else if ("object" === typeof sig && sig.r && sig.s) {
|
|
r = sig.r;
|
|
s = sig.s;
|
|
} else {
|
|
throw "Invalid value for signature";
|
|
}
|
|
|
|
var Q;
|
|
if (pubkey instanceof ECPointFp) {
|
|
Q = pubkey;
|
|
} else if (Bitcoin.Util.isArray(pubkey)) {
|
|
Q = ECPointFp.decodeFrom(ecparams.getCurve(), pubkey);
|
|
} else {
|
|
throw "Invalid format for pubkey value, must be byte array or ECPointFp";
|
|
}
|
|
var e = BigInteger.fromByteArrayUnsigned(hash);
|
|
|
|
return ECDSA.verifyRaw(e, r, s, Q);
|
|
},
|
|
|
|
verifyRaw: function (e, r, s, Q) {
|
|
var n = ecparams.getN();
|
|
var G = ecparams.getG();
|
|
|
|
if (r.compareTo(BigInteger.ONE) < 0 ||
|
|
r.compareTo(n) >= 0)
|
|
return false;
|
|
|
|
if (s.compareTo(BigInteger.ONE) < 0 ||
|
|
s.compareTo(n) >= 0)
|
|
return false;
|
|
|
|
var c = s.modInverse(n);
|
|
|
|
var u1 = e.multiply(c).mod(n);
|
|
var u2 = r.multiply(c).mod(n);
|
|
|
|
// TODO(!!!): For some reason Shamir's trick isn't working with
|
|
// signed message verification!? Probably an implementation
|
|
// error!
|
|
//var point = implShamirsTrick(G, u1, Q, u2);
|
|
var point = G.multiply(u1).add(Q.multiply(u2));
|
|
|
|
var v = point.getX().toBigInteger().mod(n);
|
|
|
|
return v.equals(r);
|
|
},
|
|
|
|
/**
|
|
* Serialize a signature into DER format.
|
|
*
|
|
* Takes two BigIntegers representing r and s and returns a byte array.
|
|
*/
|
|
serializeSig: function (r, s) {
|
|
var rBa = r.toByteArraySigned();
|
|
var sBa = s.toByteArraySigned();
|
|
|
|
var sequence = [];
|
|
sequence.push(0x02); // INTEGER
|
|
sequence.push(rBa.length);
|
|
sequence = sequence.concat(rBa);
|
|
|
|
sequence.push(0x02); // INTEGER
|
|
sequence.push(sBa.length);
|
|
sequence = sequence.concat(sBa);
|
|
|
|
sequence.unshift(sequence.length);
|
|
sequence.unshift(0x30); // SEQUENCE
|
|
|
|
return sequence;
|
|
},
|
|
|
|
/**
|
|
* Parses a byte array containing a DER-encoded signature.
|
|
*
|
|
* This function will return an object of the form:
|
|
*
|
|
* {
|
|
* r: BigInteger,
|
|
* s: BigInteger
|
|
* }
|
|
*/
|
|
parseSig: function (sig) {
|
|
var cursor;
|
|
if (sig[0] != 0x30)
|
|
throw new Error("Signature not a valid DERSequence");
|
|
|
|
cursor = 2;
|
|
if (sig[cursor] != 0x02)
|
|
throw new Error("First element in signature must be a DERInteger");;
|
|
var rBa = sig.slice(cursor+2, cursor+2+sig[cursor+1]);
|
|
|
|
cursor += 2+sig[cursor+1];
|
|
if (sig[cursor] != 0x02)
|
|
throw new Error("Second element in signature must be a DERInteger");
|
|
var sBa = sig.slice(cursor+2, cursor+2+sig[cursor+1]);
|
|
|
|
cursor += 2+sig[cursor+1];
|
|
|
|
//if (cursor != sig.length)
|
|
// throw new Error("Extra bytes in signature");
|
|
|
|
var r = BigInteger.fromByteArrayUnsigned(rBa);
|
|
var s = BigInteger.fromByteArrayUnsigned(sBa);
|
|
|
|
return {r: r, s: s};
|
|
},
|
|
|
|
parseSigCompact: function (sig) {
|
|
if (sig.length !== 65) {
|
|
throw "Signature has the wrong length";
|
|
}
|
|
|
|
// Signature is prefixed with a type byte storing three bits of
|
|
// information.
|
|
var i = sig[0] - 27;
|
|
if (i < 0 || i > 7) {
|
|
throw "Invalid signature type";
|
|
}
|
|
|
|
var n = ecparams.getN();
|
|
var r = BigInteger.fromByteArrayUnsigned(sig.slice(1, 33)).mod(n);
|
|
var s = BigInteger.fromByteArrayUnsigned(sig.slice(33, 65)).mod(n);
|
|
|
|
return {r: r, s: s, i: i};
|
|
},
|
|
|
|
/**
|
|
* Recover a public key from a signature.
|
|
*
|
|
* See SEC 1: Elliptic Curve Cryptography, section 4.1.6, "Public
|
|
* Key Recovery Operation".
|
|
*
|
|
* http://www.secg.org/download/aid-780/sec1-v2.pdf
|
|
*/
|
|
recoverPubKey: function (r, s, hash, i) {
|
|
// The recovery parameter i has two bits.
|
|
i = i & 3;
|
|
|
|
// The less significant bit specifies whether the y coordinate
|
|
// of the compressed point is even or not.
|
|
var isYEven = i & 1;
|
|
|
|
// The more significant bit specifies whether we should use the
|
|
// first or second candidate key.
|
|
var isSecondKey = i >> 1;
|
|
|
|
var n = ecparams.getN();
|
|
var G = ecparams.getG();
|
|
var curve = ecparams.getCurve();
|
|
var p = curve.getQ();
|
|
var a = curve.getA().toBigInteger();
|
|
var b = curve.getB().toBigInteger();
|
|
|
|
// We precalculate (p + 1) / 4 where p is if the field order
|
|
if (!P_OVER_FOUR) {
|
|
P_OVER_FOUR = p.add(BigInteger.ONE).divide(BigInteger.valueOf(4));
|
|
}
|
|
|
|
// 1.1 Compute x
|
|
var x = isSecondKey ? r.add(n) : r;
|
|
|
|
// 1.3 Convert x to point
|
|
var alpha = x.multiply(x).multiply(x).add(a.multiply(x)).add(b).mod(p);
|
|
var beta = alpha.modPow(P_OVER_FOUR, p);
|
|
|
|
var xorOdd = beta.isEven() ? (i % 2) : ((i+1) % 2);
|
|
// If beta is even, but y isn't or vice versa, then convert it,
|
|
// otherwise we're done and y == beta.
|
|
var y = (beta.isEven() ? !isYEven : isYEven) ? beta : p.subtract(beta);
|
|
|
|
// 1.4 Check that nR is at infinity
|
|
var R = new ECPointFp(curve,
|
|
curve.fromBigInteger(x),
|
|
curve.fromBigInteger(y));
|
|
R.validate();
|
|
|
|
// 1.5 Compute e from M
|
|
var e = BigInteger.fromByteArrayUnsigned(hash);
|
|
var eNeg = BigInteger.ZERO.subtract(e).mod(n);
|
|
|
|
// 1.6 Compute Q = r^-1 (sR - eG)
|
|
var rInv = r.modInverse(n);
|
|
var Q = implShamirsTrick(R, s, G, eNeg).multiply(rInv);
|
|
|
|
Q.validate();
|
|
if (!ECDSA.verifyRaw(e, r, s, Q)) {
|
|
throw "Pubkey recovery unsuccessful";
|
|
}
|
|
|
|
var pubKey = new Bitcoin.ECKey();
|
|
pubKey.pub = Q;
|
|
return pubKey;
|
|
},
|
|
|
|
/**
|
|
* Calculate pubkey extraction parameter.
|
|
*
|
|
* When extracting a pubkey from a signature, we have to
|
|
* distinguish four different cases. Rather than putting this
|
|
* burden on the verifier, Bitcoin includes a 2-bit value with the
|
|
* signature.
|
|
*
|
|
* This function simply tries all four cases and returns the value
|
|
* that resulted in a successful pubkey recovery.
|
|
*/
|
|
calcPubkeyRecoveryParam: function (address, r, s, hash)
|
|
{
|
|
for (var i = 0; i < 4; i++) {
|
|
try {
|
|
var pubkey = Bitcoin.ECDSA.recoverPubKey(r, s, hash, i);
|
|
if (pubkey.getBitcoinAddress().toString() == address) {
|
|
return i;
|
|
}
|
|
} catch (e) {}
|
|
}
|
|
throw "Unable to find valid recovery factor";
|
|
}
|
|
};
|
|
|
|
return ECDSA;
|
|
})();
|
|
Bitcoin.ECKey = (function () {
|
|
var ECDSA = Bitcoin.ECDSA;
|
|
var ecparams = getSECCurveByName("secp256k1");
|
|
var rng = new SecureRandom();
|
|
|
|
var ECKey = function (input) {
|
|
if (!input) {
|
|
// Generate new key
|
|
var n = ecparams.getN();
|
|
//this.priv = ECDSA.getBigRandom(n);
|
|
} else if (input instanceof BigInteger) {
|
|
// Input is a private key value
|
|
this.priv = input;
|
|
} else if (Bitcoin.Util.isArray(input)) {
|
|
// Prepend zero byte to prevent interpretation as negative integer
|
|
this.priv = BigInteger.fromByteArrayUnsigned(input);
|
|
} else if ("string" == typeof input) {
|
|
if (input.length == 51 && input[0] == '5') {
|
|
// Base58 encoded private key
|
|
this.priv = BigInteger.fromByteArrayUnsigned(ECKey.decodeString(input));
|
|
} else {
|
|
// Prepend zero byte to prevent interpretation as negative integer
|
|
this.priv = BigInteger.fromByteArrayUnsigned(Crypto.util.hexToBytes(input));
|
|
}
|
|
}
|
|
this.compressed = !!ECKey.compressByDefault;
|
|
};
|
|
|
|
/**
|
|
* Whether public keys should be returned compressed by default.
|
|
*/
|
|
ECKey.compressByDefault = false;
|
|
|
|
/**
|
|
* Set whether the public key should be returned compressed or not.
|
|
*/
|
|
ECKey.prototype.setCompressed = function (v) {
|
|
this.compressed = !!v;
|
|
};
|
|
|
|
/**
|
|
* Return public key in DER encoding.
|
|
*/
|
|
ECKey.prototype.getPub = function () {
|
|
return this.getPubPoint().getEncoded(this.compressed);
|
|
};
|
|
|
|
/**
|
|
* Return public point as ECPoint object.
|
|
*/
|
|
ECKey.prototype.getPubPoint = function () {
|
|
if (!this.pub) this.pub = ecparams.getG().multiply(this.priv);
|
|
|
|
return this.pub;
|
|
};
|
|
|
|
/**
|
|
* Get the pubKeyHash for this key.
|
|
*
|
|
* This is calculated as RIPE160(SHA256([encoded pubkey])) and returned as
|
|
* a byte array.
|
|
*/
|
|
ECKey.prototype.getPubKeyHash = function () {
|
|
if (this.pubKeyHash) return this.pubKeyHash;
|
|
|
|
return this.pubKeyHash = Bitcoin.Util.sha256ripe160(this.getPub());
|
|
};
|
|
|
|
ECKey.prototype.getBitcoinAddress = function () {
|
|
var hash = this.getPubKeyHash();
|
|
var addr = new Bitcoin.Address(hash);
|
|
return addr;
|
|
};
|
|
|
|
ECKey.prototype.getExportedPrivateKey = function () {
|
|
var hash = this.priv.toByteArrayUnsigned();
|
|
while (hash.length < 32) hash.unshift(0);
|
|
hash.unshift(0x80);
|
|
var checksum = Crypto.SHA256(Crypto.SHA256(hash, {asBytes: true}), {asBytes: true});
|
|
var bytes = hash.concat(checksum.slice(0,4));
|
|
return Bitcoin.Base58.encode(bytes);
|
|
};
|
|
|
|
ECKey.prototype.setPub = function (pub) {
|
|
this.pub = ECPointFp.decodeFrom(ecparams.getCurve(), pub);
|
|
};
|
|
|
|
ECKey.prototype.toString = function (format) {
|
|
if (format === "base64") {
|
|
return Crypto.util.bytesToBase64(this.priv.toByteArrayUnsigned());
|
|
} else {
|
|
return Crypto.util.bytesToHex(this.priv.toByteArrayUnsigned());
|
|
}
|
|
};
|
|
|
|
ECKey.prototype.sign = function (hash) {
|
|
return ECDSA.sign(hash, this.priv);
|
|
};
|
|
|
|
ECKey.prototype.verify = function (hash, sig) {
|
|
return ECDSA.verify(hash, sig, this.getPub());
|
|
};
|
|
|
|
/**
|
|
* Parse an exported private key contained in a string.
|
|
*/
|
|
ECKey.decodeString = function (string) {
|
|
var bytes = Bitcoin.Base58.decode(string);
|
|
|
|
var hash = bytes.slice(0, 33);
|
|
|
|
var checksum = Crypto.SHA256(Crypto.SHA256(hash, {asBytes: true}), {asBytes: true});
|
|
|
|
if (checksum[0] != bytes[33] ||
|
|
checksum[1] != bytes[34] ||
|
|
checksum[2] != bytes[35] ||
|
|
checksum[3] != bytes[36]) {
|
|
throw "Checksum validation failed!";
|
|
}
|
|
|
|
var version = hash.shift();
|
|
|
|
if (version != 0x80) {
|
|
throw "Version "+version+" not supported!";
|
|
}
|
|
|
|
return hash;
|
|
};
|
|
|
|
return ECKey;
|
|
})();
|
|
|
|
|
|
module.exports.ECKey = Bitcoin.ECKey;
|
|
|
|
},{}],"./config":[function(require,module,exports){
|
|
module.exports=require('4itQ50');
|
|
},{}],"4itQ50":[function(require,module,exports){
|
|
module.exports = {
|
|
network: 'livenet',
|
|
logger: 'normal' // none, normal, debug
|
|
};
|
|
|
|
},{}],"./const":[function(require,module,exports){
|
|
module.exports=require('f08cvL');
|
|
},{}],"f08cvL":[function(require,module,exports){
|
|
|
|
MSG = {
|
|
TX: 1,
|
|
BLOCK: 2,
|
|
FILTERED_BLOCK: 3,
|
|
};
|
|
|
|
MSG.to_str = function(t) {
|
|
switch(t) {
|
|
case MSG.TX: return 'transaction';
|
|
case MSG.BLOCK: return 'block';
|
|
case MSG.FILTERED_BLOCK: return 'filtered block';
|
|
default: return 'unknown';
|
|
}
|
|
}
|
|
|
|
exports.MSG = MSG;
|
|
|
|
|
|
},{}],"G+CcXD":[function(require,module,exports){
|
|
(function (Buffer){
|
|
// Address
|
|
// =======
|
|
//
|
|
// Handles a bitcoin address
|
|
//
|
|
//
|
|
// Synopsis
|
|
// --------
|
|
// ```
|
|
// var address = new Address('1A1zP1eP5QGefi2DMPTfTL5SLmv7DivfNa');
|
|
// if (address.isValid()) {
|
|
// //...
|
|
// }
|
|
//
|
|
// // Also an address can be created from
|
|
// // public keys
|
|
// var address = Address.fromPubKey(myPubkey);
|
|
//
|
|
// // Or from a ScriptPubKey (from a transaction output)
|
|
// var address = Address.fromScriptPubKey(scriptPubKey);
|
|
//
|
|
// // Multisig address p2sh handling
|
|
// var myPukeys = [pubkey0, pubkey1, pubkey2];
|
|
// var p2shAddress = Address.fromPubKeys(2, myPubkeys);
|
|
// if (p2shAddress.isScript()) { //true
|
|
// }
|
|
//
|
|
//
|
|
// ```
|
|
|
|
|
|
'use strict';
|
|
var imports = require('soop').imports();
|
|
var coinUtil = imports.coinUtil || require('../util');
|
|
var parent = imports.parent || require('../util/VersionedData');
|
|
var networks = imports.networks || require('../networks');
|
|
var Script = imports.Script || require('./Script');
|
|
|
|
function Address() {
|
|
Address.super(this, arguments);
|
|
}
|
|
|
|
Address.parent = parent;
|
|
parent.applyEncodingsTo(Address);
|
|
|
|
// create a pubKeyHash address
|
|
Address.fromPubKey = function(pubKey, network) {
|
|
if (!network)
|
|
network = 'livenet';
|
|
|
|
if (pubKey.length !== 33 && pubKey.length !== 65)
|
|
throw new Error('Invalid public key');
|
|
|
|
var version = networks[network].addressVersion;
|
|
var hash = coinUtil.sha256ripe160(pubKey);
|
|
|
|
return new Address(version, hash);
|
|
};
|
|
|
|
// create an address from a Key object
|
|
Address.fromKey = function(key, network) {
|
|
return Address.fromPubKey(key.public, network);
|
|
};
|
|
|
|
// create a p2sh m-of-n multisig address
|
|
Address.fromPubKeys = function(mReq, pubKeys, network, opts) {
|
|
if (!network)
|
|
network = 'livenet';
|
|
|
|
for (var i in pubKeys) {
|
|
var pubKey = pubKeys[i];
|
|
if (pubKey.length != 33 && pubKey.length != 65)
|
|
throw new Error('Invalid public key');
|
|
}
|
|
|
|
var script = Script.createMultisig(mReq, pubKeys, opts);
|
|
return Address.fromScript(script, network);
|
|
};
|
|
|
|
//create a p2sh address from redeemScript
|
|
Address.fromScript = function(script, network) {
|
|
if (!network)
|
|
network = 'livenet';
|
|
|
|
if (typeof script === 'string') {
|
|
script = new Script(new Buffer(script,'hex'));
|
|
}
|
|
|
|
var version = networks[network].P2SHVersion;
|
|
var buf = script.getBuffer();
|
|
var hash = coinUtil.sha256ripe160(buf);
|
|
|
|
return new Address(version, hash);
|
|
};
|
|
|
|
//extract and address from scriptPubKey
|
|
Address.fromScriptPubKey = function(scriptPubKey, network) {
|
|
|
|
if (typeof scriptPubKey === 'string') {
|
|
scriptPubKey = new Script(new Buffer(scriptPubKey,'hex'));
|
|
}
|
|
|
|
if (!network)
|
|
network = 'livenet';
|
|
|
|
var ret=[], version;
|
|
var payload = scriptPubKey.capture();
|
|
|
|
if (payload) {
|
|
var txType = scriptPubKey.classify();
|
|
switch (txType) {
|
|
case Script.TX_PUBKEY:
|
|
payload[0] = coinUtil.sha256ripe160(payload[0]);
|
|
version = networks[network].addressVersion;
|
|
break;
|
|
case Script.TX_PUBKEYHASH:
|
|
version = networks[network].addressVersion;
|
|
break;
|
|
case Script.TX_MULTISIG:
|
|
version = networks[network].addressVersion;
|
|
for(var i in payload)
|
|
payload[i] = coinUtil.sha256ripe160(payload[i]);
|
|
break;
|
|
case Script.TX_SCRIPTHASH:
|
|
version = networks[network].P2SHVersion;
|
|
break;
|
|
}
|
|
for(var i in payload)
|
|
ret.push(new Address(version,payload[i]));
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
// validates the address
|
|
Address.prototype.validate = function() {
|
|
this.doAsBinary(function() {
|
|
Address.super(this, 'validate', arguments);
|
|
if(this.data.length !== 21) throw new Error('invalid data length');
|
|
});
|
|
if (typeof this.network() === 'undefined') throw new Error('invalid network');
|
|
};
|
|
|
|
Address.prototype.isValid = function() {
|
|
var answer = Address.super(this, 'isValid', arguments);
|
|
return answer;
|
|
};
|
|
|
|
// returns the network information (livenet or testnet, as described on networks.js) of the address
|
|
Address.prototype.network = function() {
|
|
var version = this.version();
|
|
|
|
var livenet = networks.livenet;
|
|
var testnet = networks.testnet;
|
|
|
|
var answer;
|
|
if (version === livenet.addressVersion || version === livenet.P2SHVersion)
|
|
answer = livenet;
|
|
else if (version === testnet.addressVersion || version === testnet.P2SHVersion)
|
|
answer = testnet;
|
|
|
|
return answer;
|
|
};
|
|
|
|
// returns true is the address is a pay-to-script (P2SH) address type.
|
|
Address.prototype.isScript = function() {
|
|
return this.isValid() && this.version() === this.network().P2SHVersion;
|
|
};
|
|
|
|
|
|
module.exports = require('soop')(Address);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../networks":"ULNIu2","../util":140,"../util/VersionedData":"QLzNQg","./Script":"hQ0t76","buffer":80,"soop":125}],"./lib/Address":[function(require,module,exports){
|
|
module.exports=require('G+CcXD');
|
|
},{}],"YL/05i":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var Point = require('./Point'),
|
|
Key = require('./Key'),
|
|
sha256 = require('../util').sha256,
|
|
twoSha256 = require('../util').twoSha256;
|
|
|
|
/**
|
|
* For now, this class can only supports derivation from public key
|
|
* It doesn't support private key derivation (TODO).
|
|
*
|
|
* @example examples/Armory.js
|
|
*/
|
|
function Armory (chaincode, pubkey) {
|
|
this.chaincode = new Buffer(chaincode, 'hex');
|
|
this.pubkey = new Buffer(pubkey, 'hex');
|
|
}
|
|
|
|
Armory.prototype.generatePubKey = function () {
|
|
var pubKey = this.pubkey;
|
|
var chainCode = this.chaincode;
|
|
var chainXor = twoSha256(pubKey);
|
|
|
|
for (var i = 0; i < 32; i++)
|
|
chainXor[i] ^= chainCode[i];
|
|
|
|
var pt = Point.fromUncompressedPubKey(pubKey);
|
|
pt = Point.multiply(pt, chainXor);
|
|
|
|
var new_pubkey = pt.toUncompressedPubKey();
|
|
|
|
return new_pubkey;
|
|
};
|
|
|
|
Armory.prototype.next = function () {
|
|
var next_pubkey = this.generatePubKey();
|
|
return new Armory(this.chaincode, next_pubkey);
|
|
};
|
|
|
|
/**
|
|
* PS: MPK here represents the pubkey concatenated
|
|
* with the chain code. It is an unofficial standard.
|
|
*
|
|
* Armory will soon release an officially supported
|
|
* format:
|
|
*
|
|
* https://github.com/etotheipi/BitcoinArmory/issues/204#issuecomment-42217801
|
|
*/
|
|
Armory.fromMasterPublicKey = function (mpk) {
|
|
var pubkey = mpk.substr(0, 130);
|
|
var chaincode = mpk.substr(130, mpk.length);
|
|
return new Armory(chaincode, pubkey);
|
|
};
|
|
|
|
function decode (str) {
|
|
var from = '0123456789abcdef';
|
|
var to = 'asdfghjkwertuion';
|
|
var res = '';
|
|
for (var i = 0; i < str.length; i++)
|
|
res += from.charAt(to.indexOf(str.charAt(i)));
|
|
return res;
|
|
}
|
|
|
|
Armory.decodeSeed = function (seed) {
|
|
var keys = seed.trim().split('\n');
|
|
var lines = [];
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
var k = keys[i].replace(' ','');
|
|
var raw = new Buffer(decode(k), 'hex');
|
|
var data = raw.slice(0, 16);
|
|
lines.push(data);
|
|
}
|
|
|
|
var privKey = Buffer.concat([ lines[0], lines[1] ]);
|
|
var chainCode = (lines.length==4) ?
|
|
Buffer.concat([ lines[2], lines[3] ]) : Armory.deriveChaincode(privKey);
|
|
|
|
return {
|
|
privKey: privKey,
|
|
chainCode: chainCode
|
|
};
|
|
};
|
|
|
|
// Derive chain code from root key
|
|
Armory.fromSeed = function (seed) {
|
|
var res = Armory.decodeSeed(seed);
|
|
// generate first public key
|
|
var key = new Key();
|
|
key.private = res.privKey;
|
|
key.compressed = false;
|
|
key.regenerateSync();
|
|
|
|
return new Armory(res.chainCode, key.public);
|
|
};
|
|
|
|
Armory.deriveChaincode = function (root) {
|
|
var msg = 'Derive Chaincode from Root Key';
|
|
var hash = twoSha256(root);
|
|
|
|
var okey = [];
|
|
var ikey = [];
|
|
for (var i = 0; i < hash.length; i++) {
|
|
okey.push(0x5c ^ hash[i]);
|
|
ikey.push(0x36 ^ hash[i]);
|
|
}
|
|
|
|
okey = new Buffer(okey);
|
|
ikey = new Buffer(ikey);
|
|
|
|
var m = new Buffer(msg, 'utf8');
|
|
var a = sha256(Buffer.concat([ ikey, m ]));
|
|
var b = sha256(Buffer.concat([ okey, a ]));
|
|
return b;
|
|
};
|
|
|
|
module.exports = Armory;
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util":140,"./Key":"ALJ4PS","./Point":"6tXgqr","buffer":80}],"./lib/Armory":[function(require,module,exports){
|
|
module.exports=require('YL/05i');
|
|
},{}],"./lib/Base58":[function(require,module,exports){
|
|
module.exports=require('6VqyzY');
|
|
},{}],"6VqyzY":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var crypto = require('crypto');
|
|
var bignum = require('bignum');
|
|
|
|
var globalBuffer = new Buffer(1024);
|
|
var zerobuf = new Buffer(0);
|
|
var ALPHABET = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz';
|
|
var ALPHABET_ZERO = ALPHABET[0];
|
|
var ALPHABET_BUF = new Buffer(ALPHABET, 'ascii');
|
|
var ALPHABET_INV = {};
|
|
for(var i=0; i < ALPHABET.length; i++) {
|
|
ALPHABET_INV[ALPHABET[i]] = i;
|
|
};
|
|
|
|
// Vanilla Base58 Encoding
|
|
var base58 = {
|
|
encode: function(buf) {
|
|
var str;
|
|
var x = bignum.fromBuffer(buf);
|
|
var r;
|
|
|
|
if(buf.length < 512) {
|
|
str = globalBuffer;
|
|
} else {
|
|
str = new Buffer(buf.length << 1);
|
|
}
|
|
var i = str.length - 1;
|
|
while(x.gt(0)) {
|
|
r = x.mod(58);
|
|
x = x.div(58);
|
|
str[i] = ALPHABET_BUF[r.toNumber()];
|
|
i--;
|
|
}
|
|
|
|
// deal with leading zeros
|
|
var j=0;
|
|
while(buf[j] == 0) {
|
|
str[i] = ALPHABET_BUF[0];
|
|
j++; i--;
|
|
}
|
|
|
|
return str.slice(i+1,str.length).toString('ascii');
|
|
},
|
|
|
|
decode: function(str) {
|
|
if(str.length == 0) return zerobuf;
|
|
var answer = bignum(0);
|
|
for(var i=0; i<str.length; i++) {
|
|
answer.mul(58)
|
|
answer = answer.mul(58);
|
|
answer = answer.add(ALPHABET_INV[str[i]]);
|
|
};
|
|
var i = 0;
|
|
while(i < str.length && str[i] == ALPHABET_ZERO) {
|
|
i++;
|
|
}
|
|
if(i > 0) {
|
|
var zb = new Buffer(i);
|
|
zb.fill(0);
|
|
if(i == str.length) return zb;
|
|
answer = answer.toBuffer();
|
|
return Buffer.concat([zb, answer], i+answer.length);
|
|
} else {
|
|
return answer.toBuffer();
|
|
}
|
|
},
|
|
};
|
|
|
|
// Base58Check Encoding
|
|
function sha256(data) {
|
|
return new Buffer(crypto.createHash('sha256').update(data).digest('binary'), 'binary');
|
|
};
|
|
|
|
function doubleSHA256(data) {
|
|
return sha256(sha256(data));
|
|
};
|
|
|
|
var base58Check = {
|
|
encode: function(buf) {
|
|
var checkedBuf = new Buffer(buf.length + 4);
|
|
var hash = doubleSHA256(buf);
|
|
buf.copy(checkedBuf);
|
|
hash.copy(checkedBuf, buf.length);
|
|
return base58.encode(checkedBuf);
|
|
},
|
|
|
|
decode: function(s) {
|
|
var buf = base58.decode(s);
|
|
if (buf.length < 4) {
|
|
throw new Error("invalid input: too short");
|
|
}
|
|
|
|
var data = buf.slice(0, -4);
|
|
var csum = buf.slice(-4);
|
|
|
|
var hash = doubleSHA256(data);
|
|
var hash4 = hash.slice(0, 4);
|
|
|
|
if (csum.toString() != hash4.toString()) {
|
|
throw new Error("checksum mismatch");
|
|
}
|
|
|
|
return data;
|
|
},
|
|
};
|
|
|
|
// if you frequently do base58 encodings with data larger
|
|
// than 512 bytes, you can use this method to expand the
|
|
// size of the reusable buffer
|
|
exports.setBuffer = function(buf) {
|
|
globalBuffer = buf;
|
|
};
|
|
|
|
exports.base58 = base58;
|
|
exports.base58Check = base58Check;
|
|
exports.encode = base58.encode;
|
|
exports.decode = base58.decode;
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"bignum":58,"buffer":80,"crypto":84}],"./lib/Block":[function(require,module,exports){
|
|
module.exports=require('pJEQEB');
|
|
},{}],"pJEQEB":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
|
|
var util = imports.util || require('../util');
|
|
var Debug1 = imports.Debug1 || function() {};
|
|
var Script = imports.Script || require('./Script');
|
|
var Bignum = imports.Bignum || require('bignum');
|
|
var Binary = imports.Binary || require('binary');
|
|
var Step = imports.Step || require('step');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
var Transaction = imports.Transaction || require('./Transaction');
|
|
var TransactionIn = Transaction.In;
|
|
var TransactionOut = Transaction.Out;
|
|
var COINBASE_OP = Transaction.COINBASE_OP;
|
|
var VerificationError = imports.VerificationError || require('../util/error').VerificationError;
|
|
var BlockRules = {
|
|
maxTimeOffset: 2 * 60 * 60, // How far block timestamps can be into the future
|
|
largestHash: Bignum(2).pow(256)
|
|
};
|
|
|
|
function Block(data)
|
|
{
|
|
if ("object" !== typeof data) {
|
|
data = {};
|
|
}
|
|
this.hash = data.hash || null;
|
|
this.prev_hash = data.prev_hash || util.NULL_HASH;
|
|
this.merkle_root = data.merkle_root || util.NULL_HASH;
|
|
this.timestamp = data.timestamp || 0;
|
|
this.bits = data.bits || 0;
|
|
this.nonce = data.nonce || 0;
|
|
this.version = data.version || 0;
|
|
this.height = data.height || 0;
|
|
this.size = data.size || 0;
|
|
this.active = data.active || false;
|
|
this.chainWork = data.chainWork || util.EMPTY_BUFFER;
|
|
this.txs = data.txs || [];
|
|
}
|
|
|
|
Block.prototype.getHeader = function getHeader() {
|
|
var buf = new Buffer(80);
|
|
var ofs = 0;
|
|
buf.writeUInt32LE(this.version, ofs); ofs += 4;
|
|
this.prev_hash.copy(buf, ofs); ofs += 32;
|
|
this.merkle_root.copy(buf, ofs); ofs += 32;
|
|
buf.writeUInt32LE(this.timestamp, ofs); ofs += 4;
|
|
buf.writeUInt32LE(this.bits, ofs); ofs += 4;
|
|
buf.writeUInt32LE(this.nonce, ofs); ofs += 4;
|
|
return buf;
|
|
};
|
|
|
|
Block.prototype.parse = function parse(parser, headerOnly) {
|
|
this.version = parser.word32le();
|
|
this.prev_hash = parser.buffer(32);
|
|
this.merkle_root = parser.buffer(32);
|
|
this.timestamp = parser.word32le();
|
|
this.bits = parser.word32le();
|
|
this.nonce = parser.word32le();
|
|
|
|
this.txs = [];
|
|
this.size = 0;
|
|
|
|
if (headerOnly)
|
|
return;
|
|
|
|
var txCount = parser.varInt();
|
|
|
|
for (var i = 0; i < txCount; i++) {
|
|
var tx = new Transaction();
|
|
tx.parse(parser);
|
|
this.txs.push(tx);
|
|
}
|
|
};
|
|
|
|
Block.prototype.calcHash = function calcHash() {
|
|
var header = this.getHeader();
|
|
|
|
return util.twoSha256(header);
|
|
};
|
|
|
|
Block.prototype.checkHash = function checkHash() {
|
|
if (!this.hash || !this.hash.length) return false;
|
|
return buffertools.compare(this.calcHash(), this.hash) == 0;
|
|
};
|
|
|
|
Block.prototype.getHash = function getHash() {
|
|
if (!this.hash || !this.hash.length) this.hash = this.calcHash();
|
|
|
|
return this.hash;
|
|
};
|
|
|
|
Block.prototype.checkProofOfWork = function checkProofOfWork() {
|
|
var target = util.decodeDiffBits(this.bits);
|
|
|
|
// TODO: Create a compare method in node-buffertools that uses the correct
|
|
// endian so we don't have to reverse both buffers before comparing.
|
|
var reverseHash = buffertools.reverse(this.hash);
|
|
if (buffertools.compare(reverseHash, target) > 0) {
|
|
throw new VerificationError('Difficulty target not met');
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Returns the amount of work that went into this block.
|
|
*
|
|
* Work is defined as the average number of tries required to meet this
|
|
* block's difficulty target. For example a target that is greater than 5%
|
|
* of all possible hashes would mean that 20 "work" is required to meet it.
|
|
*/
|
|
Block.prototype.getWork = function getWork() {
|
|
var target = util.decodeDiffBits(this.bits, true);
|
|
return BlockRules.largestHash.div(target.add(1));
|
|
};
|
|
|
|
Block.prototype.checkTimestamp = function checkTimestamp() {
|
|
var currentTime = new Date().getTime() / 1000;
|
|
if (this.timestamp > currentTime + BlockRules.maxTimeOffset) {
|
|
throw new VerificationError('Timestamp too far into the future');
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
Block.prototype.checkTransactions = function checkTransactions(txs) {
|
|
if (!Array.isArray(txs) || txs.length <= 0) {
|
|
throw new VerificationError('No transactions');
|
|
}
|
|
if (!txs[0].isCoinBase()) {
|
|
throw new VerificationError('First tx must be coinbase');
|
|
}
|
|
for (var i = 1; i < txs.length; i++) {
|
|
if (txs[i].isCoinBase()) {
|
|
throw new VerificationError('Tx index '+i+' must not be coinbase');
|
|
}
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Build merkle tree.
|
|
*
|
|
* Ported from Java. Original code: BitcoinJ by Mike Hearn
|
|
* Copyright (c) 2011 Google Inc.
|
|
*/
|
|
Block.prototype.getMerkleTree = function getMerkleTree(txs) {
|
|
// The merkle hash is based on a tree of hashes calculated from the transactions:
|
|
//
|
|
// merkleHash
|
|
// /\
|
|
// / \
|
|
// A B
|
|
// / \ / \
|
|
// tx1 tx2 tx3 tx4
|
|
//
|
|
// Basically transactions are hashed, then the hashes of the transactions are hashed
|
|
// again and so on upwards into the tree. The point of this scheme is to allow for
|
|
// disk space savings later on.
|
|
//
|
|
// This function is a direct translation of CBlock::BuildMerkleTree().
|
|
|
|
if (txs.length == 0) {
|
|
return [util.NULL_HASH.slice(0)];
|
|
}
|
|
|
|
// Start by adding all the hashes of the transactions as leaves of the tree.
|
|
var tree = txs.map(function (tx) {
|
|
return tx instanceof Transaction ? tx.getHash() : tx;
|
|
});
|
|
|
|
var j = 0;
|
|
// Now step through each level ...
|
|
for (var size = txs.length; size > 1; size = Math.floor((size + 1) / 2)) {
|
|
// and for each leaf on that level ..
|
|
for (var i = 0; i < size; i += 2) {
|
|
var i2 = Math.min(i + 1, size - 1);
|
|
var a = tree[j + i];
|
|
var b = tree[j + i2];
|
|
tree.push(util.twoSha256(Buffer.concat([a,b])));
|
|
}
|
|
j += size;
|
|
}
|
|
|
|
return tree;
|
|
};
|
|
|
|
Block.prototype.calcMerkleRoot = function calcMerkleRoot(txs) {
|
|
var tree = this.getMerkleTree(txs);
|
|
return tree[tree.length - 1];
|
|
};
|
|
|
|
Block.prototype.checkMerkleRoot = function checkMerkleRoot(txs) {
|
|
if (!this.merkle_root || !this.merkle_root.length) {
|
|
throw new VerificationError('No merkle root');
|
|
}
|
|
|
|
if (buffertools.compare(this.calcMerkleRoot(txs), new Buffer(this.merkle_root)) !== 0) {
|
|
throw new VerificationError('Merkle root incorrect');
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
Block.prototype.checkBlock = function checkBlock(txs) {
|
|
if (!this.checkHash()) {
|
|
throw new VerificationError("Block hash invalid");
|
|
}
|
|
this.checkProofOfWork();
|
|
this.checkTimestamp();
|
|
|
|
if (txs) {
|
|
this.checkTransactions(txs);
|
|
if (!this.checkMerkleRoot(txs)) {
|
|
throw new VerificationError("Merkle hash invalid");
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
Block.getBlockValue = function getBlockValue(height) {
|
|
var subsidy = Bignum(50).mul(util.COIN);
|
|
subsidy = subsidy.div(Bignum(2).pow(Math.floor(height / 210000)));
|
|
return subsidy;
|
|
};
|
|
|
|
Block.prototype.getBlockValue = function getBlockValue() {
|
|
return Block.getBlockValue(this.height);
|
|
};
|
|
|
|
Block.prototype.toString = function toString() {
|
|
return "<Block " + util.formatHashAlt(this.hash) + " height="+this.height+">";
|
|
};
|
|
|
|
|
|
Block.prototype.createCoinbaseTx =
|
|
function createCoinbaseTx(beneficiary)
|
|
{
|
|
var tx = new Transaction();
|
|
tx.ins.push(new TransactionIn({
|
|
s: util.EMPTY_BUFFER,
|
|
q: 0xffffffff,
|
|
o: COINBASE_OP
|
|
}));
|
|
tx.outs.push(new TransactionOut({
|
|
v: util.bigIntToValue(this.getBlockValue()),
|
|
s: Script.createPubKeyOut(beneficiary).getBuffer()
|
|
}));
|
|
return tx;
|
|
};
|
|
|
|
Block.prototype.solve = function solve(miner, callback) {
|
|
var header = this.getHeader();
|
|
var target = util.decodeDiffBits(this.bits);
|
|
miner.solve(header, target, callback);
|
|
};
|
|
|
|
/**
|
|
* Returns an object with the same field names as jgarzik's getblock patch.
|
|
*/
|
|
Block.prototype.getStandardizedObject =
|
|
function getStandardizedObject(txs)
|
|
{
|
|
var block = {
|
|
hash: util.formatHashFull(this.getHash()),
|
|
version: this.version,
|
|
prev_block: util.formatHashFull(this.prev_hash),
|
|
mrkl_root: util.formatHashFull(this.merkle_root),
|
|
time: this.timestamp,
|
|
bits: this.bits,
|
|
nonce: this.nonce,
|
|
height: this.height
|
|
};
|
|
|
|
|
|
if (txs) {
|
|
var mrkl_tree = this.getMerkleTree(txs).map(function (buffer) {
|
|
return util.formatHashFull(buffer);
|
|
});
|
|
block.mrkl_root = mrkl_tree[mrkl_tree.length - 1];
|
|
|
|
block.n_tx = txs.length;
|
|
var totalSize = 80; // Block header
|
|
totalSize += util.getVarIntSize(txs.length); // txn_count
|
|
txs = txs.map(function (tx) {
|
|
tx = tx.getStandardizedObject();
|
|
totalSize += tx.size;
|
|
return tx;
|
|
});
|
|
block.size = totalSize;
|
|
block.tx = txs;
|
|
|
|
block.mrkl_tree = mrkl_tree;
|
|
} else {
|
|
block.size = this.size;
|
|
}
|
|
return block;
|
|
};
|
|
|
|
module.exports = require('soop')(Block);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util":140,"../util/error":139,"./Script":"hQ0t76","./Transaction":"LJhYtm","bignum":58,"binary":69,"buffer":80,"buffertools":"fugeBw","soop":125,"step":126}],"./lib/Bloom":[function(require,module,exports){
|
|
module.exports=require('KifRG4');
|
|
},{}],"KifRG4":[function(require,module,exports){
|
|
var MAX_BLOOM_FILTER_SIZE = 36000; // bytes
|
|
var MAX_HASH_FUNCS = 50;
|
|
var LN2SQUARED = 0.4804530139182014246671025263266649717305529515945455;
|
|
var LN2 = 0.6931471805599453094172321214581765680755001343602552;
|
|
var bit_mask = [0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80];
|
|
|
|
function Bloom() {
|
|
this.data = '';
|
|
this.hashFuncs = 0;
|
|
};
|
|
|
|
function ROTL32(x, r) {
|
|
return (x << r) | (x >> (32 - r));
|
|
};
|
|
|
|
function getBlockU32(blockIdx, data) {
|
|
var idx = blockIdx * 4;
|
|
var v = (data[idx + 0] << (0 * 8)) |
|
|
(data[idx + 1] << (1 * 8)) |
|
|
(data[idx + 2] << (2 * 8)) |
|
|
(data[idx + 3] << (3 * 8));
|
|
return v;
|
|
};
|
|
|
|
Bloom.prototype.hash = function(hashNum, data) {
|
|
var h1 = hashNum * (0xffffffff / (this.hashFuncs - 1));
|
|
var c1 = 0xcc9e2d51;
|
|
var c2 = 0x1b873593;
|
|
var nBlocks = data.length / 4;
|
|
|
|
// data body
|
|
for (var i = -nBlocks; i; i++) {
|
|
var k1 = getBlockU32(i);
|
|
|
|
k1 *= c1;
|
|
k1 = ROTLF32(k1, 15);
|
|
k1 *= c2;
|
|
|
|
h1 ^= k1;
|
|
h1 = ROTFL(h1, 13);
|
|
h1 = h1 * 5 + 0xe6546b64;
|
|
}
|
|
|
|
// tail (trailing 1-3 bytes)
|
|
var tail = data.slice(nBlocks * 4);
|
|
|
|
var k1 = 0;
|
|
|
|
switch (data.length & 3) {
|
|
case 3: k1 ^= tail[2] << 16;
|
|
case 2: k1 ^= tail[1] << 8;
|
|
case 1: k1 ^= tail[0];
|
|
k1 *= c1;
|
|
k1 = ROTL32(k1, 15);
|
|
k1 *= c2;
|
|
h1 ^= k1;
|
|
}
|
|
|
|
// finalize
|
|
h1 ^= data.length;
|
|
h1 ^= h1 >> 16;
|
|
h1 *= 0x85ebca6b;
|
|
h1 ^= h1 >> 13;
|
|
h1 *= 0xc2b2ae35;
|
|
h1 ^= h1 >> 16;
|
|
|
|
return h1 % (this.data.length * 8);
|
|
};
|
|
|
|
Bloom.prototype.insert = function(data) {
|
|
for (var i = 0; i < this.hashFuncs; i++) {
|
|
var index = this.hash(i, data);
|
|
this.data[index >> 3] |= bit_mask[7 & index];
|
|
}
|
|
};
|
|
|
|
Bloom.prototype.contains = function(data) {
|
|
for (var i = 0; i < this.hashFuncs; i++) {
|
|
var index = this.hash(i, data);
|
|
if (!(this.data[index >> 3] & bit_mask[7 & index]))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
Bloom.prototype.sizeOk = function() {
|
|
return this.data.length <= MAX_BLOOM_FILTER_SIZE &&
|
|
this.hashFuncs <= MAX_HASH_FUNCS;
|
|
};
|
|
|
|
function toInt(v) {
|
|
return ~~v;
|
|
}
|
|
|
|
function min(a, b) {
|
|
if (a < b)
|
|
return a;
|
|
return b;
|
|
}
|
|
|
|
Bloom.prototype.init = function(elements, FPRate) {
|
|
var filterSize = min(toInt(-1.0 / LN2SQUARED * elements * Math.log(FPRate)),
|
|
MAX_BLOOM_FILTER_SIZE * 8) / 8;
|
|
this.data[filterSize] = 0;
|
|
this.hashFuncs = min(toInt(this.data.length * 8 / elements * LN2),
|
|
MAX_HASH_FUNCS);
|
|
};
|
|
|
|
|
|
module.exports = require('soop')(Bloom);
|
|
|
|
},{"soop":125}],"./lib/Connection":[function(require,module,exports){
|
|
module.exports=require('DB/p3X');
|
|
},{}],"DB/p3X":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
|
|
var log = imports.log || require('../util/log');
|
|
|
|
var MAX_RECEIVE_BUFFER = 10000000;
|
|
var PROTOCOL_VERSION = 70000;
|
|
|
|
var Put = imports.Put || require('bufferput');
|
|
var Buffers = imports.Buffers || require('buffers');
|
|
require('../patches/Buffers.monkey').patch(Buffers);
|
|
|
|
var bitcoreDefaults = imports.config || require('../config');
|
|
var networks = imports.networks || require('../networks');
|
|
var Block = imports.Block || require('./Block');
|
|
var Transaction = imports.Transaction || require('./Transaction');
|
|
var util = imports.util || require('../util');
|
|
var Parser = imports.Parser || require('../util/BinaryParser');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
var doubleSha256 = imports.doubleSha256 || util.twoSha256;
|
|
var SecureRandom = imports.SecureRandom || require('./SecureRandom');
|
|
var nonce = SecureRandom.getPseudoRandomBuffer(8);
|
|
|
|
var BIP0031_VERSION = 60000;
|
|
|
|
function Connection(socket, peer, opts) {
|
|
Connection.super(this, arguments);
|
|
|
|
this.config = opts || bitcoreDefaults;
|
|
|
|
this.network = networks[this.config.network] || networks.livenet;
|
|
this.socket = socket;
|
|
this.peer = peer;
|
|
|
|
// check for socks5 proxy options and construct a proxied socket
|
|
if (this.config.proxy) {
|
|
var Socks5Client = imports.Socks5Client || require('socks5-client');
|
|
this.socket = new Socks5Client(this.config.proxy.host, this.config.proxy.port);
|
|
}
|
|
|
|
// A connection is considered "active" once we have received verack
|
|
this.active = false;
|
|
// The version incoming packages are interpreted as
|
|
this.recvVer = 0;
|
|
// The version outgoing packages are sent as
|
|
this.sendVer = 0;
|
|
// The (claimed) height of the remote peer's block chain
|
|
this.bestHeight = 0;
|
|
// Is this an inbound connection?
|
|
this.inbound = !!this.socket.server;
|
|
// Have we sent a getaddr on this connection?
|
|
this.getaddr = false;
|
|
|
|
// Receive buffer
|
|
this.buffers = new Buffers();
|
|
|
|
// Starting 20 Feb 2012, Version 0.2 is obsolete
|
|
// This is the same behavior as the official client
|
|
if (new Date().getTime() > 1329696000000) {
|
|
this.recvVer = 209;
|
|
this.sendVer = 209;
|
|
}
|
|
|
|
this.setupHandlers();
|
|
}
|
|
Connection.parent = imports.parent || require('events').EventEmitter;
|
|
|
|
Connection.prototype.open = function(callback) {
|
|
if (typeof callback === 'function') this.once('connect', callback);
|
|
this.socket.connect(this.peer.port, this.peer.host);
|
|
return this;
|
|
};
|
|
|
|
Connection.prototype.setupHandlers = function () {
|
|
this.socket.addListener('connect', this.handleConnect.bind(this));
|
|
this.socket.addListener('error', this.handleError.bind(this));
|
|
this.socket.addListener('end', this.handleDisconnect.bind(this));
|
|
this.socket.addListener('data', (function (data) {
|
|
var dumpLen = 35;
|
|
log.debug('['+this.peer+'] '+
|
|
'Recieved '+data.length+' bytes of data:');
|
|
log.debug('... '+ buffertools.toHex(data.slice(0, dumpLen > data.length ?
|
|
data.length : dumpLen)) +
|
|
(data.length > dumpLen ? '...' : ''));
|
|
}).bind(this));
|
|
this.socket.addListener('data', this.handleData.bind(this));
|
|
};
|
|
|
|
Connection.prototype.handleConnect = function () {
|
|
if (!this.inbound) {
|
|
this.sendVersion();
|
|
}
|
|
this.emit('connect', {
|
|
conn: this,
|
|
socket: this.socket,
|
|
peer: this.peer
|
|
});
|
|
};
|
|
|
|
Connection.prototype.handleError = function(err) {
|
|
if (err.errno == 110 || err.errno == 'ETIMEDOUT') {
|
|
log.info('connection timed out for '+this.peer);
|
|
} else if (err.errno == 111 || err.errno == 'ECONNREFUSED') {
|
|
log.info('connection refused for '+this.peer);
|
|
} else {
|
|
log.warn('connection with '+this.peer+' '+err.toString());
|
|
}
|
|
this.emit('error', {
|
|
conn: this,
|
|
socket: this.socket,
|
|
peer: this.peer,
|
|
err: err
|
|
});
|
|
};
|
|
|
|
Connection.prototype.handleDisconnect = function () {
|
|
this.emit('disconnect', {
|
|
conn: this,
|
|
socket: this.socket,
|
|
peer: this.peer
|
|
});
|
|
};
|
|
|
|
Connection.prototype.handleMessage = function(message) {
|
|
if (!message) {
|
|
// Parser was unable to make sense of the message, drop it
|
|
return;
|
|
}
|
|
|
|
try {
|
|
switch (message.command) {
|
|
case 'version':
|
|
// Did we connect to ourself?
|
|
if (buffertools.compare(nonce, message.nonce) === 0) {
|
|
this.socket.end();
|
|
return;
|
|
}
|
|
|
|
if (this.inbound) {
|
|
this.sendVersion();
|
|
}
|
|
|
|
if (message.version >= 209) {
|
|
this.sendMessage('verack', new Buffer([]));
|
|
}
|
|
this.sendVer = Math.min(message.version, PROTOCOL_VERSION);
|
|
if (message.version < 209) {
|
|
this.recvVer = Math.min(message.version, PROTOCOL_VERSION);
|
|
} else {
|
|
// We won't start expecting a checksum until after we've received
|
|
// the 'verack' message.
|
|
this.once('verack', (function () {
|
|
this.recvVer = message.version;
|
|
}).bind(this));
|
|
}
|
|
this.bestHeight = message.start_height;
|
|
break;
|
|
|
|
case 'verack':
|
|
this.recvVer = Math.min(message.version, PROTOCOL_VERSION);
|
|
this.active = true;
|
|
break;
|
|
|
|
case 'ping':
|
|
if ('object' === typeof message.nonce) {
|
|
this.sendPong(message.nonce);
|
|
}
|
|
break;
|
|
}
|
|
} catch (e) {
|
|
log.err('Error while handling "'+message.command+'" message from ' +
|
|
this.peer + ':\n' +
|
|
(e.stack ? e.stack : e.toString()));
|
|
return;
|
|
}
|
|
this.emit(message.command, {
|
|
conn: this,
|
|
socket: this.socket,
|
|
peer: this.peer,
|
|
message: message
|
|
});
|
|
};
|
|
|
|
Connection.prototype.sendPong = function (nonce) {
|
|
this.sendMessage('pong', nonce);
|
|
};
|
|
|
|
Connection.prototype.sendVersion = function () {
|
|
var subversion = '/BitcoinX:0.1/';
|
|
|
|
var put = new Put();
|
|
put.word32le(PROTOCOL_VERSION); // version
|
|
put.word64le(1); // services
|
|
put.word64le(Math.round(new Date().getTime()/1000)); // timestamp
|
|
put.pad(26); // addr_me
|
|
put.pad(26); // addr_you
|
|
put.put(nonce);
|
|
put.varint(subversion.length);
|
|
put.put(new Buffer(subversion, 'ascii'));
|
|
put.word32le(0);
|
|
|
|
this.sendMessage('version', put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendGetBlocks = function (starts, stop, wantHeaders) {
|
|
// Default value for stop is 0 to get as many blocks as possible (500)
|
|
stop = stop || util.NULL_HASH;
|
|
|
|
var put = new Put();
|
|
|
|
// https://en.bitcoin.it/wiki/Protocol_specification#getblocks
|
|
put.word32le(this.sendVer);
|
|
put.varint(starts.length);
|
|
|
|
for (var i = 0; i < starts.length; i++) {
|
|
if (starts[i].length != 32) {
|
|
throw new Error('Invalid hash length');
|
|
}
|
|
|
|
put.put(starts[i]);
|
|
}
|
|
|
|
var stopBuffer = new Buffer(stop, 'binary');
|
|
if (stopBuffer.length != 32) {
|
|
throw new Error('Invalid hash length');
|
|
}
|
|
|
|
put.put(stopBuffer);
|
|
|
|
var command = 'getblocks';
|
|
if (wantHeaders)
|
|
command = 'getheaders';
|
|
this.sendMessage(command, put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendGetHeaders = function(starts, stop) {
|
|
this.sendGetBlocks(starts, stop, true);
|
|
};
|
|
|
|
Connection.prototype.sendGetData = function (invs) {
|
|
var put = new Put();
|
|
put.varint(invs.length);
|
|
for (var i = 0; i < invs.length; i++) {
|
|
put.word32le(invs[i].type);
|
|
put.put(invs[i].hash);
|
|
}
|
|
this.sendMessage('getdata', put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendGetAddr = function (invs) {
|
|
var put = new Put();
|
|
this.sendMessage('getaddr', put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendInv = function(data) {
|
|
if(!Array.isArray(data)) data = [data];
|
|
var put = new Put();
|
|
put.varint(data.length);
|
|
data.forEach(function (value) {
|
|
if (value instanceof Block) {
|
|
// Block
|
|
put.word32le(2); // MSG_BLOCK
|
|
} else {
|
|
// Transaction
|
|
put.word32le(1); // MSG_TX
|
|
}
|
|
put.put(value.getHash());
|
|
});
|
|
this.sendMessage('inv', put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendHeaders = function (headers) {
|
|
var put = new Put();
|
|
put.varint(headers.length);
|
|
headers.forEach(function (header) {
|
|
put.put(header);
|
|
|
|
// Indicate 0 transactions
|
|
put.word8(0);
|
|
});
|
|
this.sendMessage('headers', put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendTx = function (tx) {
|
|
this.sendMessage('tx', tx.serialize());
|
|
};
|
|
|
|
Connection.prototype.sendBlock = function (block, txs) {
|
|
var put = new Put();
|
|
|
|
// Block header
|
|
put.put(block.getHeader());
|
|
|
|
// List of transactions
|
|
put.varint(txs.length);
|
|
txs.forEach(function (tx) {
|
|
put.put(tx.serialize());
|
|
});
|
|
|
|
this.sendMessage('block', put.buffer());
|
|
};
|
|
|
|
Connection.prototype.sendMessage = function (command, payload) {
|
|
try {
|
|
var magic = this.network.magic;
|
|
var commandBuf = new Buffer(command, 'ascii');
|
|
if (commandBuf.length > 12) throw 'Command name too long';
|
|
|
|
var checksum;
|
|
if (this.sendVer >= 209) {
|
|
checksum = doubleSha256(payload).slice(0, 4);
|
|
} else {
|
|
checksum = new Buffer([]);
|
|
}
|
|
|
|
var message = new Put(); // -- HEADER --
|
|
message.put(magic); // magic bytes
|
|
message.put(commandBuf); // command name
|
|
message.pad(12 - commandBuf.length); // zero-padded
|
|
message.word32le(payload.length); // payload length
|
|
message.put(checksum); // checksum
|
|
// -- BODY --
|
|
message.put(payload); // payload data
|
|
|
|
var buffer = message.buffer();
|
|
|
|
log.debug('['+this.peer+'] '+
|
|
'Sending message '+command+' ('+payload.length+' bytes)');
|
|
|
|
this.socket.write(buffer);
|
|
} catch (err) {
|
|
// TODO: We should catch this error one level higher in order to better
|
|
// determine how to react to it. For now though, ignoring it will do.
|
|
log.err('Error while sending message to peer '+this.peer+': '+
|
|
(err.stack ? err.stack : err.toString()));
|
|
}
|
|
};
|
|
|
|
Connection.prototype.handleData = function (data) {
|
|
this.buffers.push(data);
|
|
|
|
if (this.buffers.length > MAX_RECEIVE_BUFFER) {
|
|
log.err('Peer '+this.peer+' exceeded maxreceivebuffer, disconnecting.'+
|
|
(err.stack ? err.stack : err.toString()));
|
|
this.socket.destroy();
|
|
return;
|
|
}
|
|
|
|
this.processData();
|
|
};
|
|
|
|
Connection.prototype.processData = function () {
|
|
// If there are less than 20 bytes there can't be a message yet.
|
|
if (this.buffers.length < 20) return;
|
|
|
|
var magic = this.network.magic;
|
|
var i = 0;
|
|
for (;;) {
|
|
if (this.buffers.get(i ) === magic[0] &&
|
|
this.buffers.get(i+1) === magic[1] &&
|
|
this.buffers.get(i+2) === magic[2] &&
|
|
this.buffers.get(i+3) === magic[3]) {
|
|
if (i !== 0) {
|
|
log.debug('['+this.peer+'] '+
|
|
'Received '+i+
|
|
' bytes of inter-message garbage: ');
|
|
log.debug('... '+this.buffers.slice(0,i));
|
|
|
|
this.buffers.skip(i);
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (i > (this.buffers.length - 4)) {
|
|
this.buffers.skip(i);
|
|
return;
|
|
}
|
|
i++;
|
|
}
|
|
|
|
var payloadLen = (this.buffers.get(16) ) +
|
|
(this.buffers.get(17) << 8) +
|
|
(this.buffers.get(18) << 16) +
|
|
(this.buffers.get(19) << 24);
|
|
|
|
var startPos = (this.recvVer >= 209) ? 24 : 20;
|
|
var endPos = startPos + payloadLen;
|
|
|
|
if (this.buffers.length < endPos) return;
|
|
|
|
var command = this.buffers.slice(4, 16).toString('ascii').replace(/\0+$/,'');
|
|
var payload = this.buffers.slice(startPos, endPos);
|
|
var checksum = (this.recvVer >= 209) ? this.buffers.slice(20, 24) : null;
|
|
|
|
log.debug('['+this.peer+'] ' +
|
|
'Received message ' + command +
|
|
' (' + payloadLen + ' bytes)');
|
|
|
|
if (checksum !== null) {
|
|
var checksumConfirm = doubleSha256(payload).slice(0, 4);
|
|
if (buffertools.compare(checksumConfirm, checksum) !== 0) {
|
|
log.err('['+this.peer+'] '+
|
|
'Checksum failed',
|
|
{ cmd: command,
|
|
expected: checksumConfirm.toString('hex'),
|
|
actual: checksum.toString('hex') });
|
|
return;
|
|
}
|
|
}
|
|
|
|
var message;
|
|
try {
|
|
message = this.parseMessage(command, payload);
|
|
} catch (e) {
|
|
log.err('Error while parsing message '+command+' from ' +
|
|
this.peer + ':\n' +
|
|
(e.stack ? e.stack : e.toString()));
|
|
}
|
|
|
|
if (message) {
|
|
this.handleMessage(message);
|
|
}
|
|
|
|
this.buffers.skip(endPos);
|
|
this.processData();
|
|
};
|
|
|
|
Connection.prototype.parseMessage = function (command, payload) {
|
|
var parser = new Parser(payload);
|
|
|
|
var data = {
|
|
command: command
|
|
};
|
|
|
|
var i;
|
|
|
|
switch (command) {
|
|
case 'version': // https://en.bitcoin.it/wiki/Protocol_specification#version
|
|
data.version = parser.word32le();
|
|
data.services = parser.word64le();
|
|
data.timestamp = parser.word64le();
|
|
data.addr_me = parser.buffer(26);
|
|
data.addr_you = parser.buffer(26);
|
|
data.nonce = parser.buffer(8);
|
|
data.subversion = parser.varStr();
|
|
data.start_height = parser.word32le();
|
|
break;
|
|
|
|
case 'inv':
|
|
case 'getdata':
|
|
data.count = parser.varInt();
|
|
|
|
data.invs = [];
|
|
for (i = 0; i < data.count; i++) {
|
|
data.invs.push({
|
|
type: parser.word32le(),
|
|
hash: parser.buffer(32)
|
|
});
|
|
}
|
|
break;
|
|
|
|
case 'headers':
|
|
data.count = parser.varInt();
|
|
|
|
data.headers = [];
|
|
for (i = 0; i < data.count; i++) {
|
|
var header = new Block();
|
|
header.parse(parser);
|
|
data.headers.push(header);
|
|
}
|
|
break;
|
|
|
|
case 'block':
|
|
var block = new Block();
|
|
block.parse(parser);
|
|
|
|
data.block = block;
|
|
data.version = block.version;
|
|
data.prev_hash = block.prev_hash;
|
|
data.merkle_root = block.merkle_root;
|
|
data.timestamp = block.timestamp;
|
|
data.bits = block.bits;
|
|
data.nonce = block.nonce;
|
|
|
|
data.txs = block.txs;
|
|
|
|
data.size = payload.length;
|
|
break;
|
|
|
|
case 'tx':
|
|
var tx = new Transaction();
|
|
tx.parse(parser);
|
|
return {
|
|
command: command,
|
|
version: tx.version,
|
|
lock_time: tx.lock_time,
|
|
ins: tx.ins,
|
|
outs: tx.outs,
|
|
tx: tx,
|
|
};
|
|
|
|
case 'getblocks':
|
|
case 'getheaders':
|
|
// parse out the version
|
|
data.version = parser.word32le();
|
|
|
|
// TODO: Limit block locator size?
|
|
// reference implementation limits to 500 results
|
|
var startCount = parser.varInt();
|
|
|
|
data.starts = [];
|
|
for (i = 0; i < startCount; i++) {
|
|
data.starts.push(parser.buffer(32));
|
|
}
|
|
data.stop = parser.buffer(32);
|
|
break;
|
|
|
|
case 'addr':
|
|
var addrCount = parser.varInt();
|
|
|
|
// Enforce a maximum number of addresses per message
|
|
if (addrCount > 1000) {
|
|
addrCount = 1000;
|
|
}
|
|
|
|
data.addrs = [];
|
|
for (i = 0; i < addrCount; i++) {
|
|
// TODO: Time actually depends on the version of the other peer (>=31402)
|
|
data.addrs.push({
|
|
time: parser.word32le(),
|
|
services: parser.word64le(),
|
|
ip: parser.buffer(16),
|
|
port: parser.word16be()
|
|
});
|
|
}
|
|
break;
|
|
|
|
case 'alert':
|
|
data.payload = parser.varStr();
|
|
data.signature = parser.varStr();
|
|
break;
|
|
|
|
case 'ping':
|
|
if (this.recvVer > BIP0031_VERSION) {
|
|
data.nonce = parser.buffer(8);
|
|
}
|
|
break;
|
|
|
|
case 'getaddr':
|
|
case 'verack':
|
|
case 'reject':
|
|
// Empty message, nothing to parse
|
|
break;
|
|
|
|
default:
|
|
log.err('Connection.parseMessage(): Command not implemented',
|
|
{cmd: command});
|
|
|
|
// This tells the calling function not to issue an event
|
|
return null;
|
|
}
|
|
|
|
return data;
|
|
};
|
|
|
|
module.exports = require('soop')(Connection);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../config":"4itQ50","../networks":"ULNIu2","../patches/Buffers.monkey":"kytKTK","../util":140,"../util/BinaryParser":"b3ZSD7","../util/log":"AdF7pF","./Block":"pJEQEB","./SecureRandom":"p4SiC2","./Transaction":"LJhYtm","buffer":80,"bufferput":"aXRuS6","buffers":"OBo3aV","buffertools":"fugeBw","events":89,"socks5-client":118,"soop":125}],"./lib/Curve":[function(require,module,exports){
|
|
module.exports=require('Ynul1S');
|
|
},{}],"Ynul1S":[function(require,module,exports){
|
|
(function (Buffer){
|
|
"use strict";
|
|
var imports = require('soop');
|
|
var bignum = imports.bignum || require('bignum');
|
|
var Point = imports.Point || require('./Point');
|
|
|
|
var n = bignum.fromBuffer(new Buffer("FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141", 'hex'), {size: 32});
|
|
|
|
|
|
var Curve = function() {
|
|
};
|
|
|
|
/* secp256k1 curve */
|
|
var G;
|
|
Curve.getG = function() {
|
|
// don't use Point in top scope, causes exception in browser
|
|
// when Point is not loaded yet
|
|
|
|
// use cached version if available
|
|
G = G || new Point(bignum.fromBuffer(new Buffer("79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798", 'hex'), {size: 32}),
|
|
bignum.fromBuffer(new Buffer("483ADA7726A3C4655DA4FBFC0E1108A8FD17B448A68554199C47D08FFB10D4B8", 'hex'), {size: 32}));
|
|
return G;
|
|
};
|
|
|
|
Curve.getN = function() {
|
|
return n;
|
|
};
|
|
|
|
module.exports = require('soop')(Curve);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"./Point":"6tXgqr","bignum":58,"buffer":80,"soop":125}],"ez/meX":[function(require,module,exports){
|
|
|
|
exports.intFromCompact = function(c)
|
|
{
|
|
var bytes = ((c >>> 24) & 0xff) >>> 0;
|
|
var v = ((c & 0xffffff) << (8 * (bytes - 3))) >>> 0;
|
|
return v;
|
|
}
|
|
|
|
|
|
},{}],"./lib/Deserialize":[function(require,module,exports){
|
|
module.exports=require('ez/meX');
|
|
},{}],"hdzBvq":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var Key = require('./Key'),
|
|
Point = require('./Point'),
|
|
twoSha256 = require('../util').twoSha256,
|
|
buffertools = require('buffertools'),
|
|
bignum = require('bignum');
|
|
|
|
/**
|
|
* Pre-BIP32 Electrum public key derivation (electrum <2.0)
|
|
*
|
|
* For now, this class can only understands master public keys.
|
|
* It doesn't support derivation from a private master key (TODO).
|
|
*
|
|
* @example examples/ElectrumMPK.js
|
|
*/
|
|
function Electrum (master_public_key) {
|
|
this.mpk = new Buffer(master_public_key, 'hex');
|
|
}
|
|
|
|
Electrum.prototype.getSequence = function (for_change, n) {
|
|
var mode = for_change ? 1 : 0;
|
|
var buf = Buffer.concat([ new Buffer(n + ':' + mode + ':', 'utf8'), this.mpk ]);
|
|
return bignum.fromBuffer(twoSha256(buf));
|
|
};
|
|
|
|
Electrum.prototype.generatePubKey = function (n, for_change) {
|
|
var x = bignum.fromBuffer(this.mpk.slice(0, 32), { size: 32 });
|
|
var y = bignum.fromBuffer(this.mpk.slice(32, 64), { size: 32 });
|
|
var mpk_pt = new Point(x, y);
|
|
|
|
var sequence = this.getSequence(for_change, n);
|
|
var sequence_key = new Key();
|
|
sequence_key.private = sequence.toBuffer();
|
|
sequence_key.regenerateSync();
|
|
sequence_key.compressed = false;
|
|
|
|
var sequence_pt = Point.fromUncompressedPubKey(sequence_key.public);
|
|
|
|
pt = Point.add(mpk_pt, sequence_pt);
|
|
|
|
var xbuf = pt.x.toBuffer({ size: 32 });
|
|
var ybuf = pt.y.toBuffer({ size: 32 });
|
|
var prefix = new Buffer([0x04]);
|
|
|
|
var key = new Key();
|
|
key.compressed = false;
|
|
key.public = Buffer.concat([prefix, xbuf, ybuf]);
|
|
|
|
return key.public;
|
|
};
|
|
|
|
Electrum.prototype.generateChangePubKey = function (sequence) {
|
|
return this.generatePubKey(sequence, true);
|
|
};
|
|
|
|
module.exports = Electrum;
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util":140,"./Key":"ALJ4PS","./Point":"6tXgqr","bignum":58,"buffer":80,"buffertools":"fugeBw"}],"./lib/Electrum":[function(require,module,exports){
|
|
module.exports=require('hdzBvq');
|
|
},{}],"./lib/HierarchicalKey":[function(require,module,exports){
|
|
module.exports=require('x1O6JW');
|
|
},{}],"x1O6JW":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
var base58 = imports.base58 || require('./Base58').base58;
|
|
var coinUtil = imports.coinUtil || require('../util');
|
|
var Key = imports.Key || require('./Key');
|
|
var Point = imports.Point || require('./Point');
|
|
var SecureRandom = imports.SecureRandom || require('./SecureRandom');
|
|
var bignum = imports.bignum || require('bignum');
|
|
var networks = require('../networks');
|
|
|
|
var secp256k1_n = new bignum('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141', 16);
|
|
var secp256k1_Gx = new bignum('79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798', 16);
|
|
|
|
/*
|
|
random new HierarchicalKey: new HierarchicalKey();
|
|
from extended public or private key: new HierarchicalKey(str);
|
|
new blank HierarchicalKey: new HierarchicalKey(null);
|
|
*/
|
|
var HierarchicalKey = function(bytes) {
|
|
if (typeof bytes == 'undefined' || bytes == 'mainnet' || bytes == 'livenet') {
|
|
bytes = 'livenet';
|
|
this.version = networks['livenet'].hkeyPrivateVersion;
|
|
}
|
|
else if (bytes == 'testnet') {
|
|
this.version = networks['testnet'].hkeyPrivateVersion;
|
|
}
|
|
if (bytes == 'livenet' || bytes == 'testnet') {
|
|
this.depth = 0x00;
|
|
this.parentFingerprint = new Buffer([0, 0, 0, 0]);
|
|
this.childIndex = new Buffer([0, 0, 0, 0]);
|
|
this.chainCode = SecureRandom.getRandomBuffer(32);
|
|
this.eckey = Key.generateSync();
|
|
this.hasPrivateKey = true;
|
|
this.pubKeyHash = coinUtil.sha256ripe160(this.eckey.public);
|
|
this.buildExtendedPublicKey();
|
|
this.buildExtendedPrivateKey();
|
|
return;
|
|
}
|
|
|
|
// decode base58
|
|
if (typeof bytes === 'string') {
|
|
var decoded = base58.decode(bytes);
|
|
if (decoded.length != 82)
|
|
throw new Error('Not enough data, expected 82 and received '+decoded.length);
|
|
var checksum = decoded.slice(78, 82);
|
|
bytes = decoded.slice(0, 78);
|
|
|
|
var hash = coinUtil.sha256(coinUtil.sha256(bytes));
|
|
|
|
if (hash[0] != checksum[0] || hash[1] != checksum[1] || hash[2] != checksum[2] || hash[3] != checksum[3]) {
|
|
throw new Error('Invalid checksum');
|
|
}
|
|
}
|
|
|
|
if (bytes !== undefined && bytes !== null)
|
|
this.initFromBytes(bytes);
|
|
}
|
|
|
|
HierarchicalKey.seed = function(bytes, network) {
|
|
if (!network)
|
|
network = 'livenet';
|
|
|
|
if (!Buffer.isBuffer(bytes))
|
|
bytes = new Buffer(bytes, 'hex'); //if not buffer, assume hex
|
|
if (bytes.length < 128/8)
|
|
return false; //need more entropy
|
|
var hash = coinUtil.sha512hmac(bytes, new Buffer('Bitcoin seed'));
|
|
|
|
var hkey = new HierarchicalKey(null);
|
|
hkey.depth = 0x00;
|
|
hkey.parentFingerprint = new Buffer([0, 0, 0, 0]);
|
|
hkey.childIndex = new Buffer([0, 0, 0, 0]);
|
|
hkey.chainCode = hash.slice(32, 64);
|
|
hkey.version = networks[network].hkeyPrivateVersion;
|
|
hkey.eckey = new Key();
|
|
hkey.eckey.private = hash.slice(0, 32);
|
|
hkey.eckey.regenerateSync();
|
|
hkey.hasPrivateKey = true;
|
|
hkey.pubKeyHash = coinUtil.sha256ripe160(hkey.eckey.public);
|
|
|
|
hkey.buildExtendedPublicKey();
|
|
hkey.buildExtendedPrivateKey();
|
|
|
|
return hkey;
|
|
};
|
|
|
|
HierarchicalKey.prototype.initFromBytes = function(bytes) {
|
|
// Both pub and private extended keys are 78 bytes
|
|
if(bytes.length != 78) throw new Error('not enough data');
|
|
|
|
this.version = u32(bytes.slice(0, 4));
|
|
this.depth = u8(bytes.slice(4, 5));
|
|
this.parentFingerprint = bytes.slice(5, 9);
|
|
this.childIndex = u32(bytes.slice(9, 13));
|
|
this.chainCode = bytes.slice(13, 45);
|
|
|
|
var keyBytes = bytes.slice(45, 78);
|
|
|
|
var isPrivate =
|
|
(this.version == networks['livenet'].hkeyPrivateVersion ||
|
|
this.version == networks['testnet'].hkeyPrivateVersion );
|
|
|
|
var isPublic =
|
|
(this.version == networks['livenet'].hkeyPublicVersion ||
|
|
this.version == networks['testnet'].hkeyPublicVersion );
|
|
|
|
if (isPrivate && keyBytes[0] == 0) {
|
|
this.eckey = new Key();
|
|
this.eckey.private = keyBytes.slice(1, 33);
|
|
this.eckey.compressed = true;
|
|
this.eckey.regenerateSync();
|
|
this.pubKeyHash = coinUtil.sha256ripe160(this.eckey.public);
|
|
this.hasPrivateKey = true;
|
|
} else if (isPublic && (keyBytes[0] == 0x02 || keyBytes[0] == 0x03)) {
|
|
this.eckey = new Key();
|
|
this.eckey.public = keyBytes;
|
|
this.pubKeyHash = coinUtil.sha256ripe160(this.eckey.public);
|
|
this.hasPrivateKey = false;
|
|
} else {
|
|
throw new Error('Invalid key');
|
|
}
|
|
|
|
this.buildExtendedPublicKey();
|
|
this.buildExtendedPrivateKey();
|
|
}
|
|
|
|
HierarchicalKey.prototype.buildExtendedPublicKey = function() {
|
|
this.extendedPublicKey = new Buffer([]);
|
|
|
|
var v = null;
|
|
switch(this.version) {
|
|
case networks['livenet'].hkeyPublicVersion:
|
|
case networks['livenet'].hkeyPrivateVersion:
|
|
v = networks['livenet'].hkeyPublicVersion;
|
|
break;
|
|
case networks['testnet'].hkeyPublicVersion:
|
|
case networks['testnet'].hkeyPrivateVersion:
|
|
v = networks['testnet'].hkeyPublicVersion;
|
|
break;
|
|
default:
|
|
throw new Error('Unknown version');
|
|
}
|
|
|
|
// Version
|
|
this.extendedPublicKey = Buffer.concat([
|
|
new Buffer([v >> 24]),
|
|
new Buffer([(v >> 16) & 0xff]),
|
|
new Buffer([(v >> 8) & 0xff]),
|
|
new Buffer([v & 0xff]),
|
|
new Buffer([this.depth]),
|
|
this.parentFingerprint,
|
|
new Buffer([this.childIndex >>> 24]),
|
|
new Buffer([(this.childIndex >>> 16) & 0xff]),
|
|
new Buffer([(this.childIndex >>> 8) & 0xff]),
|
|
new Buffer([this.childIndex & 0xff]),
|
|
this.chainCode,
|
|
this.eckey.public
|
|
]);
|
|
}
|
|
|
|
HierarchicalKey.prototype.extendedPublicKeyString = function(format) {
|
|
if (format === undefined || format === 'base58') {
|
|
var hash = coinUtil.sha256(coinUtil.sha256(this.extendedPublicKey));
|
|
var checksum = hash.slice(0, 4);
|
|
var data = Buffer.concat([this.extendedPublicKey, checksum]);
|
|
return base58.encode(data);
|
|
} else if (format === 'hex') {
|
|
return this.extendedPublicKey.toString('hex');;
|
|
} else {
|
|
throw new Error('bad format');
|
|
}
|
|
}
|
|
|
|
HierarchicalKey.prototype.buildExtendedPrivateKey = function() {
|
|
if (!this.hasPrivateKey) return;
|
|
this.extendedPrivateKey = new Buffer([]);
|
|
|
|
var v = this.version;
|
|
|
|
this.extendedPrivateKey = Buffer.concat([
|
|
new Buffer([v >> 24]),
|
|
new Buffer([(v >> 16) & 0xff]),
|
|
new Buffer([(v >> 8) & 0xff]),
|
|
new Buffer([v & 0xff]),
|
|
new Buffer([this.depth]),
|
|
this.parentFingerprint,
|
|
new Buffer([this.childIndex >>> 24]),
|
|
new Buffer([(this.childIndex >>> 16) & 0xff]),
|
|
new Buffer([(this.childIndex >>> 8) & 0xff]),
|
|
new Buffer([this.childIndex & 0xff]),
|
|
this.chainCode,
|
|
new Buffer([0]),
|
|
this.eckey.private
|
|
]);
|
|
}
|
|
|
|
HierarchicalKey.prototype.extendedPrivateKeyString = function(format) {
|
|
if (format === undefined || format === 'base58') {
|
|
var hash = coinUtil.sha256(coinUtil.sha256(this.extendedPrivateKey));
|
|
var checksum = hash.slice(0, 4);
|
|
var data = Buffer.concat([this.extendedPrivateKey, checksum]);
|
|
return base58.encode(data);
|
|
} else if (format === 'hex') {
|
|
return this.extendedPrivateKey.toString('hex');
|
|
} else {
|
|
throw new Error('bad format');
|
|
}
|
|
}
|
|
|
|
|
|
HierarchicalKey.prototype.derive = function(path) {
|
|
var e = path.split('/');
|
|
|
|
// Special cases:
|
|
if (path == 'm' || path == 'M' || path == 'm\'' || path == 'M\'')
|
|
return this;
|
|
|
|
var hkey = this;
|
|
for (var i in e) {
|
|
var c = e[i];
|
|
|
|
if (i == 0 ) {
|
|
if (c != 'm') throw new Error('invalid path');
|
|
continue;
|
|
}
|
|
|
|
var usePrivate = (c.length > 1) && (c[c.length-1] == '\'');
|
|
var childIndex = parseInt(usePrivate ? c.slice(0, c.length - 1) : c) & 0x7fffffff;
|
|
|
|
if (usePrivate)
|
|
childIndex += 0x80000000;
|
|
|
|
hkey = hkey.deriveChild(childIndex);
|
|
}
|
|
|
|
return hkey;
|
|
}
|
|
|
|
HierarchicalKey.prototype.deriveChild = function(i) {
|
|
var ib = [];
|
|
ib.push((i >> 24) & 0xff);
|
|
ib.push((i >> 16) & 0xff);
|
|
ib.push((i >> 8) & 0xff);
|
|
ib.push(i & 0xff);
|
|
ib = new Buffer(ib);
|
|
|
|
var usePrivate = (i & 0x80000000) != 0;
|
|
|
|
var isPrivate =
|
|
(this.version == networks['livenet'].hkeyPrivateVersion ||
|
|
this.version == networks['testnet'].hkeyPrivateVersion );
|
|
|
|
if (usePrivate && (!this.hasPrivateKey || !isPrivate))
|
|
throw new Error('Cannot do private key derivation without private key');
|
|
|
|
var ret = null;
|
|
if (this.hasPrivateKey) {
|
|
var data = null;
|
|
|
|
if (usePrivate) {
|
|
data = Buffer.concat([new Buffer([0]), this.eckey.private, ib]);
|
|
} else {
|
|
data = Buffer.concat([this.eckey.public, ib]);
|
|
}
|
|
|
|
var hash = coinUtil.sha512hmac(data, this.chainCode);
|
|
var il = bignum.fromBuffer(hash.slice(0, 32), {size: 32});
|
|
var ir = hash.slice(32, 64);
|
|
|
|
// ki = IL + kpar (mod n).
|
|
var priv = bignum.fromBuffer(this.eckey.private, {size: 32});
|
|
var k = il.add(priv).mod(secp256k1_n);
|
|
|
|
ret = new HierarchicalKey(null);
|
|
ret.chainCode = ir;
|
|
|
|
ret.eckey = new Key();
|
|
ret.eckey.private = k.toBuffer({size: 32});
|
|
ret.eckey.regenerateSync();
|
|
ret.hasPrivateKey = true;
|
|
|
|
} else {
|
|
var data = Buffer.concat([this.eckey.public, ib]);
|
|
var hash = coinUtil.sha512hmac(data, this.chainCode);
|
|
var il = hash.slice(0, 32);
|
|
var ir = hash.slice(32, 64);
|
|
|
|
// Ki = (IL + kpar)*G = IL*G + Kpar
|
|
var ilGkey = new Key();
|
|
ilGkey.private = il;
|
|
ilGkey.regenerateSync();
|
|
ilGkey.compressed = false;
|
|
var ilG = Point.fromUncompressedPubKey(ilGkey.public);
|
|
var oldkey = new Key();
|
|
oldkey.public = this.eckey.public;
|
|
oldkey.compressed = false;
|
|
var Kpar = Point.fromUncompressedPubKey(oldkey.public);
|
|
var newpub = Point.add(ilG, Kpar).toUncompressedPubKey();
|
|
|
|
ret = new HierarchicalKey(null);
|
|
ret.chainCode = new Buffer(ir);
|
|
|
|
var eckey = new Key();
|
|
eckey.public = newpub;
|
|
eckey.compressed = true;
|
|
ret.eckey = eckey;
|
|
ret.hasPrivateKey = false;
|
|
}
|
|
|
|
ret.childIndex = i;
|
|
ret.parentFingerprint = this.pubKeyHash.slice(0,4);
|
|
ret.version = this.version;
|
|
ret.depth = this.depth + 1;
|
|
|
|
ret.eckey.compressed = true;
|
|
ret.pubKeyHash = coinUtil.sha256ripe160(ret.eckey.public);
|
|
|
|
ret.buildExtendedPublicKey();
|
|
ret.buildExtendedPrivateKey();
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
function uint(f, size) {
|
|
if (f.length < size)
|
|
throw new Error('not enough data');
|
|
var n = 0;
|
|
for (var i = 0; i < size; i++) {
|
|
n *= 256;
|
|
n += f[i];
|
|
}
|
|
return n;
|
|
}
|
|
|
|
function u8(f) {return uint(f,1);}
|
|
function u16(f) {return uint(f,2);}
|
|
function u32(f) {return uint(f,4);}
|
|
function u64(f) {return uint(f,8);}
|
|
|
|
module.exports = require('soop')(HierarchicalKey);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../networks":"ULNIu2","../util":140,"./Base58":"6VqyzY","./Key":"ALJ4PS","./Point":"6tXgqr","./SecureRandom":"p4SiC2","bignum":58,"buffer":80,"soop":125}],"CBDCgz":[function(require,module,exports){
|
|
(function (Buffer){
|
|
'use strict';
|
|
var imports = require('soop').imports();
|
|
var coinUtil = imports.coinUtil || require('../util');
|
|
var Key = imports.Key || require('./Key');
|
|
|
|
var Message = function() {
|
|
};
|
|
|
|
Message.sign = function(str, key) {
|
|
var hash = Message.magicHash(str);
|
|
var sig = key.signSync(hash);
|
|
return sig;
|
|
};
|
|
|
|
Message.verifyWithPubKey = function(pubkey, message, sig) {
|
|
var hash = Message.magicHash(message);
|
|
var key = new Key();
|
|
if (pubkey.length == 65)
|
|
key.compressed = false;
|
|
key.public = pubkey;
|
|
|
|
return key.verifySignatureSync(hash, sig);
|
|
};
|
|
|
|
//TODO: Message.verify ... with address, not pubkey
|
|
|
|
Message.magicBytes = new Buffer('Bitcoin Signed Message:\n');
|
|
|
|
Message.magicHash = function(str) {
|
|
var magicBytes = Message.magicBytes;
|
|
var prefix1 = coinUtil.varIntBuf(magicBytes.length);
|
|
var message = new Buffer(str);
|
|
var prefix2 = coinUtil.varIntBuf(message.length);
|
|
|
|
var buf = Buffer.concat([prefix1, magicBytes, prefix2, message]);
|
|
|
|
var hash = coinUtil.twoSha256(buf);
|
|
|
|
return hash;
|
|
};
|
|
|
|
module.exports = require('soop')(Message);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util":140,"./Key":"ALJ4PS","buffer":80,"soop":125}],"./lib/Message":[function(require,module,exports){
|
|
module.exports=require('CBDCgz');
|
|
},{}],"./lib/Opcode":[function(require,module,exports){
|
|
module.exports=require('Zm7/h9');
|
|
},{}],"Zm7/h9":[function(require,module,exports){
|
|
var imports = require('soop').imports();
|
|
|
|
function Opcode(num) {
|
|
this.code = num;
|
|
};
|
|
|
|
Opcode.prototype.toString = function () {
|
|
return Opcode.reverseMap[this.code];
|
|
};
|
|
|
|
Opcode.map = {
|
|
// push value
|
|
OP_FALSE : 0,
|
|
OP_0 : 0,
|
|
OP_PUSHDATA1 : 76,
|
|
OP_PUSHDATA2 : 77,
|
|
OP_PUSHDATA4 : 78,
|
|
OP_1NEGATE : 79,
|
|
OP_RESERVED : 80,
|
|
OP_TRUE : 81,
|
|
OP_1 : 81,
|
|
OP_2 : 82,
|
|
OP_3 : 83,
|
|
OP_4 : 84,
|
|
OP_5 : 85,
|
|
OP_6 : 86,
|
|
OP_7 : 87,
|
|
OP_8 : 88,
|
|
OP_9 : 89,
|
|
OP_10 : 90,
|
|
OP_11 : 91,
|
|
OP_12 : 92,
|
|
OP_13 : 93,
|
|
OP_14 : 94,
|
|
OP_15 : 95,
|
|
OP_16 : 96,
|
|
|
|
// control
|
|
OP_NOP : 97,
|
|
OP_VER : 98,
|
|
OP_IF : 99,
|
|
OP_NOTIF : 100,
|
|
OP_VERIF : 101,
|
|
OP_VERNOTIF : 102,
|
|
OP_ELSE : 103,
|
|
OP_ENDIF : 104,
|
|
OP_VERIFY : 105,
|
|
OP_RETURN : 106,
|
|
|
|
// stack ops
|
|
OP_TOALTSTACK : 107,
|
|
OP_FROMALTSTACK : 108,
|
|
OP_2DROP : 109,
|
|
OP_2DUP : 110,
|
|
OP_3DUP : 111,
|
|
OP_2OVER : 112,
|
|
OP_2ROT : 113,
|
|
OP_2SWAP : 114,
|
|
OP_IFDUP : 115,
|
|
OP_DEPTH : 116,
|
|
OP_DROP : 117,
|
|
OP_DUP : 118,
|
|
OP_NIP : 119,
|
|
OP_OVER : 120,
|
|
OP_PICK : 121,
|
|
OP_ROLL : 122,
|
|
OP_ROT : 123,
|
|
OP_SWAP : 124,
|
|
OP_TUCK : 125,
|
|
|
|
// splice ops
|
|
OP_CAT : 126,
|
|
OP_SUBSTR : 127,
|
|
OP_LEFT : 128,
|
|
OP_RIGHT : 129,
|
|
OP_SIZE : 130,
|
|
|
|
// bit logic
|
|
OP_INVERT : 131,
|
|
OP_AND : 132,
|
|
OP_OR : 133,
|
|
OP_XOR : 134,
|
|
OP_EQUAL : 135,
|
|
OP_EQUALVERIFY : 136,
|
|
OP_RESERVED1 : 137,
|
|
OP_RESERVED2 : 138,
|
|
|
|
// numeric
|
|
OP_1ADD : 139,
|
|
OP_1SUB : 140,
|
|
OP_2MUL : 141,
|
|
OP_2DIV : 142,
|
|
OP_NEGATE : 143,
|
|
OP_ABS : 144,
|
|
OP_NOT : 145,
|
|
OP_0NOTEQUAL : 146,
|
|
|
|
OP_ADD : 147,
|
|
OP_SUB : 148,
|
|
OP_MUL : 149,
|
|
OP_DIV : 150,
|
|
OP_MOD : 151,
|
|
OP_LSHIFT : 152,
|
|
OP_RSHIFT : 153,
|
|
|
|
OP_BOOLAND : 154,
|
|
OP_BOOLOR : 155,
|
|
OP_NUMEQUAL : 156,
|
|
OP_NUMEQUALVERIFY : 157,
|
|
OP_NUMNOTEQUAL : 158,
|
|
OP_LESSTHAN : 159,
|
|
OP_GREATERTHAN : 160,
|
|
OP_LESSTHANOREQUAL : 161,
|
|
OP_GREATERTHANOREQUAL : 162,
|
|
OP_MIN : 163,
|
|
OP_MAX : 164,
|
|
|
|
OP_WITHIN : 165,
|
|
|
|
// crypto
|
|
OP_RIPEMD160 : 166,
|
|
OP_SHA1 : 167,
|
|
OP_SHA256 : 168,
|
|
OP_HASH160 : 169,
|
|
OP_HASH256 : 170,
|
|
OP_CODESEPARATOR : 171,
|
|
OP_CHECKSIG : 172,
|
|
OP_CHECKSIGVERIFY : 173,
|
|
OP_CHECKMULTISIG : 174,
|
|
OP_CHECKMULTISIGVERIFY : 175,
|
|
|
|
// expansion
|
|
OP_NOP1 : 176,
|
|
OP_NOP2 : 177,
|
|
OP_NOP3 : 178,
|
|
OP_NOP4 : 179,
|
|
OP_NOP5 : 180,
|
|
OP_NOP6 : 181,
|
|
OP_NOP7 : 182,
|
|
OP_NOP8 : 183,
|
|
OP_NOP9 : 184,
|
|
OP_NOP10 : 185,
|
|
|
|
// template matching params
|
|
OP_PUBKEYHASH : 253,
|
|
OP_PUBKEY : 254,
|
|
OP_INVALIDOPCODE : 255
|
|
};
|
|
|
|
Opcode.reverseMap = [];
|
|
|
|
for (var k in Opcode.map) {
|
|
if(Opcode.map.hasOwnProperty(k)) {
|
|
Opcode.reverseMap[Opcode.map[k]] = k.substr(3);
|
|
}
|
|
}
|
|
|
|
Opcode.asList = function() {
|
|
var keys = [];
|
|
for (var prop in Opcode.map) {
|
|
if (Opcode.map.hasOwnProperty(prop)) {
|
|
keys.push(prop);
|
|
}
|
|
}
|
|
return keys;
|
|
};
|
|
|
|
module.exports = require('soop')(Opcode);
|
|
|
|
},{"soop":125}],"oolY81":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
|
|
var Net = imports.Net || require('net');
|
|
var Binary = imports.Binary || require('binary');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
|
|
function Peer(host, port, services) {
|
|
if ("string" === typeof host) {
|
|
if (host.indexOf(':') && !port) {
|
|
var parts = host.split(':');
|
|
host = parts[0];
|
|
port = parts[1];
|
|
}
|
|
this.host = host;
|
|
this.port = +port || 8333;
|
|
} else if (host instanceof Peer) {
|
|
this.host = host.host;
|
|
this.port = host.port;
|
|
} else if (Buffer.isBuffer(host)) {
|
|
if (buffertools.compare(Peer.IPV6_IPV4_PADDING, host.slice(0, 12)) != 0) {
|
|
throw new Error('IPV6 not supported yet! Cannot instantiate host.');
|
|
}
|
|
this.host = Array.prototype.slice.apply(host.slice(12)).join('.');
|
|
this.port = +port || 8333;
|
|
} else {
|
|
throw new Error('Could not instantiate peer, invalid parameter type: ' +
|
|
typeof host);
|
|
}
|
|
|
|
this.services = (services) ? services : null;
|
|
this.lastSeen = 0;
|
|
};
|
|
|
|
Peer.IPV6_IPV4_PADDING = new Buffer([0,0,0,0,0,0,0,0,0,0,255,255]);
|
|
|
|
Peer.prototype.createConnection = function () {
|
|
this.connection = Net.createConnection(this.port, this.host);
|
|
return this.connection;
|
|
};
|
|
|
|
Peer.prototype.getHostAsBuffer = function () {
|
|
return new Buffer(this.host.split('.'));
|
|
};
|
|
|
|
Peer.prototype.toString = function () {
|
|
return this.host + ":" + this.port;
|
|
};
|
|
|
|
Peer.prototype.toBuffer = function () {
|
|
var put = Binary.put();
|
|
put.word32le(this.lastSeen);
|
|
put.word64le(this.services);
|
|
put.put(this.getHostAsBuffer());
|
|
put.word16be(this.port);
|
|
return put.buffer();
|
|
};
|
|
|
|
module.exports = require('soop')(Peer);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"binary":69,"buffer":80,"buffertools":"fugeBw","net":76,"soop":125}],"./lib/Peer":[function(require,module,exports){
|
|
module.exports=require('oolY81');
|
|
},{}],"nsqKeP":[function(require,module,exports){
|
|
var imports = require('soop').imports();
|
|
var _ = imports._ || require('underscore');
|
|
var log = imports.log || require('../util/log');
|
|
var bitcoreDefaults = imports.config || require('../config');
|
|
var Connection = imports.Connection || require ('./Connection');
|
|
var Peer = imports.Peer || require('./Peer');
|
|
|
|
GetAdjustedTime = imports.GetAdjustedTime || function () {
|
|
// TODO: Implement actual adjustment
|
|
return Math.floor(new Date().getTime() / 1000);
|
|
};
|
|
|
|
function PeerManager(config) {
|
|
// extend defaults with config
|
|
this.config = _.extend(bitcoreDefaults, config || {});
|
|
this.active = false;
|
|
this.timer = null;
|
|
|
|
this.peers = [];
|
|
this.pool = [];
|
|
this.connections = [];
|
|
this.isConnected = false;
|
|
this.peerDiscovery = false;
|
|
|
|
// Move these to the Node's settings object
|
|
this.interval = 5000;
|
|
this.minConnections = 8;
|
|
this.minKnownPeers = 10;
|
|
|
|
// keep track of tried seeds and results
|
|
this.seeds = {
|
|
resolved: [],
|
|
failed: []
|
|
};
|
|
}
|
|
|
|
PeerManager.parent = imports.parent || require('events').EventEmitter;
|
|
PeerManager.Connection = Connection;
|
|
|
|
PeerManager.prototype.start = function() {
|
|
this.active = true;
|
|
if(!this.timer) {
|
|
this.timer = setInterval(this.checkStatus.bind(this), this.interval);
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.stop = function() {
|
|
this.active = false;
|
|
if(this.timer) {
|
|
clearInterval(this.timer);
|
|
this.timer = null;
|
|
}
|
|
for(var i=0; i<this.connections.length; i++) {
|
|
this.connections[i].socket.end();
|
|
};
|
|
};
|
|
|
|
PeerManager.prototype.addPeer = function(peer, port) {
|
|
if(peer instanceof Peer) {
|
|
this.peers.push(peer);
|
|
} else if ("string" == typeof peer) {
|
|
this.addPeer(new Peer(peer, port));
|
|
} else {
|
|
log.err('Node.addPeer(): Invalid value provided for peer',
|
|
{val: peer});
|
|
throw 'Node.addPeer(): Invalid value provided for peer.';
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.removePeer = function(peer) {
|
|
var index = this.peers.indexOf(peer);
|
|
var exists = !!~index;
|
|
if (exists) this.peers.splice(index, 1);
|
|
return exists;
|
|
};
|
|
|
|
PeerManager.prototype.checkStatus = function checkStatus() {
|
|
// Make sure we are connected to all forcePeers
|
|
if(this.peers.length) {
|
|
var peerIndex = {};
|
|
this.peers.forEach(function(peer) {
|
|
peerIndex[peer.toString()] = peer;
|
|
});
|
|
|
|
// Ignore the ones we're already connected to
|
|
this.connections.forEach(function(conn) {
|
|
var peerName = conn.peer.toString();
|
|
if("undefined" !== peerIndex[peerName]) {
|
|
delete peerIndex[peerName];
|
|
}
|
|
});
|
|
|
|
// for debug purposes, print how many of our peers are actually connected
|
|
var connected = 0
|
|
this.peers.forEach(function(p) {
|
|
if (p.connection && !p.connection._connecting) connected++
|
|
});
|
|
log.info(connected + ' of ' + this.peers.length + ' peers connected');
|
|
|
|
Object.keys(peerIndex).forEach(function(i) {
|
|
this.connectTo(peerIndex[i]);
|
|
}.bind(this));
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.connectTo = function(peer) {
|
|
log.info('connecting to ' + peer);
|
|
try {
|
|
return this.addConnection(peer.createConnection(), peer);
|
|
} catch (e) {
|
|
log.err('creating connection',e);
|
|
return null;
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.addConnection = function(socketConn, peer) {
|
|
var conn = new Connection(socketConn, peer, this.config);
|
|
this.connections.push(conn);
|
|
this.emit('connection', conn);
|
|
|
|
conn.addListener('version', this.handleVersion.bind(this));
|
|
conn.addListener('verack', this.handleReady.bind(this));
|
|
conn.addListener('addr', this.handleAddr.bind(this));
|
|
conn.addListener('getaddr', this.handleGetAddr.bind(this));
|
|
conn.addListener('error', this.handleError.bind(this));
|
|
conn.addListener('disconnect', this.handleDisconnect.bind(this));
|
|
|
|
return conn;
|
|
};
|
|
|
|
PeerManager.prototype.handleVersion = function(e) {
|
|
e.peer.version = e.message.version;
|
|
e.peer.start_height = e.message.start_height;
|
|
|
|
if (!e.conn.inbound) {
|
|
// TODO: Advertise our address (if listening)
|
|
}
|
|
// Get recent addresses
|
|
if(this.peerDiscovery &&
|
|
(e.message.version >= 31402 || this.peers.length < 1000)) {
|
|
e.conn.sendGetAddr();
|
|
e.conn.getaddr = true;
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.handleReady = function (e) {
|
|
log.info('connected to '+e.conn.peer.host+':'+e.conn.peer.port);
|
|
this.emit('connect', {
|
|
pm: this,
|
|
conn: e.conn,
|
|
socket: e.socket,
|
|
peer: e.peer
|
|
});
|
|
|
|
if(this.isConnected == false) {
|
|
this.emit('netConnected', e);
|
|
this.isConnected = true;
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.handleAddr = function (e) {
|
|
if(!this.peerDiscovery) return;
|
|
|
|
var now = GetAdjustedTime();
|
|
e.message.addrs.forEach(function (addr) {
|
|
try {
|
|
// In case of an invalid time, assume "5 days ago"
|
|
if (addr.time <= 100000000 || addr.time > (now + 10 * 60)) {
|
|
addr.time = now - 5 * 24 * 60 * 60;
|
|
}
|
|
var peer = new Peer(addr.ip, addr.port, addr.services);
|
|
peer.lastSeen = addr.time;
|
|
|
|
// TODO: Handle duplicate peers
|
|
this.peers.push(peer);
|
|
|
|
// TODO: Handle addr relay
|
|
} catch(e) {
|
|
log.warn("Invalid addr received: "+e.message);
|
|
}
|
|
}.bind(this));
|
|
if (e.message.addrs.length < 1000 ) {
|
|
e.conn.getaddr = false;
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.handleGetAddr = function(e) {
|
|
// TODO: Reply with addr message.
|
|
};
|
|
|
|
PeerManager.prototype.handleError = function(e) {
|
|
log.err('unkown error with peer '+e.peer+' (disconnecting): '+e.err);
|
|
this.handleDisconnect.apply(this, [].slice.call(arguments));
|
|
};
|
|
|
|
PeerManager.prototype.handleDisconnect = function(e) {
|
|
log.info('disconnected from peer ' + e.peer);
|
|
var i = this.connections.indexOf(e.conn);
|
|
if(i != -1) this.connections.splice(i, 1);
|
|
|
|
this.removePeer(e.peer);
|
|
if (this.pool.length) {
|
|
log.info('replacing peer using the pool of ' + this.pool.length + ' seeds');
|
|
this.addPeer(this.pool.pop());
|
|
}
|
|
|
|
if(!this.connections.length) {
|
|
this.emit('netDisconnected');
|
|
this.isConnected = false;
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.getActiveConnection = function () {
|
|
var activeConnections = this.connections.filter(function (conn) {
|
|
return conn.active;
|
|
});
|
|
|
|
if (activeConnections.length) {
|
|
var randomIndex = Math.floor(Math.random()*activeConnections.length);
|
|
var candidate = activeConnections[randomIndex];
|
|
if (candidate.socket.writable) {
|
|
return candidate;
|
|
} else {
|
|
// Socket is not writable, remove it from active connections
|
|
activeConnections.splice(randomIndex, 1);
|
|
|
|
// Then try again
|
|
// TODO: This causes an infinite recursion when all connections are dead,
|
|
// although it shouldn't.
|
|
return this.getActiveConnection();
|
|
}
|
|
} else {
|
|
return null;
|
|
}
|
|
};
|
|
|
|
PeerManager.prototype.getActiveConnections = function () {
|
|
return this.connections.slice(0);
|
|
};
|
|
|
|
PeerManager.prototype.discover = function(options, callback) {
|
|
var self = this;
|
|
var async = imports.async || require('async');
|
|
var dns = imports.dns || require('dns');
|
|
var networks = imports.networks || require('../networks');
|
|
var seeds = networks[self.config.network].dnsSeeds;
|
|
|
|
self.limit = options.limit || 12;
|
|
|
|
var dnsExecutor = seeds.map(function(seed) {
|
|
return function(done) {
|
|
// have we already resolved this seed?
|
|
if (~self.seeds.resolved.indexOf(seed)) {
|
|
// if so, just pass back cached peer list
|
|
return done(null, self.seeds.results[seed]);
|
|
}
|
|
|
|
// has this seed failed to resolve?
|
|
if (~self.seeds.failed.indexOf(seed)) {
|
|
// if so, pass back empty results
|
|
return done(null, []);
|
|
}
|
|
|
|
log.info('resolving dns seed '+ seed);
|
|
|
|
dns.resolve(seed, function(err, peers) {
|
|
if (err) {
|
|
log.err('failed to resolve dns seed '+ seed, err);
|
|
self.seeds.failed.push(seed);
|
|
return done(null, []);
|
|
}
|
|
|
|
log.info('found '+ peers.length + ' peers from ' + seed);
|
|
self.seeds.resolved.push(seed);
|
|
|
|
// transform that list into a list of Peer instances
|
|
peers = peers.map(function(ip) {
|
|
return new Peer(ip, networks.defaultClientPort);
|
|
});
|
|
|
|
peers.forEach(function(p) {
|
|
if (self.peers.length < self.limit) self.addPeer(p);
|
|
else self.pool.push(p);
|
|
});
|
|
|
|
self.emit('peers', peers);
|
|
|
|
return done(null, peers);
|
|
});
|
|
|
|
};
|
|
});
|
|
|
|
// try resolving all seeds
|
|
async.parallel(dnsExecutor, function(err, results) {
|
|
var peers = [];
|
|
|
|
// consolidate all resolved peers into one list
|
|
results.forEach(function(peerlist) {
|
|
peers = peers.concat(peerlist);
|
|
});
|
|
|
|
if (typeof callback === 'function') callback(null, peers);
|
|
});
|
|
|
|
return self;
|
|
};
|
|
|
|
module.exports = require('soop')(PeerManager);
|
|
|
|
},{"../config":"4itQ50","../networks":"ULNIu2","../util/log":"AdF7pF","./Connection":"DB/p3X","./Peer":"oolY81","async":68,"dns":76,"events":89,"soop":125,"underscore":127}],"./lib/PeerManager":[function(require,module,exports){
|
|
module.exports=require('nsqKeP');
|
|
},{}],"izTl9z":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
|
|
var parent = imports.parent || require('../util/VersionedData');
|
|
var networks= imports.networks || require('../networks');
|
|
|
|
//compressed is true if public key is compressed; false otherwise
|
|
function PrivateKey(version, buf, compressed) {
|
|
PrivateKey.super(this, arguments);
|
|
if (compressed !== undefined)
|
|
this.compressed(compressed);
|
|
};
|
|
|
|
PrivateKey.parent = parent;
|
|
parent.applyEncodingsTo(PrivateKey);
|
|
|
|
PrivateKey.prototype.validate = function() {
|
|
this.doAsBinary(function() {
|
|
PrivateKey.super(this, 'validate', arguments);
|
|
if (this.data.length < 32 || (this.data.length > 1+32 && !this.compressed()) || (this.data.length==1+32+1 && this.data[1+32+1-1]!=1) || this.data.length>1+32+1)
|
|
throw new Error('invalid data length');
|
|
});
|
|
if (typeof this.network() === 'undefined') throw new Error('invalid network');
|
|
};
|
|
|
|
// get or set the payload data (as a Buffer object)
|
|
// overloaded from VersionedData
|
|
PrivateKey.prototype.payload = function(data) {
|
|
if(data) {
|
|
this.doAsBinary(function() {data.copy(this.data,1);});
|
|
return data;
|
|
}
|
|
var buf=this.as('binary');
|
|
if (buf.length==1+32+1)
|
|
return buf.slice(1,1+32);
|
|
else if (buf.length==1+32)
|
|
return buf.slice(1);
|
|
};
|
|
|
|
// get or set whether the corresponding public key is compressed
|
|
PrivateKey.prototype.compressed = function(compressed) {
|
|
if (compressed !== undefined) {
|
|
this.doAsBinary(function(){
|
|
var len=1+32+1;
|
|
if (compressed) {
|
|
var data=new Buffer(len);
|
|
this.data.copy(data);
|
|
this.data=data;
|
|
this.data[len-1]=1;
|
|
} else {
|
|
this.data=this.data.slice(0,len-1);
|
|
}
|
|
});
|
|
}
|
|
else {
|
|
var len=1+32+1;
|
|
var data=this.as('binary');
|
|
if (data.length==len && data[len-1]==1)
|
|
return true;
|
|
else if (data.length==len-1)
|
|
return false;
|
|
else
|
|
throw new Error('invalid private key');
|
|
}
|
|
};
|
|
|
|
PrivateKey.prototype.network = function() {
|
|
var version = this.version();
|
|
|
|
var livenet = networks.livenet;
|
|
var testnet = networks.testnet;
|
|
|
|
var answer;
|
|
if (version === livenet.privKeyVersion)
|
|
answer = livenet;
|
|
else if (version === testnet.privKeyVersion)
|
|
answer = testnet;
|
|
|
|
return answer;
|
|
};
|
|
|
|
module.exports = require('soop')(PrivateKey);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../networks":"ULNIu2","../util/VersionedData":"QLzNQg","buffer":80,"soop":125}],"./lib/PrivateKey":[function(require,module,exports){
|
|
module.exports=require('izTl9z');
|
|
},{}],"./lib/RpcClient":[function(require,module,exports){
|
|
module.exports=require('7siE1N');
|
|
},{}],"7siE1N":[function(require,module,exports){
|
|
(function (Buffer){
|
|
// RpcClient.js
|
|
// MIT/X11-like license. See LICENSE.txt.
|
|
// Copyright 2013 BitPay, Inc.
|
|
//
|
|
var imports = require('soop').imports();
|
|
var http = imports.http || require('http');
|
|
var https = imports.https || require('https');
|
|
var log = imports.log || require('../util/log');
|
|
|
|
function RpcClient(opts) {
|
|
opts = opts || {};
|
|
this.host = opts.host || '127.0.0.1';
|
|
this.port = opts.port || 8332;
|
|
this.user = opts.user || 'user';
|
|
this.pass = opts.pass || 'pass';
|
|
this.protocol = (opts.protocol == 'http') ? http : https;
|
|
this.batchedCalls = null;
|
|
this.disableAgent = opts.disableAgent || false;
|
|
}
|
|
|
|
RpcClient.prototype.batch = function(batchCallback, resultCallback) {
|
|
this.batchedCalls = [];
|
|
batchCallback();
|
|
rpc.call(this, this.batchedCalls, resultCallback);
|
|
this.batchedCalls = null;
|
|
}
|
|
|
|
var callspec = {
|
|
addMultiSigAddress: '',
|
|
addNode: '',
|
|
backupWallet: '',
|
|
createMultiSig: '',
|
|
createRawTransaction: '',
|
|
decodeRawTransaction: '',
|
|
dumpPrivKey: '',
|
|
encryptWallet: '',
|
|
getAccount: '',
|
|
getAccountAddress: 'str',
|
|
getAddedNodeInfo: '',
|
|
getAddressesByAccount: '',
|
|
getBalance: 'str int',
|
|
getBestBlockHash: '',
|
|
getBlock: '',
|
|
getBlockCount: '',
|
|
getBlockHash: 'int',
|
|
getBlockNumber: '',
|
|
getBlockTemplate: '',
|
|
getConnectionCount: '',
|
|
getDifficulty: '',
|
|
getGenerate: '',
|
|
getHashesPerSec: '',
|
|
getInfo: '',
|
|
getMemoryPool: '',
|
|
getMiningInfo: '',
|
|
getNewAddress: '',
|
|
getPeerInfo: '',
|
|
getRawMemPool: '',
|
|
getRawTransaction: 'str int',
|
|
getReceivedByAccount: 'str int',
|
|
getReceivedByAddress: 'str int',
|
|
getTransaction: '',
|
|
getTxOut: 'str int bool',
|
|
getTxOutSetInfo: '',
|
|
getWork: '',
|
|
help: '',
|
|
importAddress: 'str str bool',
|
|
importPrivKey: 'str str bool',
|
|
keyPoolRefill: '',
|
|
listAccounts: 'int',
|
|
listAddressGroupings: '',
|
|
listReceivedByAccount: 'int bool',
|
|
listReceivedByAddress: 'int bool',
|
|
listSinceBlock: 'str int',
|
|
listTransactions: 'str int int',
|
|
listUnspent: 'int int',
|
|
listLockUnspent: 'bool',
|
|
lockUnspent: '',
|
|
move: 'str str float int str',
|
|
sendFrom: 'str str float int str str',
|
|
sendMany: 'str str int str', //not sure this is will work
|
|
sendRawTransaction: '',
|
|
sendToAddress: 'str float str str',
|
|
setAccount: '',
|
|
setGenerate: 'bool int',
|
|
setTxFee: 'float',
|
|
signMessage: '',
|
|
signRawTransaction: '',
|
|
stop: '',
|
|
submitBlock: '',
|
|
validateAddress: '',
|
|
verifyMessage: '',
|
|
walletLock: '',
|
|
walletPassPhrase: 'string int',
|
|
walletPassphraseChange: '',
|
|
};
|
|
|
|
var slice = function(arr, start, end) {
|
|
return Array.prototype.slice.call(arr, start, end);
|
|
};
|
|
|
|
function generateRPCMethods(constructor, apiCalls, rpc) {
|
|
function createRPCMethod(methodName, argMap) {
|
|
return function() {
|
|
var limit = arguments.length - 1;
|
|
if(this.batchedCalls) var limit = arguments.length;
|
|
for (var i=0; i<limit; i++) {
|
|
if(argMap[i]) arguments[i] = argMap[i](arguments[i]);
|
|
};
|
|
if(this.batchedCalls) {
|
|
this.batchedCalls.push({jsonrpc: '2.0', method: methodName, params: slice(arguments)});
|
|
} else {
|
|
rpc.call(this, {method: methodName, params: slice(arguments, 0, arguments.length - 1)}, arguments[arguments.length - 1]);
|
|
}
|
|
};
|
|
};
|
|
|
|
var types = {
|
|
str: function(arg) {return arg.toString();},
|
|
int: function(arg) {return parseFloat(arg);},
|
|
float: function(arg) {return parseFloat(arg);},
|
|
bool: function(arg) {return (arg === true || arg == '1' || arg == 'true' || arg.toString().toLowerCase() == 'true');},
|
|
};
|
|
|
|
for(var k in apiCalls) {
|
|
if (apiCalls.hasOwnProperty(k)) {
|
|
var spec = apiCalls[k].split(' ');
|
|
for (var i = 0; i < spec.length; i++) {
|
|
if(types[spec[i]]) {
|
|
spec[i] = types[spec[i]];
|
|
} else {
|
|
spec[i] = types.string;
|
|
}
|
|
}
|
|
var methodName = k.toLowerCase();
|
|
constructor.prototype[k] = createRPCMethod(methodName, spec);
|
|
constructor.prototype[methodName] = constructor.prototype[k];
|
|
}
|
|
}
|
|
}
|
|
|
|
function rpc(request, callback) {
|
|
var self = this;
|
|
var request;
|
|
request = JSON.stringify(request);
|
|
var auth = Buffer(self.user + ':' + self.pass).toString('base64');
|
|
|
|
var options = {
|
|
host: self.host,
|
|
path: '/',
|
|
method: 'POST',
|
|
port: self.port,
|
|
agent: self.disableAgent ? false : undefined,
|
|
};
|
|
if(self.httpOptions) {
|
|
for(var k in self.httpOptions) {
|
|
options[k] = self.httpOptions[k];
|
|
}
|
|
}
|
|
var err = null;
|
|
var req = this.protocol.request(options, function(res) {
|
|
|
|
var buf = '';
|
|
res.on('data', function(data) {
|
|
buf += data;
|
|
});
|
|
res.on('end', function() {
|
|
if(res.statusCode == 401) {
|
|
callback(new Error('bitcoin JSON-RPC connection rejected: 401 unauthorized'));
|
|
return;
|
|
}
|
|
if(res.statusCode == 403) {
|
|
callback(new Error('bitcoin JSON-RPC connection rejected: 403 forbidden'));
|
|
return;
|
|
}
|
|
|
|
if(err) {
|
|
callback(err);
|
|
return;
|
|
}
|
|
try {
|
|
var parsedBuf = JSON.parse(buf);
|
|
} catch(e) {
|
|
log.err(e.stack);
|
|
log.err(buf);
|
|
log.err('HTTP Status code:' + res.statusCode);
|
|
callback(e);
|
|
return;
|
|
}
|
|
callback(parsedBuf.error, parsedBuf);
|
|
});
|
|
});
|
|
req.on('error', function(e) {
|
|
var err = new Error('Could not connect to bitcoin via RPC: '+e.message);
|
|
log.err(err);
|
|
callback(err);
|
|
});
|
|
|
|
req.setHeader('Content-Length', request.length);
|
|
req.setHeader('Content-Type', 'application/json');
|
|
req.setHeader('Authorization', 'Basic ' + auth);
|
|
req.write(request);
|
|
req.end();
|
|
};
|
|
|
|
generateRPCMethods(RpcClient, callspec, rpc);
|
|
|
|
module.exports = require('soop')(RpcClient);
|
|
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util/log":"AdF7pF","buffer":80,"http":90,"https":94,"soop":125}],"tBM27q":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
var parent = imports.parent || require('../util/VersionedData');
|
|
|
|
function SIN(type, payload) {
|
|
if (typeof type != 'number') {
|
|
SIN.super(this, arguments);
|
|
return;
|
|
};
|
|
this.data = new Buffer(1 + 1 + payload.length);
|
|
this.__proto__ = this.encodings['binary'];
|
|
this.prefix(0x0F); // SIN magic number, in numberspace
|
|
this.type(type);
|
|
this.payload(payload);
|
|
};
|
|
SIN.parent = parent;
|
|
parent.applyEncodingsTo(SIN);
|
|
|
|
SIN.SIN_PERSIST_MAINNET = 0x01; // associated with sacrifice TX
|
|
SIN.SIN_PERSIST_TESTNET = 0x11; // associated with sacrifice TX
|
|
SIN.SIN_EPHEM = 0x02; // generate off-net at any time
|
|
|
|
// get or set the prefix data (the first byte of the address)
|
|
SIN.prototype.prefix = function(num) {
|
|
if(num || (num === 0)) {
|
|
this.doAsBinary(function() {this.data.writeUInt8(num, 0);});
|
|
return num;
|
|
}
|
|
return this.as('binary').readUInt8(0);
|
|
};
|
|
|
|
// get or set the SIN-type data (the second byte of the address)
|
|
SIN.prototype.type = function(num) {
|
|
if(num || (num === 0)) {
|
|
this.doAsBinary(function() {this.data.writeUInt8(num, 1);});
|
|
return num;
|
|
}
|
|
return this.as('binary').readUInt8(1);
|
|
};
|
|
|
|
// get or set the payload data (as a Buffer object)
|
|
SIN.prototype.payload = function(data) {
|
|
if(data) {
|
|
this.doAsBinary(function() {data.copy(this.data, 2);});
|
|
return data;
|
|
}
|
|
return this.as('binary').slice(1);
|
|
};
|
|
|
|
SIN.prototype.validate = function() {
|
|
this.doAsBinary(function() {
|
|
SIN.super(this, 'validate', arguments);
|
|
if (this.data.length != 22) throw new Error('invalid data length');
|
|
});
|
|
};
|
|
module.exports = require('soop')(SIN);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util/VersionedData":"QLzNQg","buffer":80,"soop":125}],"./lib/SIN":[function(require,module,exports){
|
|
module.exports=require('tBM27q');
|
|
},{}],"EyghZQ":[function(require,module,exports){
|
|
var coinUtil = require('../util');
|
|
var timeUtil = require('../util/time');
|
|
var Key = require('./Key');
|
|
var SIN = require('./SIN');
|
|
|
|
function SINKey(cfg) {
|
|
if (typeof cfg != 'object')
|
|
cfg = {};
|
|
|
|
this.created = cfg.created;
|
|
this.privKey = cfg.privKey;
|
|
};
|
|
|
|
SINKey.prototype.generate = function() {
|
|
this.privKey = Key.generateSync();
|
|
this.created = timeUtil.curtime();
|
|
};
|
|
|
|
SINKey.prototype.pubkeyHash = function() {
|
|
return coinUtil.sha256ripe160(this.privKey.public);
|
|
};
|
|
|
|
SINKey.prototype.storeObj = function() {
|
|
var pubKey = this.privKey.public.toString('hex');
|
|
var pubKeyHash = this.pubkeyHash();
|
|
var sin = new SIN(SIN.SIN_EPHEM, pubKeyHash);
|
|
var obj = {
|
|
created: this.created,
|
|
priv: this.privKey.private.toString('hex'),
|
|
pub: pubKey,
|
|
sin: sin.toString(),
|
|
};
|
|
|
|
return obj;
|
|
};
|
|
|
|
module.exports = require('soop')(SINKey);
|
|
|
|
},{"../util":140,"../util/time":143,"./Key":"ALJ4PS","./SIN":"tBM27q","soop":125}],"./lib/SINKey":[function(require,module,exports){
|
|
module.exports=require('EyghZQ');
|
|
},{}],"hQ0t76":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
var config = imports.config || require('../config');
|
|
var log = imports.log || require('../util/log');
|
|
var Opcode = imports.Opcode || require('./Opcode');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
|
|
var util = imports.util || require('../util/util');
|
|
var Parser = imports.Parser || require('../util/BinaryParser');
|
|
var Put = imports.Put || require('bufferput');
|
|
|
|
var TX_UNKNOWN = 0;
|
|
var TX_PUBKEY = 1;
|
|
var TX_PUBKEYHASH = 2;
|
|
var TX_MULTISIG = 3;
|
|
var TX_SCRIPTHASH = 4;
|
|
|
|
var TX_TYPES = [
|
|
'unknown',
|
|
'pubkey',
|
|
'pubkeyhash',
|
|
'multisig',
|
|
'scripthash'
|
|
];
|
|
|
|
function Script(buffer) {
|
|
if (buffer) {
|
|
this.buffer = buffer;
|
|
} else {
|
|
this.buffer = util.EMPTY_BUFFER;
|
|
}
|
|
this.chunks = [];
|
|
this.parse();
|
|
}
|
|
|
|
Script.TX_UNKNOWN = TX_UNKNOWN;
|
|
Script.TX_PUBKEY = TX_PUBKEY;
|
|
Script.TX_PUBKEYHASH = TX_PUBKEYHASH;
|
|
Script.TX_MULTISIG = TX_MULTISIG;
|
|
Script.TX_SCRIPTHASH = TX_SCRIPTHASH;
|
|
|
|
Script.prototype.parse = function() {
|
|
this.chunks = [];
|
|
|
|
var parser = new Parser(this.buffer);
|
|
while (!parser.eof()) {
|
|
var opcode = parser.word8();
|
|
|
|
var len, chunk;
|
|
if (opcode > 0 && opcode < Opcode.map.OP_PUSHDATA1) {
|
|
// Read some bytes of data, opcode value is the length of data
|
|
this.chunks.push(parser.buffer(opcode));
|
|
} else if (opcode === Opcode.map.OP_PUSHDATA1) {
|
|
len = parser.word8();
|
|
chunk = parser.buffer(len);
|
|
this.chunks.push(chunk);
|
|
} else if (opcode === Opcode.map.OP_PUSHDATA2) {
|
|
len = parser.word16le();
|
|
chunk = parser.buffer(len);
|
|
this.chunks.push(chunk);
|
|
} else if (opcode === Opcode.map.OP_PUSHDATA4) {
|
|
len = parser.word32le();
|
|
chunk = parser.buffer(len);
|
|
this.chunks.push(chunk);
|
|
} else {
|
|
this.chunks.push(opcode);
|
|
}
|
|
}
|
|
};
|
|
|
|
Script.prototype.isPushOnly = function() {
|
|
for (var i = 0; i < this.chunks.length; i++) {
|
|
var op = this.chunks[i];
|
|
if (!Buffer.isBuffer(op) && op > Opcode.map.OP_16) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
Script.prototype.isP2SH = function() {
|
|
return (this.chunks.length == 3 &&
|
|
this.chunks[0] == Opcode.map.OP_HASH160 &&
|
|
Buffer.isBuffer(this.chunks[1]) &&
|
|
this.chunks[1].length == 20 &&
|
|
this.chunks[2] == Opcode.map.OP_EQUAL);
|
|
};
|
|
|
|
Script.prototype.isPubkey = function() {
|
|
return (this.chunks.length == 2 &&
|
|
Buffer.isBuffer(this.chunks[0]) &&
|
|
this.chunks[1] == Opcode.map.OP_CHECKSIG);
|
|
};
|
|
|
|
Script.prototype.isPubkeyHash = function() {
|
|
return (this.chunks.length == 5 &&
|
|
this.chunks[0] == Opcode.map.OP_DUP &&
|
|
this.chunks[1] == Opcode.map.OP_HASH160 &&
|
|
Buffer.isBuffer(this.chunks[2]) &&
|
|
this.chunks[2].length == 20 &&
|
|
this.chunks[3] == Opcode.map.OP_EQUALVERIFY &&
|
|
this.chunks[4] == Opcode.map.OP_CHECKSIG);
|
|
};
|
|
|
|
function isSmallIntOp(opcode) {
|
|
return ((opcode == Opcode.map.OP_0) ||
|
|
((opcode >= Opcode.map.OP_1) && (opcode <= Opcode.map.OP_16)));
|
|
};
|
|
|
|
Script.prototype.isMultiSig = function() {
|
|
return (this.chunks.length > 3 &&
|
|
isSmallIntOp(this.chunks[0]) &&
|
|
isSmallIntOp(this.chunks[this.chunks.length - 2]) &&
|
|
this.chunks[this.chunks.length - 1] == Opcode.map.OP_CHECKMULTISIG);
|
|
};
|
|
|
|
Script.prototype.isP2shScriptSig = function() {
|
|
if( !isSmallIntOp(this.chunks[0]) || this.chunks[0] !==0 )
|
|
return false;
|
|
|
|
var redeemScript = new Script(this.chunks[this.chunks.length-1]);
|
|
var type=redeemScript.classify();
|
|
return type !== TX_UNKNOWN;
|
|
};
|
|
|
|
Script.prototype.isMultiSigScriptSig = function() {
|
|
if( !isSmallIntOp(this.chunks[0]) || this.chunks[0] !==0 )
|
|
return false;
|
|
return !this.isP2shScriptSig();
|
|
};
|
|
|
|
Script.prototype.countSignatures = function() {
|
|
var ret = 0;
|
|
var l =this.chunks.length;
|
|
|
|
// Multisig?
|
|
if (this.isMultiSigScriptSig()){
|
|
ret = l - 1;
|
|
}
|
|
else if (this.isP2shScriptSig()) {
|
|
ret = l - 2;
|
|
}
|
|
// p2pubkey or p2pubkeyhash
|
|
else {
|
|
ret = buffertools.compare(this.getBuffer(), util.EMPTY_BUFFER)===0?0:1;
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
Script.prototype.countMissingSignatures = function() {
|
|
if (this.isMultiSig()) {
|
|
log.debug("Can not count missing signatures on normal Multisig script");
|
|
return null;
|
|
}
|
|
|
|
var ret = 0;
|
|
var l =this.chunks.length;
|
|
// P2SH?
|
|
if (isSmallIntOp(this.chunks[0]) && this.chunks[0] ===0) {
|
|
var redeemScript = new Script(this.chunks[l-1]);
|
|
if (!isSmallIntOp(redeemScript.chunks[0])) {
|
|
log.debug("Unrecognized script type");
|
|
}
|
|
else {
|
|
var nreq = redeemScript.chunks[0] - 80; //see OP_2-OP_16
|
|
ret = nreq - (l - 2); // 2-> marked 0 + redeemScript
|
|
}
|
|
}
|
|
// p2pubkey or p2pubkeyhash
|
|
else {
|
|
if (buffertools.compare(this.getBuffer(), util.EMPTY_BUFFER) === 0) {
|
|
ret = 1;
|
|
}
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
Script.prototype.finishedMultiSig = function() {
|
|
var missing = this.countMissingSignatures();
|
|
if (missing === null) return null;
|
|
|
|
return missing === 0;
|
|
};
|
|
|
|
Script.prototype.getMultiSigInfo = function() {
|
|
if (!this.isMultiSig()) {
|
|
throw new Error("Script.getMultiSigInfo(): Not a multiSig script.");
|
|
}
|
|
|
|
var nsigs = this.chunks[0] - 80; //see OP_2-OP_16;
|
|
var npubkeys = this.chunks[this.chunks.length - 2] - 80; //see OP_2-OP_16;
|
|
|
|
var pubkeys = [];
|
|
for (var i = 1; i < this.chunks.length - 2; i++) {
|
|
pubkeys.push(this.chunks[i]);
|
|
}
|
|
|
|
if (pubkeys.length != npubkeys) {
|
|
throw new Error("Script.getMultiSigInfo(): Amount of PKs does not match what the script specifies.");
|
|
}
|
|
|
|
return {
|
|
nsigs : nsigs,
|
|
npubkeys : npubkeys,
|
|
pubkeys : pubkeys
|
|
}
|
|
};
|
|
|
|
Script.prototype.prependOp0 = function() {
|
|
var chunks = [0];
|
|
for (i in this.chunks) {
|
|
if (this.chunks.hasOwnProperty(i)) {
|
|
chunks.push(this.chunks[i]);
|
|
}
|
|
}
|
|
this.chunks = chunks;
|
|
this.updateBuffer();
|
|
return this;
|
|
};
|
|
|
|
// is this a script form we know?
|
|
Script.prototype.classify = function() {
|
|
if (this.isPubkeyHash())
|
|
return TX_PUBKEYHASH;
|
|
if (this.isP2SH())
|
|
return TX_SCRIPTHASH;
|
|
if (this.isMultiSig())
|
|
return TX_MULTISIG;
|
|
if (this.isPubkey())
|
|
return TX_PUBKEY;
|
|
return TX_UNKNOWN;
|
|
};
|
|
|
|
// extract useful data items from known scripts
|
|
Script.prototype.capture = function() {
|
|
var txType = this.classify();
|
|
var res = [];
|
|
switch (txType) {
|
|
case TX_PUBKEY:
|
|
res.push(this.chunks[0]);
|
|
break;
|
|
case TX_PUBKEYHASH:
|
|
res.push(this.chunks[2]);
|
|
break;
|
|
case TX_MULTISIG:
|
|
for (var i = 1; i < (this.chunks.length - 2); i++)
|
|
res.push(this.chunks[i]);
|
|
break;
|
|
case TX_SCRIPTHASH:
|
|
res.push(this.chunks[1]);
|
|
break;
|
|
|
|
case TX_UNKNOWN:
|
|
default:
|
|
// do nothing
|
|
break;
|
|
}
|
|
|
|
return res;
|
|
};
|
|
|
|
// return first extracted data item from script
|
|
Script.prototype.captureOne = function() {
|
|
var arr = this.capture();
|
|
return arr[0];
|
|
};
|
|
|
|
Script.prototype.getOutType = function() {
|
|
var txType = this.classify();
|
|
switch (txType) {
|
|
case TX_PUBKEY:
|
|
return 'Pubkey';
|
|
case TX_PUBKEYHASH:
|
|
return 'Address';
|
|
default:
|
|
return 'Strange';
|
|
}
|
|
};
|
|
|
|
Script.prototype.getRawOutType = function() {
|
|
return TX_TYPES[this.classify()];
|
|
};
|
|
|
|
Script.prototype.simpleOutHash = function() {
|
|
switch (this.getOutType()) {
|
|
case 'Address':
|
|
return this.chunks[2];
|
|
case 'Pubkey':
|
|
return util.sha256ripe160(this.chunks[0]);
|
|
default:
|
|
log.debug("Encountered non-standard scriptPubKey");
|
|
log.debug("Strange script was: " + this.toString());
|
|
return null;
|
|
}
|
|
};
|
|
|
|
Script.prototype.getInType = function() {
|
|
if (this.chunks.length == 1) {
|
|
// Direct IP to IP transactions only have the public key in their scriptSig.
|
|
return 'Pubkey';
|
|
} else if (this.chunks.length == 2 &&
|
|
Buffer.isBuffer(this.chunks[0]) &&
|
|
Buffer.isBuffer(this.chunks[1])) {
|
|
return 'Address';
|
|
} else {
|
|
return 'Strange';
|
|
}
|
|
};
|
|
|
|
Script.prototype.simpleInPubKey = function() {
|
|
switch (this.getInType()) {
|
|
case 'Address':
|
|
return this.chunks[1];
|
|
case 'Pubkey':
|
|
return null;
|
|
default:
|
|
log.debug("Encountered non-standard scriptSig");
|
|
log.debug("Strange script was: " + this.toString());
|
|
return null;
|
|
}
|
|
};
|
|
|
|
Script.prototype.getBuffer = function() {
|
|
return this.buffer;
|
|
};
|
|
|
|
Script.prototype.serialize = Script.prototype.getBuffer;
|
|
|
|
Script.prototype.getStringContent = function(truncate, maxEl) {
|
|
if (truncate === null) {
|
|
truncate = true;
|
|
}
|
|
|
|
if ('undefined' === typeof maxEl) {
|
|
maxEl = 15;
|
|
}
|
|
|
|
var s = '';
|
|
for (var i = 0, l = this.chunks.length; i < l; i++) {
|
|
var chunk = this.chunks[i];
|
|
|
|
if (i > 0) {
|
|
s += ' ';
|
|
}
|
|
|
|
if (Buffer.isBuffer(chunk)) {
|
|
s += '0x' + util.formatBuffer(chunk, truncate ? null : 0);
|
|
} else {
|
|
s += Opcode.reverseMap[chunk];
|
|
}
|
|
|
|
if (maxEl && i > maxEl) {
|
|
s += ' ...';
|
|
break;
|
|
}
|
|
}
|
|
return s;
|
|
};
|
|
|
|
Script.prototype.toString = function(truncate, maxEl) {
|
|
var script = "<Script ";
|
|
script += this.getStringContent(truncate, maxEl);
|
|
script += ">";
|
|
return script;
|
|
};
|
|
|
|
Script.prototype.writeOp = function(opcode) {
|
|
var buf = Buffer(this.buffer.length + 1);
|
|
this.buffer.copy(buf);
|
|
buf.writeUInt8(opcode, this.buffer.length);
|
|
|
|
this.buffer = buf;
|
|
|
|
this.chunks.push(opcode);
|
|
};
|
|
|
|
Script.prototype.writeN = function(n) {
|
|
if (n < 0 || n > 16)
|
|
throw new Error("writeN: out of range value " + n);
|
|
|
|
if (n == 0)
|
|
this.writeOp(Opcode.map.OP_0);
|
|
else
|
|
this.writeOp(Opcode.map.OP_1 + n - 1);
|
|
};
|
|
|
|
function prefixSize(data_length) {
|
|
if (data_length < Opcode.map.OP_PUSHDATA1) {
|
|
return 1;
|
|
} else if (data_length <= 0xff) {
|
|
return 1 + 1;
|
|
} else if (data_length <= 0xffff) {
|
|
return 1 + 2;
|
|
} else {
|
|
return 1 + 4;
|
|
}
|
|
};
|
|
|
|
function encodeLen(data_length) {
|
|
var buf = undefined;
|
|
if (data_length < Opcode.map.OP_PUSHDATA1) {
|
|
buf = new Buffer(1);
|
|
buf.writeUInt8(data_length, 0);
|
|
} else if (data_length <= 0xff) {
|
|
buf = new Buffer(1 + 1);
|
|
buf.writeUInt8(Opcode.map.OP_PUSHDATA1, 0);
|
|
buf.writeUInt8(data_length, 1);
|
|
} else if (data_length <= 0xffff) {
|
|
buf = new Buffer(1 + 2);
|
|
buf.writeUInt8(Opcode.map.OP_PUSHDATA2, 0);
|
|
buf.writeUInt16LE(data_length, 1);
|
|
} else {
|
|
buf = new Buffer(1 + 4);
|
|
buf.writeUInt8(Opcode.map.OP_PUSHDATA4, 0);
|
|
buf.writeUInt32LE(data_length, 1);
|
|
}
|
|
|
|
return buf;
|
|
};
|
|
|
|
Script.prototype.writeBytes = function(data) {
|
|
var newSize = this.buffer.length + prefixSize(data.length) + data.length;
|
|
this.buffer = Buffer.concat([this.buffer, encodeLen(data.length), data]);
|
|
this.chunks.push(data);
|
|
};
|
|
|
|
Script.prototype.updateBuffer = function() {
|
|
this.buffer = Script.chunksToBuffer(this.chunks);
|
|
};
|
|
|
|
Script.prototype.findAndDelete = function(chunk) {
|
|
var dirty = false;
|
|
if (Buffer.isBuffer(chunk)) {
|
|
for (var i = 0, l = this.chunks.length; i < l; i++) {
|
|
if (Buffer.isBuffer(this.chunks[i]) &&
|
|
buffertools.compare(this.chunks[i], chunk) === 0) {
|
|
this.chunks.splice(i, 1);
|
|
i--;
|
|
dirty = true;
|
|
}
|
|
}
|
|
} else if ("number" === typeof chunk) {
|
|
for (var i = 0, l = this.chunks.length; i < l; i++) {
|
|
if (this.chunks[i] === chunk) {
|
|
this.chunks.splice(i, 1);
|
|
i--;
|
|
dirty = true;
|
|
}
|
|
}
|
|
} else {
|
|
throw new Error("Invalid chunk datatype.");
|
|
}
|
|
if (dirty) {
|
|
this.updateBuffer();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Creates a simple OP_CHECKSIG with pubkey output script.
|
|
*
|
|
* These are used for coinbase transactions and at some point were used for
|
|
* IP-based transactions as well.
|
|
*/
|
|
Script.createPubKeyOut = function(pubkey) {
|
|
var script = new Script();
|
|
script.writeBytes(pubkey);
|
|
script.writeOp(Opcode.map.OP_CHECKSIG);
|
|
return script;
|
|
};
|
|
|
|
/**
|
|
* Creates a standard txout script.
|
|
*/
|
|
Script.createPubKeyHashOut = function(pubKeyHash) {
|
|
var script = new Script();
|
|
script.writeOp(Opcode.map.OP_DUP);
|
|
script.writeOp(Opcode.map.OP_HASH160);
|
|
script.writeBytes(pubKeyHash);
|
|
script.writeOp(Opcode.map.OP_EQUALVERIFY);
|
|
script.writeOp(Opcode.map.OP_CHECKSIG);
|
|
return script;
|
|
};
|
|
|
|
Script._sortKeys = function(keys) {
|
|
return keys.sort(function(buf1, buf2) {
|
|
var len = buf1.length > buf1.length ? buf1.length : buf2.length;
|
|
for (var i = 0; i <= len; i++) {
|
|
if (buf1[i] === undefined)
|
|
return -1; //shorter strings come first
|
|
if (buf2[i] === undefined)
|
|
return 1;
|
|
if (buf1[i] < buf2[i])
|
|
return -1;
|
|
if (buf1[i] > buf2[i])
|
|
return 1;
|
|
else
|
|
continue;
|
|
}
|
|
return 0;
|
|
});
|
|
};
|
|
|
|
Script.createMultisig = function(n_required, inKeys, opts) {
|
|
opts = opts || {};
|
|
var keys = opts.noSorting ? inKeys : this._sortKeys(inKeys);
|
|
var script = new Script();
|
|
script.writeN(n_required);
|
|
keys.forEach(function(key) {
|
|
script.writeBytes(key);
|
|
});
|
|
script.writeN(keys.length);
|
|
script.writeOp(Opcode.map.OP_CHECKMULTISIG);
|
|
return script;
|
|
};
|
|
|
|
Script.createP2SH = function(scriptHash) {
|
|
var script = new Script();
|
|
script.writeOp(Opcode.map.OP_HASH160);
|
|
script.writeBytes(scriptHash);
|
|
script.writeOp(Opcode.map.OP_EQUAL);
|
|
return script;
|
|
};
|
|
|
|
Script.fromTestData = function(testData) {
|
|
testData = testData.map(function(chunk) {
|
|
if ("string" === typeof chunk) {
|
|
return new Buffer(chunk, 'hex');
|
|
} else {
|
|
return chunk;
|
|
}
|
|
});
|
|
|
|
var script = new Script();
|
|
script.chunks = testData;
|
|
script.updateBuffer();
|
|
return script;
|
|
};
|
|
|
|
Script.fromChunks = function(chunks) {
|
|
var script = new Script();
|
|
script.chunks = chunks;
|
|
script.updateBuffer();
|
|
return script;
|
|
};
|
|
|
|
Script.fromHumanReadable = function(s) {
|
|
return new Script(Script.stringToBuffer(s));
|
|
};
|
|
|
|
Script.prototype.toHumanReadable = function() {
|
|
var s = '';
|
|
for (var i = 0, l = this.chunks.length; i < l; i++) {
|
|
var chunk = this.chunks[i];
|
|
|
|
if (i > 0) {
|
|
s += ' ';
|
|
}
|
|
|
|
if (Buffer.isBuffer(chunk)) {
|
|
if (chunk.length === 0) {
|
|
s += '0';
|
|
} else {
|
|
s += '0x' + util.formatBuffer(encodeLen(chunk.length), 0) + ' ';
|
|
s += '0x' + util.formatBuffer(chunk, 0);
|
|
}
|
|
} else {
|
|
var opcode = Opcode.reverseMap[chunk];
|
|
if (typeof opcode === 'undefined') {
|
|
opcode = '0x'+chunk.toString(16);
|
|
}
|
|
s += opcode;
|
|
}
|
|
}
|
|
return s;
|
|
};
|
|
|
|
Script.stringToBuffer = function(s) {
|
|
var buf = new Put();
|
|
var split = s.split(' ');
|
|
for (var i = 0; i < split.length; i++) {
|
|
var word = split[i];
|
|
if (word === '') continue;
|
|
if (word.length > 2 && word.substring(0, 2) === '0x') {
|
|
// raw hex value
|
|
//console.log('hex value');
|
|
buf.put(new Buffer(word.substring(2, word.length), 'hex'));
|
|
} else {
|
|
var opcode = Opcode.map['OP_' + word] || Opcode.map[word];
|
|
if (typeof opcode !== 'undefined') {
|
|
// op code in string form
|
|
//console.log('opcode');
|
|
buf.word8(opcode);
|
|
} else {
|
|
var integer = parseInt(word);
|
|
if (!isNaN(integer)) {
|
|
// integer
|
|
//console.log('integer');
|
|
var data = util.intToBufferSM(integer);
|
|
buf.put(Script.chunksToBuffer([data]));
|
|
} else if (word[0] === '\'' && word[word.length-1] === '\'') {
|
|
// string
|
|
//console.log('string');
|
|
word = word.substring(1,word.length-1);
|
|
var hexString = '';
|
|
for(var c=0;c<word.length;c++) {
|
|
hexString += ''+word.charCodeAt(c).toString(16);
|
|
}
|
|
buf.put(Script.chunksToBuffer([new Buffer(word)]));
|
|
} else {
|
|
throw new Error('Could not parse word "' +word+'" from script "'+s+'"');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return buf.buffer();
|
|
};
|
|
|
|
Script.chunksToBuffer = function(chunks) {
|
|
var buf = new Put();
|
|
|
|
for (var i = 0, l = chunks.length; i < l; i++) {
|
|
var data = chunks[i];
|
|
if (Buffer.isBuffer(data)) {
|
|
if (data.length < Opcode.map.OP_PUSHDATA1) {
|
|
buf.word8(data.length);
|
|
} else if (data.length <= 0xff) {
|
|
buf.word8(Opcode.map.OP_PUSHDATA1);
|
|
buf.word8(data.length);
|
|
} else if (data.length <= 0xffff) {
|
|
buf.word8(Opcode.map.OP_PUSHDATA2);
|
|
buf.word16le(data.length);
|
|
} else {
|
|
buf.word8(Opcode.map.OP_PUSHDATA4);
|
|
buf.word32le(data.length);
|
|
}
|
|
buf.put(data);
|
|
} else if ("number" === typeof data) {
|
|
buf.word8(data);
|
|
} else {
|
|
throw new Error("Script.chunksToBuffer(): Invalid chunk datatype");
|
|
}
|
|
}
|
|
return buf.buffer();
|
|
};
|
|
|
|
|
|
|
|
module.exports = require('soop')(Script);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../config":"4itQ50","../util/BinaryParser":"b3ZSD7","../util/log":"AdF7pF","../util/util":"ACyo5H","./Opcode":"Zm7/h9","buffer":80,"bufferput":"aXRuS6","buffertools":"fugeBw","soop":125}],"./lib/Script":[function(require,module,exports){
|
|
module.exports=require('hQ0t76');
|
|
},{}],"./lib/ScriptInterpreter":[function(require,module,exports){
|
|
module.exports=require('Q/ZWXW');
|
|
},{}],"Q/ZWXW":[function(require,module,exports){
|
|
(function (process,Buffer){
|
|
var imports = require('soop').imports();
|
|
var config = imports.config || require('../config');
|
|
var log = imports.log || require('../util/log');
|
|
var util = imports.util || require('../util');
|
|
var Opcode = imports.Opcode || require('./Opcode');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
var bignum = imports.bignum || require('bignum');
|
|
var Util = imports.Util || require('../util');
|
|
var Script = require('./Script');
|
|
var Key = require('./Key');
|
|
|
|
var SIGHASH_ALL = 1;
|
|
var SIGHASH_NONE = 2;
|
|
var SIGHASH_SINGLE = 3;
|
|
var SIGHASH_ANYONECANPAY = 80;
|
|
|
|
var intToBufferSM = Util.intToBufferSM
|
|
var bufferSMToInt = Util.bufferSMToInt;
|
|
|
|
function ScriptInterpreter(opts) {
|
|
this.opts = opts || {};
|
|
this.stack = [];
|
|
this.disableUnsafeOpcodes = true;
|
|
};
|
|
|
|
ScriptInterpreter.prototype.eval = function eval(script, tx, inIndex, hashType, callback) {
|
|
if ("function" !== typeof callback) {
|
|
throw new Error("ScriptInterpreter.eval() requires a callback");
|
|
}
|
|
|
|
var pc = 0;
|
|
var execStack = [];
|
|
var altStack = [];
|
|
var hashStart = 0;
|
|
var opCount = 0;
|
|
|
|
if (script.buffer.length > 10000) {
|
|
callback(new Error("Oversized script (> 10k bytes)"));
|
|
return this;
|
|
}
|
|
|
|
// Start execution by running the first step
|
|
executeStep.call(this, callback);
|
|
|
|
function executeStep(cb) {
|
|
try {
|
|
// Once all chunks have been processed, execution ends
|
|
if (pc >= script.chunks.length) {
|
|
// Execution stack must be empty at the end of the script
|
|
if (execStack.length) {
|
|
cb(new Error("Execution stack ended non-empty"));
|
|
return;
|
|
}
|
|
|
|
// Execution successful (Note that we still have to check whether the
|
|
// final stack contains a truthy value.)
|
|
cb(null);
|
|
return;
|
|
}
|
|
|
|
// The execution bit is true if there are no "false" values in the
|
|
// execution stack. (A "false" value indicates that we're in the
|
|
// inactive branch of an if statement.)
|
|
var exec = !~execStack.indexOf(false);
|
|
|
|
var opcode = script.chunks[pc++];
|
|
|
|
if (opcode.length > 520) {
|
|
throw new Error("Max push value size exceeded (>520)");
|
|
}
|
|
|
|
if (opcode > Opcode.map.OP_16 && ++opCount > 201) {
|
|
throw new Error("Opcode limit exceeded (>200)");
|
|
}
|
|
|
|
if (this.disableUnsafeOpcodes &&
|
|
"number" === typeof opcode &&
|
|
(opcode === Opcode.map.OP_CAT ||
|
|
opcode === Opcode.map.OP_SUBSTR ||
|
|
opcode === Opcode.map.OP_LEFT ||
|
|
opcode === Opcode.map.OP_RIGHT ||
|
|
opcode === Opcode.map.OP_INVERT ||
|
|
opcode === Opcode.map.OP_AND ||
|
|
opcode === Opcode.map.OP_OR ||
|
|
opcode === Opcode.map.OP_XOR ||
|
|
opcode === Opcode.map.OP_2MUL ||
|
|
opcode === Opcode.map.OP_2DIV ||
|
|
opcode === Opcode.map.OP_MUL ||
|
|
opcode === Opcode.map.OP_DIV ||
|
|
opcode === Opcode.map.OP_MOD ||
|
|
opcode === Opcode.map.OP_LSHIFT ||
|
|
opcode === Opcode.map.OP_RSHIFT)) {
|
|
throw new Error("Encountered a disabled opcode");
|
|
}
|
|
|
|
if (exec && Buffer.isBuffer(opcode)) {
|
|
this.stack.push(opcode);
|
|
} else if (exec || (Opcode.map.OP_IF <= opcode && opcode <= Opcode.map.OP_ENDIF))
|
|
switch (opcode) {
|
|
case Opcode.map.OP_0:
|
|
this.stack.push(new Buffer([]));
|
|
break;
|
|
|
|
case Opcode.map.OP_1NEGATE:
|
|
case Opcode.map.OP_1:
|
|
case Opcode.map.OP_2:
|
|
case Opcode.map.OP_3:
|
|
case Opcode.map.OP_4:
|
|
case Opcode.map.OP_5:
|
|
case Opcode.map.OP_6:
|
|
case Opcode.map.OP_7:
|
|
case Opcode.map.OP_8:
|
|
case Opcode.map.OP_9:
|
|
case Opcode.map.OP_10:
|
|
case Opcode.map.OP_11:
|
|
case Opcode.map.OP_12:
|
|
case Opcode.map.OP_13:
|
|
case Opcode.map.OP_14:
|
|
case Opcode.map.OP_15:
|
|
case Opcode.map.OP_16:
|
|
var opint = opcode - Opcode.map.OP_1 + 1;
|
|
var opbuf = intToBufferSM(opint);
|
|
this.stack.push(opbuf);
|
|
break;
|
|
|
|
case Opcode.map.OP_NOP:
|
|
case Opcode.map.OP_NOP1:
|
|
case Opcode.map.OP_NOP2:
|
|
case Opcode.map.OP_NOP3:
|
|
case Opcode.map.OP_NOP4:
|
|
case Opcode.map.OP_NOP5:
|
|
case Opcode.map.OP_NOP6:
|
|
case Opcode.map.OP_NOP7:
|
|
case Opcode.map.OP_NOP8:
|
|
case Opcode.map.OP_NOP9:
|
|
case Opcode.map.OP_NOP10:
|
|
break;
|
|
|
|
case Opcode.map.OP_IF:
|
|
case Opcode.map.OP_NOTIF:
|
|
// <expression> if [statements] [else [statements]] endif
|
|
var value = false;
|
|
if (exec) {
|
|
value = castBool(this.stackPop());
|
|
if (opcode === Opcode.map.OP_NOTIF) {
|
|
value = !value;
|
|
}
|
|
}
|
|
execStack.push(value);
|
|
break;
|
|
|
|
case Opcode.map.OP_ELSE:
|
|
if (execStack.length < 1) {
|
|
throw new Error("Unmatched OP_ELSE");
|
|
}
|
|
execStack[execStack.length - 1] = !execStack[execStack.length - 1];
|
|
break;
|
|
|
|
case Opcode.map.OP_ENDIF:
|
|
if (execStack.length < 1) {
|
|
throw new Error("Unmatched OP_ENDIF");
|
|
}
|
|
execStack.pop();
|
|
break;
|
|
|
|
case Opcode.map.OP_VERIFY:
|
|
var value = castBool(this.stackTop());
|
|
if (value) {
|
|
this.stackPop();
|
|
} else {
|
|
throw new Error("OP_VERIFY negative");
|
|
}
|
|
break;
|
|
|
|
case Opcode.map.OP_RETURN:
|
|
throw new Error("OP_RETURN");
|
|
|
|
case Opcode.map.OP_TOALTSTACK:
|
|
altStack.push(this.stackPop());
|
|
break;
|
|
|
|
case Opcode.map.OP_FROMALTSTACK:
|
|
if (altStack.length < 1) {
|
|
throw new Error("OP_FROMALTSTACK with alt stack empty");
|
|
}
|
|
this.stack.push(altStack.pop());
|
|
break;
|
|
|
|
case Opcode.map.OP_2DROP:
|
|
// (x1 x2 -- )
|
|
this.stackPop();
|
|
this.stackPop();
|
|
break;
|
|
|
|
case Opcode.map.OP_2DUP:
|
|
// (x1 x2 -- x1 x2 x1 x2)
|
|
var v1 = this.stackTop(2);
|
|
var v2 = this.stackTop(1);
|
|
this.stack.push(v1);
|
|
this.stack.push(v2);
|
|
break;
|
|
|
|
case Opcode.map.OP_3DUP:
|
|
// (x1 x2 -- x1 x2 x1 x2)
|
|
var v1 = this.stackTop(3);
|
|
var v2 = this.stackTop(2);
|
|
var v3 = this.stackTop(1);
|
|
this.stack.push(v1);
|
|
this.stack.push(v2);
|
|
this.stack.push(v3);
|
|
break;
|
|
|
|
case Opcode.map.OP_2OVER:
|
|
// (x1 x2 x3 x4 -- x1 x2 x3 x4 x1 x2)
|
|
var v1 = this.stackTop(4);
|
|
var v2 = this.stackTop(3);
|
|
this.stack.push(v1);
|
|
this.stack.push(v2);
|
|
break;
|
|
|
|
case Opcode.map.OP_2ROT:
|
|
// (x1 x2 x3 x4 x5 x6 -- x3 x4 x5 x6 x1 x2)
|
|
var v1 = this.stackTop(6);
|
|
var v2 = this.stackTop(5);
|
|
this.stack.splice(this.stack.length - 6, 2);
|
|
this.stack.push(v1);
|
|
this.stack.push(v2);
|
|
break;
|
|
|
|
case Opcode.map.OP_2SWAP:
|
|
// (x1 x2 x3 x4 -- x3 x4 x1 x2)
|
|
this.stackSwap(4, 2);
|
|
this.stackSwap(3, 1);
|
|
break;
|
|
|
|
case Opcode.map.OP_IFDUP:
|
|
// (x - 0 | x x)
|
|
var value = this.stackTop();
|
|
if (castBool(value)) {
|
|
this.stack.push(value);
|
|
}
|
|
break;
|
|
|
|
case Opcode.map.OP_DEPTH:
|
|
// -- stacksize
|
|
var value = bignum(this.stack.length);
|
|
this.stack.push(intToBufferSM(value));
|
|
break;
|
|
|
|
case Opcode.map.OP_DROP:
|
|
// (x -- )
|
|
this.stackPop();
|
|
break;
|
|
|
|
case Opcode.map.OP_DUP:
|
|
// (x -- x x)
|
|
this.stack.push(this.stackTop());
|
|
break;
|
|
|
|
case Opcode.map.OP_NIP:
|
|
// (x1 x2 -- x2)
|
|
if (this.stack.length < 2) {
|
|
throw new Error("OP_NIP insufficient stack size");
|
|
}
|
|
this.stack.splice(this.stack.length - 2, 1);
|
|
break;
|
|
|
|
case Opcode.map.OP_OVER:
|
|
// (x1 x2 -- x1 x2 x1)
|
|
this.stack.push(this.stackTop(2));
|
|
break;
|
|
|
|
case Opcode.map.OP_PICK:
|
|
case Opcode.map.OP_ROLL:
|
|
// (xn ... x2 x1 x0 n - xn ... x2 x1 x0 xn)
|
|
// (xn ... x2 x1 x0 n - ... x2 x1 x0 xn)
|
|
var n = castInt(this.stackPop());
|
|
if (n < 0 || n >= this.stack.length) {
|
|
throw new Error("OP_PICK/OP_ROLL insufficient stack size");
|
|
}
|
|
var value = this.stackTop(n + 1);
|
|
if (opcode === Opcode.map.OP_ROLL) {
|
|
this.stack.splice(this.stack.length - n - 1, 1);
|
|
}
|
|
this.stack.push(value);
|
|
break;
|
|
|
|
case Opcode.map.OP_ROT:
|
|
// (x1 x2 x3 -- x2 x3 x1)
|
|
// x2 x1 x3 after first swap
|
|
// x2 x3 x1 after second swap
|
|
this.stackSwap(3, 2);
|
|
this.stackSwap(2, 1);
|
|
break;
|
|
|
|
case Opcode.map.OP_SWAP:
|
|
// (x1 x2 -- x2 x1)
|
|
this.stackSwap(2, 1);
|
|
break;
|
|
|
|
case Opcode.map.OP_TUCK:
|
|
// (x1 x2 -- x2 x1 x2)
|
|
if (this.stack.length < 2) {
|
|
throw new Error("OP_TUCK insufficient stack size");
|
|
}
|
|
this.stack.splice(this.stack.length - 2, 0, this.stackTop());
|
|
break;
|
|
|
|
case Opcode.map.OP_CAT:
|
|
// (x1 x2 -- out)
|
|
var v1 = this.stackTop(2);
|
|
var v2 = this.stackTop(1);
|
|
this.stackPop();
|
|
this.stackPop();
|
|
this.stack.push(Buffer.concat([v1, v2]));
|
|
break;
|
|
|
|
case Opcode.map.OP_SUBSTR:
|
|
// (in begin size -- out)
|
|
var buf = this.stackTop(3);
|
|
var start = castInt(this.stackTop(2));
|
|
var len = castInt(this.stackTop(1));
|
|
if (start < 0 || len < 0) {
|
|
throw new Error("OP_SUBSTR start < 0 or len < 0");
|
|
}
|
|
if ((start + len) >= buf.length) {
|
|
throw new Error("OP_SUBSTR range out of bounds");
|
|
}
|
|
this.stackPop();
|
|
this.stackPop();
|
|
this.stack[this.stack.length - 1] = buf.slice(start, start + len);
|
|
break;
|
|
|
|
case Opcode.map.OP_LEFT:
|
|
case Opcode.map.OP_RIGHT:
|
|
// (in size -- out)
|
|
var buf = this.stackTop(2);
|
|
var size = castInt(this.stackTop(1));
|
|
if (size < 0) {
|
|
throw new Error("OP_LEFT/OP_RIGHT size < 0");
|
|
}
|
|
if (size > buf.length) {
|
|
size = buf.length;
|
|
}
|
|
this.stackPop();
|
|
if (opcode === Opcode.map.OP_LEFT) {
|
|
this.stack[this.stack.length - 1] = buf.slice(0, size);
|
|
} else {
|
|
this.stack[this.stack.length - 1] = buf.slice(buf.length - size);
|
|
}
|
|
break;
|
|
|
|
case Opcode.map.OP_SIZE:
|
|
// (in -- in size)
|
|
var value = bignum(this.stackTop().length);
|
|
this.stack.push(intToBufferSM(value));
|
|
break;
|
|
|
|
case Opcode.map.OP_INVERT:
|
|
// (in - out)
|
|
var buf = this.stackTop();
|
|
for (var i = 0, l = buf.length; i < l; i++) {
|
|
buf[i] = ~buf[i];
|
|
}
|
|
break;
|
|
|
|
case Opcode.map.OP_AND:
|
|
case Opcode.map.OP_OR:
|
|
case Opcode.map.OP_XOR:
|
|
// (x1 x2 - out)
|
|
var v1 = this.stackTop(2);
|
|
var v2 = this.stackTop(1);
|
|
this.stackPop();
|
|
this.stackPop();
|
|
var out = new Buffer(Math.max(v1.length, v2.length));
|
|
if (opcode === Opcode.map.OP_AND) {
|
|
for (var i = 0, l = out.length; i < l; i++) {
|
|
out[i] = v1[i] & v2[i];
|
|
}
|
|
} else if (opcode === Opcode.map.OP_OR) {
|
|
for (var i = 0, l = out.length; i < l; i++) {
|
|
out[i] = v1[i] | v2[i];
|
|
}
|
|
} else if (opcode === Opcode.map.OP_XOR) {
|
|
for (var i = 0, l = out.length; i < l; i++) {
|
|
out[i] = v1[i] ^ v2[i];
|
|
}
|
|
}
|
|
this.stack.push(out);
|
|
break;
|
|
|
|
case Opcode.map.OP_EQUAL:
|
|
case Opcode.map.OP_EQUALVERIFY:
|
|
//case OP_NOTEQUAL: // use OP_NUMNOTEQUAL
|
|
// (x1 x2 - bool)
|
|
var v1 = this.stackTop(2);
|
|
var v2 = this.stackTop(1);
|
|
|
|
var value = buffertools.compare(v1, v2) === 0;
|
|
|
|
// OP_NOTEQUAL is disabled because it would be too easy to say
|
|
// something like n != 1 and have some wiseguy pass in 1 with extra
|
|
// zero bytes after it (numerically, 0x01 == 0x0001 == 0x000001)
|
|
//if (opcode == OP_NOTEQUAL)
|
|
// fEqual = !fEqual;
|
|
|
|
this.stackPop();
|
|
this.stackPop();
|
|
this.stack.push(new Buffer([value ? 1 : 0]));
|
|
if (opcode === Opcode.map.OP_EQUALVERIFY) {
|
|
if (value) {
|
|
this.stackPop();
|
|
} else {
|
|
throw new Error("OP_EQUALVERIFY negative");
|
|
}
|
|
}
|
|
break;
|
|
|
|
case Opcode.map.OP_1ADD:
|
|
case Opcode.map.OP_1SUB:
|
|
case Opcode.map.OP_2MUL:
|
|
case Opcode.map.OP_2DIV:
|
|
case Opcode.map.OP_NEGATE:
|
|
case Opcode.map.OP_ABS:
|
|
case Opcode.map.OP_NOT:
|
|
case Opcode.map.OP_0NOTEQUAL:
|
|
// (in -- out)
|
|
var num = bufferSMToInt(this.stackTop());
|
|
switch (opcode) {
|
|
case Opcode.map.OP_1ADD:
|
|
num = num.add(bignum(1));
|
|
break;
|
|
case Opcode.map.OP_1SUB:
|
|
num = num.sub(bignum(1));
|
|
break;
|
|
case Opcode.map.OP_2MUL:
|
|
num = num.mul(bignum(2));
|
|
break;
|
|
case Opcode.map.OP_2DIV:
|
|
num = num.div(bignum(2));
|
|
break;
|
|
case Opcode.map.OP_NEGATE:
|
|
num = num.neg();
|
|
break;
|
|
case Opcode.map.OP_ABS:
|
|
num = num.abs();
|
|
break;
|
|
case Opcode.map.OP_NOT:
|
|
num = bignum(num.cmp(0) == 0 ? 1 : 0);
|
|
break;
|
|
case Opcode.map.OP_0NOTEQUAL:
|
|
num = bignum(num.cmp(0) == 0 ? 0 : 1);
|
|
break;
|
|
}
|
|
this.stack[this.stack.length - 1] = intToBufferSM(num);
|
|
break;
|
|
|
|
case Opcode.map.OP_ADD:
|
|
case Opcode.map.OP_SUB:
|
|
case Opcode.map.OP_MUL:
|
|
case Opcode.map.OP_DIV:
|
|
case Opcode.map.OP_MOD:
|
|
case Opcode.map.OP_LSHIFT:
|
|
case Opcode.map.OP_RSHIFT:
|
|
case Opcode.map.OP_BOOLAND:
|
|
case Opcode.map.OP_BOOLOR:
|
|
case Opcode.map.OP_NUMEQUAL:
|
|
case Opcode.map.OP_NUMEQUALVERIFY:
|
|
case Opcode.map.OP_NUMNOTEQUAL:
|
|
case Opcode.map.OP_LESSTHAN:
|
|
case Opcode.map.OP_GREATERTHAN:
|
|
case Opcode.map.OP_LESSTHANOREQUAL:
|
|
case Opcode.map.OP_GREATERTHANOREQUAL:
|
|
case Opcode.map.OP_MIN:
|
|
case Opcode.map.OP_MAX:
|
|
// (x1 x2 -- out)
|
|
var v1 = bufferSMToInt(this.stackTop(2));
|
|
var v2 = bufferSMToInt(this.stackTop(1));
|
|
var num;
|
|
switch (opcode) {
|
|
case Opcode.map.OP_ADD:
|
|
num = v1.add(v2);
|
|
break;
|
|
case Opcode.map.OP_SUB:
|
|
num = v1.sub(v2);
|
|
break;
|
|
case Opcode.map.OP_MUL:
|
|
num = v1.mul(v2);
|
|
break;
|
|
case Opcode.map.OP_DIV:
|
|
num = v1.div(v2);
|
|
break;
|
|
case Opcode.map.OP_MOD:
|
|
num = v1.mod(v2);
|
|
break;
|
|
|
|
case Opcode.map.OP_LSHIFT:
|
|
if (v2.cmp(0) < 0 || v2.cmp(2048) > 0) {
|
|
throw new Error("OP_LSHIFT parameter out of bounds");
|
|
}
|
|
num = v1.shiftLeft(v2);
|
|
break;
|
|
|
|
case Opcode.map.OP_RSHIFT:
|
|
if (v2.cmp(0) < 0 || v2.cmp(2048) > 0) {
|
|
throw new Error("OP_RSHIFT parameter out of bounds");
|
|
}
|
|
num = v1.shiftRight(v2);
|
|
break;
|
|
|
|
case Opcode.map.OP_BOOLAND:
|
|
num = bignum((v1.cmp(0) != 0 && v2.cmp(0) != 0) ? 1 : 0);
|
|
break;
|
|
|
|
case Opcode.map.OP_BOOLOR:
|
|
num = bignum((v1.cmp(0) != 0 || v2.cmp(0) != 0) ? 1 : 0);
|
|
break;
|
|
|
|
case Opcode.map.OP_NUMEQUAL:
|
|
case Opcode.map.OP_NUMEQUALVERIFY:
|
|
num = bignum(v1.cmp(v2) == 0 ? 1 : 0);
|
|
break;
|
|
|
|
case Opcode.map.OP_NUMNOTEQUAL:
|
|
;
|
|
num = bignum(v1.cmp(v2) != 0 ? 1 : 0);
|
|
break;
|
|
|
|
case Opcode.map.OP_LESSTHAN:
|
|
num = bignum(v1.lt(v2) ? 1 : 0);
|
|
break;
|
|
|
|
case Opcode.map.OP_GREATERTHAN:
|
|
num = bignum(v1.gt(v2) ? 1 : 0);
|
|
break;
|
|
|
|
case Opcode.map.OP_LESSTHANOREQUAL:
|
|
num = bignum(v1.gt(v2) ? 0 : 1);
|
|
break;
|
|
|
|
case Opcode.map.OP_GREATERTHANOREQUAL:
|
|
num = bignum(v1.lt(v2) ? 0 : 1);
|
|
break;
|
|
|
|
case Opcode.map.OP_MIN:
|
|
num = (v1.lt(v2) ? v1 : v2);
|
|
break;
|
|
case Opcode.map.OP_MAX:
|
|
num = (v1.gt(v2) ? v1 : v2);
|
|
break;
|
|
}
|
|
this.stackPop();
|
|
this.stackPop();
|
|
this.stack.push(intToBufferSM(num));
|
|
|
|
if (opcode === Opcode.map.OP_NUMEQUALVERIFY) {
|
|
if (castBool(this.stackTop())) {
|
|
this.stackPop();
|
|
} else {
|
|
throw new Error("OP_NUMEQUALVERIFY negative");
|
|
}
|
|
}
|
|
break;
|
|
|
|
case Opcode.map.OP_WITHIN:
|
|
// (x min max -- out)
|
|
var v1 = bufferSMToInt(this.stackTop(3));
|
|
var v2 = bufferSMToInt(this.stackTop(2));
|
|
var v3 = bufferSMToInt(this.stackTop(1));
|
|
this.stackPop();
|
|
this.stackPop();
|
|
this.stackPop();
|
|
var value = v1.cmp(v2) >= 0 && v1.cmp(v3) < 0;
|
|
this.stack.push(intToBufferSM(value ? 1 : 0));
|
|
break;
|
|
|
|
case Opcode.map.OP_RIPEMD160:
|
|
case Opcode.map.OP_SHA1:
|
|
case Opcode.map.OP_SHA256:
|
|
case Opcode.map.OP_HASH160:
|
|
case Opcode.map.OP_HASH256:
|
|
// (in -- hash)
|
|
var value = this.stackPop();
|
|
var hash;
|
|
if (opcode === Opcode.map.OP_RIPEMD160) {
|
|
hash = Util.ripe160(value);
|
|
} else if (opcode === Opcode.map.OP_SHA1) {
|
|
hash = Util.sha1(value);
|
|
} else if (opcode === Opcode.map.OP_SHA256) {
|
|
hash = Util.sha256(value);
|
|
} else if (opcode === Opcode.map.OP_HASH160) {
|
|
hash = Util.sha256ripe160(value);
|
|
} else if (opcode === Opcode.map.OP_HASH256) {
|
|
hash = Util.twoSha256(value);
|
|
}
|
|
this.stack.push(hash);
|
|
break;
|
|
|
|
case Opcode.map.OP_CODESEPARATOR:
|
|
// Hash starts after the code separator
|
|
hashStart = pc;
|
|
break;
|
|
|
|
case Opcode.map.OP_CHECKSIG:
|
|
case Opcode.map.OP_CHECKSIGVERIFY:
|
|
// (sig pubkey -- bool)
|
|
var sig = this.stackTop(2);
|
|
var pubkey = this.stackTop(1);
|
|
|
|
// Get the part of this script since the last OP_CODESEPARATOR
|
|
var scriptChunks = script.chunks.slice(hashStart);
|
|
|
|
// Convert to binary
|
|
var scriptCode = Script.fromChunks(scriptChunks);
|
|
|
|
// Remove signature if present (a signature can't sign itself)
|
|
scriptCode.findAndDelete(sig);
|
|
|
|
// check canonical signature
|
|
this.isCanonicalSignature(new Buffer(sig));
|
|
|
|
// Verify signature
|
|
checkSig(sig, pubkey, scriptCode, tx, inIndex, hashType, function(e, result) {
|
|
var success;
|
|
|
|
if (e) {
|
|
// We intentionally ignore errors during signature verification and
|
|
// treat these cases as an invalid signature.
|
|
success = false;
|
|
} else {
|
|
success = result;
|
|
}
|
|
|
|
// Update stack
|
|
this.stackPop();
|
|
this.stackPop();
|
|
this.stack.push(new Buffer([success ? 1 : 0]));
|
|
if (opcode === Opcode.map.OP_CHECKSIGVERIFY) {
|
|
if (success) {
|
|
this.stackPop();
|
|
} else {
|
|
throw new Error("OP_CHECKSIGVERIFY negative");
|
|
}
|
|
}
|
|
|
|
// Run next step
|
|
executeStep.call(this, cb);
|
|
}.bind(this));
|
|
|
|
// Note that for asynchronous opcodes we have to return here to prevent
|
|
// the next opcode from being executed.
|
|
return;
|
|
|
|
case Opcode.map.OP_CHECKMULTISIG:
|
|
case Opcode.map.OP_CHECKMULTISIGVERIFY:
|
|
// ([sig ...] num_of_signatures [pubkey ...] num_of_pubkeys -- bool)
|
|
var keysCount = castInt(this.stackPop());
|
|
if (keysCount < 0 || keysCount > 20) {
|
|
throw new Error("OP_CHECKMULTISIG keysCount out of bounds");
|
|
}
|
|
opCount += keysCount;
|
|
if (opCount > 201) {
|
|
throw new Error("Opcode limit exceeded (>200)");
|
|
}
|
|
var keys = [];
|
|
for (var i = 0, l = keysCount; i < l; i++) {
|
|
var pubkey = this.stackPop()
|
|
keys.push(pubkey);
|
|
}
|
|
|
|
var sigsCount = castInt(this.stackPop());
|
|
if (sigsCount < 0 || sigsCount > keysCount) {
|
|
throw new Error("OP_CHECKMULTISIG sigsCount out of bounds");
|
|
}
|
|
var sigs = [];
|
|
for (var i = 0, l = sigsCount; i < l; i++) {
|
|
sigs.push(this.stackPop());
|
|
}
|
|
|
|
// The original client has a bug where it pops an extra element off the
|
|
// stack. It can't be fixed without causing a chain split and we need to
|
|
// imitate this behavior as well.
|
|
this.stackPop();
|
|
|
|
// Get the part of this script since the last OP_CODESEPARATOR
|
|
var scriptChunks = script.chunks.slice(hashStart);
|
|
|
|
// Convert to binary
|
|
var scriptCode = Script.fromChunks(scriptChunks);
|
|
|
|
var that = this;
|
|
sigs.forEach(function(sig) {
|
|
// check each signature is canonical
|
|
that.isCanonicalSignature(new Buffer(sig));
|
|
// Drop the signatures for the subscript, since a signature can't sign itself
|
|
scriptCode.findAndDelete(sig);
|
|
});
|
|
|
|
var success = true,
|
|
isig = 0,
|
|
ikey = 0;
|
|
|
|
function checkMultiSigStep() {
|
|
if (success && sigsCount > 0) {
|
|
var sig = sigs[isig];
|
|
var pubkey = keys[ikey];
|
|
|
|
checkSig(sig, pubkey, scriptCode, tx, inIndex, hashType, function(e, result) {
|
|
if (!e && result) {
|
|
isig++;
|
|
sigsCount--;
|
|
} else {
|
|
ikey++;
|
|
keysCount--;
|
|
|
|
// If there are more signatures than keys left, then too many
|
|
// signatures have failed
|
|
if (sigsCount > keysCount) {
|
|
success = false;
|
|
}
|
|
}
|
|
|
|
checkMultiSigStep.call(this);
|
|
}.bind(this));
|
|
} else {
|
|
this.stack.push(new Buffer([success ? 1 : 0]));
|
|
if (opcode === Opcode.map.OP_CHECKMULTISIGVERIFY) {
|
|
if (success) {
|
|
this.stackPop();
|
|
} else {
|
|
throw new Error("OP_CHECKMULTISIGVERIFY negative");
|
|
}
|
|
}
|
|
|
|
// Run next step
|
|
executeStep.call(this, cb);
|
|
}
|
|
};
|
|
checkMultiSigStep.call(this);
|
|
|
|
// Note that for asynchronous opcodes we have to return here to prevent
|
|
// the next opcode from being executed.
|
|
return;
|
|
|
|
default:
|
|
throw new Error("Unknown opcode encountered");
|
|
}
|
|
|
|
// Size limits
|
|
if ((this.stack.length + altStack.length) > 1000) {
|
|
throw new Error("Maximum stack size exceeded");
|
|
}
|
|
|
|
// Run next step
|
|
if (false && pc % 100) {
|
|
// V8 allows for much deeper stacks than Bitcoin's scripting language,
|
|
// but just to be safe, we'll reset the stack every 100 steps
|
|
process.nextTick(executeStep.bind(this, cb));
|
|
} else {
|
|
executeStep.call(this, cb);
|
|
}
|
|
} catch (e) {
|
|
cb(e);
|
|
}
|
|
}
|
|
};
|
|
|
|
ScriptInterpreter.prototype.evalTwo =
|
|
function evalTwo(scriptSig, scriptPubkey, tx, n, hashType, callback) {
|
|
var self = this;
|
|
|
|
self.eval(scriptSig, tx, n, hashType, function(e) {
|
|
if (e) {
|
|
callback(e)
|
|
return;
|
|
}
|
|
|
|
self.eval(scriptPubkey, tx, n, hashType, callback);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Get the top element of the stack.
|
|
*
|
|
* Using the offset parameter this function can also access lower elements
|
|
* from the stack.
|
|
*/
|
|
ScriptInterpreter.prototype.stackTop = function stackTop(offset) {
|
|
offset = +offset || 1;
|
|
if (offset < 1) offset = 1;
|
|
|
|
if (offset > this.stack.length) {
|
|
throw new Error('ScriptInterpreter.stackTop(): Stack underrun');
|
|
}
|
|
|
|
return this.stack[this.stack.length - offset];
|
|
};
|
|
|
|
ScriptInterpreter.prototype.stackBack = function stackBack() {
|
|
return this.stack[this.stack.length - 1];
|
|
};
|
|
|
|
/**
|
|
* Pop the top element off the stack and return it.
|
|
*/
|
|
ScriptInterpreter.prototype.stackPop = function stackPop() {
|
|
if (this.stack.length < 1) {
|
|
throw new Error('ScriptInterpreter.stackTop(): Stack underrun');
|
|
}
|
|
|
|
return this.stack.pop();
|
|
};
|
|
|
|
ScriptInterpreter.prototype.stackSwap = function stackSwap(a, b) {
|
|
if (this.stack.length < a || this.stack.length < b) {
|
|
throw new Error('ScriptInterpreter.stackTop(): Stack underrun');
|
|
}
|
|
|
|
var s = this.stack,
|
|
l = s.length;
|
|
|
|
var tmp = s[l - a];
|
|
s[l - a] = s[l - b];
|
|
s[l - b] = tmp;
|
|
};
|
|
|
|
/**
|
|
* Returns a version of the stack with only primitive types.
|
|
*
|
|
* The return value is an array. Any single byte buffer is converted to an
|
|
* integer. Any longer Buffer is converted to a hex string.
|
|
*/
|
|
ScriptInterpreter.prototype.getPrimitiveStack = function getPrimitiveStack() {
|
|
return this.stack.map(function(chunk) {
|
|
if (chunk.length > 2) {
|
|
return buffertools.toHex(chunk.slice(0));
|
|
}
|
|
var num = bufferSMToInt(chunk);
|
|
if (num.cmp(-128) >= 0 && num.cmp(127) <= 0) {
|
|
return num.toNumber();
|
|
} else {
|
|
return buffertools.toHex(chunk.slice(0));
|
|
}
|
|
});
|
|
};
|
|
|
|
var castBool = ScriptInterpreter.castBool = function castBool(v) {
|
|
for (var i = 0, l = v.length; i < l; i++) {
|
|
if (v[i] != 0) {
|
|
// Negative zero is still zero
|
|
if (i == (l - 1) && v[i] == 0x80) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
var castInt = ScriptInterpreter.castInt = function castInt(v) {
|
|
return bufferSMToInt(v).toNumber();
|
|
};
|
|
|
|
ScriptInterpreter.prototype.getResult = function getResult() {
|
|
if (this.stack.length === 0) {
|
|
throw new Error("Empty stack after script evaluation");
|
|
}
|
|
|
|
return castBool(this.stack[this.stack.length - 1]);
|
|
};
|
|
|
|
// WARN: Use ScriptInterpreter.verifyFull instead
|
|
ScriptInterpreter.verify =
|
|
function verify(scriptSig, scriptPubKey, tx, n, hashType, callback) {
|
|
if ("function" !== typeof callback) {
|
|
throw new Error("ScriptInterpreter.verify() requires a callback");
|
|
}
|
|
|
|
// Create execution environment
|
|
var si = new ScriptInterpreter();
|
|
|
|
// Evaluate scripts
|
|
si.evalTwo(scriptSig, scriptPubKey, tx, n, hashType, function(err) {
|
|
if (err) {
|
|
callback(err);
|
|
return;
|
|
}
|
|
|
|
// Cast result to bool
|
|
var result = si.getResult();
|
|
|
|
callback(null, result);
|
|
});
|
|
|
|
return si;
|
|
};
|
|
|
|
ScriptInterpreter.prototype.verifyStep4 = function(callback, siCopy) {
|
|
// 4th step, check P2SH subscript evaluated to true
|
|
if (siCopy.stack.length == 0) {
|
|
callback(null, false);
|
|
return;
|
|
}
|
|
|
|
callback(null, castBool(siCopy.stackBack()));
|
|
}
|
|
|
|
ScriptInterpreter.prototype.verifyStep3 = function(scriptSig,
|
|
scriptPubKey, tx, nIn, hashType, callback, siCopy) {
|
|
|
|
// 3rd step, check result (stack should contain true)
|
|
|
|
// if stack is empty, script considered invalid
|
|
if (this.stack.length === 0) {
|
|
callback(null, false);
|
|
return;
|
|
}
|
|
|
|
// if top of stack contains false, script evaluated to false
|
|
if (castBool(this.stackBack()) == false) {
|
|
callback(null, false);
|
|
return;
|
|
}
|
|
|
|
// if not P2SH, script evaluated to true
|
|
if (!this.opts.verifyP2SH || !scriptPubKey.isP2SH()) {
|
|
callback(null, true);
|
|
return;
|
|
}
|
|
|
|
// if P2SH, scriptSig should be push-only
|
|
if (!scriptSig.isPushOnly()) {
|
|
callback(null, false);
|
|
return;
|
|
}
|
|
|
|
// P2SH script should exist
|
|
if (siCopy.length === 0) {
|
|
throw new Error('siCopy should have length != 0');
|
|
}
|
|
|
|
var subscript = new Script(siCopy.stackPop());
|
|
var that = this;
|
|
// evaluate the P2SH subscript
|
|
siCopy.eval(subscript, tx, nIn, hashType, function(err) {
|
|
if (err) return callback(err);
|
|
that.verifyStep4(callback, siCopy);
|
|
});
|
|
};
|
|
|
|
ScriptInterpreter.prototype.verifyStep2 = function(scriptSig, scriptPubKey,
|
|
tx, nIn, hashType, callback, siCopy) {
|
|
var siCopy;
|
|
if (this.opts.verifyP2SH) {
|
|
siCopy = new ScriptInterpreter(this.opts);
|
|
this.stack.forEach(function(item) {
|
|
siCopy.stack.push(item);
|
|
});
|
|
}
|
|
|
|
var that = this;
|
|
// 2nd step, evaluate scriptPubKey
|
|
this.eval(scriptPubKey, tx, nIn, hashType, function(err) {
|
|
if (err) return callback(err);
|
|
that.verifyStep3(scriptSig, scriptPubKey, tx, nIn,
|
|
hashType, callback, siCopy);
|
|
});
|
|
};
|
|
|
|
ScriptInterpreter.prototype.verifyFull = function(scriptSig, scriptPubKey,
|
|
tx, nIn, hashType, callback) {
|
|
var that = this;
|
|
|
|
// 1st step, evaluate scriptSig
|
|
this.eval(scriptSig, tx, nIn, hashType, function(err) {
|
|
if (err) return callback(err);
|
|
that.verifyStep2(scriptSig, scriptPubKey, tx, nIn,
|
|
hashType, callback);
|
|
});
|
|
};
|
|
|
|
ScriptInterpreter.verifyFull =
|
|
function verifyFull(scriptSig, scriptPubKey, tx, nIn, hashType,
|
|
opts, callback) {
|
|
var si = new ScriptInterpreter(opts);
|
|
si.verifyFull(scriptSig, scriptPubKey,
|
|
tx, nIn, hashType, callback);
|
|
};
|
|
|
|
|
|
var checkSig = ScriptInterpreter.checkSig =
|
|
function(sig, pubkey, scriptCode, tx, n, hashType, callback) {
|
|
// https://en.bitcoin.it/wiki/OP_CHECKSIG#How_it_works
|
|
if (!sig.length) {
|
|
callback(null, false);
|
|
return;
|
|
}
|
|
|
|
// If the hash-type value is 0, then it is replaced by the last_byte of the signature.
|
|
if (hashType === 0) {
|
|
hashType = sig[sig.length - 1];
|
|
} else if (hashType != sig[sig.length - 1]) {
|
|
callback(null, false);
|
|
return;
|
|
}
|
|
|
|
// Then the last byte of the signature is always deleted. (hashType removed)
|
|
sig = sig.slice(0, sig.length - 1);
|
|
|
|
// Signature verification requires a special hash procedure
|
|
var hash = tx.hashForSignature(scriptCode, n, hashType);
|
|
|
|
// Verify signature
|
|
var key = new Key();
|
|
if (pubkey.length === 0) pubkey = new Buffer('00', 'hex');
|
|
key.public = pubkey;
|
|
|
|
key.verifySignature(hash, sig, callback);
|
|
};
|
|
|
|
ScriptInterpreter.prototype.isCanonicalSignature = function(sig) {
|
|
// See https://bitcointalk.org/index.php?topic=8392.msg127623#msg127623
|
|
// A canonical signature exists of: <30> <total len> <02> <len R> <R> <02> <len S> <S> <hashtype>
|
|
// Where R and S are not negative (their first byte has its highest bit not set), and not
|
|
// excessively padded (do not start with a 0 byte, unless an otherwise negative number follows,
|
|
// in which case a single 0 byte is necessary and even required).
|
|
|
|
if (!Buffer.isBuffer(sig))
|
|
throw new Error("arg should be a Buffer");
|
|
|
|
// TODO: change to opts.verifyStrictEnc to make the default
|
|
// behavior not verify, as in bitcoin core
|
|
if (this.opts.dontVerifyStrictEnc) return true;
|
|
|
|
var l = sig.length;
|
|
if (l < 9) throw new Error("Non-canonical signature: too short");
|
|
if (l > 73) throw new Error("Non-canonical signature: too long");
|
|
|
|
var nHashType = sig[l - 1] & (~(SIGHASH_ANYONECANPAY));
|
|
if (nHashType < SIGHASH_ALL || nHashType > SIGHASH_SINGLE)
|
|
throw new Error("Non-canonical signature: unknown hashtype byte");
|
|
|
|
if (sig[0] !== 0x30)
|
|
throw new Error("Non-canonical signature: wrong type");
|
|
if (sig[1] !== l - 3)
|
|
throw new Error("Non-canonical signature: wrong length marker");
|
|
|
|
var nLenR = sig[3];
|
|
if (5 + nLenR >= l)
|
|
throw new Error("Non-canonical signature: S length misplaced");
|
|
|
|
var nLenS = sig[5 + nLenR];
|
|
if ((nLenR + nLenS + 7) !== l)
|
|
throw new Error("Non-canonical signature: R+S length mismatch");
|
|
|
|
var rPos = 4;
|
|
var R = new Buffer(nLenR);
|
|
sig.copy(R, 0, rPos, rPos + nLenR);
|
|
if (sig[rPos - 2] !== 0x02)
|
|
throw new Error("Non-canonical signature: R value type mismatch");
|
|
if (nLenR == 0)
|
|
throw new Error("Non-canonical signature: R length is zero");
|
|
if (R[0] & 0x80)
|
|
throw new Error("Non-canonical signature: R value negative");
|
|
if (nLenR > 1 && (R[0] == 0x00) && !(R[1] & 0x80))
|
|
throw new Error("Non-canonical signature: R value excessively padded");
|
|
|
|
var sPos = 6 + nLenR;
|
|
var S = new Buffer(nLenS);
|
|
sig.copy(S, 0, sPos, sPos + nLenS);
|
|
if (sig[sPos - 2] != 0x02)
|
|
throw new Error("Non-canonical signature: S value type mismatch");
|
|
if (nLenS == 0)
|
|
throw new Error("Non-canonical signature: S length is zero");
|
|
if (S[0] & 0x80)
|
|
throw new Error("Non-canonical signature: S value negative");
|
|
if (nLenS > 1 && (S[0] == 0x00) && !(S[1] & 0x80))
|
|
throw new Error("Non-canonical signature: S value excessively padded");
|
|
|
|
if (this.opts.verifyEvenS) {
|
|
if (S[nLenS - 1] & 1)
|
|
throw new Error("Non-canonical signature: S value odd");
|
|
}
|
|
return true;
|
|
};
|
|
|
|
module.exports = require('soop')(ScriptInterpreter);
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"),require("buffer").Buffer)
|
|
},{"../config":"4itQ50","../util":140,"../util/log":"AdF7pF","./Key":"ALJ4PS","./Opcode":"Zm7/h9","./Script":"hQ0t76","/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"bignum":58,"buffer":80,"buffertools":"fugeBw","soop":125}],"./lib/Sign":[function(require,module,exports){
|
|
module.exports=require('V3JdDp');
|
|
},{}],"V3JdDp":[function(require,module,exports){
|
|
(function (Buffer){
|
|
|
|
function signOne(hash, addrStr, keys)
|
|
{
|
|
var keyObj = keys[addrStr];
|
|
var rawPrivKey = new Buffer(keyObj.priv, 'hex');
|
|
var key = new KeyModule.Key();
|
|
key.private = rawPrivKey;
|
|
var signature = key.signSync(hash);
|
|
|
|
return signature;
|
|
}
|
|
|
|
function signTxIn(nIn, tx, txInputs, network, keys, scripts)
|
|
{
|
|
// locate TX input needing a signature
|
|
var txin = tx.ins[nIn];
|
|
var scriptSig = txin.getScript();
|
|
|
|
// locate TX output, within txInputs
|
|
var txoutHash = txin.getOutpointHash();
|
|
if (!(txoutHash in txInputs))
|
|
throw new Error("signTxIn missing input hash");
|
|
var txFrom = txInputs[txoutHash];
|
|
var txoutIndex = txin.getOutpointIndex();
|
|
if (txFrom.outs.length >= txoutIndex)
|
|
throw new Error("signTxIn missing input index");
|
|
var txout = txFrom.outs[txoutIndex];
|
|
var scriptPubKey = txout.getScript();
|
|
|
|
// detect type of transaction, and extract useful elements
|
|
var txType = scriptPubKey.classify();
|
|
if (txType == TX_UNKNOWN)
|
|
throw new Error("unknown TX type");
|
|
var scriptData = scriptPubKey.capture();
|
|
|
|
// if P2SH, lookup the script
|
|
var subscriptRaw = undefined;
|
|
var subscript = undefined;
|
|
var subType = undefined;
|
|
var subData = undefined;
|
|
if (txType == TX_SCRIPTHASH) {
|
|
var addr = new Address(network.P2SHVersion, scriptData[0]);
|
|
var addrStr = addr.toString();
|
|
if (!(addrStr in scripts))
|
|
throw new Error("unknown script hash address");
|
|
|
|
subscriptRaw = new Buffer(scripts[addrStr], 'hex');
|
|
subscript = new Script(subscriptRaw);
|
|
subType = subscript.classify();
|
|
if (subType == TX_UNKNOWN)
|
|
throw new Error("unknown subscript TX type");
|
|
subData = subscript.capture();
|
|
}
|
|
|
|
var hash = tx.hashForSignature(scriptPubKey, i, 0);
|
|
|
|
switch (txType) {
|
|
case TX_PUBKEY:
|
|
// already signed
|
|
if (scriptSig.chunks.length > 0)
|
|
return;
|
|
|
|
var pubkeyhash = util.sha256ripe160(scriptData[0]);
|
|
var addr = new Address(network.addressVersion, pubkeyhash);
|
|
var addrStr = addr.toString();
|
|
if (!(addrStr in keys))
|
|
throw new Error("unknown pubkey");
|
|
|
|
var signature = signOne(hash, addrStr, keys);
|
|
scriptSig.writeBytes(signature);
|
|
break;
|
|
|
|
case TX_PUBKEYHASH:
|
|
// already signed
|
|
if (scriptSig.chunks.length > 0)
|
|
return;
|
|
|
|
var addr = new Address(network.addressVersion, scriptData[0]);
|
|
var addrStr = addr.toString();
|
|
if (!(addrStr in keys))
|
|
throw new Error("unknown pubkey hash address");
|
|
|
|
var signature = signOne(hash, addrStr, keys);
|
|
scriptSig.writeBytes(signature);
|
|
scriptSig.writeBytes(key.public);
|
|
break;
|
|
|
|
case TX_SCRIPTHASH:
|
|
// already signed
|
|
if (scriptSig.chunks.length > 0)
|
|
return;
|
|
|
|
var addr = new Address(network.addressVersion, subData[0]);
|
|
var addrStr = addr.toString();
|
|
if (!(addrStr in keys))
|
|
throw new Error("unknown script(pubkey hash) address");
|
|
|
|
var signature = signOne(hash, addrStr, keys);
|
|
scriptSig.writeBytes(signature);
|
|
scriptSig.writeBytes(key.public);
|
|
break;
|
|
|
|
case TX_MULTISIG:
|
|
while (scriptSig.chunks.length < scriptData.length) {
|
|
scriptSig.writeBytes(util.EMPTY_BUFFER);
|
|
}
|
|
for (var i = 0; i < scriptData.length; i++) {
|
|
// skip already signed
|
|
if (scriptSig.chunks[i].length > 0)
|
|
continue;
|
|
|
|
var pubkeyhash = util.sha256ripe160(scriptSig.chunks[i]);
|
|
var addr = new Address(network.addressVersion, pubkeyhash);
|
|
var addrStr = addr.toString();
|
|
if (!(addrStr in keys))
|
|
continue;
|
|
|
|
var signature = signOne(hash, addrStr, keys);
|
|
scriptSig.chunks[i] = signature;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (txtype == TX_SCRIPTHASH)
|
|
scriptSig.writeBytes(subscriptRaw);
|
|
}
|
|
|
|
exports.Transaction = function Transaction(tx, txInputs, network, keys, scripts)
|
|
{
|
|
for (var i = 0; i < tx.ins.length; i++)
|
|
signTxIn(i, tx, txInputs, network, keys, scripts);
|
|
};
|
|
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80}],"./lib/Transaction":[function(require,module,exports){
|
|
module.exports=require('LJhYtm');
|
|
},{}],"LJhYtm":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
var config = imports.config || require('../config');
|
|
var log = imports.log || require('../util/log');
|
|
var Address = imports.Address || require('./Address');
|
|
var Script = imports.Script || require('./Script');
|
|
var ScriptInterpreter = imports.ScriptInterpreter || require('./ScriptInterpreter');
|
|
var util = imports.util || require('../util');
|
|
var bignum = imports.bignum || require('bignum');
|
|
var Put = imports.Put || require('bufferput');
|
|
var Parser = imports.Parser || require('../util/BinaryParser');
|
|
var Step = imports.Step || require('step');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
var error = imports.error || require('../util/error');
|
|
var networks = imports.networks || require('../networks');
|
|
var WalletKey = imports.WalletKey || require('./WalletKey');
|
|
var PrivateKey = imports.PrivateKey || require('./PrivateKey');
|
|
|
|
var COINBASE_OP = Buffer.concat([util.NULL_HASH, new Buffer('FFFFFFFF', 'hex')]);
|
|
var FEE_PER_1000B_SAT = parseInt(0.0001 * util.COIN);
|
|
|
|
Transaction.COINBASE_OP = COINBASE_OP;
|
|
|
|
function TransactionIn(data) {
|
|
if ("object" !== typeof data) {
|
|
data = {};
|
|
}
|
|
if (data.o) {
|
|
this.o = data.o;
|
|
} else {
|
|
if (data.oTxHash && typeof data.oIndex !== 'undefined' && data.oIndex >= 0) {
|
|
var hash = new Buffer(data.oTxHash, 'hex');
|
|
hash = buffertools.reverse(hash);
|
|
var voutBuf = new Buffer(4);
|
|
voutBuf.writeUInt32LE(data.oIndex, 0);
|
|
this.o = Buffer.concat([hash, voutBuf]);
|
|
}
|
|
}
|
|
this.s = Buffer.isBuffer(data.s) ? data.s :
|
|
Buffer.isBuffer(data.script) ? data.script : util.EMPTY_BUFFER;
|
|
this.q = data.q ? data.q : data.sequence;
|
|
}
|
|
|
|
TransactionIn.MAX_SEQUENCE = 0xffffffff;
|
|
|
|
TransactionIn.prototype.getScript = function getScript() {
|
|
return new Script(this.s);
|
|
};
|
|
|
|
TransactionIn.prototype.isCoinBase = function isCoinBase() {
|
|
if (!this.o) return false;
|
|
|
|
//The new Buffer is for Firefox compatibility
|
|
return buffertools.compare(new Buffer(this.o), COINBASE_OP) === 0;
|
|
};
|
|
|
|
TransactionIn.prototype.serialize = function serialize() {
|
|
var slen = util.varIntBuf(this.s.length);
|
|
var qbuf = new Buffer(4);
|
|
qbuf.writeUInt32LE(this.q, 0);
|
|
|
|
var ret = Buffer.concat([this.o, slen, this.s, qbuf]);
|
|
return ret;
|
|
};
|
|
|
|
TransactionIn.prototype.getOutpointHash = function getOutpointHash() {
|
|
if ("undefined" !== typeof this.o.outHashCache) {
|
|
return this.o.outHashCache;
|
|
}
|
|
return this.o.outHashCache = this.o.slice(0, 32);
|
|
};
|
|
|
|
TransactionIn.prototype.getOutpointIndex = function getOutpointIndex() {
|
|
return (this.o[32]) +
|
|
(this.o[33] << 8) +
|
|
(this.o[34] << 16) +
|
|
(this.o[35] << 24);
|
|
};
|
|
|
|
TransactionIn.prototype.setOutpointIndex = function setOutpointIndex(n) {
|
|
this.o[32] = n & 0xff;
|
|
this.o[33] = n >> 8 & 0xff;
|
|
this.o[34] = n >> 16 & 0xff;
|
|
this.o[35] = n >> 24 & 0xff;
|
|
};
|
|
|
|
|
|
function TransactionOut(data) {
|
|
if ("object" !== typeof data) {
|
|
data = {};
|
|
}
|
|
this.v = data.v ? data.v : data.value;
|
|
this.s = data.s ? data.s : data.script;
|
|
};
|
|
|
|
TransactionOut.prototype.getValue = function getValue() {
|
|
return new Parser(this.v).word64lu();
|
|
};
|
|
|
|
TransactionOut.prototype.getScript = function getScript() {
|
|
return new Script(this.s);
|
|
};
|
|
|
|
TransactionOut.prototype.serialize = function serialize() {
|
|
var slen = util.varIntBuf(this.s.length);
|
|
return Buffer.concat([this.v, slen, this.s]);
|
|
};
|
|
|
|
function Transaction(data) {
|
|
if ("object" !== typeof data) {
|
|
data = {};
|
|
}
|
|
this.hash = data.hash || null;
|
|
this.version = data.version;
|
|
this.lock_time = data.lock_time;
|
|
this.ins = Array.isArray(data.ins) ? data.ins.map(function(data) {
|
|
var txin = new TransactionIn();
|
|
txin.s = data.s;
|
|
txin.q = data.q;
|
|
txin.o = data.o;
|
|
return txin;
|
|
}) : [];
|
|
this.outs = Array.isArray(data.outs) ? data.outs.map(function(data) {
|
|
var txout = new TransactionOut();
|
|
txout.v = data.v;
|
|
txout.s = data.s;
|
|
return txout;
|
|
}) : [];
|
|
if (data.buffer) this._buffer = data.buffer;
|
|
};
|
|
Transaction.In = TransactionIn;
|
|
Transaction.Out = TransactionOut;
|
|
|
|
Transaction.prototype.isCoinBase = function() {
|
|
return this.ins.length == 1 && this.ins[0].isCoinBase();
|
|
};
|
|
|
|
Transaction.prototype.isStandard = function isStandard() {
|
|
var i;
|
|
for (i = 0; i < this.ins.length; i++) {
|
|
if (this.ins[i].getScript().getInType() == "Strange") {
|
|
return false;
|
|
}
|
|
}
|
|
for (i = 0; i < this.outs.length; i++) {
|
|
if (this.outs[i].getScript().getOutType() == "Strange") {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
Transaction.prototype.serialize = function serialize() {
|
|
var bufs = [];
|
|
|
|
var buf = new Buffer(4);
|
|
buf.writeUInt32LE(this.version, 0);
|
|
bufs.push(buf);
|
|
|
|
bufs.push(util.varIntBuf(this.ins.length));
|
|
this.ins.forEach(function(txin) {
|
|
bufs.push(txin.serialize());
|
|
});
|
|
|
|
bufs.push(util.varIntBuf(this.outs.length));
|
|
this.outs.forEach(function(txout) {
|
|
bufs.push(txout.serialize());
|
|
});
|
|
|
|
var buf = new Buffer(4);
|
|
buf.writeUInt32LE(this.lock_time, 0);
|
|
bufs.push(buf);
|
|
|
|
this._buffer = Buffer.concat(bufs);
|
|
return this._buffer;
|
|
};
|
|
|
|
Transaction.prototype.getBuffer = function getBuffer() {
|
|
if (this._buffer) return this._buffer;
|
|
|
|
return this.serialize();
|
|
};
|
|
|
|
Transaction.prototype.calcHash = function calcHash() {
|
|
this.hash = util.twoSha256(this.getBuffer());
|
|
return this.hash;
|
|
};
|
|
|
|
Transaction.prototype.checkHash = function checkHash() {
|
|
if (!this.hash || !this.hash.length) return false;
|
|
|
|
return buffertools.compare(this.calcHash(), this.hash) === 0;
|
|
};
|
|
|
|
Transaction.prototype.getHash = function getHash() {
|
|
if (!this.hash || !this.hash.length) {
|
|
this.hash = this.calcHash();
|
|
}
|
|
return this.hash;
|
|
};
|
|
|
|
|
|
Transaction.prototype.calcNormalizedHash = function () {
|
|
this.normalizedHash = this.hashForSignature(new Script(),0, SIGHASH_ALL);
|
|
return this.normalizedHash;
|
|
};
|
|
|
|
|
|
Transaction.prototype.getNormalizedHash = function () {
|
|
if (!this.normalizedHash || !this.normalizedHash.length) {
|
|
this.normalizedHash = this.calcNormalizedHash();
|
|
}
|
|
return this.normalizedHash;
|
|
};
|
|
|
|
|
|
|
|
// convert encoded list of inputs to easy-to-use JS list-of-lists
|
|
Transaction.prototype.inputs = function inputs() {
|
|
var res = [];
|
|
for (var i = 0; i < this.ins.length; i++) {
|
|
var txin = this.ins[i];
|
|
var outHash = txin.getOutpointHash();
|
|
var outIndex = txin.getOutpointIndex();
|
|
res.push([outHash, outIndex]);
|
|
}
|
|
|
|
return res;
|
|
};
|
|
|
|
Transaction.prototype.verifyInput = function verifyInput(n, scriptPubKey, opts, callback) {
|
|
var scriptSig = this.ins[n].getScript();
|
|
return ScriptInterpreter.verifyFull(
|
|
scriptSig,
|
|
scriptPubKey,
|
|
this, n, 0,
|
|
opts,
|
|
callback);
|
|
};
|
|
|
|
/**
|
|
* Returns an object containing all pubkey hashes affected by this transaction.
|
|
*
|
|
* The return object contains the base64-encoded pubKeyHash values as keys
|
|
* and the original pubKeyHash buffers as values.
|
|
*/
|
|
Transaction.prototype.getAffectedKeys = function getAffectedKeys(txCache) {
|
|
// TODO: Function won't consider results cached if there are no affected
|
|
// accounts.
|
|
if (!(this.affects && this.affects.length)) {
|
|
this.affects = [];
|
|
|
|
// Index any pubkeys affected by the outputs of this transaction
|
|
for (var i = 0, l = this.outs.length; i < l; i++) {
|
|
var txout = this.outs[i];
|
|
var script = txout.getScript();
|
|
|
|
var outPubKey = script.simpleOutPubKeyHash();
|
|
if (outPubKey) {
|
|
this.affects.push(outPubKey);
|
|
}
|
|
};
|
|
|
|
// Index any pubkeys affected by the inputs of this transaction
|
|
var txIndex = txCache.txIndex;
|
|
for (var i = 0, l = this.ins.length; i < l; i++) {
|
|
var txin = this.ins[i];
|
|
|
|
if (txin.isCoinBase()) continue;
|
|
|
|
// In the case of coinbase or IP transactions, the txin doesn't
|
|
// actually contain the pubkey, so we look at the referenced txout
|
|
// instead.
|
|
var outHash = txin.getOutpointHash();
|
|
var outIndex = txin.getOutpointIndex();
|
|
var outHashBase64 = outHash.toString('base64');
|
|
var fromTxOuts = txIndex[outHashBase64];
|
|
|
|
if (!fromTxOuts) {
|
|
throw new Error("Input not found!");
|
|
}
|
|
|
|
var txout = fromTxOuts[outIndex];
|
|
var script = txout.getScript();
|
|
|
|
var outPubKey = script.simpleOutPubKeyHash();
|
|
if (outPubKey) {
|
|
this.affects.push(outPubKey);
|
|
}
|
|
}
|
|
}
|
|
|
|
var affectedKeys = {};
|
|
|
|
this.affects.forEach(function(pubKeyHash) {
|
|
affectedKeys[pubKeyHash.toString('base64')] = pubKeyHash;
|
|
});
|
|
|
|
return affectedKeys;
|
|
};
|
|
|
|
var OP_CODESEPARATOR = 171;
|
|
|
|
var SIGHASH_ALL = 1;
|
|
var SIGHASH_NONE = 2;
|
|
var SIGHASH_SINGLE = 3;
|
|
var SIGHASH_ANYONECANPAY = 0x80;
|
|
|
|
Transaction.SIGHASH_ALL = SIGHASH_ALL;
|
|
Transaction.SIGHASH_NONE = SIGHASH_NONE;
|
|
Transaction.SIGHASH_SINGLE = SIGHASH_SINGLE;
|
|
Transaction.SIGHASH_ANYONECANPAY = SIGHASH_ANYONECANPAY;
|
|
|
|
var TransactionSignatureSerializer = function(txTo, scriptCode, nIn, nHashType) {
|
|
this.txTo = txTo;
|
|
this.scriptCode = scriptCode;
|
|
this.nIn = nIn;
|
|
this.anyoneCanPay = !!(nHashType & SIGHASH_ANYONECANPAY);
|
|
var hashTypeMode = nHashType & 0x1f;
|
|
this.hashSingle = hashTypeMode === SIGHASH_SINGLE;
|
|
this.hashNone = hashTypeMode === SIGHASH_NONE;
|
|
this.bytes = new Put();
|
|
};
|
|
|
|
// serialize an output of txTo
|
|
TransactionSignatureSerializer.prototype.serializeOutput = function(nOutput) {
|
|
if (this.hashSingle && nOutput != this.nIn) {
|
|
// Do not lock-in the txout payee at other indices as txin
|
|
// ::Serialize(s, CTxOut(), nType, nVersion);
|
|
this.bytes.put(util.INT64_MAX);
|
|
this.bytes.varint(0);
|
|
} else {
|
|
//::Serialize(s, txTo.vout[nOutput], nType, nVersion);
|
|
var out = this.txTo.outs[nOutput];
|
|
this.bytes.put(out.v);
|
|
this.bytes.varint(out.s.length);
|
|
this.bytes.put(out.s);
|
|
}
|
|
};
|
|
|
|
// serialize the script
|
|
TransactionSignatureSerializer.prototype.serializeScriptCode = function() {
|
|
this.scriptCode.findAndDelete(OP_CODESEPARATOR);
|
|
this.bytes.varint(this.scriptCode.buffer.length);
|
|
this.bytes.put(this.scriptCode.buffer);
|
|
};
|
|
|
|
// serialize an input of txTo
|
|
TransactionSignatureSerializer.prototype.serializeInput = function(nInput) {
|
|
// In case of SIGHASH_ANYONECANPAY, only the input being signed is serialized
|
|
if (this.anyoneCanPay) nInput = this.nIn;
|
|
|
|
// Serialize the prevout
|
|
this.bytes.put(this.txTo.ins[nInput].o);
|
|
|
|
// Serialize the script
|
|
if (nInput !== this.nIn) {
|
|
// Blank out other inputs' signatures
|
|
this.bytes.varint(0);
|
|
} else {
|
|
this.serializeScriptCode();
|
|
}
|
|
// Serialize the nSequence
|
|
if (nInput !== this.nIn && (this.hashSingle || this.hashNone)) {
|
|
// let the others update at will
|
|
this.bytes.word32le(0);
|
|
} else {
|
|
this.bytes.word32le(this.txTo.ins[nInput].q);
|
|
}
|
|
|
|
};
|
|
|
|
|
|
// serialize txTo for signature
|
|
TransactionSignatureSerializer.prototype.serialize = function() {
|
|
// serialize nVersion
|
|
this.bytes.word32le(this.txTo.version);
|
|
// serialize vin
|
|
var nInputs = this.anyoneCanPay ? 1 : this.txTo.ins.length;
|
|
this.bytes.varint(nInputs);
|
|
for (var nInput = 0; nInput < nInputs; nInput++) {
|
|
this.serializeInput(nInput);
|
|
}
|
|
// serialize vout
|
|
var nOutputs = this.hashNone ? 0 : (this.hashSingle ? this.nIn + 1 : this.txTo.outs.length);
|
|
this.bytes.varint(nOutputs);
|
|
for (var nOutput = 0; nOutput < nOutputs; nOutput++) {
|
|
this.serializeOutput(nOutput);
|
|
}
|
|
|
|
// serialize nLockTime
|
|
this.bytes.word32le(this.txTo.lock_time);
|
|
};
|
|
|
|
TransactionSignatureSerializer.prototype.buffer = function() {
|
|
this.serialize();
|
|
return this.bytes.buffer();
|
|
};
|
|
|
|
Transaction.Serializer = TransactionSignatureSerializer;
|
|
|
|
var oneBuffer = function() {
|
|
// bug present in bitcoind which must be also present in bitcore
|
|
// see https://bitcointalk.org/index.php?topic=260595
|
|
var ret = new Buffer(32);
|
|
ret.writeUInt8(1, 0);
|
|
for (var i=1; i<32; i++) ret.writeUInt8(0, i);
|
|
return ret; // return 1 bug
|
|
};
|
|
|
|
Transaction.prototype.hashForSignature =
|
|
function hashForSignature(script, inIndex, hashType) {
|
|
|
|
if (+inIndex !== inIndex ||
|
|
inIndex < 0 || inIndex >= this.ins.length) {
|
|
return oneBuffer();
|
|
}
|
|
// Check for invalid use of SIGHASH_SINGLE
|
|
var hashTypeMode = hashType & 0x1f;
|
|
if (hashTypeMode === SIGHASH_SINGLE) {
|
|
if (inIndex >= this.outs.length) {
|
|
return oneBuffer();
|
|
}
|
|
}
|
|
|
|
// Wrapper to serialize only the necessary parts of the transaction being signed
|
|
var serializer = new TransactionSignatureSerializer(this, script, inIndex, hashType);
|
|
// Serialize
|
|
var buffer = serializer.buffer();
|
|
// Append hashType
|
|
var hashBuf = new Put().word32le(hashType).buffer();
|
|
buffer = Buffer.concat([buffer, hashBuf]);
|
|
return util.twoSha256(buffer);
|
|
};
|
|
|
|
/**
|
|
* Returns an object with the same field names as jgarzik's getblock patch.
|
|
*/
|
|
Transaction.prototype.getStandardizedObject = function getStandardizedObject() {
|
|
var tx = {
|
|
hash: util.formatHashFull(this.getHash()),
|
|
version: this.version,
|
|
lock_time: this.lock_time
|
|
};
|
|
|
|
var totalSize = 8; // version + lock_time
|
|
totalSize += util.getVarIntSize(this.ins.length); // tx_in count
|
|
var ins = this.ins.map(function(txin) {
|
|
var txinObj = {
|
|
prev_out: {
|
|
hash: buffertools.reverse(new Buffer(txin.getOutpointHash())).toString('hex'),
|
|
n: txin.getOutpointIndex()
|
|
},
|
|
sequence: txin.q
|
|
};
|
|
if (txin.isCoinBase()) {
|
|
txinObj.coinbase = txin.s.toString('hex');
|
|
} else {
|
|
txinObj.scriptSig = new Script(txin.s).getStringContent(false, 0);
|
|
}
|
|
totalSize += 36 + util.getVarIntSize(txin.s.length) +
|
|
txin.s.length + 4; // outpoint + script_len + script + sequence
|
|
return txinObj;
|
|
});
|
|
|
|
totalSize += util.getVarIntSize(this.outs.length);
|
|
var outs = this.outs.map(function(txout) {
|
|
totalSize += util.getVarIntSize(txout.s.length) +
|
|
txout.s.length + 8; // script_len + script + value
|
|
return {
|
|
value: util.formatValue(txout.v),
|
|
scriptPubKey: new Script(txout.s).getStringContent(false, 0)
|
|
};
|
|
});
|
|
|
|
tx.size = totalSize;
|
|
|
|
tx["in"] = ins;
|
|
tx["out"] = outs;
|
|
|
|
return tx;
|
|
};
|
|
|
|
// Add some Mongoose compatibility functions to the plain object
|
|
Transaction.prototype.toObject = function toObject() {
|
|
return this;
|
|
};
|
|
|
|
Transaction.prototype.fromObj = function fromObj(obj) {
|
|
var txobj = {};
|
|
txobj.version = obj.version || 1;
|
|
txobj.lock_time = obj.lock_time || 0;
|
|
txobj.ins = [];
|
|
txobj.outs = [];
|
|
|
|
obj.inputs.forEach(function(inputobj) {
|
|
var txin = new TransactionIn();
|
|
txin.s = util.EMPTY_BUFFER;
|
|
txin.q = 0xffffffff;
|
|
|
|
var hash = new Buffer(inputobj.txid, 'hex');
|
|
hash = buffertools.reverse(hash);
|
|
var vout = parseInt(inputobj.vout);
|
|
var voutBuf = new Buffer(4);
|
|
voutBuf.writeUInt32LE(vout, 0);
|
|
|
|
txin.o = Buffer.concat([hash, voutBuf]);
|
|
|
|
txobj.ins.push(txin);
|
|
});
|
|
|
|
var keys = Object.keys(obj.outputs);
|
|
keys.forEach(function(addrStr) {
|
|
var addr = new Address(addrStr);
|
|
var script = Script.createPubKeyHashOut(addr.payload());
|
|
|
|
var valueNum = bignum(obj.outputs[addrStr]);
|
|
var value = util.bigIntToValue(valueNum);
|
|
|
|
var txout = new TransactionOut();
|
|
txout.v = value;
|
|
txout.s = script.getBuffer();
|
|
|
|
txobj.outs.push(txout);
|
|
});
|
|
|
|
this.lock_time = txobj.lock_time;
|
|
this.version = txobj.version;
|
|
this.ins = txobj.ins;
|
|
this.outs = txobj.outs;
|
|
}
|
|
|
|
Transaction.prototype.parse = function(parser) {
|
|
if (Buffer.isBuffer(parser)) {
|
|
this._buffer = parser;
|
|
parser = new Parser(parser);
|
|
}
|
|
|
|
var i, sLen, startPos = parser.pos;
|
|
|
|
this.version = parser.word32le();
|
|
|
|
var txinCount = parser.varInt();
|
|
|
|
this.ins = [];
|
|
for (j = 0; j < txinCount; j++) {
|
|
var txin = new TransactionIn();
|
|
txin.o = parser.buffer(36); // outpoint
|
|
sLen = parser.varInt(); // script_len
|
|
txin.s = parser.buffer(sLen); // script
|
|
txin.q = parser.word32le(); // sequence
|
|
this.ins.push(txin);
|
|
}
|
|
|
|
var txoutCount = parser.varInt();
|
|
|
|
this.outs = [];
|
|
for (j = 0; j < txoutCount; j++) {
|
|
var txout = new TransactionOut();
|
|
txout.v = parser.buffer(8); // value
|
|
sLen = parser.varInt(); // script_len
|
|
txout.s = parser.buffer(sLen); // script
|
|
this.outs.push(txout);
|
|
}
|
|
|
|
this.lock_time = parser.word32le();
|
|
this.calcHash();
|
|
};
|
|
|
|
|
|
|
|
|
|
Transaction.prototype.calcSize = function() {
|
|
var totalSize = 8; // version + lock_time
|
|
totalSize += util.getVarIntSize(this.ins.length); // tx_in count
|
|
this.ins.forEach(function(txin) {
|
|
totalSize += 36 + util.getVarIntSize(txin.s.length) +
|
|
txin.s.length + 4; // outpoint + script_len + script + sequence
|
|
});
|
|
|
|
totalSize += util.getVarIntSize(this.outs.length);
|
|
this.outs.forEach(function(txout) {
|
|
totalSize += util.getVarIntSize(txout.s.length) +
|
|
txout.s.length + 8; // script_len + script + value
|
|
});
|
|
this.size = totalSize;
|
|
return totalSize;
|
|
};
|
|
|
|
Transaction.prototype.getSize = function () {
|
|
if (!this.size) {
|
|
this.size = this.calcSize();
|
|
}
|
|
return this.size;
|
|
};
|
|
|
|
Transaction.prototype.countInputSignatures = function(index) {
|
|
var ret = 0;
|
|
var script = new Script(this.ins[index].s);
|
|
return script.countSignatures();
|
|
};
|
|
|
|
// Works on p2pubkey, p2pubkeyhash & p2sh (no normal multisig)
|
|
Transaction.prototype.countInputMissingSignatures = function(index) {
|
|
var ret = 0;
|
|
var script = new Script(this.ins[index].s);
|
|
return script.countMissingSignatures();
|
|
};
|
|
|
|
// Works on p2pubkey, p2pubkeyhash & p2sh (no normal multisig)
|
|
Transaction.prototype.isInputComplete = function(index) {
|
|
var m = this.countInputMissingSignatures(index);
|
|
if (m===null) return null;
|
|
return m === 0;
|
|
};
|
|
|
|
// Works on p2pubkey, p2pubkeyhash & p2sh (no normal multisig)
|
|
Transaction.prototype.isComplete = function() {
|
|
var ret = true;
|
|
var l = this.ins.length;
|
|
|
|
for (var i = 0; i < l; i++) {
|
|
if (!this.isInputComplete(i)){
|
|
ret = false;
|
|
break;
|
|
}
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
|
|
module.exports = require('soop')(Transaction);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../config":"4itQ50","../networks":"ULNIu2","../util":140,"../util/BinaryParser":"b3ZSD7","../util/error":139,"../util/log":"AdF7pF","./Address":"G+CcXD","./PrivateKey":"izTl9z","./Script":"hQ0t76","./ScriptInterpreter":"Q/ZWXW","./WalletKey":"wWje7g","bignum":58,"buffer":80,"bufferput":"aXRuS6","buffertools":"fugeBw","soop":125,"step":126}],"./lib/TransactionBuilder":[function(require,module,exports){
|
|
module.exports=require('D1Ge6m');
|
|
},{}],"D1Ge6m":[function(require,module,exports){
|
|
(function (Buffer){
|
|
// TransactionBuilder
|
|
// ==================
|
|
//
|
|
// Creates a bitcore Transaction object
|
|
//
|
|
//
|
|
// Synopsis
|
|
// --------
|
|
// ```
|
|
// var tx = (new TransactionBuilder(opts))
|
|
// .setUnspent(utxos)
|
|
// .setOutputs(outs)
|
|
// .sign(keys)
|
|
// .build();
|
|
//
|
|
//
|
|
// var builder = (new TransactionBuilder(opts))
|
|
// .setUnspent(spent)
|
|
// .setOutputs(outs);
|
|
//
|
|
// // Uncomplete tx (no signed or partially signed)
|
|
// var tx = builder.build();
|
|
//
|
|
// ..later..
|
|
//
|
|
// builder.sign(keys);
|
|
// while ( builder.isFullySigned() ) {
|
|
//
|
|
// ... get new keys ...
|
|
//
|
|
// builder.sign(keys);
|
|
// }
|
|
//
|
|
// var tx = builder.build();
|
|
// broadcast(tx.serialize());
|
|
//
|
|
// //Serialize it and pass it around...
|
|
// var string = JSON.stringify(builder.toObj());
|
|
// // then...
|
|
// var builder = TransactionBuilder.fromObj(JSON.parse(str);
|
|
// builder.sign(keys);
|
|
// // Also
|
|
// var builder2 = TransactionBuilder.fromObj(JSON.parse(str2);
|
|
// builder2.merge(builder); // Will merge signatures for p2sh mulsig txs.
|
|
//
|
|
//
|
|
// ```
|
|
//
|
|
//
|
|
//
|
|
|
|
|
|
'use strict';
|
|
|
|
var imports = require('soop').imports();
|
|
var Address = imports.Address || require('./Address');
|
|
var Script = imports.Script || require('./Script');
|
|
var util = imports.util || require('../util');
|
|
var bignum = imports.bignum || require('bignum');
|
|
var buffertools = imports.buffertools || require('buffertools');
|
|
var networks = imports.networks || require('../networks');
|
|
var WalletKey = imports.WalletKey || require('./WalletKey');
|
|
var PrivateKey = imports.PrivateKey || require('./PrivateKey');
|
|
var Key = imports.Key || require('./Key');
|
|
var log = imports.log || require('../util/log');
|
|
|
|
var Transaction = imports.Transaction || require('./Transaction');
|
|
var FEE_PER_1000B_SAT = parseInt(0.0001 * util.COIN);
|
|
|
|
// Methods
|
|
// -------
|
|
//
|
|
// TransactionBuilder
|
|
// ------------------
|
|
// Creates a TransactionBuilder instance
|
|
// `opts`
|
|
// ```
|
|
// {
|
|
// remainderOut: null,
|
|
// fee: 0.001,
|
|
// lockTime: null,
|
|
// spendUnconfirmed: false,
|
|
// signhash: SIGHASH_ALL
|
|
// }
|
|
// ```
|
|
// Amounts are in BTC. instead of fee and amount; feeSat and amountSat can be given,
|
|
// repectively, to provide amounts in satoshis.
|
|
//
|
|
// If no remainderOut is given, and there are remainder coins, the
|
|
// first IN out will be used to return the coins. remainderOut has the form:
|
|
// ```
|
|
// remainderOut = { address: 1xxxxx}
|
|
// ```
|
|
// or
|
|
// ```
|
|
// remainderOut = { pubkeys: ['hex1','hex2',...} for multisig
|
|
// ```
|
|
|
|
function TransactionBuilder(opts) {
|
|
opts = opts || {};
|
|
this.lockTime = opts.lockTime || 0;
|
|
this.spendUnconfirmed = opts.spendUnconfirmed || false;
|
|
|
|
if (opts.fee || opts.feeSat) {
|
|
this.givenFeeSat = opts.fee ? opts.fee * util.COIN : opts.feeSat;
|
|
}
|
|
this.remainderOut = opts.remainderOut;
|
|
this.signhash = opts.signhash || Transaction.SIGHASH_ALL;
|
|
|
|
this.tx = {};
|
|
this.inputsSigned= 0;
|
|
this.signaturesAdded= 0;
|
|
|
|
return this;
|
|
}
|
|
|
|
/*
|
|
* scriptForAddress
|
|
*
|
|
* Returns a scriptPubKey for the given address type
|
|
*/
|
|
|
|
TransactionBuilder.scriptForAddress = function(addressString) {
|
|
|
|
var livenet = networks.livenet;
|
|
var testnet = networks.testnet;
|
|
var address = new Address(addressString);
|
|
|
|
var version = address.version();
|
|
var script;
|
|
if (version === livenet.addressVersion || version === testnet.addressVersion)
|
|
script = Script.createPubKeyHashOut(address.payload());
|
|
else if (version === livenet.P2SHVersion || version === testnet.P2SHVersion)
|
|
script = Script.createP2SH(address.payload());
|
|
else
|
|
throw new Error('invalid output address');
|
|
|
|
return script;
|
|
};
|
|
|
|
|
|
TransactionBuilder._scriptForPubkeys = function(out) {
|
|
|
|
var l = out.pubkeys.length;
|
|
var pubKeyBuf=[];
|
|
|
|
for (var i=0; i<l; i++) {
|
|
pubKeyBuf.push(new Buffer(out.pubkeys[i],'hex'));
|
|
}
|
|
|
|
return Script.createMultisig(out.nreq, pubKeyBuf);
|
|
};
|
|
|
|
TransactionBuilder._scriptForOut = function(out) {
|
|
var ret;
|
|
if (out.address)
|
|
ret = this.scriptForAddress(out.address);
|
|
else if (out.pubkeys || out.nreq || out.nreq > 1)
|
|
ret = this._scriptForPubkeys(out);
|
|
else
|
|
throw new Error('unknown out type');
|
|
|
|
return ret;
|
|
};
|
|
|
|
|
|
TransactionBuilder.infoForP2sh = function(opts, networkName) {
|
|
var script = this._scriptForOut(opts);
|
|
var hash = util.sha256ripe160(script.getBuffer());
|
|
|
|
var version = networkName === 'testnet' ?
|
|
networks.testnet.P2SHVersion : networks.livenet.P2SHVersion;
|
|
|
|
var addr = new Address(version, hash);
|
|
var addrStr = addr.as('base58');
|
|
return {
|
|
script: script,
|
|
scriptBufHex: script.getBuffer().toString('hex'),
|
|
hash: hash,
|
|
address: addrStr,
|
|
};
|
|
};
|
|
|
|
// setUnspent
|
|
// ----------
|
|
// Sets the `unspent` available for the transaction. Some (or all)
|
|
// of them to fullfil the transaction's outputs and fee.
|
|
// The expected format is:
|
|
// ```
|
|
// [{
|
|
// address: "mqSjTad2TKbPcKQ3Jq4kgCkKatyN44UMgZ",
|
|
// txid: "2ac165fa7a3a2b535d106a0041c7568d03b531e58aeccdd3199d7289ab12cfc1",
|
|
// scriptPubKey: "76a9146ce4e1163eb18939b1440c42844d5f0261c0338288ac",
|
|
// vout: 1,
|
|
// amount: 0.01,
|
|
// confirmations: 3
|
|
// }, ...
|
|
// ]
|
|
// ```
|
|
// This is compatible con insight's utxo API.
|
|
// That amount is in BTCs (as returned in insight and bitcoind).
|
|
// amountSat (instead of amount) can be given to provide amount in satochis.
|
|
TransactionBuilder.prototype.setUnspent = function(unspent) {
|
|
this.utxos = unspent;
|
|
return this;
|
|
};
|
|
|
|
TransactionBuilder.prototype._setInputMap = function() {
|
|
var inputMap = [];
|
|
|
|
var l = this.selectedUtxos.length;
|
|
for (var i = 0; i < l; i++) {
|
|
var utxo = this.selectedUtxos[i];
|
|
var scriptBuf = new Buffer(utxo.scriptPubKey, 'hex');
|
|
var scriptPubKey = new Script(scriptBuf);
|
|
var scriptType = scriptPubKey.classify();
|
|
|
|
if (scriptType === Script.TX_UNKNOWN)
|
|
throw new Error('unkown output type at:' + i +
|
|
' Type:' + scriptPubKey.getRawOutType());
|
|
|
|
inputMap.push({
|
|
address: utxo.address,
|
|
scriptPubKey: scriptPubKey,
|
|
scriptType: scriptType,
|
|
i: i,
|
|
});
|
|
}
|
|
this.inputMap = inputMap;
|
|
return this;
|
|
};
|
|
|
|
|
|
// getSelectedUnspent
|
|
// ------------------
|
|
//
|
|
// Returns the selected unspent outputs, to be used in the transaction.
|
|
|
|
TransactionBuilder.prototype.getSelectedUnspent = function() {
|
|
return this.selectedUtxos;
|
|
};
|
|
|
|
/* _selectUnspent
|
|
* TODO(?): sort sel (at the end) and check is some inputs can be avoided.
|
|
* If the initial utxos are sorted, this step would be necesary only if
|
|
* utxos were selected from different minConfirmationSteps.
|
|
*/
|
|
|
|
TransactionBuilder.prototype._selectUnspent = function(neededAmountSat) {
|
|
|
|
if (!this.utxos || !this.utxos.length)
|
|
throw new Error('unspent not set');
|
|
|
|
var minConfirmationSteps = [6, 1];
|
|
if (this.spendUnconfirmed) minConfirmationSteps.push(0);
|
|
|
|
var sel = [],
|
|
totalSat = bignum(0),
|
|
fulfill = false,
|
|
maxConfirmations = null,
|
|
l = this.utxos.length;
|
|
|
|
do {
|
|
var minConfirmations = minConfirmationSteps.shift();
|
|
for (var i = 0; i < l; i++) {
|
|
var u = this.utxos[i];
|
|
var c = u.confirmations || 0;
|
|
|
|
if (c < minConfirmations || (maxConfirmations && c >= maxConfirmations))
|
|
continue;
|
|
|
|
var sat = u.amountSat || util.parseValue(u.amount);
|
|
totalSat = totalSat.add(sat);
|
|
sel.push(u);
|
|
if (totalSat.cmp(neededAmountSat) >= 0) {
|
|
fulfill = true;
|
|
break;
|
|
}
|
|
}
|
|
maxConfirmations = minConfirmations;
|
|
} while (!fulfill && minConfirmationSteps.length);
|
|
|
|
if (!fulfill)
|
|
throw new Error('no enough unspent to fulfill totalNeededAmount [SAT]:' +
|
|
neededAmountSat);
|
|
|
|
this.selectedUtxos = sel;
|
|
this._setInputMap();
|
|
return this;
|
|
};
|
|
|
|
TransactionBuilder.prototype._setInputs = function(txobj) {
|
|
var ins = this.selectedUtxos;
|
|
var l = ins.length;
|
|
var valueInSat = bignum(0);
|
|
|
|
txobj.ins=[];
|
|
for (var i = 0; i < l; i++) {
|
|
valueInSat = valueInSat.add(util.parseValue(ins[i].amount));
|
|
|
|
var txin = {};
|
|
txin.s = util.EMPTY_BUFFER;
|
|
txin.q = 0xffffffff;
|
|
|
|
var hash = new Buffer(ins[i].txid, 'hex');
|
|
var hashReversed = buffertools.reverse(hash);
|
|
|
|
var vout = parseInt(ins[i].vout);
|
|
var voutBuf = new Buffer(4);
|
|
voutBuf.writeUInt32LE(vout, 0);
|
|
|
|
txin.o = Buffer.concat([hashReversed, voutBuf]);
|
|
txobj.ins.push(txin);
|
|
}
|
|
this.valueInSat = valueInSat;
|
|
return this;
|
|
};
|
|
|
|
TransactionBuilder.prototype._setFee = function(feeSat) {
|
|
if ( typeof this.valueOutSat === 'undefined')
|
|
throw new Error('valueOutSat undefined');
|
|
|
|
|
|
var valueOutSat = this.valueOutSat.add(feeSat);
|
|
|
|
if (this.valueInSat.cmp(valueOutSat) < 0) {
|
|
var inv = this.valueInSat.toString();
|
|
var ouv = valueOutSat.toString();
|
|
throw new Error('transaction input amount is less than outputs: ' +
|
|
inv + ' < ' + ouv + ' [SAT]');
|
|
}
|
|
this.feeSat = feeSat;
|
|
return this;
|
|
};
|
|
|
|
TransactionBuilder.prototype._setRemainder = function(txobj, remainderIndex) {
|
|
|
|
if ( typeof this.valueInSat === 'undefined' ||
|
|
typeof this.valueOutSat === 'undefined')
|
|
throw new Error('valueInSat / valueOutSat undefined');
|
|
|
|
/* add remainder (without modifying outs[]) */
|
|
var remainderSat = this.valueInSat.sub(this.valueOutSat).sub(this.feeSat);
|
|
var l =txobj.outs.length;
|
|
this.remainderSat = bignum(0);
|
|
|
|
/*remove old remainder? */
|
|
if (l > remainderIndex) {
|
|
txobj.outs.pop();
|
|
}
|
|
|
|
if (remainderSat.cmp(0) > 0) {
|
|
var remainderOut = this.remainderOut || this.selectedUtxos[0];
|
|
var value = util.bigIntToValue(remainderSat);
|
|
var script = TransactionBuilder._scriptForOut(remainderOut);
|
|
var txout = {
|
|
v: value,
|
|
s: script.getBuffer(),
|
|
};
|
|
txobj.outs.push(txout);
|
|
this.remainderSat = remainderSat;
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
TransactionBuilder.prototype._setFeeAndRemainder = function(txobj) {
|
|
|
|
/* starting size estimation */
|
|
var size = 500, maxSizeK, remainderIndex = txobj.outs.length;
|
|
do {
|
|
/* based on https://en.bitcoin.it/wiki/Transaction_fees */
|
|
maxSizeK = parseInt(size / 1000) + 1;
|
|
|
|
var feeSat = this.givenFeeSat ?
|
|
this.givenFeeSat : maxSizeK * FEE_PER_1000B_SAT;
|
|
|
|
var neededAmountSat = this.valueOutSat.add(feeSat);
|
|
|
|
this._selectUnspent(neededAmountSat)
|
|
._setInputs(txobj)
|
|
._setFee(feeSat)
|
|
._setRemainder(txobj, remainderIndex);
|
|
|
|
|
|
size = new Transaction(txobj).getSize();
|
|
} while (size > (maxSizeK + 1) * 1000);
|
|
return this;
|
|
};
|
|
|
|
// setOutputs
|
|
// ----------
|
|
// Sets the outputs for the transaction. Format is:
|
|
// ```
|
|
// an array of [{
|
|
// address: xx,
|
|
// amount:0.001
|
|
// },...]
|
|
// ```
|
|
//
|
|
// Note that only some of this outputs will be selected
|
|
// to create the transaction. The selected ones can be checked
|
|
// after calling `setOutputs`, with `.getSelectedUnspent`
|
|
//
|
|
|
|
TransactionBuilder.prototype.setOutputs = function(outs) {
|
|
var valueOutSat = bignum(0);
|
|
|
|
var txobj = {};
|
|
txobj.version = 1;
|
|
txobj.lock_time = this.lockTime || 0;
|
|
txobj.ins = [];
|
|
txobj.outs = [];
|
|
|
|
var l =outs.length;
|
|
for (var i = 0; i < l; i++) {
|
|
var amountSat = outs[i].amountSat || util.parseValue(outs[i].amount);
|
|
var value = util.bigIntToValue(amountSat);
|
|
var script = TransactionBuilder._scriptForOut(outs[i]);
|
|
var txout = {
|
|
v: value,
|
|
s: script.getBuffer(),
|
|
};
|
|
txobj.outs.push(txout);
|
|
|
|
var sat = outs[i].amountSat || util.parseValue(outs[i].amount);
|
|
valueOutSat = valueOutSat.add(sat);
|
|
}
|
|
|
|
this.valueOutSat = valueOutSat;
|
|
|
|
this._setFeeAndRemainder(txobj);
|
|
|
|
this.tx = new Transaction(txobj);
|
|
return this;
|
|
};
|
|
|
|
TransactionBuilder._mapKeys = function(keys) {
|
|
/* prepare keys */
|
|
var walletKeyMap = {};
|
|
var l = keys.length;
|
|
var wk;
|
|
for (var i = 0; i < l; i++) {
|
|
var k = keys[i];
|
|
|
|
if (typeof k === 'string') {
|
|
var pk = new PrivateKey(k);
|
|
wk = new WalletKey({ network: pk.network() });
|
|
wk.fromObj({ priv: k });
|
|
}
|
|
else if (k instanceof WalletKey) {
|
|
wk = k;
|
|
}
|
|
else {
|
|
throw new Error('argument must be an array of strings (WIF format) or WalletKey objects');
|
|
}
|
|
walletKeyMap[wk.storeObj().addr] = wk;
|
|
}
|
|
return walletKeyMap;
|
|
};
|
|
|
|
TransactionBuilder._signHashAndVerify = function(wk, txSigHash) {
|
|
var triesLeft = 10, sigRaw;
|
|
|
|
do {
|
|
sigRaw = wk.privKey.signSync(txSigHash);
|
|
} while (wk.privKey.verifySignatureSync(txSigHash, sigRaw) === false &&
|
|
triesLeft--);
|
|
|
|
if (triesLeft<0)
|
|
throw new Error('could not sign input: verification failed');
|
|
|
|
return sigRaw;
|
|
};
|
|
|
|
TransactionBuilder.prototype._checkTx = function() {
|
|
if (! this.tx || !this.tx.ins.length || !this.tx.outs.length)
|
|
throw new Error('tx is not defined');
|
|
};
|
|
|
|
|
|
TransactionBuilder.prototype._multiFindKey = function(walletKeyMap,pubKeyHash) {
|
|
var wk;
|
|
[ networks.livenet, networks.testnet].forEach(function(n) {
|
|
[ n.addressVersion, n.P2SHVersion].forEach(function(v) {
|
|
var a = new Address(v,pubKeyHash);
|
|
if (!wk && walletKeyMap[a]) {
|
|
wk = walletKeyMap[a];
|
|
}
|
|
});
|
|
});
|
|
|
|
return wk;
|
|
};
|
|
|
|
TransactionBuilder.prototype._findWalletKey = function(walletKeyMap, input) {
|
|
var wk;
|
|
|
|
if (input.address) {
|
|
wk = walletKeyMap[input.address];
|
|
}
|
|
else if (input.pubKeyHash) {
|
|
wk = this._multiFindKey(walletKeyMap, input.pubKeyHash);
|
|
}
|
|
else if (input.pubKeyBuf) {
|
|
var pubKeyHash = util.sha256ripe160(input.pubKeyBuf);
|
|
wk = this._multiFindKey(walletKeyMap, pubKeyHash);
|
|
} else {
|
|
throw new Error('no infomation at input to find keys');
|
|
}
|
|
return wk;
|
|
};
|
|
|
|
TransactionBuilder.prototype._signPubKey = function(walletKeyMap, input, txSigHash) {
|
|
if (this.tx.ins[input.i].s.length > 0) return {};
|
|
|
|
var wk = this._findWalletKey(walletKeyMap, input);
|
|
if (!wk) return;
|
|
|
|
var sigRaw = TransactionBuilder._signHashAndVerify(wk, txSigHash);
|
|
var sigType = new Buffer(1);
|
|
sigType[0] = this.signhash;
|
|
var sig = Buffer.concat([sigRaw, sigType]);
|
|
|
|
var scriptSig = new Script();
|
|
scriptSig.chunks.push(sig);
|
|
scriptSig.updateBuffer();
|
|
return {inputFullySigned: true, signaturesAdded: 1, script: scriptSig.getBuffer()};
|
|
};
|
|
|
|
TransactionBuilder.prototype._signPubKeyHash = function(walletKeyMap, input, txSigHash) {
|
|
|
|
if (this.tx.ins[input.i].s.length > 0) return {};
|
|
|
|
var wk = this._findWalletKey(walletKeyMap, input);
|
|
if (!wk) return;
|
|
|
|
var sigRaw = TransactionBuilder._signHashAndVerify(wk, txSigHash);
|
|
var sigType = new Buffer(1);
|
|
sigType[0] = this.signhash;
|
|
var sig = Buffer.concat([sigRaw, sigType]);
|
|
|
|
var scriptSig = new Script();
|
|
scriptSig.chunks.push(sig);
|
|
scriptSig.chunks.push(wk.privKey.public);
|
|
scriptSig.updateBuffer();
|
|
return {inputFullySigned: true, signaturesAdded: 1, script: scriptSig.getBuffer()};
|
|
};
|
|
|
|
/* FOR TESTING
|
|
var _dumpChunks = function (scriptSig, label) {
|
|
console.log('## DUMP: ' + label + ' ##');
|
|
for(var i=0; i<scriptSig.chunks.length; i++) {
|
|
console.log('\tCHUNK ', i, Buffer.isBuffer(scriptSig.chunks[i])
|
|
?scriptSig.chunks[i].toString('hex'):scriptSig.chunks[i] );
|
|
}
|
|
};
|
|
*/
|
|
|
|
TransactionBuilder.prototype._chunkSignedWithKey = function(scriptSig, txSigHash, publicKey) {
|
|
var ret;
|
|
var k = new Key();
|
|
k.public =publicKey;
|
|
|
|
for(var i=1; i<= scriptSig.countSignatures(); i++) {
|
|
var chunk = scriptSig.chunks[i];
|
|
var sigRaw = new Buffer(chunk.slice(0,chunk.length-1));
|
|
if (k.verifySignatureSync(txSigHash, sigRaw) ) {
|
|
ret=chunk;
|
|
}
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
|
|
TransactionBuilder.prototype._getSignatureOrder = function(sigPrio, sigRaw, txSigHash, pubkeys) {
|
|
var l=pubkeys.length;
|
|
for(var j=0; j<l; j++) {
|
|
var k = new Key();
|
|
k.public = new Buffer(pubkeys[j],'hex');
|
|
if (k.verifySignatureSync(txSigHash, sigRaw))
|
|
break;
|
|
}
|
|
return j;
|
|
};
|
|
|
|
TransactionBuilder.prototype._getNewSignatureOrder = function(sigPrio, scriptSig, txSigHash, pubkeys) {
|
|
var iPrio;
|
|
for(var i=1; i<= scriptSig.countSignatures(); i++) {
|
|
var chunk = scriptSig.chunks[i];
|
|
var sigRaw = new Buffer(chunk.slice(0,chunk.length-1));
|
|
iPrio = this._getSignatureOrder(sigPrio, sigRaw, txSigHash, pubkeys);
|
|
if (sigPrio <= iPrio) break;
|
|
}
|
|
return (sigPrio === iPrio? -1: i-1);
|
|
};
|
|
|
|
TransactionBuilder.prototype._chunkIsEmpty = function(chunk) {
|
|
return chunk === 0 || // when serializing and back, EMPTY_BUFFER becomes 0
|
|
buffertools.compare(chunk, util.EMPTY_BUFFER) === 0;
|
|
};
|
|
|
|
TransactionBuilder.prototype._initMultiSig = function(script) {
|
|
var wasUpdated = false;
|
|
if (script.chunks[0] !== 0) {
|
|
script.prependOp0();
|
|
wasUpdated = true;
|
|
}
|
|
return wasUpdated;
|
|
};
|
|
|
|
TransactionBuilder.prototype._updateMultiSig = function(sigPrio, wk, scriptSig, txSigHash, pubkeys) {
|
|
var wasUpdated = this._initMultiSig(scriptSig);
|
|
|
|
if (this._chunkSignedWithKey(scriptSig, txSigHash, wk.privKey.public))
|
|
return null;
|
|
|
|
// Create signature
|
|
var sigRaw = TransactionBuilder._signHashAndVerify(wk, txSigHash);
|
|
var sigType = new Buffer(1);
|
|
sigType[0] = this.signhash;
|
|
var sig = Buffer.concat([sigRaw, sigType]);
|
|
|
|
// Add signature
|
|
var order = this._getNewSignatureOrder(sigPrio,scriptSig,txSigHash,pubkeys);
|
|
scriptSig.chunks.splice(order+1,0,sig);
|
|
scriptSig.updateBuffer();
|
|
wasUpdated=true;
|
|
|
|
return wasUpdated ? scriptSig : null;
|
|
};
|
|
|
|
|
|
TransactionBuilder.prototype._signMultiSig = function(walletKeyMap, input, txSigHash) {
|
|
var pubkeys = input.scriptPubKey.capture(),
|
|
nreq = input.scriptPubKey.chunks[0] - 80, //see OP_2-OP_16
|
|
l = pubkeys.length,
|
|
originalScriptBuf = this.tx.ins[input.i].s;
|
|
|
|
var scriptSig = new Script (originalScriptBuf);
|
|
var signaturesAdded = 0;
|
|
|
|
for(var j=0; j<l && scriptSig.countSignatures() < nreq ; j++) {
|
|
var wk = this._findWalletKey(walletKeyMap, {pubKeyBuf: pubkeys[j]});
|
|
if (!wk) continue;
|
|
|
|
var newScriptSig = this._updateMultiSig(j, wk, scriptSig, txSigHash, pubkeys);
|
|
if (newScriptSig) {
|
|
scriptSig = newScriptSig;
|
|
signaturesAdded++;
|
|
}
|
|
}
|
|
|
|
var ret = {
|
|
inputFullySigned: scriptSig.countSignatures() === nreq,
|
|
signaturesAdded: signaturesAdded,
|
|
script: scriptSig.getBuffer(),
|
|
};
|
|
return ret;
|
|
};
|
|
|
|
var fnToSign = {};
|
|
TransactionBuilder.prototype._scriptIsAppended = function(script, scriptToAddBuf) {
|
|
var len = script.chunks.length;
|
|
|
|
if (script.chunks[len-1] === undefined)
|
|
return false;
|
|
if (typeof script.chunks[len-1] === 'number')
|
|
return false;
|
|
if (buffertools.compare(script.chunks[len-1] , scriptToAddBuf) !==0 )
|
|
return false;
|
|
|
|
return true;
|
|
};
|
|
|
|
TransactionBuilder.prototype._addScript = function(scriptBuf, scriptToAddBuf) {
|
|
var s = new Script(scriptBuf);
|
|
|
|
if (!this._scriptIsAppended(s, scriptToAddBuf)) {
|
|
s.chunks.push(scriptToAddBuf);
|
|
s.updateBuffer();
|
|
}
|
|
return s.getBuffer();
|
|
};
|
|
|
|
TransactionBuilder.prototype._getInputForP2sh = function(script, index) {
|
|
var scriptType = script.classify();
|
|
/* pubKeyHash is needed for TX_PUBKEYHASH and TX_PUBKEY to retrieve the keys. */
|
|
var pubKeyHash;
|
|
switch(scriptType) {
|
|
case Script.TX_PUBKEYHASH:
|
|
pubKeyHash = script.captureOne();
|
|
break;
|
|
case Script.TX_PUBKEY:
|
|
var chunk = script.captureOne();
|
|
pubKeyHash = util.sha256ripe160(chunk);
|
|
}
|
|
|
|
return {
|
|
i: index,
|
|
pubKeyHash: pubKeyHash,
|
|
scriptPubKey: script,
|
|
scriptType: scriptType,
|
|
isP2sh: true,
|
|
};
|
|
};
|
|
|
|
TransactionBuilder.prototype._p2shInput = function(input) {
|
|
if (!this.hashToScriptMap)
|
|
throw new Error('hashToScriptMap not set');
|
|
|
|
var scriptHex = this.hashToScriptMap[input.address];
|
|
if (!scriptHex) return;
|
|
|
|
var scriptBuf = new Buffer(scriptHex,'hex');
|
|
var script = new Script(scriptBuf);
|
|
var scriptType = script.classify();
|
|
|
|
if (!fnToSign[scriptType] || scriptType === Script.TX_SCRIPTHASH)
|
|
throw new Error('dont know how to sign p2sh script type:'+ script.getRawOutType());
|
|
|
|
return {
|
|
input: this._getInputForP2sh(script, input.i),
|
|
txSigHash: this.tx.hashForSignature( script, input.i, this.signhash),
|
|
scriptType: script.classify(),
|
|
scriptBuf: scriptBuf,
|
|
};
|
|
};
|
|
|
|
TransactionBuilder.prototype._signScriptHash = function(walletKeyMap, input, txSigHash) {
|
|
|
|
var p2sh = this._p2shInput(input);
|
|
|
|
var ret = fnToSign[p2sh.scriptType].call(this, walletKeyMap, p2sh.input, p2sh.txSigHash);
|
|
if (ret && ret.script && ret.signaturesAdded) {
|
|
ret.script = this._addScript(ret.script, p2sh.scriptBuf);
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
fnToSign[Script.TX_PUBKEYHASH] = TransactionBuilder.prototype._signPubKeyHash;
|
|
fnToSign[Script.TX_PUBKEY] = TransactionBuilder.prototype._signPubKey;
|
|
fnToSign[Script.TX_MULTISIG] = TransactionBuilder.prototype._signMultiSig;
|
|
fnToSign[Script.TX_SCRIPTHASH] = TransactionBuilder.prototype._signScriptHash;
|
|
|
|
// sign
|
|
// ----
|
|
// Signs a transaction.
|
|
// `keys`: an array of strings representing private keys to sign the
|
|
// transaction in WIF private key format OR bitcore's `WalletKey` objects
|
|
//
|
|
// If multiple keys are given, each will be tested against the transaction's
|
|
// scriptPubKeys. Only the valid private keys will be used to sign.
|
|
// This method is fully compatible with *multisig* transactions.
|
|
//
|
|
// `.isFullySigned` can be queried to check is the transactions have all the needed
|
|
// signatures.
|
|
//
|
|
//
|
|
TransactionBuilder.prototype.sign = function(keys) {
|
|
this._checkTx();
|
|
var tx = this.tx,
|
|
ins = tx.ins,
|
|
l = ins.length,
|
|
walletKeyMap = TransactionBuilder._mapKeys(keys);
|
|
|
|
for (var i = 0; i < l; i++) {
|
|
var input = this.inputMap[i];
|
|
|
|
var txSigHash = this.tx.hashForSignature(
|
|
input.scriptPubKey, i, this.signhash);
|
|
|
|
var ret = fnToSign[input.scriptType].call(this, walletKeyMap, input, txSigHash);
|
|
if (ret && ret.script) {
|
|
tx.ins[i].s = ret.script;
|
|
if (ret.inputFullySigned) this.inputsSigned++;
|
|
if (ret.signaturesAdded) this.signaturesAdded +=ret.signaturesAdded;
|
|
}
|
|
}
|
|
return this;
|
|
};
|
|
|
|
// setHashToScriptMap
|
|
// ------------------
|
|
// Needed for setup Address to Script maps
|
|
// for p2sh transactions. See `.infoForP2sh`
|
|
// for generate the input for this call.
|
|
//
|
|
TransactionBuilder.prototype.setHashToScriptMap = function(hashToScriptMap) {
|
|
this.hashToScriptMap= hashToScriptMap;
|
|
return this;
|
|
};
|
|
|
|
|
|
// isFullySigned
|
|
// -------------
|
|
// Checks if the transaction have all the necesary signatures.
|
|
// Also, `.signaturesAdded` and `.inputsSigned` can be queried
|
|
// for more information about the transaction signature status.
|
|
//
|
|
TransactionBuilder.prototype.isFullySigned = function() {
|
|
return this.inputsSigned === this.tx.ins.length;
|
|
};
|
|
|
|
TransactionBuilder.prototype.build = function() {
|
|
this._checkTx();
|
|
return this.tx;
|
|
};
|
|
|
|
// toObj
|
|
// -----
|
|
// Returns a plain Javascript object that contains
|
|
// the full status of the TransactionBuilder instance,
|
|
// suitable for serialization, storage and transmition.
|
|
// See `.fromObj`
|
|
//
|
|
TransactionBuilder.prototype.toObj = function() {
|
|
var data = {
|
|
valueInSat : this.valueInSat.toString(),
|
|
valueOutSat : this.valueOutSat.toString(),
|
|
feeSat : this.feeSat.toString(),
|
|
remainderSat : this.remainderSat.toString(),
|
|
|
|
hashToScriptMap : this.hashToScriptMap,
|
|
selectedUtxos : this.selectedUtxos,
|
|
|
|
inputsSigned : this.inputsSigned,
|
|
signaturesAdded : this.signaturesAdded,
|
|
|
|
signhash : this.signhash,
|
|
spendUnconfirmed : this.spendUnconfirmed,
|
|
};
|
|
if (this.tx) {
|
|
data.tx =this.tx.serialize().toString('hex');
|
|
}
|
|
return data;
|
|
};
|
|
|
|
// fromObj
|
|
// -------
|
|
// Returns a TransactionBuilder instance given
|
|
// a plain Javascript object created previously
|
|
// with `.toObj`. See `.toObj`.
|
|
|
|
TransactionBuilder.fromObj = function(data) {
|
|
var b = new TransactionBuilder();
|
|
b.valueInSat = data.valueInSat.toString();
|
|
b.valueOutSat = data.valueOutSat.toString();
|
|
b.feeSat = data.feeSat.toString();
|
|
b.remainderSat = data.remainderSat.toString();
|
|
|
|
b.hashToScriptMap = data.hashToScriptMap;
|
|
b.selectedUtxos = data.selectedUtxos;
|
|
|
|
b.inputsSigned = data.inputsSigned;
|
|
b.signaturesAdded = data.signaturesAdded;
|
|
|
|
b.signhash = data.signhash;
|
|
b.spendUnconfirmed = data.spendUnconfirmed;
|
|
|
|
b._setInputMap();
|
|
|
|
if (data.tx) {
|
|
// Tx may have signatures, that are not on txobj
|
|
var t = new Transaction();
|
|
t.parse(new Buffer(data.tx,'hex'));
|
|
b.tx = t;
|
|
}
|
|
return b;
|
|
};
|
|
|
|
|
|
TransactionBuilder.prototype._checkMergeability = function(b) {
|
|
var self=this;
|
|
|
|
// Builder should have the same params
|
|
['valueInSat', 'valueOutSat', 'feeSat', 'remainderSat', 'signhash', 'spendUnconfirmed']
|
|
.forEach(function (k) {
|
|
|
|
if (self[k].toString() !== b[k].toString()) {
|
|
throw new Error('mismatch at TransactionBuilder match: '
|
|
+ k + ': ' + self[k] + ' vs. ' + b[k]);
|
|
}
|
|
});
|
|
|
|
if (self.hashToScriptMap) {
|
|
var err = 0;
|
|
if(! b.hashToScriptMap) err=1;
|
|
Object.keys(self.hashToScriptMap).forEach(function(k) {
|
|
if (!b.hashToScriptMap[k]) err=1;
|
|
if (self.hashToScriptMap[k] !== b.hashToScriptMap[k]) err=1;
|
|
});
|
|
if (err)
|
|
throw new Error('mismatch at TransactionBuilder hashToScriptMap');
|
|
}
|
|
|
|
|
|
var err = 0, i=0;;
|
|
self.selectedUtxos.forEach(function(u) {
|
|
if (!err) {
|
|
var v=b.selectedUtxos[i++];
|
|
if (!v) err=1;
|
|
// confirmations could differ
|
|
['address', 'hash', 'scriptPubKey', 'vout', 'amount'].forEach(function(k) {
|
|
if (u[k] !== v[k])
|
|
err=k;
|
|
});
|
|
}
|
|
});
|
|
if (err)
|
|
throw new Error('mismatch at TransactionBuilder selectedUtxos #' + i-1+ ' Key:' + err);
|
|
|
|
|
|
err = 0; i=0;;
|
|
self.inputMap.forEach(function(u) {
|
|
if (!err) {
|
|
var v=b.inputMap[i++];
|
|
if (!v) err=1;
|
|
// confirmations could differ
|
|
['address', 'scriptType', 'scriptPubKey', 'i'].forEach(function(k) {
|
|
if (u[k].toString() !== v[k].toString())
|
|
err=k;
|
|
});
|
|
}
|
|
});
|
|
if (err)
|
|
throw new Error('mismatch at TransactionBuilder inputMap #' + i-1 + ' Key:' + err);
|
|
|
|
};
|
|
|
|
// TODO this could be on Script class
|
|
TransactionBuilder.prototype._mergeInputSigP2sh = function(input,s0,s1) {
|
|
var p2sh = this._p2shInput(input);
|
|
var redeemScript = new Script(p2sh.scriptBuf);
|
|
var pubkeys = redeemScript.capture();
|
|
|
|
// Look for differences
|
|
var s0keys = {};
|
|
var l = pubkeys.length;
|
|
for (var j=0; j<l; j++) {
|
|
if ( this._chunkSignedWithKey(s0, p2sh.txSigHash, pubkeys[j]))
|
|
s0keys[pubkeys[j].toString('hex')] = 1;
|
|
}
|
|
|
|
var diff = [];
|
|
for (var j=0; j<l; j++) {
|
|
var chunk = this._chunkSignedWithKey(s1, p2sh.txSigHash, pubkeys[j]);
|
|
var pubHex = pubkeys[j].toString('hex');
|
|
if (chunk && !s0keys[pubHex]) {
|
|
diff.push({
|
|
prio: j,
|
|
chunk: chunk,
|
|
pubHex: pubHex,
|
|
});
|
|
}
|
|
}
|
|
|
|
// Add signatures
|
|
for(var j in diff) {
|
|
var newSig = diff[j];
|
|
var order = this._getNewSignatureOrder(newSig.prio,s0,p2sh.txSigHash,pubkeys);
|
|
s0.chunks.splice(order+1,0,newSig.chunk);
|
|
this.signaturesAdded++;
|
|
}
|
|
s0.updateBuffer();
|
|
return s0.getBuffer();
|
|
};
|
|
|
|
// TODO this could be on Script class
|
|
TransactionBuilder.prototype._mergeInputSig = function(index, s0buf, s1buf) {
|
|
if (buffertools.compare(s0buf,s1buf) === 0)
|
|
return s0buf;
|
|
|
|
var s0 = new Script(s0buf);
|
|
var s1 = new Script(s1buf);
|
|
var l0 = s0.chunks.length;
|
|
var l1 = s1.chunks.length;
|
|
var s0map = {};
|
|
|
|
if (l0 && l1 && ((l0<2 && l1>2) || (l1<2 && l0>2 )))
|
|
throw new Error('TX sig types mismatch in merge');
|
|
|
|
if ((!l0 && !l1) || ( l0 && !l1) || (!l0 && l1))
|
|
return s1buf;
|
|
|
|
// Get the pubkeys
|
|
var input = this.inputMap[index];
|
|
var type = input.scriptPubKey.classify();
|
|
|
|
//p2pubkey or p2pubkeyhash
|
|
if (type === Script.TX_PUBKEYHASH || type === Script.TX_PUBKEY) {
|
|
log.debug('Merging two signed inputs type:' +
|
|
input.scriptPubKey.getRawOutType() + '. Signatures differs. Using the first version.');
|
|
return s0buf;
|
|
}
|
|
else if (type!== Script.TX_SCRIPTHASH) {
|
|
// No support for normal multisig or strange txs.
|
|
throw new Error('Script type:'+input.scriptPubKey.getRawOutType()+'not supported at #merge');
|
|
}
|
|
return this._mergeInputSigP2sh(input,s0, s1);
|
|
};
|
|
|
|
// TODO this could be on Transaction class
|
|
TransactionBuilder.prototype._mergeTx = function(tx) {
|
|
var v0 = this.tx;
|
|
var v1 = tx;
|
|
|
|
var l = v0.ins.length;
|
|
if (l !== v1.ins.length)
|
|
throw new Error('TX in length mismatch in merge');
|
|
|
|
this.inputsSigned =0;
|
|
for(var i=0; i<l; i++) {
|
|
var i0 = v0.ins[i];
|
|
var i1 = v1.ins[i];
|
|
|
|
if (i0.q !== i1.q)
|
|
throw new Error('TX sequence ins mismatch in merge. Input:',i);
|
|
|
|
if (buffertools.compare(i0.o,i1.o) !== 0)
|
|
throw new Error('TX .o in mismatch in merge. Input:',i);
|
|
|
|
i0.s=this._mergeInputSig(i, i0.s,i1.s);
|
|
|
|
if (v0.isInputComplete(i)) this.inputsSigned++;
|
|
}
|
|
};
|
|
|
|
// merge
|
|
// -----
|
|
// Merge to TransactionBuilder objects, merging inputs signatures.
|
|
// This function supports multisig p2sh inputs.
|
|
|
|
TransactionBuilder.prototype.merge = function(b) {
|
|
this._checkMergeability(b);
|
|
|
|
// Does this tX have any signature already?
|
|
if (this.tx || b.tx) {
|
|
if (this.tx.getNormalizedHash().toString('hex')
|
|
!== b.tx.getNormalizedHash().toString('hex'))
|
|
throw new Error('mismatch at TransactionBuilder NTXID');
|
|
|
|
this._mergeTx(b.tx);
|
|
}
|
|
};
|
|
|
|
module.exports = require('soop')(TransactionBuilder);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../networks":"ULNIu2","../util":140,"../util/log":"AdF7pF","./Address":"G+CcXD","./Key":"ALJ4PS","./PrivateKey":"izTl9z","./Script":"hQ0t76","./Transaction":"LJhYtm","./WalletKey":"wWje7g","bignum":58,"buffer":80,"buffertools":"fugeBw","soop":125}],"./lib/Wallet":[function(require,module,exports){
|
|
module.exports=require('yUY4WV');
|
|
},{}],"yUY4WV":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
|
|
var hex = function(hex) {return new Buffer(hex, 'hex');};
|
|
|
|
var fs = require('fs');
|
|
var EncFile = require('../util/EncFile');
|
|
var Address = require('./Address');
|
|
var networks = require('../networks');
|
|
var util = imports.util || require('../util');
|
|
var ENC_METHOD = 'aes-256-cbc';
|
|
|
|
var skeleton = {
|
|
client: 'libcoin',
|
|
client_version: '0.0.1',
|
|
network: 'testnet',
|
|
version: 1,
|
|
best_hash: null,
|
|
best_height: -1,
|
|
keys: [],
|
|
sin: {},
|
|
scripts: {},
|
|
};
|
|
|
|
function Wallet(cfg) {
|
|
if (typeof cfg !== 'object')
|
|
cfg = {};
|
|
|
|
// deep copy (no references)
|
|
if (cfg.datastore)
|
|
this.datastore = JSON.parse(JSON.stringify(cfg.datastore));
|
|
else
|
|
this.datastore = JSON.parse(JSON.stringify(skeleton));
|
|
|
|
this.network = undefined;
|
|
this.dirty = cfg.dirty || true;
|
|
};
|
|
|
|
Wallet.prototype.readSync = function(filename, passphrase) {
|
|
this.datastore = EncFile.readJFileSync(ENC_METHOD,
|
|
passphrase, filename);
|
|
this.dirty = false;
|
|
};
|
|
|
|
Wallet.prototype.writeSync = function(filename, passphrase) {
|
|
var tmp_fn = filename + ".tmp";
|
|
|
|
EncFile.writeJFileSync(ENC_METHOD, passphrase, tmp_fn,
|
|
this.datastore);
|
|
fs.renameSync(tmp_fn, filename);
|
|
|
|
this.dirty = false;
|
|
};
|
|
|
|
Wallet.prototype.setNetwork = function(netname) {
|
|
if (!netname)
|
|
netname = this.datastore.network;
|
|
|
|
switch (netname) {
|
|
case "mainnet":
|
|
case "livenet":
|
|
this.network = networks.livenet;
|
|
break;
|
|
case "testnet":
|
|
this.network = networks.testnet;
|
|
break;
|
|
default:
|
|
throw new Error("Unsupported network");
|
|
}
|
|
|
|
// store+canonicalize name
|
|
this.datastore['network'] = this.network.name;
|
|
this.dirty = true;
|
|
};
|
|
|
|
Wallet.prototype.addKey = function(wkey) {
|
|
this.datastore.keys.push(wkey);
|
|
this.dirty = true;
|
|
};
|
|
|
|
Wallet.prototype.addSIN = function(sinObj) {
|
|
this.datastore.sin[sinObj.sin] = sinObj;
|
|
this.dirty = true;
|
|
};
|
|
|
|
Wallet.prototype.findKeyHash = function(pubKeyHash) {
|
|
var pkhStr = pubKeyHash.toString();
|
|
|
|
for (var i = 0; i < this.datastore.keys.length; i++) {
|
|
var obj = this.datastore.keys[i];
|
|
var addrStr = obj.addr;
|
|
var addr = new Address(addrStr);
|
|
if (addr.payload().toString() == pkhStr)
|
|
return obj;
|
|
}
|
|
|
|
return undefined;
|
|
};
|
|
|
|
Wallet.prototype.expandKey = function(key) {
|
|
var addr = new Address(key);
|
|
var isAddr = true;
|
|
|
|
try {
|
|
addr.validate();
|
|
var b = addr.payload();
|
|
var obj = this.findKeyHash(b);
|
|
key = obj.pub;
|
|
} catch(e) {
|
|
// do nothing
|
|
}
|
|
|
|
var re = /^[a-fA-F0-9]+$/;
|
|
if (!key.match(re))
|
|
throw new Error("Unknown key type");
|
|
return hex(key);
|
|
};
|
|
|
|
Wallet.prototype.expandKeys = function(keys) {
|
|
var res = [];
|
|
var us = this;
|
|
keys.forEach(function(key) {
|
|
var expKey = us.expandKey(key);
|
|
res.push(expKey);
|
|
});
|
|
return res;
|
|
};
|
|
|
|
Wallet.prototype.addScript = function(script) {
|
|
var buf = script.getBuffer();
|
|
var hash = util.sha256ripe160(buf);
|
|
var addr = new Address(this.network.P2SHVersion, hash);
|
|
var addrStr = addr.as('base58');
|
|
this.datastore.scripts[addrStr] = buf.toString('hex');
|
|
this.dirty = true;
|
|
|
|
return addrStr;
|
|
};
|
|
|
|
module.exports = require('soop')(Wallet);
|
|
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../networks":"ULNIu2","../util":140,"../util/EncFile":134,"./Address":"G+CcXD","buffer":80,"fs":76,"soop":125}],"./lib/WalletKey":[function(require,module,exports){
|
|
module.exports=require('wWje7g');
|
|
},{}],"wWje7g":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
|
|
var coinUtil = require('../util');
|
|
var timeUtil = require('../util/time');
|
|
var Key = require('./Key');
|
|
var PrivateKey = require('./PrivateKey');
|
|
var Address = require('./Address');
|
|
|
|
function WalletKey(cfg) {
|
|
if (!cfg) cfg = {};
|
|
if (!cfg.network) throw new Error('network parameter is required');
|
|
this.network = cfg.network; // required
|
|
this.created = cfg.created;
|
|
this.privKey = cfg.privKey;
|
|
};
|
|
|
|
WalletKey.prototype.generate = function() {
|
|
this.privKey = Key.generateSync();
|
|
this.created = timeUtil.curtime();
|
|
};
|
|
|
|
WalletKey.prototype.storeObj = function() {
|
|
var pubKey = this.privKey.public.toString('hex');
|
|
var pubKeyHash = coinUtil.sha256ripe160(this.privKey.public);
|
|
var addr = new Address(this.network.addressVersion, pubKeyHash);
|
|
var priv = new PrivateKey(this.network.privKeyVersion, this.privKey.private, this.privKey.compressed);
|
|
var obj = {
|
|
created: this.created,
|
|
priv: priv.toString(),
|
|
pub: pubKey,
|
|
addr: addr.toString(),
|
|
};
|
|
|
|
return obj;
|
|
};
|
|
|
|
WalletKey.prototype.fromObj = function(obj) {
|
|
this.created = obj.created;
|
|
this.privKey = new Key();
|
|
if (obj.priv.length == 64) {
|
|
this.privKey.private = new Buffer(obj.priv, 'hex');
|
|
this.privKey.compressed = typeof obj.compressed === 'undefined'? true: obj.compressed;
|
|
} else {
|
|
var priv = new PrivateKey(obj.priv);
|
|
priv.validate();
|
|
this.privKey.private = new Buffer(priv.payload());
|
|
this.privKey.compressed = priv.compressed();
|
|
}
|
|
this.privKey.regenerateSync();
|
|
};
|
|
|
|
module.exports = require('soop')(WalletKey);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../util":140,"../util/time":143,"./Address":"G+CcXD","./Key":"ALJ4PS","./PrivateKey":"izTl9z","buffer":80,"soop":125}],58:[function(require,module,exports){
|
|
(function (Buffer){
|
|
/* bignumber.js v1.3.0 https://github.com/MikeMcl/bignumber.js/LICENCE */
|
|
|
|
/*jslint bitwise: true, eqeq: true, plusplus: true, sub: true, white: true, maxerr: 500 */
|
|
/*global module */
|
|
|
|
/*
|
|
bignumber.js v1.3.0
|
|
A JavaScript library for arbitrary-precision arithmetic.
|
|
https://github.com/MikeMcl/bignumber.js
|
|
Copyright (c) 2012 Michael Mclaughlin <M8ch88l@gmail.com>
|
|
MIT Expat Licence
|
|
*/
|
|
|
|
/*********************************** DEFAULTS ************************************/
|
|
|
|
/*
|
|
* The default values below must be integers within the stated ranges (inclusive).
|
|
* Most of these values can be changed during run-time using BigNumber.config().
|
|
*/
|
|
|
|
/*
|
|
* The limit on the value of DECIMAL_PLACES, TO_EXP_NEG, TO_EXP_POS, MIN_EXP,
|
|
* MAX_EXP, and the argument to toFixed, toPrecision and toExponential, beyond
|
|
* which an exception is thrown (if ERRORS is true).
|
|
*/
|
|
var MAX = 1E9, // 0 to 1e+9
|
|
|
|
// Limit of magnitude of exponent argument to toPower.
|
|
MAX_POWER = 1E6, // 1 to 1e+6
|
|
|
|
// The maximum number of decimal places for operations involving division.
|
|
DECIMAL_PLACES = 20, // 0 to MAX
|
|
|
|
/*
|
|
* The rounding mode used when rounding to the above decimal places, and when
|
|
* using toFixed, toPrecision and toExponential, and round (default value).
|
|
* UP 0 Away from zero.
|
|
* DOWN 1 Towards zero.
|
|
* CEIL 2 Towards +Infinity.
|
|
* FLOOR 3 Towards -Infinity.
|
|
* HALF_UP 4 Towards nearest neighbour. If equidistant, up.
|
|
* HALF_DOWN 5 Towards nearest neighbour. If equidistant, down.
|
|
* HALF_EVEN 6 Towards nearest neighbour. If equidistant, towards even neighbour.
|
|
* HALF_CEIL 7 Towards nearest neighbour. If equidistant, towards +Infinity.
|
|
* HALF_FLOOR 8 Towards nearest neighbour. If equidistant, towards -Infinity.
|
|
*/
|
|
ROUNDING_MODE = 4, // 0 to 8
|
|
|
|
// EXPONENTIAL_AT : [TO_EXP_NEG , TO_EXP_POS]
|
|
|
|
// The exponent value at and beneath which toString returns exponential notation.
|
|
// Number type: -7
|
|
TO_EXP_NEG = -7, // 0 to -MAX
|
|
|
|
// The exponent value at and above which toString returns exponential notation.
|
|
// Number type: 21
|
|
TO_EXP_POS = 21, // 0 to MAX
|
|
|
|
// RANGE : [MIN_EXP, MAX_EXP]
|
|
|
|
// The minimum exponent value, beneath which underflow to zero occurs.
|
|
// Number type: -324 (5e-324)
|
|
MIN_EXP = -MAX, // -1 to -MAX
|
|
|
|
// The maximum exponent value, above which overflow to Infinity occurs.
|
|
// Number type: 308 (1.7976931348623157e+308)
|
|
MAX_EXP = MAX, // 1 to MAX
|
|
|
|
// Whether BigNumber Errors are ever thrown.
|
|
// CHANGE parseInt to parseFloat if changing ERRORS to false.
|
|
ERRORS = true, // true or false
|
|
parse = parseInt, // parseInt or parseFloat
|
|
|
|
/***********************************************************************************/
|
|
|
|
P = BigNumber.prototype,
|
|
DIGITS = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ$_',
|
|
outOfRange,
|
|
id = 0,
|
|
isValid = /^-?(\d+(\.\d*)?|\.\d+)(e[+-]?\d+)?$/i,
|
|
trim = String.prototype.trim || function () {return this.replace(/^\s+|\s+$/g, '')},
|
|
ONE = BigNumber(1);
|
|
|
|
|
|
// CONSTRUCTOR
|
|
|
|
|
|
/*
|
|
* The exported function.
|
|
* Create and return a new instance of a BigNumber object.
|
|
*
|
|
* n {number|string|BigNumber} A numeric value.
|
|
* [b] {number} The base of n. Integer, 2 to 64 inclusive.
|
|
*/
|
|
function BigNumber( n, b ) {
|
|
var e, i, isNum, digits, valid, orig,
|
|
x = this;
|
|
|
|
// Enable constructor usage without new.
|
|
if ( !(x instanceof BigNumber) ) {
|
|
return new BigNumber( n, b )
|
|
}
|
|
|
|
// Duplicate.
|
|
if ( n instanceof BigNumber ) {
|
|
id = 0;
|
|
|
|
// e is undefined.
|
|
if ( b !== e ) {
|
|
n += ''
|
|
} else {
|
|
x['s'] = n['s'];
|
|
x['e'] = n['e'];
|
|
x['c'] = ( n = n['c'] ) ? n.slice() : n;
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Accept empty string as zero
|
|
if (n === '') n = 0;
|
|
|
|
// If number, check if minus zero.
|
|
if ( typeof n != 'string' ) {
|
|
n = ( isNum = typeof n == 'number' ||
|
|
Object.prototype.toString.call(n) == '[object Number]' ) &&
|
|
n === 0 && 1 / n < 0 ? '-0' : n + '';
|
|
}
|
|
|
|
orig = n;
|
|
|
|
if ( b === e && isValid.test(n) ) {
|
|
|
|
// Determine sign.
|
|
x['s'] = n.charAt(0) == '-' ? ( n = n.slice(1), -1 ) : 1;
|
|
|
|
// Either n is not a valid BigNumber or a base has been specified.
|
|
} else {
|
|
|
|
// Enable exponential notation to be used with base 10 argument.
|
|
// Ensure return value is rounded to DECIMAL_PLACES as with other bases.
|
|
if ( b == 10 ) {
|
|
|
|
return setMode( n, DECIMAL_PLACES, ROUNDING_MODE );
|
|
}
|
|
|
|
n = trim.call(n).replace( /^\+(?!-)/, '' );
|
|
|
|
x['s'] = n.charAt(0) == '-' ? ( n = n.replace( /^-(?!-)/, '' ), -1 ) : 1;
|
|
|
|
if ( b != null ) {
|
|
|
|
if ( ( b == (b | 0) || !ERRORS ) &&
|
|
!( outOfRange = !( b >= 2 && b < 65 ) ) ) {
|
|
|
|
digits = '[' + DIGITS.slice( 0, b = b | 0 ) + ']+';
|
|
|
|
// Before non-decimal number validity test and base conversion
|
|
// remove the `.` from e.g. '1.', and replace e.g. '.1' with '0.1'.
|
|
n = n.replace( /\.$/, '' ).replace( /^\./, '0.' );
|
|
|
|
// Any number in exponential form will fail due to the e+/-.
|
|
if ( valid = new RegExp(
|
|
'^' + digits + '(?:\\.' + digits + ')?$', b < 37 ? 'i' : '' ).test(n) ) {
|
|
|
|
if ( isNum ) {
|
|
|
|
if ( n.replace( /^0\.0*|\./, '' ).length > 15 ) {
|
|
|
|
// 'new BigNumber() number type has more than 15 significant digits: {n}'
|
|
ifExceptionsThrow( orig, 0 );
|
|
}
|
|
|
|
// Prevent later check for length on converted number.
|
|
isNum = !isNum;
|
|
}
|
|
n = convert( n, 10, b, x['s'] );
|
|
|
|
} else if ( n != 'Infinity' && n != 'NaN' ) {
|
|
|
|
// 'new BigNumber() not a base {b} number: {n}'
|
|
ifExceptionsThrow( orig, 1, b );
|
|
n = 'NaN';
|
|
}
|
|
} else {
|
|
|
|
// 'new BigNumber() base not an integer: {b}'
|
|
// 'new BigNumber() base out of range: {b}'
|
|
ifExceptionsThrow( b, 2 );
|
|
|
|
// Ignore base.
|
|
valid = isValid.test(n);
|
|
}
|
|
} else {
|
|
valid = isValid.test(n);
|
|
}
|
|
|
|
if ( !valid ) {
|
|
|
|
// Infinity/NaN
|
|
x['c'] = x['e'] = null;
|
|
|
|
// NaN
|
|
if ( n != 'Infinity' ) {
|
|
|
|
// No exception on NaN.
|
|
if ( n != 'NaN' ) {
|
|
|
|
// 'new BigNumber() not a number: {n}'
|
|
ifExceptionsThrow( orig, 3 );
|
|
}
|
|
x['s'] = null;
|
|
}
|
|
id = 0;
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Decimal point?
|
|
if ( ( e = n.indexOf('.') ) > -1 ) {
|
|
n = n.replace( '.', '' );
|
|
}
|
|
|
|
// Exponential form?
|
|
if ( ( i = n.search( /e/i ) ) > 0 ) {
|
|
|
|
// Determine exponent.
|
|
if ( e < 0 ) {
|
|
e = i;
|
|
}
|
|
e += +n.slice( i + 1 );
|
|
n = n.substring( 0, i );
|
|
|
|
} else if ( e < 0 ) {
|
|
|
|
// Integer.
|
|
e = n.length;
|
|
}
|
|
|
|
// Determine leading zeros.
|
|
for ( i = 0; n.charAt(i) == '0'; i++ ) {
|
|
}
|
|
|
|
b = n.length;
|
|
|
|
// Disallow numbers with over 15 significant digits if number type.
|
|
if ( isNum && b > 15 && n.slice(i).length > 15 ) {
|
|
|
|
// 'new BigNumber() number type has more than 15 significant digits: {n}'
|
|
ifExceptionsThrow( orig, 0 );
|
|
}
|
|
id = 0;
|
|
|
|
// Overflow?
|
|
if ( ( e -= i + 1 ) > MAX_EXP ) {
|
|
|
|
// Infinity.
|
|
x['c'] = x['e'] = null;
|
|
|
|
// Zero or underflow?
|
|
} else if ( i == b || e < MIN_EXP ) {
|
|
|
|
// Zero.
|
|
x['c'] = [ x['e'] = 0 ];
|
|
} else {
|
|
|
|
// Determine trailing zeros.
|
|
for ( ; n.charAt(--b) == '0'; ) {
|
|
}
|
|
|
|
x['e'] = e;
|
|
x['c'] = [];
|
|
|
|
// Convert string to array of digits (without leading and trailing zeros).
|
|
for ( e = 0; i <= b; x['c'][e++] = +n.charAt(i++) ) {
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// CONSTRUCTOR PROPERTIES/METHODS
|
|
|
|
|
|
BigNumber['ROUND_UP'] = 0;
|
|
BigNumber['ROUND_DOWN'] = 1;
|
|
BigNumber['ROUND_CEIL'] = 2;
|
|
BigNumber['ROUND_FLOOR'] = 3;
|
|
BigNumber['ROUND_HALF_UP'] = 4;
|
|
BigNumber['ROUND_HALF_DOWN'] = 5;
|
|
BigNumber['ROUND_HALF_EVEN'] = 6;
|
|
BigNumber['ROUND_HALF_CEIL'] = 7;
|
|
BigNumber['ROUND_HALF_FLOOR'] = 8;
|
|
|
|
/*
|
|
* Create an instance from a Buffer
|
|
*/
|
|
BigNumber['fromBuffer'] = function (buf, opts) {
|
|
|
|
if (!opts) opts = {};
|
|
|
|
var endian = { 1 : 'big', '-1' : 'little' }[opts.endian]
|
|
|| opts.endian || 'big'
|
|
;
|
|
|
|
var size = opts.size === 'auto' ? Math.ceil(buf.length) : (opts.size || 1);
|
|
|
|
if (buf.length % size !== 0) {
|
|
throw new RangeError('Buffer length (' + buf.length + ')'
|
|
+ ' must be a multiple of size (' + size + ')'
|
|
);
|
|
}
|
|
|
|
var hex = [];
|
|
for (var i = 0; i < buf.length; i += size) {
|
|
var chunk = [];
|
|
for (var j = 0; j < size; j++) {
|
|
chunk.push(buf[
|
|
i + (endian === 'big' ? j : (size - j - 1))
|
|
]);
|
|
}
|
|
|
|
hex.push(chunk
|
|
.map(function (c) {
|
|
return (c < 16 ? '0' : '') + c.toString(16);
|
|
})
|
|
.join('')
|
|
);
|
|
}
|
|
|
|
return BigNumber(hex.join(''), 16);
|
|
|
|
};
|
|
|
|
/*
|
|
* Configure infrequently-changing library-wide settings.
|
|
*
|
|
* Accept an object or an argument list, with one or many of the following
|
|
* properties or parameters respectively:
|
|
* [ DECIMAL_PLACES [, ROUNDING_MODE [, EXPONENTIAL_AT [, RANGE [, ERRORS ]]]]]
|
|
*
|
|
* E.g.
|
|
* BigNumber.config(20, 4) is equivalent to
|
|
* BigNumber.config({ DECIMAL_PLACES : 20, ROUNDING_MODE : 4 })
|
|
* Ignore properties/parameters set to null or undefined.
|
|
*
|
|
* Return an object with the properties current values.
|
|
*/
|
|
BigNumber['config'] = function () {
|
|
var v, p,
|
|
i = 0,
|
|
r = {},
|
|
a = arguments,
|
|
o = a[0],
|
|
c = 'config',
|
|
inRange = function ( n, lo, hi ) {
|
|
return !( ( outOfRange = n < lo || n > hi ) ||
|
|
parse(n) != n && n !== 0 );
|
|
},
|
|
has = o && typeof o == 'object'
|
|
? function () {if ( o.hasOwnProperty(p) ) return ( v = o[p] ) != null}
|
|
: function () {if ( a.length > i ) return ( v = a[i++] ) != null};
|
|
|
|
// [DECIMAL_PLACES] {number} Integer, 0 to MAX inclusive.
|
|
if ( has( p = 'DECIMAL_PLACES' ) ) {
|
|
|
|
if ( inRange( v, 0, MAX ) ) {
|
|
DECIMAL_PLACES = v | 0;
|
|
} else {
|
|
|
|
// 'config() DECIMAL_PLACES not an integer: {v}'
|
|
// 'config() DECIMAL_PLACES out of range: {v}'
|
|
ifExceptionsThrow( v, p, c );
|
|
}
|
|
}
|
|
r[p] = DECIMAL_PLACES;
|
|
|
|
// [ROUNDING_MODE] {number} Integer, 0 to 8 inclusive.
|
|
if ( has( p = 'ROUNDING_MODE' ) ) {
|
|
|
|
if ( inRange( v, 0, 8 ) ) {
|
|
ROUNDING_MODE = v | 0;
|
|
} else {
|
|
|
|
// 'config() ROUNDING_MODE not an integer: {v}'
|
|
// 'config() ROUNDING_MODE out of range: {v}'
|
|
ifExceptionsThrow( v, p, c );
|
|
}
|
|
}
|
|
r[p] = ROUNDING_MODE;
|
|
|
|
/*
|
|
* [EXPONENTIAL_AT] {number|number[]} Integer, -MAX to MAX inclusive or
|
|
* [ integer -MAX to 0 inclusive, 0 to MAX inclusive ].
|
|
*/
|
|
if ( has( p = 'EXPONENTIAL_AT' ) ) {
|
|
|
|
if ( inRange( v, -MAX, MAX ) ) {
|
|
TO_EXP_NEG = -( TO_EXP_POS = ~~( v < 0 ? -v : +v ) );
|
|
} else if ( !outOfRange && v && inRange( v[0], -MAX, 0 ) &&
|
|
inRange( v[1], 0, MAX ) ) {
|
|
TO_EXP_NEG = ~~v[0];
|
|
TO_EXP_POS = ~~v[1];
|
|
} else {
|
|
|
|
// 'config() EXPONENTIAL_AT not an integer or not [integer, integer]: {v}'
|
|
// 'config() EXPONENTIAL_AT out of range or not [negative, positive: {v}'
|
|
ifExceptionsThrow( v, p, c, 1 );
|
|
}
|
|
}
|
|
r[p] = [ TO_EXP_NEG, TO_EXP_POS ];
|
|
|
|
/*
|
|
* [RANGE][ {number|number[]} Non-zero integer, -MAX to MAX inclusive or
|
|
* [ integer -MAX to -1 inclusive, integer 1 to MAX inclusive ].
|
|
*/
|
|
if ( has( p = 'RANGE' ) ) {
|
|
|
|
if ( inRange( v, -MAX, MAX ) && ~~v ) {
|
|
MIN_EXP = -( MAX_EXP = ~~( v < 0 ? -v : +v ) );
|
|
} else if ( !outOfRange && v && inRange( v[0], -MAX, -1 ) &&
|
|
inRange( v[1], 1, MAX ) ) {
|
|
MIN_EXP = ~~v[0], MAX_EXP = ~~v[1];
|
|
} else {
|
|
|
|
// 'config() RANGE not a non-zero integer or not [integer, integer]: {v}'
|
|
// 'config() RANGE out of range or not [negative, positive: {v}'
|
|
ifExceptionsThrow( v, p, c, 1, 1 );
|
|
}
|
|
}
|
|
r[p] = [ MIN_EXP, MAX_EXP ];
|
|
|
|
// [ERRORS] {boolean|number} true, false, 1 or 0.
|
|
if ( has( p = 'ERRORS' ) ) {
|
|
|
|
if ( v === !!v || v === 1 || v === 0 ) {
|
|
parse = ( outOfRange = id = 0, ERRORS = !!v )
|
|
? parseInt
|
|
: parseFloat;
|
|
} else {
|
|
|
|
// 'config() ERRORS not a boolean or binary digit: {v}'
|
|
ifExceptionsThrow( v, p, c, 0, 0, 1 );
|
|
}
|
|
}
|
|
r[p] = ERRORS;
|
|
|
|
return r;
|
|
};
|
|
|
|
|
|
// PRIVATE FUNCTIONS
|
|
|
|
|
|
// Assemble error messages. Throw BigNumber Errors.
|
|
function ifExceptionsThrow( arg, i, j, isArray, isRange, isErrors) {
|
|
|
|
if ( ERRORS ) {
|
|
var error,
|
|
method = ['new BigNumber', 'cmp', 'div', 'eq', 'gt', 'gte', 'lt',
|
|
'lte', 'minus', 'mod', 'plus', 'times', 'toFr'
|
|
][ id ? id < 0 ? -id : id : 1 / id < 0 ? 1 : 0 ] + '()',
|
|
message = outOfRange ? ' out of range' : ' not a' +
|
|
( isRange ? ' non-zero' : 'n' ) + ' integer';
|
|
|
|
message = ( [
|
|
method + ' number type has more than 15 significant digits',
|
|
method + ' not a base ' + j + ' number',
|
|
method + ' base' + message,
|
|
method + ' not a number' ][i] ||
|
|
j + '() ' + i + ( isErrors
|
|
? ' not a boolean or binary digit'
|
|
: message + ( isArray
|
|
? ' or not [' + ( outOfRange
|
|
? ' negative, positive'
|
|
: ' integer, integer' ) + ' ]'
|
|
: '' ) ) ) + ': ' + arg;
|
|
|
|
outOfRange = id = 0;
|
|
error = new Error(message);
|
|
error['name'] = 'BigNumber Error';
|
|
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Convert a numeric string of baseIn to a numeric string of baseOut.
|
|
*/
|
|
function convert( nStr, baseOut, baseIn, sign ) {
|
|
var e, dvs, dvd, nArr, fracArr, fracBN;
|
|
|
|
// Convert string of base bIn to an array of numbers of baseOut.
|
|
// Eg. strToArr('255', 10) where baseOut is 16, returns [15, 15].
|
|
// Eg. strToArr('ff', 16) where baseOut is 10, returns [2, 5, 5].
|
|
function strToArr( str, bIn ) {
|
|
var j,
|
|
i = 0,
|
|
strL = str.length,
|
|
arrL,
|
|
arr = [0];
|
|
|
|
for ( bIn = bIn || baseIn; i < strL; i++ ) {
|
|
|
|
for ( arrL = arr.length, j = 0; j < arrL; arr[j] *= bIn, j++ ) {
|
|
}
|
|
|
|
for ( arr[0] += DIGITS.indexOf( str.charAt(i) ), j = 0;
|
|
j < arr.length;
|
|
j++ ) {
|
|
|
|
if ( arr[j] > baseOut - 1 ) {
|
|
|
|
if ( arr[j + 1] == null ) {
|
|
arr[j + 1] = 0;
|
|
}
|
|
arr[j + 1] += arr[j] / baseOut ^ 0;
|
|
arr[j] %= baseOut;
|
|
}
|
|
}
|
|
}
|
|
|
|
return arr.reverse();
|
|
}
|
|
|
|
// Convert array to string.
|
|
// E.g. arrToStr( [9, 10, 11] ) becomes '9ab' (in bases above 11).
|
|
function arrToStr( arr ) {
|
|
var i = 0,
|
|
arrL = arr.length,
|
|
str = '';
|
|
|
|
for ( ; i < arrL; str += DIGITS.charAt( arr[i++] ) ) {
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
if ( baseIn < 37 ) {
|
|
nStr = nStr.toLowerCase();
|
|
}
|
|
|
|
/*
|
|
* If non-integer convert integer part and fraction part separately.
|
|
* Convert the fraction part as if it is an integer than use division to
|
|
* reduce it down again to a value less than one.
|
|
*/
|
|
if ( ( e = nStr.indexOf( '.' ) ) > -1 ) {
|
|
|
|
/*
|
|
* Calculate the power to which to raise the base to get the number
|
|
* to divide the fraction part by after it has been converted as an
|
|
* integer to the required base.
|
|
*/
|
|
e = nStr.length - e - 1;
|
|
|
|
// Use toFixed to avoid possible exponential notation.
|
|
dvs = strToArr( new BigNumber(baseIn)['pow'](e)['toF'](), 10 );
|
|
|
|
nArr = nStr.split('.');
|
|
|
|
// Convert the base of the fraction part (as integer).
|
|
dvd = strToArr( nArr[1] );
|
|
|
|
// Convert the base of the integer part.
|
|
nArr = strToArr( nArr[0] );
|
|
|
|
// Result will be a BigNumber with a value less than 1.
|
|
fracBN = divide( dvd, dvs, dvd.length - dvs.length, sign, baseOut,
|
|
// Is least significant digit of integer part an odd number?
|
|
nArr[nArr.length - 1] & 1 );
|
|
|
|
fracArr = fracBN['c'];
|
|
|
|
// e can be <= 0 ( if e == 0, fracArr is [0] or [1] ).
|
|
if ( e = fracBN['e'] ) {
|
|
|
|
// Append zeros according to the exponent of the result.
|
|
for ( ; ++e; fracArr.unshift(0) ) {
|
|
}
|
|
|
|
// Append the fraction part to the converted integer part.
|
|
nStr = arrToStr(nArr) + '.' + arrToStr(fracArr);
|
|
|
|
// fracArr is [1].
|
|
// Fraction digits rounded up, so increment last digit of integer part.
|
|
} else if ( fracArr[0] ) {
|
|
|
|
if ( nArr[ e = nArr.length - 1 ] < baseOut - 1 ) {
|
|
++nArr[e];
|
|
nStr = arrToStr(nArr);
|
|
} else {
|
|
nStr = new BigNumber( arrToStr(nArr),
|
|
baseOut )['plus'](ONE)['toS'](baseOut);
|
|
}
|
|
|
|
// fracArr is [0]. No fraction digits.
|
|
} else {
|
|
nStr = arrToStr(nArr);
|
|
}
|
|
} else {
|
|
|
|
// Simple integer. Convert base.
|
|
nStr = arrToStr( strToArr(nStr) );
|
|
}
|
|
|
|
return nStr;
|
|
}
|
|
|
|
|
|
// Perform division in the specified base. Called by div and convert.
|
|
function divide( dvd, dvs, exp, s, base, isOdd ) {
|
|
var dvsL, dvsT, next, cmp, remI,
|
|
dvsZ = dvs.slice(),
|
|
dvdI = dvsL = dvs.length,
|
|
dvdL = dvd.length,
|
|
rem = dvd.slice( 0, dvsL ),
|
|
remL = rem.length,
|
|
quo = new BigNumber(ONE),
|
|
qc = quo['c'] = [],
|
|
qi = 0,
|
|
dig = DECIMAL_PLACES + ( quo['e'] = exp ) + 1;
|
|
|
|
quo['s'] = s;
|
|
s = dig < 0 ? 0 : dig;
|
|
|
|
// Add zeros to make remainder as long as divisor.
|
|
for ( ; remL++ < dvsL; rem.push(0) ) {
|
|
}
|
|
|
|
// Create version of divisor with leading zero.
|
|
dvsZ.unshift(0);
|
|
|
|
do {
|
|
|
|
// 'next' is how many times the divisor goes into the current remainder.
|
|
for ( next = 0; next < base; next++ ) {
|
|
|
|
// Compare divisor and remainder.
|
|
if ( dvsL != ( remL = rem.length ) ) {
|
|
cmp = dvsL > remL ? 1 : -1;
|
|
} else {
|
|
for ( remI = -1, cmp = 0; ++remI < dvsL; ) {
|
|
|
|
if ( dvs[remI] != rem[remI] ) {
|
|
cmp = dvs[remI] > rem[remI] ? 1 : -1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Subtract divisor from remainder (if divisor < remainder).
|
|
if ( cmp < 0 ) {
|
|
|
|
// Remainder cannot be more than one digit longer than divisor.
|
|
// Equalise lengths using divisor with extra leading zero?
|
|
for ( dvsT = remL == dvsL ? dvs : dvsZ; remL; ) {
|
|
|
|
if ( rem[--remL] < dvsT[remL] ) {
|
|
|
|
for ( remI = remL;
|
|
remI && !rem[--remI];
|
|
rem[remI] = base - 1 ) {
|
|
}
|
|
--rem[remI];
|
|
rem[remL] += base;
|
|
}
|
|
rem[remL] -= dvsT[remL];
|
|
}
|
|
for ( ; !rem[0]; rem.shift() ) {
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add the 'next' digit to the result array.
|
|
qc[qi++] = cmp ? next : ++next;
|
|
|
|
// Update the remainder.
|
|
rem[0] && cmp
|
|
? ( rem[remL] = dvd[dvdI] || 0 )
|
|
: ( rem = [ dvd[dvdI] ] );
|
|
|
|
} while ( ( dvdI++ < dvdL || rem[0] != null ) && s-- );
|
|
|
|
// Leading zero? Do not remove if result is simply zero (qi == 1).
|
|
if ( !qc[0] && qi != 1 ) {
|
|
|
|
// There can't be more than one zero.
|
|
--quo['e'];
|
|
qc.shift();
|
|
}
|
|
|
|
// Round?
|
|
if ( qi > dig ) {
|
|
rnd( quo, DECIMAL_PLACES, base, isOdd, rem[0] != null );
|
|
}
|
|
|
|
// Overflow?
|
|
if ( quo['e'] > MAX_EXP ) {
|
|
|
|
// Infinity.
|
|
quo['c'] = quo['e'] = null;
|
|
|
|
// Underflow?
|
|
} else if ( quo['e'] < MIN_EXP ) {
|
|
|
|
// Zero.
|
|
quo['c'] = [quo['e'] = 0];
|
|
}
|
|
|
|
return quo;
|
|
}
|
|
|
|
|
|
/*
|
|
* Return a string representing the value of BigNumber n in normal or
|
|
* exponential notation rounded to the specified decimal places or
|
|
* significant digits.
|
|
* Called by toString, toExponential (exp 1), toFixed, and toPrecision (exp 2).
|
|
* d is the index (with the value in normal notation) of the digit that may be
|
|
* rounded up.
|
|
*/
|
|
function format( n, d, exp ) {
|
|
|
|
// Initially, i is the number of decimal places required.
|
|
var i = d - (n = new BigNumber(n))['e'],
|
|
c = n['c'];
|
|
|
|
// +-Infinity or NaN?
|
|
if ( !c ) {
|
|
return n['toS']();
|
|
}
|
|
|
|
// Round?
|
|
if ( c.length > ++d ) {
|
|
rnd( n, i, 10 );
|
|
}
|
|
|
|
// Recalculate d if toFixed as n['e'] may have changed if value rounded up.
|
|
i = c[0] == 0 ? i + 1 : exp ? d : n['e'] + i + 1;
|
|
|
|
// Append zeros?
|
|
for ( ; c.length < i; c.push(0) ) {
|
|
}
|
|
i = n['e'];
|
|
|
|
/*
|
|
* toPrecision returns exponential notation if the number of significant
|
|
* digits specified is less than the number of digits necessary to
|
|
* represent the integer part of the value in normal notation.
|
|
*/
|
|
return exp == 1 || exp == 2 && ( --d < i || i <= TO_EXP_NEG )
|
|
|
|
// Exponential notation.
|
|
? ( n['s'] < 0 && c[0] ? '-' : '' ) + ( c.length > 1
|
|
? ( c.splice( 1, 0, '.' ), c.join('') )
|
|
: c[0] ) + ( i < 0 ? 'e' : 'e+' ) + i
|
|
|
|
// Normal notation.
|
|
: n['toS']();
|
|
}
|
|
|
|
|
|
// Round if necessary.
|
|
// Called by divide, format, setMode and sqrt.
|
|
function rnd( x, dp, base, isOdd, r ) {
|
|
var xc = x['c'],
|
|
isNeg = x['s'] < 0,
|
|
half = base / 2,
|
|
i = x['e'] + dp + 1,
|
|
|
|
// 'next' is the digit after the digit that may be rounded up.
|
|
next = xc[i],
|
|
|
|
/*
|
|
* 'more' is whether there are digits after 'next'.
|
|
* E.g.
|
|
* 0.005 (e = -3) to be rounded to 0 decimal places (dp = 0) gives i = -2
|
|
* The 'next' digit is zero, and there ARE 'more' digits after it.
|
|
* 0.5 (e = -1) dp = 0 gives i = 0
|
|
* The 'next' digit is 5 and there are no 'more' digits after it.
|
|
*/
|
|
more = r || i < 0 || xc[i + 1] != null;
|
|
|
|
r = ROUNDING_MODE < 4
|
|
? ( next != null || more ) &&
|
|
( ROUNDING_MODE == 0 ||
|
|
ROUNDING_MODE == 2 && !isNeg ||
|
|
ROUNDING_MODE == 3 && isNeg )
|
|
: next > half || next == half &&
|
|
( ROUNDING_MODE == 4 || more ||
|
|
|
|
/*
|
|
* isOdd is used in base conversion and refers to the least significant
|
|
* digit of the integer part of the value to be converted. The fraction
|
|
* part is rounded by this method separately from the integer part.
|
|
*/
|
|
ROUNDING_MODE == 6 && ( xc[i - 1] & 1 || !dp && isOdd ) ||
|
|
ROUNDING_MODE == 7 && !isNeg ||
|
|
ROUNDING_MODE == 8 && isNeg );
|
|
|
|
if ( i < 1 || !xc[0] ) {
|
|
xc.length = 0;
|
|
xc.push(0);
|
|
|
|
if ( r ) {
|
|
|
|
// 1, 0.1, 0.01, 0.001, 0.0001 etc.
|
|
xc[0] = 1;
|
|
x['e'] = -dp;
|
|
} else {
|
|
|
|
// Zero.
|
|
x['e'] = 0;
|
|
}
|
|
|
|
return x;
|
|
}
|
|
|
|
// Remove any digits after the required decimal places.
|
|
xc.length = i--;
|
|
|
|
// Round up?
|
|
if ( r ) {
|
|
|
|
// Rounding up may mean the previous digit has to be rounded up and so on.
|
|
for ( --base; ++xc[i] > base; ) {
|
|
xc[i] = 0;
|
|
|
|
if ( !i-- ) {
|
|
++x['e'];
|
|
xc.unshift(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove trailing zeros.
|
|
for ( i = xc.length; !xc[--i]; xc.pop() ) {
|
|
}
|
|
|
|
return x;
|
|
}
|
|
|
|
|
|
// Round after setting the appropriate rounding mode.
|
|
// Handles ceil, floor and round.
|
|
function setMode( x, dp, rm ) {
|
|
var r = ROUNDING_MODE;
|
|
|
|
ROUNDING_MODE = rm;
|
|
x = new BigNumber(x);
|
|
x['c'] && rnd( x, dp, 10 );
|
|
ROUNDING_MODE = r;
|
|
|
|
return x;
|
|
}
|
|
|
|
|
|
// PROTOTYPE/INSTANCE METHODS
|
|
|
|
|
|
/*
|
|
* Return a new BigNumber whose value is the absolute value of this BigNumber.
|
|
*/
|
|
P['abs'] = P['absoluteValue'] = function () {
|
|
var x = new BigNumber(this);
|
|
|
|
if ( x['s'] < 0 ) {
|
|
x['s'] = 1;
|
|
}
|
|
|
|
return x;
|
|
};
|
|
|
|
/*
|
|
* Return the bit length of the number.
|
|
*/
|
|
P['bitLength'] = function () {
|
|
return this.toString(2).length;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a new BigNumber whose value is the value of this BigNumber
|
|
* rounded to a whole number in the direction of Infinity.
|
|
*/
|
|
P['ceil'] = function () {
|
|
return setMode( this, 0, 2 );
|
|
};
|
|
|
|
|
|
/*
|
|
* Return
|
|
* 1 if the value of this BigNumber is greater than the value of BigNumber(y, b),
|
|
* -1 if the value of this BigNumber is less than the value of BigNumber(y, b),
|
|
* 0 if they have the same value,
|
|
* or null if the value of either is NaN.
|
|
*/
|
|
P['comparedTo'] = P['cmp'] = function ( y, b ) {
|
|
var a,
|
|
x = this,
|
|
xc = x['c'],
|
|
yc = ( id = -id, y = new BigNumber( y, b ) )['c'],
|
|
i = x['s'],
|
|
j = y['s'],
|
|
k = x['e'],
|
|
l = y['e'];
|
|
|
|
// Either NaN?
|
|
if ( !i || !j ) {
|
|
return null;
|
|
}
|
|
|
|
a = xc && !xc[0], b = yc && !yc[0];
|
|
|
|
// Either zero?
|
|
if ( a || b ) {
|
|
return a ? b ? 0 : -j : i;
|
|
}
|
|
|
|
// Signs differ?
|
|
if ( i != j ) {
|
|
return i;
|
|
}
|
|
|
|
// Either Infinity?
|
|
if ( a = i < 0, b = k == l, !xc || !yc ) {
|
|
return b ? 0 : !xc ^ a ? 1 : -1;
|
|
}
|
|
|
|
// Compare exponents.
|
|
if ( !b ) {
|
|
return k > l ^ a ? 1 : -1;
|
|
}
|
|
|
|
// Compare digit by digit.
|
|
for ( i = -1,
|
|
j = ( k = xc.length ) < ( l = yc.length ) ? k : l;
|
|
++i < j; ) {
|
|
|
|
if ( xc[i] != yc[i] ) {
|
|
return xc[i] > yc[i] ^ a ? 1 : -1;
|
|
}
|
|
}
|
|
// Compare lengths.
|
|
return k == l ? 0 : k > l ^ a ? 1 : -1;
|
|
};
|
|
|
|
|
|
/*
|
|
* n / 0 = I
|
|
* n / N = N
|
|
* n / I = 0
|
|
* 0 / n = 0
|
|
* 0 / 0 = N
|
|
* 0 / N = N
|
|
* 0 / I = 0
|
|
* N / n = N
|
|
* N / 0 = N
|
|
* N / N = N
|
|
* N / I = N
|
|
* I / n = I
|
|
* I / 0 = I
|
|
* I / N = N
|
|
* I / I = N
|
|
*
|
|
* Return a new BigNumber whose value is the value of this BigNumber
|
|
* divided by the value of BigNumber(y, b), rounded according to
|
|
* DECIMAL_PLACES and ROUNDING_MODE.
|
|
*/
|
|
P['dividedBy'] = P['div'] = function ( y, b ) {
|
|
var xc = this['c'],
|
|
xe = this['e'],
|
|
xs = this['s'],
|
|
yc = ( id = 2, y = new BigNumber( y, b ) )['c'],
|
|
ye = y['e'],
|
|
ys = y['s'],
|
|
s = xs == ys ? 1 : -1;
|
|
|
|
// Either NaN/Infinity/0?
|
|
return !xe && ( !xc || !xc[0] ) || !ye && ( !yc || !yc[0] )
|
|
|
|
// Either NaN?
|
|
? new BigNumber( !xs || !ys ||
|
|
|
|
// Both 0 or both Infinity?
|
|
( xc ? yc && xc[0] == yc[0] : !yc )
|
|
|
|
// Return NaN.
|
|
? NaN
|
|
|
|
// x is 0 or y is Infinity?
|
|
: xc && xc[0] == 0 || !yc
|
|
|
|
// Return +-0.
|
|
? s * 0
|
|
|
|
// y is 0. Return +-Infinity.
|
|
: s / 0 )
|
|
|
|
: divide( xc, yc, xe - ye, s, 10 );
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is equal to the value of
|
|
* BigNumber(n, b), otherwise returns false.
|
|
*/
|
|
P['equals'] = P['eq'] = function ( n, b ) {
|
|
id = 3;
|
|
return this['cmp']( n, b ) === 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a new BigNumber whose value is the value of this BigNumber
|
|
* rounded to a whole number in the direction of -Infinity.
|
|
*/
|
|
P['floor'] = function () {
|
|
return setMode( this, 0, 3 );
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is greater than the value of
|
|
* BigNumber(n, b), otherwise returns false.
|
|
*/
|
|
P['greaterThan'] = P['gt'] = function ( n, b ) {
|
|
id = 4;
|
|
return this['cmp']( n, b ) > 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is greater than or equal to
|
|
* the value of BigNumber(n, b), otherwise returns false.
|
|
*/
|
|
P['greaterThanOrEqualTo'] = P['gte'] = function ( n, b ) {
|
|
id = 5;
|
|
return ( b = this['cmp']( n, b ) ) == 1 || b === 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is a finite number, otherwise
|
|
* returns false.
|
|
*/
|
|
P['isFinite'] = P['isF'] = function () {
|
|
return !!this['c'];
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is NaN, otherwise returns
|
|
* false.
|
|
*/
|
|
P['isNaN'] = function () {
|
|
return !this['s'];
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is negative, otherwise
|
|
* returns false.
|
|
*/
|
|
P['isNegative'] = P['isNeg'] = function () {
|
|
return this['s'] < 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is 0 or -0, otherwise returns
|
|
* false.
|
|
*/
|
|
P['isZero'] = P['isZ'] = function () {
|
|
return !!this['c'] && this['c'][0] == 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is less than the value of
|
|
* BigNumber(n, b), otherwise returns false.
|
|
*/
|
|
P['lessThan'] = P['lt'] = function ( n, b ) {
|
|
id = 6;
|
|
return this['cmp']( n, b ) < 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return true if the value of this BigNumber is less than or equal to the
|
|
* value of BigNumber(n, b), otherwise returns false.
|
|
*/
|
|
P['lessThanOrEqualTo'] = P['lte'] = P['le'] = function ( n, b ) {
|
|
id = 7;
|
|
return ( b = this['cmp']( n, b ) ) == -1 || b === 0;
|
|
};
|
|
|
|
|
|
/*
|
|
* n - 0 = n
|
|
* n - N = N
|
|
* n - I = -I
|
|
* 0 - n = -n
|
|
* 0 - 0 = 0
|
|
* 0 - N = N
|
|
* 0 - I = -I
|
|
* N - n = N
|
|
* N - 0 = N
|
|
* N - N = N
|
|
* N - I = N
|
|
* I - n = I
|
|
* I - 0 = I
|
|
* I - N = N
|
|
* I - I = N
|
|
*
|
|
* Return a new BigNumber whose value is the value of this BigNumber minus
|
|
* the value of BigNumber(y, b).
|
|
*/
|
|
P['minus'] = P['sub'] = function ( y, b ) {
|
|
var d, i, j, xLTy,
|
|
x = this,
|
|
a = x['s'];
|
|
|
|
b = ( id = 8, y = new BigNumber( y, b ) )['s'];
|
|
|
|
// Either NaN?
|
|
if ( !a || !b ) {
|
|
return new BigNumber(NaN);
|
|
}
|
|
|
|
// Signs differ?
|
|
if ( a != b ) {
|
|
return y['s'] = -b, x['plus'](y);
|
|
}
|
|
|
|
var xc = x['c'],
|
|
xe = x['e'],
|
|
yc = y['c'],
|
|
ye = y['e'];
|
|
|
|
if ( !xe || !ye ) {
|
|
|
|
// Either Infinity?
|
|
if ( !xc || !yc ) {
|
|
return xc ? ( y['s'] = -b, y ) : new BigNumber( yc ? x : NaN );
|
|
}
|
|
|
|
// Either zero?
|
|
if ( !xc[0] || !yc[0] ) {
|
|
|
|
// y is non-zero?
|
|
return yc[0]
|
|
? ( y['s'] = -b, y )
|
|
|
|
// x is non-zero?
|
|
: new BigNumber( xc[0]
|
|
? x
|
|
|
|
// Both are zero.
|
|
// IEEE 754 (2008) 6.3: n - n = -0 when rounding to -Infinity
|
|
: ROUNDING_MODE == 3 ? -0 : 0 );
|
|
}
|
|
}
|
|
|
|
// Determine which is the bigger number.
|
|
// Prepend zeros to equalise exponents.
|
|
if ( xc = xc.slice(), a = xe - ye ) {
|
|
d = ( xLTy = a < 0 ) ? ( a = -a, xc ) : ( ye = xe, yc );
|
|
|
|
for ( d.reverse(), b = a; b--; d.push(0) ) {
|
|
}
|
|
d.reverse();
|
|
} else {
|
|
|
|
// Exponents equal. Check digit by digit.
|
|
j = ( ( xLTy = xc.length < yc.length ) ? xc : yc ).length;
|
|
|
|
for ( a = b = 0; b < j; b++ ) {
|
|
|
|
if ( xc[b] != yc[b] ) {
|
|
xLTy = xc[b] < yc[b];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// x < y? Point xc to the array of the bigger number.
|
|
if ( xLTy ) {
|
|
d = xc, xc = yc, yc = d;
|
|
y['s'] = -y['s'];
|
|
}
|
|
|
|
/*
|
|
* Append zeros to xc if shorter. No need to add zeros to yc if shorter
|
|
* as subtraction only needs to start at yc.length.
|
|
*/
|
|
if ( ( b = -( ( j = xc.length ) - yc.length ) ) > 0 ) {
|
|
|
|
for ( ; b--; xc[j++] = 0 ) {
|
|
}
|
|
}
|
|
|
|
// Subtract yc from xc.
|
|
for ( b = yc.length; b > a; ){
|
|
|
|
if ( xc[--b] < yc[b] ) {
|
|
|
|
for ( i = b; i && !xc[--i]; xc[i] = 9 ) {
|
|
}
|
|
--xc[i];
|
|
xc[b] += 10;
|
|
}
|
|
xc[b] -= yc[b];
|
|
}
|
|
|
|
// Remove trailing zeros.
|
|
for ( ; xc[--j] == 0; xc.pop() ) {
|
|
}
|
|
|
|
// Remove leading zeros and adjust exponent accordingly.
|
|
for ( ; xc[0] == 0; xc.shift(), --ye ) {
|
|
}
|
|
|
|
/*
|
|
* No need to check for Infinity as +x - +y != Infinity && -x - -y != Infinity
|
|
* when neither x or y are Infinity.
|
|
*/
|
|
|
|
// Underflow?
|
|
if ( ye < MIN_EXP || !xc[0] ) {
|
|
|
|
/*
|
|
* Following IEEE 754 (2008) 6.3,
|
|
* n - n = +0 but n - n = -0 when rounding towards -Infinity.
|
|
*/
|
|
if ( !xc[0] ) {
|
|
y['s'] = ROUNDING_MODE == 3 ? -1 : 1;
|
|
}
|
|
|
|
// Result is zero.
|
|
xc = [ye = 0];
|
|
}
|
|
|
|
return y['c'] = xc, y['e'] = ye, y;
|
|
};
|
|
|
|
|
|
/*
|
|
* n % 0 = N
|
|
* n % N = N
|
|
* 0 % n = 0
|
|
* -0 % n = -0
|
|
* 0 % 0 = N
|
|
* 0 % N = N
|
|
* N % n = N
|
|
* N % 0 = N
|
|
* N % N = N
|
|
*
|
|
* Return a new BigNumber whose value is the value of this BigNumber modulo
|
|
* the value of BigNumber(y, b).
|
|
*/
|
|
P['modulo'] = P['mod'] = function ( y, b ) {
|
|
var x = this,
|
|
xc = x['c'],
|
|
yc = ( id = 9, y = new BigNumber( y, b ) )['c'],
|
|
i = x['s'],
|
|
j = y['s'];
|
|
|
|
// Is x or y NaN, or y zero?
|
|
b = !i || !j || yc && !yc[0];
|
|
|
|
if ( b || xc && !xc[0] ) {
|
|
return new BigNumber( b ? NaN : x );
|
|
}
|
|
|
|
x['s'] = y['s'] = 1;
|
|
b = y['cmp'](x) == 1;
|
|
x['s'] = i, y['s'] = j;
|
|
|
|
return b
|
|
? new BigNumber(x)
|
|
: ( i = DECIMAL_PLACES, j = ROUNDING_MODE,
|
|
DECIMAL_PLACES = 0, ROUNDING_MODE = 1,
|
|
x = x['div'](y),
|
|
DECIMAL_PLACES = i, ROUNDING_MODE = j,
|
|
this['minus']( x['times'](y) ) );
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a new BigNumber whose value is the value of this BigNumber
|
|
* negated, i.e. multiplied by -1.
|
|
*/
|
|
P['negated'] = P['neg'] = function () {
|
|
var x = new BigNumber(this);
|
|
|
|
return x['s'] = -x['s'] || null, x;
|
|
};
|
|
|
|
|
|
/*
|
|
* n + 0 = n
|
|
* n + N = N
|
|
* n + I = I
|
|
* 0 + n = n
|
|
* 0 + 0 = 0
|
|
* 0 + N = N
|
|
* 0 + I = I
|
|
* N + n = N
|
|
* N + 0 = N
|
|
* N + N = N
|
|
* N + I = N
|
|
* I + n = I
|
|
* I + 0 = I
|
|
* I + N = N
|
|
* I + I = I
|
|
*
|
|
* Return a new BigNumber whose value is the value of this BigNumber plus
|
|
* the value of BigNumber(y, b).
|
|
*/
|
|
P['plus'] = P['add'] = function ( y, b ) {
|
|
var d,
|
|
x = this,
|
|
a = x['s'];
|
|
|
|
b = ( id = 10, y = new BigNumber( y, b ) )['s'];
|
|
|
|
// Either NaN?
|
|
if ( !a || !b ) {
|
|
return new BigNumber(NaN);
|
|
}
|
|
|
|
// Signs differ?
|
|
if ( a != b ) {
|
|
return y['s'] = -b, x['minus'](y);
|
|
}
|
|
|
|
var xe = x['e'],
|
|
xc = x['c'],
|
|
ye = y['e'],
|
|
yc = y['c'];
|
|
|
|
if ( !xe || !ye ) {
|
|
|
|
// Either Infinity?
|
|
if ( !xc || !yc ) {
|
|
|
|
// Return +-Infinity.
|
|
return new BigNumber( a / 0 );
|
|
}
|
|
|
|
// Either zero?
|
|
if ( !xc[0] || !yc[0] ) {
|
|
|
|
// y is non-zero?
|
|
return yc[0]
|
|
? y
|
|
|
|
// x is non-zero?
|
|
: new BigNumber( xc[0]
|
|
? x
|
|
|
|
// Both are zero. Return zero.
|
|
: a * 0 );
|
|
}
|
|
}
|
|
|
|
// Prepend zeros to equalise exponents.
|
|
// Note: Faster to use reverse then do unshifts.
|
|
if ( xc = xc.slice(), a = xe - ye ) {
|
|
d = a > 0 ? ( ye = xe, yc ) : ( a = -a, xc );
|
|
|
|
for ( d.reverse(); a--; d.push(0) ) {
|
|
}
|
|
d.reverse();
|
|
}
|
|
|
|
// Point xc to the longer array.
|
|
if ( xc.length - yc.length < 0 ) {
|
|
d = yc, yc = xc, xc = d;
|
|
}
|
|
|
|
/*
|
|
* Only start adding at yc.length - 1 as the
|
|
* further digits of xc can be left as they are.
|
|
*/
|
|
for ( a = yc.length, b = 0; a;
|
|
b = ( xc[--a] = xc[a] + yc[a] + b ) / 10 ^ 0, xc[a] %= 10 ) {
|
|
}
|
|
|
|
// No need to check for zero, as +x + +y != 0 && -x + -y != 0
|
|
|
|
if ( b ) {
|
|
xc.unshift(b);
|
|
|
|
// Overflow? (MAX_EXP + 1 possible)
|
|
if ( ++ye > MAX_EXP ) {
|
|
|
|
// Infinity.
|
|
xc = ye = null;
|
|
}
|
|
}
|
|
|
|
// Remove trailing zeros.
|
|
for ( a = xc.length; xc[--a] == 0; xc.pop() ) {
|
|
}
|
|
|
|
return y['c'] = xc, y['e'] = ye, y;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a BigNumber whose value is the value of this BigNumber raised to
|
|
* the power e. If e is negative round according to DECIMAL_PLACES and
|
|
* ROUNDING_MODE.
|
|
*
|
|
* e {number} Integer, -MAX_POWER to MAX_POWER inclusive.
|
|
*/
|
|
P['toPower'] = P['pow'] = function ( e ) {
|
|
|
|
// e to integer, avoiding NaN or Infinity becoming 0.
|
|
var i = e * 0 == 0 ? e | 0 : e,
|
|
x = new BigNumber(this),
|
|
y = new BigNumber(ONE);
|
|
|
|
// Use Math.pow?
|
|
// Pass +-Infinity for out of range exponents.
|
|
if ( ( ( ( outOfRange = e < -MAX_POWER || e > MAX_POWER ) &&
|
|
(i = e * 1 / 0) ) ||
|
|
|
|
/*
|
|
* Any exponent that fails the parse becomes NaN.
|
|
*
|
|
* Include 'e !== 0' because on Opera -0 == parseFloat(-0) is false,
|
|
* despite -0 === parseFloat(-0) && -0 == parseFloat('-0') is true.
|
|
*/
|
|
parse(e) != e && e !== 0 && !(i = NaN) ) &&
|
|
|
|
// 'pow() exponent not an integer: {e}'
|
|
// 'pow() exponent out of range: {e}'
|
|
!ifExceptionsThrow( e, 'exponent', 'pow' ) ||
|
|
|
|
// Pass zero to Math.pow, as any value to the power zero is 1.
|
|
!i ) {
|
|
|
|
// i is +-Infinity, NaN or 0.
|
|
return new BigNumber( Math.pow( x['toS'](), i ) );
|
|
}
|
|
|
|
for ( i = i < 0 ? -i : i; ; ) {
|
|
|
|
if ( i & 1 ) {
|
|
y = y['times'](x);
|
|
}
|
|
i >>= 1;
|
|
|
|
if ( !i ) {
|
|
break;
|
|
}
|
|
x = x['times'](x);
|
|
}
|
|
|
|
return e < 0 ? ONE['div'](y) : y;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a BigNumber whose value is the value of this BigNumber raised to
|
|
* the power m modulo n.
|
|
*
|
|
* m {BigNumber} the value to take the power of
|
|
* n {BigNumber} the value to modulo by
|
|
*/
|
|
P['powm'] = function ( m, n ) {
|
|
return this.pow(m).mod(n);
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a new BigNumber whose value is the value of this BigNumber
|
|
* rounded to a maximum of dp decimal places using rounding mode rm, or to
|
|
* 0 and ROUNDING_MODE respectively if omitted.
|
|
*
|
|
* [dp] {number} Integer, 0 to MAX inclusive.
|
|
* [rm] {number} Integer, 0 to 8 inclusive.
|
|
*/
|
|
P['round'] = function ( dp, rm ) {
|
|
|
|
dp = dp == null || ( ( ( outOfRange = dp < 0 || dp > MAX ) ||
|
|
parse(dp) != dp ) &&
|
|
|
|
// 'round() decimal places out of range: {dp}'
|
|
// 'round() decimal places not an integer: {dp}'
|
|
!ifExceptionsThrow( dp, 'decimal places', 'round' ) )
|
|
? 0
|
|
: dp | 0;
|
|
|
|
rm = rm == null || ( ( ( outOfRange = rm < 0 || rm > 8 ) ||
|
|
|
|
// Include '&& rm !== 0' because with Opera -0 == parseFloat(-0) is false.
|
|
parse(rm) != rm && rm !== 0 ) &&
|
|
|
|
// 'round() mode not an integer: {rm}'
|
|
// 'round() mode out of range: {rm}'
|
|
!ifExceptionsThrow( rm, 'mode', 'round' ) )
|
|
? ROUNDING_MODE
|
|
: rm | 0;
|
|
|
|
return setMode( this, dp, rm );
|
|
};
|
|
|
|
|
|
/*
|
|
* sqrt(-n) = N
|
|
* sqrt( N) = N
|
|
* sqrt(-I) = N
|
|
* sqrt( I) = I
|
|
* sqrt( 0) = 0
|
|
* sqrt(-0) = -0
|
|
*
|
|
* Return a new BigNumber whose value is the square root of the value of
|
|
* this BigNumber, rounded according to DECIMAL_PLACES and ROUNDING_MODE.
|
|
*/
|
|
P['squareRoot'] = P['sqrt'] = function () {
|
|
var n, r, re, t,
|
|
x = this,
|
|
c = x['c'],
|
|
s = x['s'],
|
|
e = x['e'],
|
|
dp = DECIMAL_PLACES,
|
|
rm = ROUNDING_MODE,
|
|
half = new BigNumber('0.5');
|
|
|
|
// Negative/NaN/Infinity/zero?
|
|
if ( s !== 1 || !c || !c[0] ) {
|
|
|
|
return new BigNumber( !s || s < 0 && ( !c || c[0] )
|
|
? NaN
|
|
: c ? x : 1 / 0 );
|
|
}
|
|
|
|
// Initial estimate.
|
|
s = Math.sqrt( x['toS']() );
|
|
ROUNDING_MODE = 1;
|
|
|
|
/*
|
|
Math.sqrt underflow/overflow?
|
|
Pass x to Math.sqrt as integer, then adjust the exponent of the result.
|
|
*/
|
|
if ( s == 0 || s == 1 / 0 ) {
|
|
n = c.join('');
|
|
|
|
if ( !( n.length + e & 1 ) ) {
|
|
n += '0';
|
|
}
|
|
r = new BigNumber( Math.sqrt(n) + '' );
|
|
|
|
// r may still not be finite.
|
|
if ( !r['c'] ) {
|
|
r['c'] = [1];
|
|
}
|
|
r['e'] = ( ( ( e + 1 ) / 2 ) | 0 ) - ( e < 0 || e & 1 );
|
|
} else {
|
|
r = new BigNumber( n = s.toString() );
|
|
}
|
|
re = r['e'];
|
|
s = re + ( DECIMAL_PLACES += 4 );
|
|
|
|
if ( s < 3 ) {
|
|
s = 0;
|
|
}
|
|
e = s;
|
|
|
|
// Newton-Raphson iteration.
|
|
for ( ; ; ) {
|
|
t = r;
|
|
r = half['times']( t['plus']( x['div'](t) ) );
|
|
|
|
if ( t['c'].slice( 0, s ).join('') === r['c'].slice( 0, s ).join('') ) {
|
|
c = r['c'];
|
|
|
|
/*
|
|
The exponent of r may here be one less than the final result
|
|
exponent (re), e.g 0.0009999 (e-4) --> 0.001 (e-3), so adjust
|
|
s so the rounding digits are indexed correctly.
|
|
*/
|
|
s = s - ( n && r['e'] < re );
|
|
|
|
/*
|
|
The 4th rounding digit may be in error by -1 so if the 4 rounding
|
|
digits are 9999 or 4999 (i.e. approaching a rounding boundary)
|
|
continue the iteration.
|
|
*/
|
|
if ( c[s] == 9 && c[s - 1] == 9 && c[s - 2] == 9 &&
|
|
( c[s - 3] == 9 || n && c[s - 3] == 4 ) ) {
|
|
|
|
/*
|
|
If 9999 on first run through, check to see if rounding up
|
|
gives the exact result as the nines may infinitely repeat.
|
|
*/
|
|
if ( n && c[s - 3] == 9 ) {
|
|
t = r['round']( dp, 0 );
|
|
|
|
if ( t['times'](t)['eq'](x) ) {
|
|
ROUNDING_MODE = rm;
|
|
DECIMAL_PLACES = dp;
|
|
|
|
return t;
|
|
}
|
|
}
|
|
DECIMAL_PLACES += 4;
|
|
s += 4;
|
|
n = '';
|
|
} else {
|
|
|
|
/*
|
|
If the rounding digits are null, 0000 or 5000, check for an
|
|
exact result. If not, then there are further digits so
|
|
increment the 1st rounding digit to ensure correct rounding.
|
|
*/
|
|
if ( !c[e] && !c[e - 1] && !c[e - 2] &&
|
|
( !c[e - 3] || c[e - 3] == 5 ) ) {
|
|
|
|
// Truncate to the first rounding digit.
|
|
if ( c.length > e - 2 ) {
|
|
c.length = e - 2;
|
|
}
|
|
|
|
if ( !r['times'](r)['eq'](x) ) {
|
|
|
|
while ( c.length < e - 3 ) {
|
|
c.push(0);
|
|
}
|
|
c[e - 3]++;
|
|
}
|
|
}
|
|
ROUNDING_MODE = rm;
|
|
rnd( r, DECIMAL_PLACES = dp, 10 );
|
|
|
|
return r;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
/*
|
|
* n * 0 = 0
|
|
* n * N = N
|
|
* n * I = I
|
|
* 0 * n = 0
|
|
* 0 * 0 = 0
|
|
* 0 * N = N
|
|
* 0 * I = N
|
|
* N * n = N
|
|
* N * 0 = N
|
|
* N * N = N
|
|
* N * I = N
|
|
* I * n = I
|
|
* I * 0 = N
|
|
* I * N = N
|
|
* I * I = I
|
|
*
|
|
* Return a new BigNumber whose value is the value of this BigNumber times
|
|
* the value of BigNumber(y, b).
|
|
*/
|
|
P['times'] = P['mul'] = function ( y, b ) {
|
|
var c,
|
|
x = this,
|
|
xc = x['c'],
|
|
yc = ( id = 11, y = new BigNumber( y, b ) )['c'],
|
|
i = x['e'],
|
|
j = y['e'],
|
|
a = x['s'];
|
|
|
|
y['s'] = a == ( b = y['s'] ) ? 1 : -1;
|
|
|
|
// Either NaN/Infinity/0?
|
|
if ( !i && ( !xc || !xc[0] ) || !j && ( !yc || !yc[0] ) ) {
|
|
|
|
// Either NaN?
|
|
return new BigNumber( !a || !b ||
|
|
|
|
// x is 0 and y is Infinity or y is 0 and x is Infinity?
|
|
xc && !xc[0] && !yc || yc && !yc[0] && !xc
|
|
|
|
// Return NaN.
|
|
? NaN
|
|
|
|
// Either Infinity?
|
|
: !xc || !yc
|
|
|
|
// Return +-Infinity.
|
|
? y['s'] / 0
|
|
|
|
// x or y is 0. Return +-0.
|
|
: y['s'] * 0 );
|
|
}
|
|
y['e'] = i + j;
|
|
|
|
if ( ( a = xc.length ) < ( b = yc.length ) ) {
|
|
c = xc, xc = yc, yc = c, j = a, a = b, b = j;
|
|
}
|
|
|
|
for ( j = a + b, c = []; j--; c.push(0) ) {
|
|
}
|
|
|
|
// Multiply!
|
|
for ( i = b - 1; i > -1; i-- ) {
|
|
|
|
for ( b = 0, j = a + i;
|
|
j > i;
|
|
b = c[j] + yc[i] * xc[j - i - 1] + b,
|
|
c[j--] = b % 10 | 0,
|
|
b = b / 10 | 0 ) {
|
|
}
|
|
|
|
if ( b ) {
|
|
c[j] = ( c[j] + b ) % 10;
|
|
}
|
|
}
|
|
|
|
b && ++y['e'];
|
|
|
|
// Remove any leading zero.
|
|
!c[0] && c.shift();
|
|
|
|
// Remove trailing zeros.
|
|
for ( j = c.length; !c[--j]; c.pop() ) {
|
|
}
|
|
|
|
// No zero check needed as only x * 0 == 0 etc.
|
|
|
|
// Overflow?
|
|
y['c'] = y['e'] > MAX_EXP
|
|
|
|
// Infinity.
|
|
? ( y['e'] = null )
|
|
|
|
// Underflow?
|
|
: y['e'] < MIN_EXP
|
|
|
|
// Zero.
|
|
? [ y['e'] = 0 ]
|
|
|
|
// Neither.
|
|
: c;
|
|
|
|
return y;
|
|
};
|
|
|
|
/*
|
|
* Return a buffer containing the
|
|
*/
|
|
P['toBuffer'] = function ( opts ) {
|
|
|
|
if (typeof opts === 'string') {
|
|
if (opts !== 'mpint') return 'Unsupported Buffer representation';
|
|
|
|
var abs = this.abs();
|
|
var buf = abs.toBuffer({ size : 1, endian : 'big' });
|
|
var len = buf.length === 1 && buf[0] === 0 ? 0 : buf.length;
|
|
if (buf[0] & 0x80) len ++;
|
|
|
|
var ret = new Buffer(4 + len);
|
|
if (len > 0) buf.copy(ret, 4 + (buf[0] & 0x80 ? 1 : 0));
|
|
if (buf[0] & 0x80) ret[4] = 0;
|
|
|
|
ret[0] = len & (0xff << 24);
|
|
ret[1] = len & (0xff << 16);
|
|
ret[2] = len & (0xff << 8);
|
|
ret[3] = len & (0xff << 0);
|
|
|
|
// two's compliment for negative integers:
|
|
var isNeg = this.lt(0);
|
|
if (isNeg) {
|
|
for (var i = 4; i < ret.length; i++) {
|
|
ret[i] = 0xff - ret[i];
|
|
}
|
|
}
|
|
ret[4] = (ret[4] & 0x7f) | (isNeg ? 0x80 : 0);
|
|
if (isNeg) ret[ret.length - 1] ++;
|
|
|
|
return ret;
|
|
}
|
|
|
|
if (!opts) opts = {};
|
|
|
|
var endian = { 1 : 'big', '-1' : 'little' }[opts.endian]
|
|
|| opts.endian || 'big'
|
|
;
|
|
|
|
var hex = this.toString(16);
|
|
if (hex.charAt(0) === '-') throw new Error(
|
|
'converting negative numbers to Buffers not supported yet'
|
|
);
|
|
|
|
var size = opts.size === 'auto' ? Math.ceil(hex.length / 2) : (opts.size || 1);
|
|
|
|
var len = Math.ceil(hex.length / (2 * size)) * size;
|
|
var buf = new Buffer(len);
|
|
|
|
// zero-pad the hex string so the chunks are all `size` long
|
|
while (hex.length < 2 * len) hex = '0' + hex;
|
|
|
|
var hx = hex
|
|
.split(new RegExp('(.{' + (2 * size) + '})'))
|
|
.filter(function (s) { return s.length > 0 })
|
|
;
|
|
|
|
hx.forEach(function (chunk, i) {
|
|
for (var j = 0; j < size; j++) {
|
|
var ix = i * size + (endian === 'big' ? j : size - j - 1);
|
|
buf[ix] = parseInt(chunk.slice(j*2,j*2+2), 16);
|
|
}
|
|
});
|
|
|
|
return buf;
|
|
};
|
|
|
|
/*
|
|
* Return a string representing the value of this BigNumber in exponential
|
|
* notation to dp fixed decimal places and rounded using ROUNDING_MODE if
|
|
* necessary.
|
|
*
|
|
* [dp] {number} Integer, 0 to MAX inclusive.
|
|
*/
|
|
P['toExponential'] = P['toE'] = function ( dp ) {
|
|
|
|
return format( this,
|
|
( dp == null || ( ( outOfRange = dp < 0 || dp > MAX ) ||
|
|
|
|
/*
|
|
* Include '&& dp !== 0' because with Opera -0 == parseFloat(-0) is
|
|
* false, despite -0 == parseFloat('-0') && 0 == -0 being true.
|
|
*/
|
|
parse(dp) != dp && dp !== 0 ) &&
|
|
|
|
// 'toE() decimal places not an integer: {dp}'
|
|
// 'toE() decimal places out of range: {dp}'
|
|
!ifExceptionsThrow( dp, 'decimal places', 'toE' ) ) && this['c']
|
|
? this['c'].length - 1
|
|
: dp | 0, 1 );
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a string representing the value of this BigNumber in normal
|
|
* notation to dp fixed decimal places and rounded using ROUNDING_MODE if
|
|
* necessary.
|
|
*
|
|
* Note: as with JavaScript's number type, (-0).toFixed(0) is '0',
|
|
* but e.g. (-0.00001).toFixed(0) is '-0'.
|
|
*
|
|
* [dp] {number} Integer, 0 to MAX inclusive.
|
|
*/
|
|
P['toFixed'] = P['toF'] = function ( dp ) {
|
|
var n, str, d,
|
|
x = this;
|
|
|
|
if ( !( dp == null || ( ( outOfRange = dp < 0 || dp > MAX ) ||
|
|
parse(dp) != dp && dp !== 0 ) &&
|
|
|
|
// 'toF() decimal places not an integer: {dp}'
|
|
// 'toF() decimal places out of range: {dp}'
|
|
!ifExceptionsThrow( dp, 'decimal places', 'toF' ) ) ) {
|
|
d = x['e'] + ( dp | 0 );
|
|
}
|
|
|
|
n = TO_EXP_NEG, dp = TO_EXP_POS;
|
|
TO_EXP_NEG = -( TO_EXP_POS = 1 / 0 );
|
|
|
|
// Note: str is initially undefined.
|
|
if ( d == str ) {
|
|
str = x['toS']();
|
|
} else {
|
|
str = format( x, d );
|
|
|
|
// (-0).toFixed() is '0', but (-0.1).toFixed() is '-0'.
|
|
// (-0).toFixed(1) is '0.0', but (-0.01).toFixed(1) is '-0.0'.
|
|
if ( x['s'] < 0 && x['c'] ) {
|
|
|
|
// As e.g. -0 toFixed(3), will wrongly be returned as -0.000 from toString.
|
|
if ( !x['c'][0] ) {
|
|
str = str.replace(/^-/, '');
|
|
|
|
// As e.g. -0.5 if rounded to -0 will cause toString to omit the minus sign.
|
|
} else if ( str.indexOf('-') < 0 ) {
|
|
str = '-' + str;
|
|
}
|
|
}
|
|
}
|
|
TO_EXP_NEG = n, TO_EXP_POS = dp;
|
|
|
|
return str;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a string array representing the value of this BigNumber as a
|
|
* simple fraction with an integer numerator and an integer denominator.
|
|
* The denominator will be a positive non-zero value less than or equal to
|
|
* the specified maximum denominator. If a maximum denominator is not
|
|
* specified, the denominator will be the lowest value necessary to
|
|
* represent the number exactly.
|
|
*
|
|
* [maxD] {number|string|BigNumber} Integer >= 1 and < Infinity.
|
|
*/
|
|
P['toFraction'] = P['toFr'] = function ( maxD ) {
|
|
var q, frac, n0, d0, d2, n, e,
|
|
n1 = d0 = new BigNumber(ONE),
|
|
d1 = n0 = new BigNumber('0'),
|
|
x = this,
|
|
xc = x['c'],
|
|
exp = MAX_EXP,
|
|
dp = DECIMAL_PLACES,
|
|
rm = ROUNDING_MODE,
|
|
d = new BigNumber(ONE);
|
|
|
|
// NaN, Infinity.
|
|
if ( !xc ) {
|
|
return x['toS']();
|
|
}
|
|
|
|
e = d['e'] = xc.length - x['e'] - 1;
|
|
|
|
// If max denominator is undefined or null...
|
|
if ( maxD == null ||
|
|
|
|
// or NaN...
|
|
( !( id = 12, n = new BigNumber(maxD) )['s'] ||
|
|
|
|
// or less than 1, or Infinity...
|
|
( outOfRange = n['cmp'](n1) < 0 || !n['c'] ) ||
|
|
|
|
// or not an integer...
|
|
( ERRORS && n['e'] < n['c'].length - 1 ) ) &&
|
|
|
|
// 'toFr() max denominator not an integer: {maxD}'
|
|
// 'toFr() max denominator out of range: {maxD}'
|
|
!ifExceptionsThrow( maxD, 'max denominator', 'toFr' ) ||
|
|
|
|
// or greater than the maxD needed to specify the value exactly...
|
|
( maxD = n )['cmp'](d) > 0 ) {
|
|
|
|
// d is e.g. 10, 100, 1000, 10000... , n1 is 1.
|
|
maxD = e > 0 ? d : n1;
|
|
}
|
|
|
|
MAX_EXP = 1 / 0;
|
|
n = new BigNumber( xc.join('') );
|
|
|
|
for ( DECIMAL_PLACES = 0, ROUNDING_MODE = 1; ; ) {
|
|
q = n['div'](d);
|
|
d2 = d0['plus']( q['times'](d1) );
|
|
|
|
if ( d2['cmp'](maxD) == 1 ) {
|
|
break;
|
|
}
|
|
|
|
d0 = d1, d1 = d2;
|
|
|
|
n1 = n0['plus']( q['times']( d2 = n1 ) );
|
|
n0 = d2;
|
|
|
|
d = n['minus']( q['times']( d2 = d ) );
|
|
n = d2;
|
|
}
|
|
|
|
d2 = maxD['minus'](d0)['div'](d1);
|
|
n0 = n0['plus']( d2['times'](n1) );
|
|
d0 = d0['plus']( d2['times'](d1) );
|
|
|
|
n0['s'] = n1['s'] = x['s'];
|
|
|
|
DECIMAL_PLACES = e * 2;
|
|
ROUNDING_MODE = rm;
|
|
|
|
// Determine which fraction is closer to x, n0 / d0 or n1 / d1?
|
|
frac = n1['div'](d1)['minus'](x)['abs']()['cmp'](
|
|
n0['div'](d0)['minus'](x)['abs']() ) < 1
|
|
? [ n1['toS'](), d1['toS']() ]
|
|
: [ n0['toS'](), d0['toS']() ];
|
|
|
|
return MAX_EXP = exp, DECIMAL_PLACES = dp, frac;
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a string representing the value of this BigNumber to sd significant
|
|
* digits and rounded using ROUNDING_MODE if necessary.
|
|
* If sd is less than the number of digits necessary to represent the integer
|
|
* part of the value in normal notation, then use exponential notation.
|
|
*
|
|
* sd {number} Integer, 1 to MAX inclusive.
|
|
*/
|
|
P['toPrecision'] = P['toP'] = function ( sd ) {
|
|
|
|
/*
|
|
* ERRORS true: Throw if sd not undefined, null or an integer in range.
|
|
* ERRORS false: Ignore sd if not a number or not in range.
|
|
* Truncate non-integers.
|
|
*/
|
|
return sd == null || ( ( ( outOfRange = sd < 1 || sd > MAX ) ||
|
|
parse(sd) != sd ) &&
|
|
|
|
// 'toP() precision not an integer: {sd}'
|
|
// 'toP() precision out of range: {sd}'
|
|
!ifExceptionsThrow( sd, 'precision', 'toP' ) )
|
|
? this['toS']()
|
|
: format( this, --sd | 0, 2 );
|
|
};
|
|
|
|
|
|
/*
|
|
* Return a string representing the value of this BigNumber in base b, or
|
|
* base 10 if b is omitted. If a base is specified, including base 10,
|
|
* round according to DECIMAL_PLACES and ROUNDING_MODE.
|
|
* If a base is not specified, and this BigNumber has a positive exponent
|
|
* that is equal to or greater than TO_EXP_POS, or a negative exponent equal
|
|
* to or less than TO_EXP_NEG, return exponential notation.
|
|
*
|
|
* [b] {number} Integer, 2 to 64 inclusive.
|
|
*/
|
|
P['toString'] = P['toS'] = function ( b ) {
|
|
var u, str, strL,
|
|
x = this,
|
|
xe = x['e'];
|
|
|
|
// Infinity or NaN?
|
|
if ( xe === null ) {
|
|
str = x['s'] ? 'Infinity' : 'NaN';
|
|
|
|
// Exponential format?
|
|
} else if ( b === u && ( xe <= TO_EXP_NEG || xe >= TO_EXP_POS ) ) {
|
|
return format( x, x['c'].length - 1, 1 );
|
|
} else {
|
|
str = x['c'].join('');
|
|
|
|
// Negative exponent?
|
|
if ( xe < 0 ) {
|
|
|
|
// Prepend zeros.
|
|
for ( ; ++xe; str = '0' + str ) {
|
|
}
|
|
str = '0.' + str;
|
|
|
|
// Positive exponent?
|
|
} else if ( strL = str.length, xe > 0 ) {
|
|
|
|
if ( ++xe > strL ) {
|
|
|
|
// Append zeros.
|
|
for ( xe -= strL; xe-- ; str += '0' ) {
|
|
}
|
|
} else if ( xe < strL ) {
|
|
str = str.slice( 0, xe ) + '.' + str.slice(xe);
|
|
}
|
|
|
|
// Exponent zero.
|
|
} else {
|
|
if ( u = str.charAt(0), strL > 1 ) {
|
|
str = u + '.' + str.slice(1);
|
|
|
|
// Avoid '-0'
|
|
} else if ( u == '0' ) {
|
|
return u;
|
|
}
|
|
}
|
|
|
|
if ( b != null ) {
|
|
|
|
if ( !( outOfRange = !( b >= 2 && b < 65 ) ) &&
|
|
( b == (b | 0) || !ERRORS ) ) {
|
|
str = convert( str, b | 0, 10, x['s'] );
|
|
|
|
// Avoid '-0'
|
|
if ( str == '0' ) {
|
|
return str;
|
|
}
|
|
} else {
|
|
|
|
// 'toS() base not an integer: {b}'
|
|
// 'toS() base out of range: {b}'
|
|
ifExceptionsThrow( b, 'base', 'toS' );
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return x['s'] < 0 ? '-' + str : str;
|
|
};
|
|
|
|
P['toNumber'] = function () {
|
|
return parseInt(this['toString'](), 10);
|
|
};
|
|
|
|
|
|
/*
|
|
* Return as toString, but do not accept a base argument.
|
|
*/
|
|
P['valueOf'] = function () {
|
|
return this['toS']();
|
|
};
|
|
|
|
|
|
// Add aliases for BigDecimal methods.
|
|
//P['add'] = P['plus'];
|
|
//P['subtract'] = P['minus'];
|
|
//P['multiply'] = P['times'];
|
|
//P['divide'] = P['div'];
|
|
//P['remainder'] = P['mod'];
|
|
//P['compareTo'] = P['cmp'];
|
|
//P['negate'] = P['neg'];
|
|
|
|
|
|
// EXPORT
|
|
BigNumber.config({EXPONENTIAL_AT: 9999999, DECIMAL_PLACES: 0, ROUNDING_MODE: 1});
|
|
module.exports = BigNumber;
|
|
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80}],"ALJ4PS":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var ECKey = require('../../browser/vendor-bundle.js').ECKey;
|
|
var SecureRandom = require('../SecureRandom');
|
|
var Curve = require('../Curve');
|
|
var bignum = require('bignum');
|
|
|
|
var Key = function() {
|
|
this._pub = null;
|
|
this._compressed = true; // default
|
|
};
|
|
|
|
var bufferToArray = Key.bufferToArray = function(buffer) {
|
|
var ret = [];
|
|
|
|
var l = buffer.length;
|
|
for(var i =0; i<l; i++) {
|
|
ret.push(buffer.readUInt8(i));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
Object.defineProperty(Key.prototype, 'public', {
|
|
set: function(p){
|
|
if (!Buffer.isBuffer(p) ) {
|
|
throw new Error('Arg should be a buffer');
|
|
}
|
|
var type = p[0];
|
|
this._compressed = type!==0x04;
|
|
this._pub = p;
|
|
},
|
|
get: function(){
|
|
return this._pub;
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(Key.prototype, 'compressed', {
|
|
set: function(c) {
|
|
var oldc = this._compressed;
|
|
this._compressed = !!c;
|
|
if (oldc == this._compressed)
|
|
return;
|
|
var oldp = this._pub;
|
|
if (this._pub) {
|
|
var eckey = new ECKey();
|
|
eckey.setPub(bufferToArray(this.public));
|
|
eckey.setCompressed(this._compressed);
|
|
this._pub = new Buffer(eckey.getPub());
|
|
}
|
|
if (!this._compressed) {
|
|
//bug in eckey
|
|
//oldp.slice(1).copy(this._pub, 1);
|
|
}
|
|
},
|
|
get: function() {
|
|
return this._compressed;
|
|
}
|
|
});
|
|
|
|
Key.generateSync = function() {
|
|
var privbuf;
|
|
|
|
while(true) {
|
|
privbuf = SecureRandom.getRandomBuffer(32);
|
|
if ((bignum.fromBuffer(privbuf, {size: 32})).cmp(Curve.getN()) < 0)
|
|
break;
|
|
}
|
|
|
|
var privhex = privbuf.toString('hex');
|
|
var eck = new ECKey(privhex);
|
|
eck.setCompressed(true);
|
|
var pub = eck.getPub();
|
|
|
|
ret = new Key();
|
|
ret.private = privbuf;
|
|
ret._compressed = true;
|
|
ret.public = new Buffer(eck.getPub());
|
|
|
|
return ret;
|
|
};
|
|
|
|
Key.prototype.regenerateSync = function() {
|
|
if (!this.private) {
|
|
throw new Error('Key does not have a private key set');
|
|
}
|
|
|
|
var eck = new ECKey(this.private.toString('hex'));
|
|
eck.setCompressed(this._compressed);
|
|
this._pub = new Buffer(eck.getPub());
|
|
return this;
|
|
};
|
|
|
|
Key.prototype.signSync = function(hash) {
|
|
var getSECCurveByName = require('../../browser/vendor-bundle.js').getSECCurveByName;
|
|
var BigInteger = require('../../browser/vendor-bundle.js').BigInteger;
|
|
var rng = new SecureRandom();
|
|
var ecparams = getSECCurveByName('secp256k1');
|
|
|
|
var rng = {};
|
|
rng.nextBytes = function(array) {
|
|
var buf = SecureRandom.getRandomBuffer(array.length);
|
|
var a = bufferToArray(SecureRandom.getRandomBuffer(array.length));
|
|
for (var i in a) {
|
|
array[i] = a[i];
|
|
}
|
|
};
|
|
|
|
var getBigRandom = function (limit) {
|
|
return new BigInteger(limit.bitLength(), rng)
|
|
.mod(limit.subtract(BigInteger.ONE))
|
|
.add(BigInteger.ONE);
|
|
};
|
|
|
|
var sign = function (hash, priv) {
|
|
var d = priv;
|
|
var n = ecparams.getN();
|
|
var e = BigInteger.fromByteArrayUnsigned(hash);
|
|
|
|
do {
|
|
var k = getBigRandom(n);
|
|
var G = ecparams.getG();
|
|
var Q = G.multiply(k);
|
|
var r = Q.getX().toBigInteger().mod(n);
|
|
} while (r.compareTo(BigInteger.ZERO) <= 0);
|
|
|
|
var s = k.modInverse(n).multiply(e.add(d.multiply(r))).mod(n);
|
|
|
|
return serializeSig(r, s);
|
|
};
|
|
|
|
var serializeSig = function (r, s) {
|
|
var rBa = r.toByteArraySigned();
|
|
var sBa = s.toByteArraySigned();
|
|
|
|
var sequence = [];
|
|
sequence.push(0x02); // INTEGER
|
|
sequence.push(rBa.length);
|
|
sequence = sequence.concat(rBa);
|
|
|
|
sequence.push(0x02); // INTEGER
|
|
sequence.push(sBa.length);
|
|
sequence = sequence.concat(sBa);
|
|
|
|
sequence.unshift(sequence.length);
|
|
sequence.unshift(0x30); // SEQUENCE
|
|
|
|
return sequence;
|
|
};
|
|
|
|
if (!this.private) {
|
|
throw new Error('Key does not have a private key set');
|
|
}
|
|
|
|
if (!Buffer.isBuffer(hash) || hash.length !== 32) {
|
|
throw new Error('Arg should be a 32 bytes hash buffer');
|
|
}
|
|
var privhex = this.private.toString('hex');
|
|
var privnum = new BigInteger(privhex, 16);
|
|
var signature = sign(bufferToArray(hash), privnum);
|
|
|
|
return new Buffer(signature);
|
|
};
|
|
|
|
Key.prototype.verifySignature = function(hash, sig, callback) {
|
|
try {
|
|
var result = this.verifySignatureSync(hash, sig);
|
|
callback(null, result);
|
|
} catch (e) {
|
|
callback(e);
|
|
}
|
|
};
|
|
|
|
Key.prototype.verifySignatureSync = function(hash, sig) {
|
|
var self = this;
|
|
|
|
if (!Buffer.isBuffer(hash) || hash.length !== 32) {
|
|
throw new Error('Arg 1 should be a 32 bytes hash buffer');
|
|
}
|
|
if (!Buffer.isBuffer(sig)) {
|
|
throw new Error('Arg 2 should be a buffer');
|
|
}
|
|
if (!self.public) {
|
|
throw new Error('Key does not have a public key set');
|
|
}
|
|
|
|
var eck = new ECKey();
|
|
eck.setPub(bufferToArray(self.public));
|
|
eck.setCompressed(self._compressed);
|
|
var sigA = bufferToArray(sig);
|
|
var ret = eck.verify(bufferToArray(hash),sigA);
|
|
return ret;
|
|
};
|
|
|
|
module.exports = Key;
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../../browser/vendor-bundle.js":3,"../Curve":"Ynul1S","../SecureRandom":"p4SiC2","bignum":58,"buffer":80}],"./lib/Key":[function(require,module,exports){
|
|
module.exports=require('ALJ4PS');
|
|
},{}],"./lib/Point":[function(require,module,exports){
|
|
module.exports=require('6tXgqr');
|
|
},{}],"6tXgqr":[function(require,module,exports){
|
|
(function (Buffer){
|
|
"use strict";
|
|
|
|
var imports = require('soop').imports();
|
|
var Key = imports.Key || require('./Key');
|
|
var bignum = imports.bignum || require('bignum');
|
|
var assert = require('assert');
|
|
var ECPointFp = require('../../browser/vendor-bundle.js').ECPointFp;
|
|
var ECFieldElementFp = require('../../browser/vendor-bundle.js').ECFieldElementFp;
|
|
var getSECCurveByName = require('../../browser/vendor-bundle.js').getSECCurveByName;
|
|
var BigInteger = require('../../browser/vendor-bundle.js').BigInteger;
|
|
|
|
//a point on the secp256k1 curve
|
|
//x and y are bignums
|
|
var Point = function(x, y) {
|
|
this.x = x;
|
|
this.y = y;
|
|
};
|
|
|
|
Point.add = function(p1, p2) {
|
|
var ecparams = getSECCurveByName('secp256k1');
|
|
|
|
var p1xhex = p1.x.toBuffer({size: 32}).toString('hex');
|
|
var p1x = new BigInteger(p1xhex, 16);
|
|
var p1yhex = p1.y.toBuffer({size: 32}).toString('hex');
|
|
var p1y = new BigInteger(p1yhex, 16);
|
|
var p1px = new ECFieldElementFp(ecparams.getCurve().getQ(), p1x);
|
|
var p1py = new ECFieldElementFp(ecparams.getCurve().getQ(), p1y);
|
|
var p1p = new ECPointFp(ecparams.getCurve(), p1px, p1py);
|
|
|
|
var p2xhex = p2.x.toBuffer({size: 32}).toString('hex');
|
|
var p2x = new BigInteger(p2xhex, 16);
|
|
var p2yhex = p2.y.toBuffer({size: 32}).toString('hex');
|
|
var p2y = new BigInteger(p2yhex, 16);
|
|
var p2px = new ECFieldElementFp(ecparams.getCurve().getQ(), p2x);
|
|
var p2py = new ECFieldElementFp(ecparams.getCurve().getQ(), p2y);
|
|
var p2p = new ECPointFp(ecparams.getCurve(), p2px, p2py);
|
|
|
|
var p = p1p.add(p2p);
|
|
|
|
var point = new Point();
|
|
var pointxbuf = new Buffer(p.getX().toBigInteger().toByteArrayUnsigned());
|
|
point.x = bignum.fromBuffer(pointxbuf, {size: pointxbuf.length});
|
|
assert(pointxbuf.length <= 32);
|
|
var pointybuf = new Buffer(p.getY().toBigInteger().toByteArrayUnsigned());
|
|
assert(pointybuf.length <= 32);
|
|
point.y = bignum.fromBuffer(pointybuf, {size: pointybuf.length});
|
|
|
|
return point;
|
|
};
|
|
|
|
Point.multiply = function(p1, x) {
|
|
var x = new BigInteger(x.toString('hex'), 16);
|
|
|
|
var ecparams = getSECCurveByName('secp256k1');
|
|
|
|
var p1xhex = p1.x.toBuffer({size: 32}).toString('hex');
|
|
var p1x = new BigInteger(p1xhex, 16);
|
|
var p1yhex = p1.y.toBuffer({size: 32}).toString('hex');
|
|
var p1y = new BigInteger(p1yhex, 16);
|
|
var p1px = new ECFieldElementFp(ecparams.getCurve().getQ(), p1x);
|
|
var p1py = new ECFieldElementFp(ecparams.getCurve().getQ(), p1y);
|
|
var p1p = new ECPointFp(ecparams.getCurve(), p1px, p1py);
|
|
|
|
var p = p1p.multiply(x);
|
|
|
|
var point = new Point();
|
|
var pointxbuf = new Buffer(p.getX().toBigInteger().toByteArrayUnsigned());
|
|
point.x = bignum.fromBuffer(pointxbuf, {size: pointxbuf.length});
|
|
assert(pointxbuf.length <= 32);
|
|
var pointybuf = new Buffer(p.getY().toBigInteger().toByteArrayUnsigned());
|
|
assert(pointybuf.length <= 32);
|
|
point.y = bignum.fromBuffer(pointybuf, {size: pointybuf.length});
|
|
|
|
return point;
|
|
};
|
|
|
|
//convert the public key of a Key into a Point
|
|
Point.fromUncompressedPubKey = function(pubkey) {
|
|
var point = new Point();
|
|
point.x = bignum.fromBuffer((new Buffer(pubkey)).slice(1, 33), {size: 32});
|
|
point.y = bignum.fromBuffer((new Buffer(pubkey)).slice(33, 65), {size: 32});
|
|
return point;
|
|
};
|
|
|
|
//convert the Point into the Key containing a compressed public key
|
|
Point.prototype.toUncompressedPubKey = function() {
|
|
var xbuf = this.x.toBuffer({size: 32});
|
|
var ybuf = this.y.toBuffer({size: 32});
|
|
var prefix = new Buffer([0x04]);
|
|
var pub = Buffer.concat([prefix, xbuf, ybuf]);
|
|
return pub;
|
|
};
|
|
|
|
module.exports = require('soop')(Point);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../../browser/vendor-bundle.js":3,"./Key":"ALJ4PS","assert":77,"bignum":58,"buffer":80,"soop":125}],"p4SiC2":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop');
|
|
|
|
var SecureRandom = require('../common/SecureRandom');
|
|
|
|
SecureRandom.getRandomBuffer = function(size) {
|
|
if (!window.crypto && !window.msCrypto)
|
|
throw new Error('window.crypto not available');
|
|
|
|
if (window.crypto && window.crypto.getRandomValues)
|
|
var crypto = window.crypto;
|
|
else if (window.msCrypto && window.msCrypto.getRandomValues) //internet explorer
|
|
var crypto = window.msCrypto;
|
|
else
|
|
throw new Error('window.crypto.getRandomValues not available');
|
|
|
|
var bbuf = new Uint8Array(size);
|
|
crypto.getRandomValues(bbuf);
|
|
var buf = new Buffer(bbuf);
|
|
|
|
return buf;
|
|
};
|
|
|
|
module.exports = require('soop')(SecureRandom);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../common/SecureRandom":65,"buffer":80,"soop":125}],"./lib/SecureRandom":[function(require,module,exports){
|
|
module.exports=require('p4SiC2');
|
|
},{}],65:[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop');
|
|
|
|
var SecureRandom = function() {
|
|
};
|
|
|
|
/* secure random bytes that sometimes throws an error due to lack of entropy */
|
|
SecureRandom.getRandomBuffer = function() {};
|
|
|
|
/* insecure random bytes, but it never fails */
|
|
SecureRandom.getPseudoRandomBuffer = function(size) {
|
|
var b32 = 0x100000000;
|
|
var b = new Buffer(size);
|
|
|
|
for (var i = 0; i <= size; i++) {
|
|
var j = Math.floor(i / 4);
|
|
var k = i - j * 4;
|
|
if (k == 0) {
|
|
r = Math.random() * b32;
|
|
b[i] = r & 0xff;
|
|
} else {
|
|
b[i] = (r = r >>> 8) & 0xff;
|
|
}
|
|
}
|
|
|
|
return b;
|
|
};
|
|
|
|
module.exports = require('soop')(SecureRandom);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80,"soop":125}],"./networks":[function(require,module,exports){
|
|
module.exports=require('ULNIu2');
|
|
},{}],"ULNIu2":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var Put = require('bufferput');
|
|
var buffertools = require('buffertools');
|
|
var hex = function(hex) {return new Buffer(hex, 'hex');};
|
|
|
|
exports.livenet = {
|
|
name: 'livenet',
|
|
magic: hex('f9beb4d9'),
|
|
addressVersion: 0x00,
|
|
privKeyVersion: 128,
|
|
P2SHVersion: 5,
|
|
hkeyPublicVersion: 0x0488b21e,
|
|
hkeyPrivateVersion: 0x0488ade4,
|
|
genesisBlock: {
|
|
hash: hex('6FE28C0AB6F1B372C1A6A246AE63F74F931E8365E15A089C68D6190000000000'),
|
|
merkle_root: hex('3BA3EDFD7A7B12B27AC72C3E67768F617FC81BC3888A51323A9FB8AA4B1E5E4A'),
|
|
height: 0,
|
|
nonce: 2083236893,
|
|
version: 1,
|
|
prev_hash: buffertools.fill(new Buffer(32), 0),
|
|
timestamp: 1231006505,
|
|
bits: 486604799,
|
|
},
|
|
dnsSeeds: [
|
|
'seed.bitcoin.sipa.be',
|
|
'dnsseed.bluematt.me',
|
|
'dnsseed.bitcoin.dashjr.org',
|
|
'seed.bitcoinstats.com',
|
|
'seed.bitnodes.io',
|
|
'bitseed.xf2.org'
|
|
],
|
|
defaultClientPort: 8333
|
|
};
|
|
|
|
exports.testnet = {
|
|
name: 'testnet',
|
|
magic: hex('0b110907'),
|
|
addressVersion: 0x6f,
|
|
privKeyVersion: 239,
|
|
P2SHVersion: 196,
|
|
hkeyPublicVersion: 0x043587cf,
|
|
hkeyPrivateVersion: 0x04358394,
|
|
genesisBlock: {
|
|
hash: hex('43497FD7F826957108F4A30FD9CEC3AEBA79972084E90EAD01EA330900000000'),
|
|
merkle_root: hex('3BA3EDFD7A7B12B27AC72C3E67768F617FC81BC3888A51323A9FB8AA4B1E5E4A'),
|
|
height: 0,
|
|
nonce: 414098458,
|
|
version: 1,
|
|
prev_hash: buffertools.fill(new Buffer(32), 0),
|
|
timestamp: 1296688602,
|
|
bits: 486604799,
|
|
},
|
|
dnsSeeds: [
|
|
'testnet-seed.bitcoin.petertodd.org',
|
|
'testnet-seed.bluematt.me'
|
|
],
|
|
defaultClientPort: 18333
|
|
};
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80,"bufferput":"aXRuS6","buffertools":"fugeBw"}],68:[function(require,module,exports){
|
|
(function (process){
|
|
/*global setImmediate: false, setTimeout: false, console: false */
|
|
(function () {
|
|
|
|
var async = {};
|
|
|
|
// global on the server, window in the browser
|
|
var root, previous_async;
|
|
|
|
root = this;
|
|
if (root != null) {
|
|
previous_async = root.async;
|
|
}
|
|
|
|
async.noConflict = function () {
|
|
root.async = previous_async;
|
|
return async;
|
|
};
|
|
|
|
function only_once(fn) {
|
|
var called = false;
|
|
return function() {
|
|
if (called) throw new Error("Callback was already called.");
|
|
called = true;
|
|
fn.apply(root, arguments);
|
|
}
|
|
}
|
|
|
|
//// cross-browser compatiblity functions ////
|
|
|
|
var _each = function (arr, iterator) {
|
|
if (arr.forEach) {
|
|
return arr.forEach(iterator);
|
|
}
|
|
for (var i = 0; i < arr.length; i += 1) {
|
|
iterator(arr[i], i, arr);
|
|
}
|
|
};
|
|
|
|
var _map = function (arr, iterator) {
|
|
if (arr.map) {
|
|
return arr.map(iterator);
|
|
}
|
|
var results = [];
|
|
_each(arr, function (x, i, a) {
|
|
results.push(iterator(x, i, a));
|
|
});
|
|
return results;
|
|
};
|
|
|
|
var _reduce = function (arr, iterator, memo) {
|
|
if (arr.reduce) {
|
|
return arr.reduce(iterator, memo);
|
|
}
|
|
_each(arr, function (x, i, a) {
|
|
memo = iterator(memo, x, i, a);
|
|
});
|
|
return memo;
|
|
};
|
|
|
|
var _keys = function (obj) {
|
|
if (Object.keys) {
|
|
return Object.keys(obj);
|
|
}
|
|
var keys = [];
|
|
for (var k in obj) {
|
|
if (obj.hasOwnProperty(k)) {
|
|
keys.push(k);
|
|
}
|
|
}
|
|
return keys;
|
|
};
|
|
|
|
//// exported async module functions ////
|
|
|
|
//// nextTick implementation with browser-compatible fallback ////
|
|
if (typeof process === 'undefined' || !(process.nextTick)) {
|
|
if (typeof setImmediate === 'function') {
|
|
async.nextTick = function (fn) {
|
|
// not a direct alias for IE10 compatibility
|
|
setImmediate(fn);
|
|
};
|
|
async.setImmediate = async.nextTick;
|
|
}
|
|
else {
|
|
async.nextTick = function (fn) {
|
|
setTimeout(fn, 0);
|
|
};
|
|
async.setImmediate = async.nextTick;
|
|
}
|
|
}
|
|
else {
|
|
async.nextTick = process.nextTick;
|
|
if (typeof setImmediate !== 'undefined') {
|
|
async.setImmediate = function (fn) {
|
|
// not a direct alias for IE10 compatibility
|
|
setImmediate(fn);
|
|
};
|
|
}
|
|
else {
|
|
async.setImmediate = async.nextTick;
|
|
}
|
|
}
|
|
|
|
async.each = function (arr, iterator, callback) {
|
|
callback = callback || function () {};
|
|
if (!arr.length) {
|
|
return callback();
|
|
}
|
|
var completed = 0;
|
|
_each(arr, function (x) {
|
|
iterator(x, only_once(function (err) {
|
|
if (err) {
|
|
callback(err);
|
|
callback = function () {};
|
|
}
|
|
else {
|
|
completed += 1;
|
|
if (completed >= arr.length) {
|
|
callback(null);
|
|
}
|
|
}
|
|
}));
|
|
});
|
|
};
|
|
async.forEach = async.each;
|
|
|
|
async.eachSeries = function (arr, iterator, callback) {
|
|
callback = callback || function () {};
|
|
if (!arr.length) {
|
|
return callback();
|
|
}
|
|
var completed = 0;
|
|
var iterate = function () {
|
|
iterator(arr[completed], function (err) {
|
|
if (err) {
|
|
callback(err);
|
|
callback = function () {};
|
|
}
|
|
else {
|
|
completed += 1;
|
|
if (completed >= arr.length) {
|
|
callback(null);
|
|
}
|
|
else {
|
|
iterate();
|
|
}
|
|
}
|
|
});
|
|
};
|
|
iterate();
|
|
};
|
|
async.forEachSeries = async.eachSeries;
|
|
|
|
async.eachLimit = function (arr, limit, iterator, callback) {
|
|
var fn = _eachLimit(limit);
|
|
fn.apply(null, [arr, iterator, callback]);
|
|
};
|
|
async.forEachLimit = async.eachLimit;
|
|
|
|
var _eachLimit = function (limit) {
|
|
|
|
return function (arr, iterator, callback) {
|
|
callback = callback || function () {};
|
|
if (!arr.length || limit <= 0) {
|
|
return callback();
|
|
}
|
|
var completed = 0;
|
|
var started = 0;
|
|
var running = 0;
|
|
|
|
(function replenish () {
|
|
if (completed >= arr.length) {
|
|
return callback();
|
|
}
|
|
|
|
while (running < limit && started < arr.length) {
|
|
started += 1;
|
|
running += 1;
|
|
iterator(arr[started - 1], function (err) {
|
|
if (err) {
|
|
callback(err);
|
|
callback = function () {};
|
|
}
|
|
else {
|
|
completed += 1;
|
|
running -= 1;
|
|
if (completed >= arr.length) {
|
|
callback();
|
|
}
|
|
else {
|
|
replenish();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
})();
|
|
};
|
|
};
|
|
|
|
|
|
var doParallel = function (fn) {
|
|
return function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
return fn.apply(null, [async.each].concat(args));
|
|
};
|
|
};
|
|
var doParallelLimit = function(limit, fn) {
|
|
return function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
return fn.apply(null, [_eachLimit(limit)].concat(args));
|
|
};
|
|
};
|
|
var doSeries = function (fn) {
|
|
return function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
return fn.apply(null, [async.eachSeries].concat(args));
|
|
};
|
|
};
|
|
|
|
|
|
var _asyncMap = function (eachfn, arr, iterator, callback) {
|
|
var results = [];
|
|
arr = _map(arr, function (x, i) {
|
|
return {index: i, value: x};
|
|
});
|
|
eachfn(arr, function (x, callback) {
|
|
iterator(x.value, function (err, v) {
|
|
results[x.index] = v;
|
|
callback(err);
|
|
});
|
|
}, function (err) {
|
|
callback(err, results);
|
|
});
|
|
};
|
|
async.map = doParallel(_asyncMap);
|
|
async.mapSeries = doSeries(_asyncMap);
|
|
async.mapLimit = function (arr, limit, iterator, callback) {
|
|
return _mapLimit(limit)(arr, iterator, callback);
|
|
};
|
|
|
|
var _mapLimit = function(limit) {
|
|
return doParallelLimit(limit, _asyncMap);
|
|
};
|
|
|
|
// reduce only has a series version, as doing reduce in parallel won't
|
|
// work in many situations.
|
|
async.reduce = function (arr, memo, iterator, callback) {
|
|
async.eachSeries(arr, function (x, callback) {
|
|
iterator(memo, x, function (err, v) {
|
|
memo = v;
|
|
callback(err);
|
|
});
|
|
}, function (err) {
|
|
callback(err, memo);
|
|
});
|
|
};
|
|
// inject alias
|
|
async.inject = async.reduce;
|
|
// foldl alias
|
|
async.foldl = async.reduce;
|
|
|
|
async.reduceRight = function (arr, memo, iterator, callback) {
|
|
var reversed = _map(arr, function (x) {
|
|
return x;
|
|
}).reverse();
|
|
async.reduce(reversed, memo, iterator, callback);
|
|
};
|
|
// foldr alias
|
|
async.foldr = async.reduceRight;
|
|
|
|
var _filter = function (eachfn, arr, iterator, callback) {
|
|
var results = [];
|
|
arr = _map(arr, function (x, i) {
|
|
return {index: i, value: x};
|
|
});
|
|
eachfn(arr, function (x, callback) {
|
|
iterator(x.value, function (v) {
|
|
if (v) {
|
|
results.push(x);
|
|
}
|
|
callback();
|
|
});
|
|
}, function (err) {
|
|
callback(_map(results.sort(function (a, b) {
|
|
return a.index - b.index;
|
|
}), function (x) {
|
|
return x.value;
|
|
}));
|
|
});
|
|
};
|
|
async.filter = doParallel(_filter);
|
|
async.filterSeries = doSeries(_filter);
|
|
// select alias
|
|
async.select = async.filter;
|
|
async.selectSeries = async.filterSeries;
|
|
|
|
var _reject = function (eachfn, arr, iterator, callback) {
|
|
var results = [];
|
|
arr = _map(arr, function (x, i) {
|
|
return {index: i, value: x};
|
|
});
|
|
eachfn(arr, function (x, callback) {
|
|
iterator(x.value, function (v) {
|
|
if (!v) {
|
|
results.push(x);
|
|
}
|
|
callback();
|
|
});
|
|
}, function (err) {
|
|
callback(_map(results.sort(function (a, b) {
|
|
return a.index - b.index;
|
|
}), function (x) {
|
|
return x.value;
|
|
}));
|
|
});
|
|
};
|
|
async.reject = doParallel(_reject);
|
|
async.rejectSeries = doSeries(_reject);
|
|
|
|
var _detect = function (eachfn, arr, iterator, main_callback) {
|
|
eachfn(arr, function (x, callback) {
|
|
iterator(x, function (result) {
|
|
if (result) {
|
|
main_callback(x);
|
|
main_callback = function () {};
|
|
}
|
|
else {
|
|
callback();
|
|
}
|
|
});
|
|
}, function (err) {
|
|
main_callback();
|
|
});
|
|
};
|
|
async.detect = doParallel(_detect);
|
|
async.detectSeries = doSeries(_detect);
|
|
|
|
async.some = function (arr, iterator, main_callback) {
|
|
async.each(arr, function (x, callback) {
|
|
iterator(x, function (v) {
|
|
if (v) {
|
|
main_callback(true);
|
|
main_callback = function () {};
|
|
}
|
|
callback();
|
|
});
|
|
}, function (err) {
|
|
main_callback(false);
|
|
});
|
|
};
|
|
// any alias
|
|
async.any = async.some;
|
|
|
|
async.every = function (arr, iterator, main_callback) {
|
|
async.each(arr, function (x, callback) {
|
|
iterator(x, function (v) {
|
|
if (!v) {
|
|
main_callback(false);
|
|
main_callback = function () {};
|
|
}
|
|
callback();
|
|
});
|
|
}, function (err) {
|
|
main_callback(true);
|
|
});
|
|
};
|
|
// all alias
|
|
async.all = async.every;
|
|
|
|
async.sortBy = function (arr, iterator, callback) {
|
|
async.map(arr, function (x, callback) {
|
|
iterator(x, function (err, criteria) {
|
|
if (err) {
|
|
callback(err);
|
|
}
|
|
else {
|
|
callback(null, {value: x, criteria: criteria});
|
|
}
|
|
});
|
|
}, function (err, results) {
|
|
if (err) {
|
|
return callback(err);
|
|
}
|
|
else {
|
|
var fn = function (left, right) {
|
|
var a = left.criteria, b = right.criteria;
|
|
return a < b ? -1 : a > b ? 1 : 0;
|
|
};
|
|
callback(null, _map(results.sort(fn), function (x) {
|
|
return x.value;
|
|
}));
|
|
}
|
|
});
|
|
};
|
|
|
|
async.auto = function (tasks, callback) {
|
|
callback = callback || function () {};
|
|
var keys = _keys(tasks);
|
|
if (!keys.length) {
|
|
return callback(null);
|
|
}
|
|
|
|
var results = {};
|
|
|
|
var listeners = [];
|
|
var addListener = function (fn) {
|
|
listeners.unshift(fn);
|
|
};
|
|
var removeListener = function (fn) {
|
|
for (var i = 0; i < listeners.length; i += 1) {
|
|
if (listeners[i] === fn) {
|
|
listeners.splice(i, 1);
|
|
return;
|
|
}
|
|
}
|
|
};
|
|
var taskComplete = function () {
|
|
_each(listeners.slice(0), function (fn) {
|
|
fn();
|
|
});
|
|
};
|
|
|
|
addListener(function () {
|
|
if (_keys(results).length === keys.length) {
|
|
callback(null, results);
|
|
callback = function () {};
|
|
}
|
|
});
|
|
|
|
_each(keys, function (k) {
|
|
var task = (tasks[k] instanceof Function) ? [tasks[k]]: tasks[k];
|
|
var taskCallback = function (err) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
if (args.length <= 1) {
|
|
args = args[0];
|
|
}
|
|
if (err) {
|
|
var safeResults = {};
|
|
_each(_keys(results), function(rkey) {
|
|
safeResults[rkey] = results[rkey];
|
|
});
|
|
safeResults[k] = args;
|
|
callback(err, safeResults);
|
|
// stop subsequent errors hitting callback multiple times
|
|
callback = function () {};
|
|
}
|
|
else {
|
|
results[k] = args;
|
|
async.setImmediate(taskComplete);
|
|
}
|
|
};
|
|
var requires = task.slice(0, Math.abs(task.length - 1)) || [];
|
|
var ready = function () {
|
|
return _reduce(requires, function (a, x) {
|
|
return (a && results.hasOwnProperty(x));
|
|
}, true) && !results.hasOwnProperty(k);
|
|
};
|
|
if (ready()) {
|
|
task[task.length - 1](taskCallback, results);
|
|
}
|
|
else {
|
|
var listener = function () {
|
|
if (ready()) {
|
|
removeListener(listener);
|
|
task[task.length - 1](taskCallback, results);
|
|
}
|
|
};
|
|
addListener(listener);
|
|
}
|
|
});
|
|
};
|
|
|
|
async.waterfall = function (tasks, callback) {
|
|
callback = callback || function () {};
|
|
if (tasks.constructor !== Array) {
|
|
var err = new Error('First argument to waterfall must be an array of functions');
|
|
return callback(err);
|
|
}
|
|
if (!tasks.length) {
|
|
return callback();
|
|
}
|
|
var wrapIterator = function (iterator) {
|
|
return function (err) {
|
|
if (err) {
|
|
callback.apply(null, arguments);
|
|
callback = function () {};
|
|
}
|
|
else {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
var next = iterator.next();
|
|
if (next) {
|
|
args.push(wrapIterator(next));
|
|
}
|
|
else {
|
|
args.push(callback);
|
|
}
|
|
async.setImmediate(function () {
|
|
iterator.apply(null, args);
|
|
});
|
|
}
|
|
};
|
|
};
|
|
wrapIterator(async.iterator(tasks))();
|
|
};
|
|
|
|
var _parallel = function(eachfn, tasks, callback) {
|
|
callback = callback || function () {};
|
|
if (tasks.constructor === Array) {
|
|
eachfn.map(tasks, function (fn, callback) {
|
|
if (fn) {
|
|
fn(function (err) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
if (args.length <= 1) {
|
|
args = args[0];
|
|
}
|
|
callback.call(null, err, args);
|
|
});
|
|
}
|
|
}, callback);
|
|
}
|
|
else {
|
|
var results = {};
|
|
eachfn.each(_keys(tasks), function (k, callback) {
|
|
tasks[k](function (err) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
if (args.length <= 1) {
|
|
args = args[0];
|
|
}
|
|
results[k] = args;
|
|
callback(err);
|
|
});
|
|
}, function (err) {
|
|
callback(err, results);
|
|
});
|
|
}
|
|
};
|
|
|
|
async.parallel = function (tasks, callback) {
|
|
_parallel({ map: async.map, each: async.each }, tasks, callback);
|
|
};
|
|
|
|
async.parallelLimit = function(tasks, limit, callback) {
|
|
_parallel({ map: _mapLimit(limit), each: _eachLimit(limit) }, tasks, callback);
|
|
};
|
|
|
|
async.series = function (tasks, callback) {
|
|
callback = callback || function () {};
|
|
if (tasks.constructor === Array) {
|
|
async.mapSeries(tasks, function (fn, callback) {
|
|
if (fn) {
|
|
fn(function (err) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
if (args.length <= 1) {
|
|
args = args[0];
|
|
}
|
|
callback.call(null, err, args);
|
|
});
|
|
}
|
|
}, callback);
|
|
}
|
|
else {
|
|
var results = {};
|
|
async.eachSeries(_keys(tasks), function (k, callback) {
|
|
tasks[k](function (err) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
if (args.length <= 1) {
|
|
args = args[0];
|
|
}
|
|
results[k] = args;
|
|
callback(err);
|
|
});
|
|
}, function (err) {
|
|
callback(err, results);
|
|
});
|
|
}
|
|
};
|
|
|
|
async.iterator = function (tasks) {
|
|
var makeCallback = function (index) {
|
|
var fn = function () {
|
|
if (tasks.length) {
|
|
tasks[index].apply(null, arguments);
|
|
}
|
|
return fn.next();
|
|
};
|
|
fn.next = function () {
|
|
return (index < tasks.length - 1) ? makeCallback(index + 1): null;
|
|
};
|
|
return fn;
|
|
};
|
|
return makeCallback(0);
|
|
};
|
|
|
|
async.apply = function (fn) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
return function () {
|
|
return fn.apply(
|
|
null, args.concat(Array.prototype.slice.call(arguments))
|
|
);
|
|
};
|
|
};
|
|
|
|
var _concat = function (eachfn, arr, fn, callback) {
|
|
var r = [];
|
|
eachfn(arr, function (x, cb) {
|
|
fn(x, function (err, y) {
|
|
r = r.concat(y || []);
|
|
cb(err);
|
|
});
|
|
}, function (err) {
|
|
callback(err, r);
|
|
});
|
|
};
|
|
async.concat = doParallel(_concat);
|
|
async.concatSeries = doSeries(_concat);
|
|
|
|
async.whilst = function (test, iterator, callback) {
|
|
if (test()) {
|
|
iterator(function (err) {
|
|
if (err) {
|
|
return callback(err);
|
|
}
|
|
async.whilst(test, iterator, callback);
|
|
});
|
|
}
|
|
else {
|
|
callback();
|
|
}
|
|
};
|
|
|
|
async.doWhilst = function (iterator, test, callback) {
|
|
iterator(function (err) {
|
|
if (err) {
|
|
return callback(err);
|
|
}
|
|
if (test()) {
|
|
async.doWhilst(iterator, test, callback);
|
|
}
|
|
else {
|
|
callback();
|
|
}
|
|
});
|
|
};
|
|
|
|
async.until = function (test, iterator, callback) {
|
|
if (!test()) {
|
|
iterator(function (err) {
|
|
if (err) {
|
|
return callback(err);
|
|
}
|
|
async.until(test, iterator, callback);
|
|
});
|
|
}
|
|
else {
|
|
callback();
|
|
}
|
|
};
|
|
|
|
async.doUntil = function (iterator, test, callback) {
|
|
iterator(function (err) {
|
|
if (err) {
|
|
return callback(err);
|
|
}
|
|
if (!test()) {
|
|
async.doUntil(iterator, test, callback);
|
|
}
|
|
else {
|
|
callback();
|
|
}
|
|
});
|
|
};
|
|
|
|
async.queue = function (worker, concurrency) {
|
|
if (concurrency === undefined) {
|
|
concurrency = 1;
|
|
}
|
|
function _insert(q, data, pos, callback) {
|
|
if(data.constructor !== Array) {
|
|
data = [data];
|
|
}
|
|
_each(data, function(task) {
|
|
var item = {
|
|
data: task,
|
|
callback: typeof callback === 'function' ? callback : null
|
|
};
|
|
|
|
if (pos) {
|
|
q.tasks.unshift(item);
|
|
} else {
|
|
q.tasks.push(item);
|
|
}
|
|
|
|
if (q.saturated && q.tasks.length === concurrency) {
|
|
q.saturated();
|
|
}
|
|
async.setImmediate(q.process);
|
|
});
|
|
}
|
|
|
|
var workers = 0;
|
|
var q = {
|
|
tasks: [],
|
|
concurrency: concurrency,
|
|
saturated: null,
|
|
empty: null,
|
|
drain: null,
|
|
push: function (data, callback) {
|
|
_insert(q, data, false, callback);
|
|
},
|
|
unshift: function (data, callback) {
|
|
_insert(q, data, true, callback);
|
|
},
|
|
process: function () {
|
|
if (workers < q.concurrency && q.tasks.length) {
|
|
var task = q.tasks.shift();
|
|
if (q.empty && q.tasks.length === 0) {
|
|
q.empty();
|
|
}
|
|
workers += 1;
|
|
var next = function () {
|
|
workers -= 1;
|
|
if (task.callback) {
|
|
task.callback.apply(task, arguments);
|
|
}
|
|
if (q.drain && q.tasks.length + workers === 0) {
|
|
q.drain();
|
|
}
|
|
q.process();
|
|
};
|
|
var cb = only_once(next);
|
|
worker(task.data, cb);
|
|
}
|
|
},
|
|
length: function () {
|
|
return q.tasks.length;
|
|
},
|
|
running: function () {
|
|
return workers;
|
|
}
|
|
};
|
|
return q;
|
|
};
|
|
|
|
async.cargo = function (worker, payload) {
|
|
var working = false,
|
|
tasks = [];
|
|
|
|
var cargo = {
|
|
tasks: tasks,
|
|
payload: payload,
|
|
saturated: null,
|
|
empty: null,
|
|
drain: null,
|
|
push: function (data, callback) {
|
|
if(data.constructor !== Array) {
|
|
data = [data];
|
|
}
|
|
_each(data, function(task) {
|
|
tasks.push({
|
|
data: task,
|
|
callback: typeof callback === 'function' ? callback : null
|
|
});
|
|
if (cargo.saturated && tasks.length === payload) {
|
|
cargo.saturated();
|
|
}
|
|
});
|
|
async.setImmediate(cargo.process);
|
|
},
|
|
process: function process() {
|
|
if (working) return;
|
|
if (tasks.length === 0) {
|
|
if(cargo.drain) cargo.drain();
|
|
return;
|
|
}
|
|
|
|
var ts = typeof payload === 'number'
|
|
? tasks.splice(0, payload)
|
|
: tasks.splice(0);
|
|
|
|
var ds = _map(ts, function (task) {
|
|
return task.data;
|
|
});
|
|
|
|
if(cargo.empty) cargo.empty();
|
|
working = true;
|
|
worker(ds, function () {
|
|
working = false;
|
|
|
|
var args = arguments;
|
|
_each(ts, function (data) {
|
|
if (data.callback) {
|
|
data.callback.apply(null, args);
|
|
}
|
|
});
|
|
|
|
process();
|
|
});
|
|
},
|
|
length: function () {
|
|
return tasks.length;
|
|
},
|
|
running: function () {
|
|
return working;
|
|
}
|
|
};
|
|
return cargo;
|
|
};
|
|
|
|
var _console_fn = function (name) {
|
|
return function (fn) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
fn.apply(null, args.concat([function (err) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
if (typeof console !== 'undefined') {
|
|
if (err) {
|
|
if (console.error) {
|
|
console.error(err);
|
|
}
|
|
}
|
|
else if (console[name]) {
|
|
_each(args, function (x) {
|
|
console[name](x);
|
|
});
|
|
}
|
|
}
|
|
}]));
|
|
};
|
|
};
|
|
async.log = _console_fn('log');
|
|
async.dir = _console_fn('dir');
|
|
/*async.info = _console_fn('info');
|
|
async.warn = _console_fn('warn');
|
|
async.error = _console_fn('error');*/
|
|
|
|
async.memoize = function (fn, hasher) {
|
|
var memo = {};
|
|
var queues = {};
|
|
hasher = hasher || function (x) {
|
|
return x;
|
|
};
|
|
var memoized = function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
var callback = args.pop();
|
|
var key = hasher.apply(null, args);
|
|
if (key in memo) {
|
|
callback.apply(null, memo[key]);
|
|
}
|
|
else if (key in queues) {
|
|
queues[key].push(callback);
|
|
}
|
|
else {
|
|
queues[key] = [callback];
|
|
fn.apply(null, args.concat([function () {
|
|
memo[key] = arguments;
|
|
var q = queues[key];
|
|
delete queues[key];
|
|
for (var i = 0, l = q.length; i < l; i++) {
|
|
q[i].apply(null, arguments);
|
|
}
|
|
}]));
|
|
}
|
|
};
|
|
memoized.memo = memo;
|
|
memoized.unmemoized = fn;
|
|
return memoized;
|
|
};
|
|
|
|
async.unmemoize = function (fn) {
|
|
return function () {
|
|
return (fn.unmemoized || fn).apply(null, arguments);
|
|
};
|
|
};
|
|
|
|
async.times = function (count, iterator, callback) {
|
|
var counter = [];
|
|
for (var i = 0; i < count; i++) {
|
|
counter.push(i);
|
|
}
|
|
return async.map(counter, iterator, callback);
|
|
};
|
|
|
|
async.timesSeries = function (count, iterator, callback) {
|
|
var counter = [];
|
|
for (var i = 0; i < count; i++) {
|
|
counter.push(i);
|
|
}
|
|
return async.mapSeries(counter, iterator, callback);
|
|
};
|
|
|
|
async.compose = function (/* functions... */) {
|
|
var fns = Array.prototype.reverse.call(arguments);
|
|
return function () {
|
|
var that = this;
|
|
var args = Array.prototype.slice.call(arguments);
|
|
var callback = args.pop();
|
|
async.reduce(fns, args, function (newargs, fn, cb) {
|
|
fn.apply(that, newargs.concat([function () {
|
|
var err = arguments[0];
|
|
var nextargs = Array.prototype.slice.call(arguments, 1);
|
|
cb(err, nextargs);
|
|
}]))
|
|
},
|
|
function (err, results) {
|
|
callback.apply(that, [err].concat(results));
|
|
});
|
|
};
|
|
};
|
|
|
|
var _applyEach = function (eachfn, fns /*args...*/) {
|
|
var go = function () {
|
|
var that = this;
|
|
var args = Array.prototype.slice.call(arguments);
|
|
var callback = args.pop();
|
|
return eachfn(fns, function (fn, cb) {
|
|
fn.apply(that, args.concat([cb]));
|
|
},
|
|
callback);
|
|
};
|
|
if (arguments.length > 2) {
|
|
var args = Array.prototype.slice.call(arguments, 2);
|
|
return go.apply(this, args);
|
|
}
|
|
else {
|
|
return go;
|
|
}
|
|
};
|
|
async.applyEach = doParallel(_applyEach);
|
|
async.applyEachSeries = doSeries(_applyEach);
|
|
|
|
async.forever = function (fn, callback) {
|
|
function next(err) {
|
|
if (err) {
|
|
if (callback) {
|
|
return callback(err);
|
|
}
|
|
throw err;
|
|
}
|
|
fn(next);
|
|
}
|
|
next();
|
|
};
|
|
|
|
// AMD / RequireJS
|
|
if (typeof define !== 'undefined' && define.amd) {
|
|
define([], function () {
|
|
return async;
|
|
});
|
|
}
|
|
// Node.js
|
|
else if (typeof module !== 'undefined' && module.exports) {
|
|
module.exports = async;
|
|
}
|
|
// included directly via <script> tag
|
|
else {
|
|
root.async = async;
|
|
}
|
|
|
|
}());
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"))
|
|
},{"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96}],69:[function(require,module,exports){
|
|
(function (Buffer){
|
|
var Chainsaw = require('chainsaw');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
var Buffers = require('buffers');
|
|
var Vars = require('./lib/vars.js');
|
|
var Stream = require('stream').Stream;
|
|
|
|
exports = module.exports = function (bufOrEm, eventName) {
|
|
if (Buffer.isBuffer(bufOrEm)) {
|
|
return exports.parse(bufOrEm);
|
|
}
|
|
|
|
var s = exports.stream();
|
|
if (bufOrEm && bufOrEm.pipe) {
|
|
bufOrEm.pipe(s);
|
|
}
|
|
else if (bufOrEm) {
|
|
bufOrEm.on(eventName || 'data', function (buf) {
|
|
s.write(buf);
|
|
});
|
|
|
|
bufOrEm.on('end', function () {
|
|
s.end();
|
|
});
|
|
}
|
|
return s;
|
|
};
|
|
|
|
exports.stream = function (input) {
|
|
if (input) return exports.apply(null, arguments);
|
|
|
|
var pending = null;
|
|
function getBytes (bytes, cb, skip) {
|
|
pending = {
|
|
bytes : bytes,
|
|
skip : skip,
|
|
cb : function (buf) {
|
|
pending = null;
|
|
cb(buf);
|
|
},
|
|
};
|
|
dispatch();
|
|
}
|
|
|
|
var offset = null;
|
|
function dispatch () {
|
|
if (!pending) {
|
|
if (caughtEnd) done = true;
|
|
return;
|
|
}
|
|
if (typeof pending === 'function') {
|
|
pending();
|
|
}
|
|
else {
|
|
var bytes = offset + pending.bytes;
|
|
|
|
if (buffers.length >= bytes) {
|
|
var buf;
|
|
if (offset == null) {
|
|
buf = buffers.splice(0, bytes);
|
|
if (!pending.skip) {
|
|
buf = buf.slice();
|
|
}
|
|
}
|
|
else {
|
|
if (!pending.skip) {
|
|
buf = buffers.slice(offset, bytes);
|
|
}
|
|
offset = bytes;
|
|
}
|
|
|
|
if (pending.skip) {
|
|
pending.cb();
|
|
}
|
|
else {
|
|
pending.cb(buf);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function builder (saw) {
|
|
function next () { if (!done) saw.next() }
|
|
|
|
var self = words(function (bytes, cb) {
|
|
return function (name) {
|
|
getBytes(bytes, function (buf) {
|
|
vars.set(name, cb(buf));
|
|
next();
|
|
});
|
|
};
|
|
});
|
|
|
|
self.tap = function (cb) {
|
|
saw.nest(cb, vars.store);
|
|
};
|
|
|
|
self.into = function (key, cb) {
|
|
if (!vars.get(key)) vars.set(key, {});
|
|
var parent = vars;
|
|
vars = Vars(parent.get(key));
|
|
|
|
saw.nest(function () {
|
|
cb.apply(this, arguments);
|
|
this.tap(function () {
|
|
vars = parent;
|
|
});
|
|
}, vars.store);
|
|
};
|
|
|
|
self.flush = function () {
|
|
vars.store = {};
|
|
next();
|
|
};
|
|
|
|
self.loop = function (cb) {
|
|
var end = false;
|
|
|
|
saw.nest(false, function loop () {
|
|
this.vars = vars.store;
|
|
cb.call(this, function () {
|
|
end = true;
|
|
next();
|
|
}, vars.store);
|
|
this.tap(function () {
|
|
if (end) saw.next()
|
|
else loop.call(this)
|
|
}.bind(this));
|
|
}, vars.store);
|
|
};
|
|
|
|
self.buffer = function (name, bytes) {
|
|
if (typeof bytes === 'string') {
|
|
bytes = vars.get(bytes);
|
|
}
|
|
|
|
getBytes(bytes, function (buf) {
|
|
vars.set(name, buf);
|
|
next();
|
|
});
|
|
};
|
|
|
|
self.skip = function (bytes) {
|
|
if (typeof bytes === 'string') {
|
|
bytes = vars.get(bytes);
|
|
}
|
|
|
|
getBytes(bytes, function () {
|
|
next();
|
|
});
|
|
};
|
|
|
|
self.scan = function find (name, search) {
|
|
if (typeof search === 'string') {
|
|
search = new Buffer(search);
|
|
}
|
|
else if (!Buffer.isBuffer(search)) {
|
|
throw new Error('search must be a Buffer or a string');
|
|
}
|
|
|
|
var taken = 0;
|
|
pending = function () {
|
|
var pos = buffers.indexOf(search, offset + taken);
|
|
var i = pos-offset-taken;
|
|
if (pos !== -1) {
|
|
pending = null;
|
|
if (offset != null) {
|
|
vars.set(
|
|
name,
|
|
buffers.slice(offset, offset + taken + i)
|
|
);
|
|
offset += taken + i + search.length;
|
|
}
|
|
else {
|
|
vars.set(
|
|
name,
|
|
buffers.slice(0, taken + i)
|
|
);
|
|
buffers.splice(0, taken + i + search.length);
|
|
}
|
|
next();
|
|
dispatch();
|
|
} else {
|
|
i = Math.max(buffers.length - search.length - offset - taken, 0);
|
|
}
|
|
taken += i;
|
|
};
|
|
dispatch();
|
|
};
|
|
|
|
self.peek = function (cb) {
|
|
offset = 0;
|
|
saw.nest(function () {
|
|
cb.call(this, vars.store);
|
|
this.tap(function () {
|
|
offset = null;
|
|
});
|
|
});
|
|
};
|
|
|
|
return self;
|
|
};
|
|
|
|
var stream = Chainsaw.light(builder);
|
|
stream.writable = true;
|
|
|
|
var buffers = Buffers();
|
|
|
|
stream.write = function (buf) {
|
|
buffers.push(buf);
|
|
dispatch();
|
|
};
|
|
|
|
var vars = Vars();
|
|
|
|
var done = false, caughtEnd = false;
|
|
stream.end = function () {
|
|
caughtEnd = true;
|
|
};
|
|
|
|
stream.pipe = Stream.prototype.pipe;
|
|
Object.getOwnPropertyNames(EventEmitter.prototype).forEach(function (name) {
|
|
stream[name] = EventEmitter.prototype[name];
|
|
});
|
|
|
|
return stream;
|
|
};
|
|
|
|
exports.parse = function parse (buffer) {
|
|
var self = words(function (bytes, cb) {
|
|
return function (name) {
|
|
if (offset + bytes <= buffer.length) {
|
|
var buf = buffer.slice(offset, offset + bytes);
|
|
offset += bytes;
|
|
vars.set(name, cb(buf));
|
|
}
|
|
else {
|
|
vars.set(name, null);
|
|
}
|
|
return self;
|
|
};
|
|
});
|
|
|
|
var offset = 0;
|
|
var vars = Vars();
|
|
self.vars = vars.store;
|
|
|
|
self.tap = function (cb) {
|
|
cb.call(self, vars.store);
|
|
return self;
|
|
};
|
|
|
|
self.into = function (key, cb) {
|
|
if (!vars.get(key)) {
|
|
vars.set(key, {});
|
|
}
|
|
var parent = vars;
|
|
vars = Vars(parent.get(key));
|
|
cb.call(self, vars.store);
|
|
vars = parent;
|
|
return self;
|
|
};
|
|
|
|
self.loop = function (cb) {
|
|
var end = false;
|
|
var ender = function () { end = true };
|
|
while (end === false) {
|
|
cb.call(self, ender, vars.store);
|
|
}
|
|
return self;
|
|
};
|
|
|
|
self.buffer = function (name, size) {
|
|
if (typeof size === 'string') {
|
|
size = vars.get(size);
|
|
}
|
|
var buf = buffer.slice(offset, Math.min(buffer.length, offset + size));
|
|
offset += size;
|
|
vars.set(name, buf);
|
|
|
|
return self;
|
|
};
|
|
|
|
self.skip = function (bytes) {
|
|
if (typeof bytes === 'string') {
|
|
bytes = vars.get(bytes);
|
|
}
|
|
offset += bytes;
|
|
|
|
return self;
|
|
};
|
|
|
|
self.scan = function (name, search) {
|
|
if (typeof search === 'string') {
|
|
search = new Buffer(search);
|
|
}
|
|
else if (!Buffer.isBuffer(search)) {
|
|
throw new Error('search must be a Buffer or a string');
|
|
}
|
|
vars.set(name, null);
|
|
|
|
// simple but slow string search
|
|
for (var i = 0; i + offset <= buffer.length - search.length + 1; i++) {
|
|
for (
|
|
var j = 0;
|
|
j < search.length && buffer[offset+i+j] === search[j];
|
|
j++
|
|
);
|
|
if (j === search.length) break;
|
|
}
|
|
|
|
vars.set(name, buffer.slice(offset, offset + i));
|
|
offset += i + search.length;
|
|
return self;
|
|
};
|
|
|
|
self.peek = function (cb) {
|
|
var was = offset;
|
|
cb.call(self, vars.store);
|
|
offset = was;
|
|
return self;
|
|
};
|
|
|
|
self.flush = function () {
|
|
vars.store = {};
|
|
return self;
|
|
};
|
|
|
|
self.eof = function () {
|
|
return offset >= buffer.length;
|
|
};
|
|
|
|
return self;
|
|
};
|
|
|
|
// convert byte strings to unsigned little endian numbers
|
|
function decodeLEu (bytes) {
|
|
var acc = 0;
|
|
for (var i = 0; i < bytes.length; i++) {
|
|
acc += Math.pow(256,i) * bytes[i];
|
|
}
|
|
return acc;
|
|
}
|
|
|
|
// convert byte strings to unsigned big endian numbers
|
|
function decodeBEu (bytes) {
|
|
var acc = 0;
|
|
for (var i = 0; i < bytes.length; i++) {
|
|
acc += Math.pow(256, bytes.length - i - 1) * bytes[i];
|
|
}
|
|
return acc;
|
|
}
|
|
|
|
// convert byte strings to signed big endian numbers
|
|
function decodeBEs (bytes) {
|
|
var val = decodeBEu(bytes);
|
|
if ((bytes[0] & 0x80) == 0x80) {
|
|
val -= Math.pow(256, bytes.length);
|
|
}
|
|
return val;
|
|
}
|
|
|
|
// convert byte strings to signed little endian numbers
|
|
function decodeLEs (bytes) {
|
|
var val = decodeLEu(bytes);
|
|
if ((bytes[bytes.length - 1] & 0x80) == 0x80) {
|
|
val -= Math.pow(256, bytes.length);
|
|
}
|
|
return val;
|
|
}
|
|
|
|
function words (decode) {
|
|
var self = {};
|
|
|
|
[ 1, 2, 4, 8 ].forEach(function (bytes) {
|
|
var bits = bytes * 8;
|
|
|
|
self['word' + bits + 'le']
|
|
= self['word' + bits + 'lu']
|
|
= decode(bytes, decodeLEu);
|
|
|
|
self['word' + bits + 'ls']
|
|
= decode(bytes, decodeLEs);
|
|
|
|
self['word' + bits + 'be']
|
|
= self['word' + bits + 'bu']
|
|
= decode(bytes, decodeBEu);
|
|
|
|
self['word' + bits + 'bs']
|
|
= decode(bytes, decodeBEs);
|
|
});
|
|
|
|
// word8be(n) == word8le(n) for all n
|
|
self.word8 = self.word8u = self.word8be;
|
|
self.word8s = self.word8bs;
|
|
|
|
return self;
|
|
}
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"./lib/vars.js":70,"buffer":80,"buffers":"OBo3aV","chainsaw":71,"events":89,"stream":103}],70:[function(require,module,exports){
|
|
module.exports = function (store) {
|
|
function getset (name, value) {
|
|
var node = vars.store;
|
|
var keys = name.split('.');
|
|
keys.slice(0,-1).forEach(function (k) {
|
|
if (node[k] === undefined) node[k] = {};
|
|
node = node[k]
|
|
});
|
|
var key = keys[keys.length - 1];
|
|
if (arguments.length == 1) {
|
|
return node[key];
|
|
}
|
|
else {
|
|
return node[key] = value;
|
|
}
|
|
}
|
|
|
|
var vars = {
|
|
get : function (name) {
|
|
return getset(name);
|
|
},
|
|
set : function (name, value) {
|
|
return getset(name, value);
|
|
},
|
|
store : store || {},
|
|
};
|
|
return vars;
|
|
};
|
|
|
|
},{}],71:[function(require,module,exports){
|
|
(function (process){
|
|
var Traverse = require('traverse');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
|
|
module.exports = Chainsaw;
|
|
function Chainsaw (builder) {
|
|
var saw = Chainsaw.saw(builder, {});
|
|
var r = builder.call(saw.handlers, saw);
|
|
if (r !== undefined) saw.handlers = r;
|
|
saw.record();
|
|
return saw.chain();
|
|
};
|
|
|
|
Chainsaw.light = function ChainsawLight (builder) {
|
|
var saw = Chainsaw.saw(builder, {});
|
|
var r = builder.call(saw.handlers, saw);
|
|
if (r !== undefined) saw.handlers = r;
|
|
return saw.chain();
|
|
};
|
|
|
|
Chainsaw.saw = function (builder, handlers) {
|
|
var saw = new EventEmitter;
|
|
saw.handlers = handlers;
|
|
saw.actions = [];
|
|
|
|
saw.chain = function () {
|
|
var ch = Traverse(saw.handlers).map(function (node) {
|
|
if (this.isRoot) return node;
|
|
var ps = this.path;
|
|
|
|
if (typeof node === 'function') {
|
|
this.update(function () {
|
|
saw.actions.push({
|
|
path : ps,
|
|
args : [].slice.call(arguments)
|
|
});
|
|
return ch;
|
|
});
|
|
}
|
|
});
|
|
|
|
process.nextTick(function () {
|
|
saw.emit('begin');
|
|
saw.next();
|
|
});
|
|
|
|
return ch;
|
|
};
|
|
|
|
saw.pop = function () {
|
|
return saw.actions.shift();
|
|
};
|
|
|
|
saw.next = function () {
|
|
var action = saw.pop();
|
|
|
|
if (!action) {
|
|
saw.emit('end');
|
|
}
|
|
else if (!action.trap) {
|
|
var node = saw.handlers;
|
|
action.path.forEach(function (key) { node = node[key] });
|
|
node.apply(saw.handlers, action.args);
|
|
}
|
|
};
|
|
|
|
saw.nest = function (cb) {
|
|
var args = [].slice.call(arguments, 1);
|
|
var autonext = true;
|
|
|
|
if (typeof cb === 'boolean') {
|
|
var autonext = cb;
|
|
cb = args.shift();
|
|
}
|
|
|
|
var s = Chainsaw.saw(builder, {});
|
|
var r = builder.call(s.handlers, s);
|
|
|
|
if (r !== undefined) s.handlers = r;
|
|
|
|
// If we are recording...
|
|
if ("undefined" !== typeof saw.step) {
|
|
// ... our children should, too
|
|
s.record();
|
|
}
|
|
|
|
cb.apply(s.chain(), args);
|
|
if (autonext !== false) s.on('end', saw.next);
|
|
};
|
|
|
|
saw.record = function () {
|
|
upgradeChainsaw(saw);
|
|
};
|
|
|
|
['trap', 'down', 'jump'].forEach(function (method) {
|
|
saw[method] = function () {
|
|
throw new Error("To use the trap, down and jump features, please "+
|
|
"call record() first to start recording actions.");
|
|
};
|
|
});
|
|
|
|
return saw;
|
|
};
|
|
|
|
function upgradeChainsaw(saw) {
|
|
saw.step = 0;
|
|
|
|
// override pop
|
|
saw.pop = function () {
|
|
return saw.actions[saw.step++];
|
|
};
|
|
|
|
saw.trap = function (name, cb) {
|
|
var ps = Array.isArray(name) ? name : [name];
|
|
saw.actions.push({
|
|
path : ps,
|
|
step : saw.step,
|
|
cb : cb,
|
|
trap : true
|
|
});
|
|
};
|
|
|
|
saw.down = function (name) {
|
|
var ps = (Array.isArray(name) ? name : [name]).join('/');
|
|
var i = saw.actions.slice(saw.step).map(function (x) {
|
|
if (x.trap && x.step <= saw.step) return false;
|
|
return x.path.join('/') == ps;
|
|
}).indexOf(true);
|
|
|
|
if (i >= 0) saw.step += i;
|
|
else saw.step = saw.actions.length;
|
|
|
|
var act = saw.actions[saw.step - 1];
|
|
if (act && act.trap) {
|
|
// It's a trap!
|
|
saw.step = act.step;
|
|
act.cb();
|
|
}
|
|
else saw.next();
|
|
};
|
|
|
|
saw.jump = function (step) {
|
|
saw.step = step;
|
|
saw.next();
|
|
};
|
|
};
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"))
|
|
},{"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"events":89,"traverse":72}],72:[function(require,module,exports){
|
|
module.exports = Traverse;
|
|
function Traverse (obj) {
|
|
if (!(this instanceof Traverse)) return new Traverse(obj);
|
|
this.value = obj;
|
|
}
|
|
|
|
Traverse.prototype.get = function (ps) {
|
|
var node = this.value;
|
|
for (var i = 0; i < ps.length; i ++) {
|
|
var key = ps[i];
|
|
if (!Object.hasOwnProperty.call(node, key)) {
|
|
node = undefined;
|
|
break;
|
|
}
|
|
node = node[key];
|
|
}
|
|
return node;
|
|
};
|
|
|
|
Traverse.prototype.set = function (ps, value) {
|
|
var node = this.value;
|
|
for (var i = 0; i < ps.length - 1; i ++) {
|
|
var key = ps[i];
|
|
if (!Object.hasOwnProperty.call(node, key)) node[key] = {};
|
|
node = node[key];
|
|
}
|
|
node[ps[i]] = value;
|
|
return value;
|
|
};
|
|
|
|
Traverse.prototype.map = function (cb) {
|
|
return walk(this.value, cb, true);
|
|
};
|
|
|
|
Traverse.prototype.forEach = function (cb) {
|
|
this.value = walk(this.value, cb, false);
|
|
return this.value;
|
|
};
|
|
|
|
Traverse.prototype.reduce = function (cb, init) {
|
|
var skip = arguments.length === 1;
|
|
var acc = skip ? this.value : init;
|
|
this.forEach(function (x) {
|
|
if (!this.isRoot || !skip) {
|
|
acc = cb.call(this, acc, x);
|
|
}
|
|
});
|
|
return acc;
|
|
};
|
|
|
|
Traverse.prototype.deepEqual = function (obj) {
|
|
if (arguments.length !== 1) {
|
|
throw new Error(
|
|
'deepEqual requires exactly one object to compare against'
|
|
);
|
|
}
|
|
|
|
var equal = true;
|
|
var node = obj;
|
|
|
|
this.forEach(function (y) {
|
|
var notEqual = (function () {
|
|
equal = false;
|
|
//this.stop();
|
|
return undefined;
|
|
}).bind(this);
|
|
|
|
//if (node === undefined || node === null) return notEqual();
|
|
|
|
if (!this.isRoot) {
|
|
/*
|
|
if (!Object.hasOwnProperty.call(node, this.key)) {
|
|
return notEqual();
|
|
}
|
|
*/
|
|
if (typeof node !== 'object') return notEqual();
|
|
node = node[this.key];
|
|
}
|
|
|
|
var x = node;
|
|
|
|
this.post(function () {
|
|
node = x;
|
|
});
|
|
|
|
var toS = function (o) {
|
|
return Object.prototype.toString.call(o);
|
|
};
|
|
|
|
if (this.circular) {
|
|
if (Traverse(obj).get(this.circular.path) !== x) notEqual();
|
|
}
|
|
else if (typeof x !== typeof y) {
|
|
notEqual();
|
|
}
|
|
else if (x === null || y === null || x === undefined || y === undefined) {
|
|
if (x !== y) notEqual();
|
|
}
|
|
else if (x.__proto__ !== y.__proto__) {
|
|
notEqual();
|
|
}
|
|
else if (x === y) {
|
|
// nop
|
|
}
|
|
else if (typeof x === 'function') {
|
|
if (x instanceof RegExp) {
|
|
// both regexps on account of the __proto__ check
|
|
if (x.toString() != y.toString()) notEqual();
|
|
}
|
|
else if (x !== y) notEqual();
|
|
}
|
|
else if (typeof x === 'object') {
|
|
if (toS(y) === '[object Arguments]'
|
|
|| toS(x) === '[object Arguments]') {
|
|
if (toS(x) !== toS(y)) {
|
|
notEqual();
|
|
}
|
|
}
|
|
else if (x instanceof Date || y instanceof Date) {
|
|
if (!(x instanceof Date) || !(y instanceof Date)
|
|
|| x.getTime() !== y.getTime()) {
|
|
notEqual();
|
|
}
|
|
}
|
|
else {
|
|
var kx = Object.keys(x);
|
|
var ky = Object.keys(y);
|
|
if (kx.length !== ky.length) return notEqual();
|
|
for (var i = 0; i < kx.length; i++) {
|
|
var k = kx[i];
|
|
if (!Object.hasOwnProperty.call(y, k)) {
|
|
notEqual();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
return equal;
|
|
};
|
|
|
|
Traverse.prototype.paths = function () {
|
|
var acc = [];
|
|
this.forEach(function (x) {
|
|
acc.push(this.path);
|
|
});
|
|
return acc;
|
|
};
|
|
|
|
Traverse.prototype.nodes = function () {
|
|
var acc = [];
|
|
this.forEach(function (x) {
|
|
acc.push(this.node);
|
|
});
|
|
return acc;
|
|
};
|
|
|
|
Traverse.prototype.clone = function () {
|
|
var parents = [], nodes = [];
|
|
|
|
return (function clone (src) {
|
|
for (var i = 0; i < parents.length; i++) {
|
|
if (parents[i] === src) {
|
|
return nodes[i];
|
|
}
|
|
}
|
|
|
|
if (typeof src === 'object' && src !== null) {
|
|
var dst = copy(src);
|
|
|
|
parents.push(src);
|
|
nodes.push(dst);
|
|
|
|
Object.keys(src).forEach(function (key) {
|
|
dst[key] = clone(src[key]);
|
|
});
|
|
|
|
parents.pop();
|
|
nodes.pop();
|
|
return dst;
|
|
}
|
|
else {
|
|
return src;
|
|
}
|
|
})(this.value);
|
|
};
|
|
|
|
function walk (root, cb, immutable) {
|
|
var path = [];
|
|
var parents = [];
|
|
var alive = true;
|
|
|
|
return (function walker (node_) {
|
|
var node = immutable ? copy(node_) : node_;
|
|
var modifiers = {};
|
|
|
|
var state = {
|
|
node : node,
|
|
node_ : node_,
|
|
path : [].concat(path),
|
|
parent : parents.slice(-1)[0],
|
|
key : path.slice(-1)[0],
|
|
isRoot : path.length === 0,
|
|
level : path.length,
|
|
circular : null,
|
|
update : function (x) {
|
|
if (!state.isRoot) {
|
|
state.parent.node[state.key] = x;
|
|
}
|
|
state.node = x;
|
|
},
|
|
'delete' : function () {
|
|
delete state.parent.node[state.key];
|
|
},
|
|
remove : function () {
|
|
if (Array.isArray(state.parent.node)) {
|
|
state.parent.node.splice(state.key, 1);
|
|
}
|
|
else {
|
|
delete state.parent.node[state.key];
|
|
}
|
|
},
|
|
before : function (f) { modifiers.before = f },
|
|
after : function (f) { modifiers.after = f },
|
|
pre : function (f) { modifiers.pre = f },
|
|
post : function (f) { modifiers.post = f },
|
|
stop : function () { alive = false }
|
|
};
|
|
|
|
if (!alive) return state;
|
|
|
|
if (typeof node === 'object' && node !== null) {
|
|
state.isLeaf = Object.keys(node).length == 0;
|
|
|
|
for (var i = 0; i < parents.length; i++) {
|
|
if (parents[i].node_ === node_) {
|
|
state.circular = parents[i];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
state.isLeaf = true;
|
|
}
|
|
|
|
state.notLeaf = !state.isLeaf;
|
|
state.notRoot = !state.isRoot;
|
|
|
|
// use return values to update if defined
|
|
var ret = cb.call(state, state.node);
|
|
if (ret !== undefined && state.update) state.update(ret);
|
|
if (modifiers.before) modifiers.before.call(state, state.node);
|
|
|
|
if (typeof state.node == 'object'
|
|
&& state.node !== null && !state.circular) {
|
|
parents.push(state);
|
|
|
|
var keys = Object.keys(state.node);
|
|
keys.forEach(function (key, i) {
|
|
path.push(key);
|
|
|
|
if (modifiers.pre) modifiers.pre.call(state, state.node[key], key);
|
|
|
|
var child = walker(state.node[key]);
|
|
if (immutable && Object.hasOwnProperty.call(state.node, key)) {
|
|
state.node[key] = child.node;
|
|
}
|
|
|
|
child.isLast = i == keys.length - 1;
|
|
child.isFirst = i == 0;
|
|
|
|
if (modifiers.post) modifiers.post.call(state, child);
|
|
|
|
path.pop();
|
|
});
|
|
parents.pop();
|
|
}
|
|
|
|
if (modifiers.after) modifiers.after.call(state, state.node);
|
|
|
|
return state;
|
|
})(root).node;
|
|
}
|
|
|
|
Object.keys(Traverse.prototype).forEach(function (key) {
|
|
Traverse[key] = function (obj) {
|
|
var args = [].slice.call(arguments, 1);
|
|
var t = Traverse(obj);
|
|
return t[key].apply(t, args);
|
|
};
|
|
});
|
|
|
|
function copy (src) {
|
|
if (typeof src === 'object' && src !== null) {
|
|
var dst;
|
|
|
|
if (Array.isArray(src)) {
|
|
dst = [];
|
|
}
|
|
else if (src instanceof Date) {
|
|
dst = new Date(src);
|
|
}
|
|
else if (src instanceof Boolean) {
|
|
dst = new Boolean(src);
|
|
}
|
|
else if (src instanceof Number) {
|
|
dst = new Number(src);
|
|
}
|
|
else if (src instanceof String) {
|
|
dst = new String(src);
|
|
}
|
|
else {
|
|
dst = Object.create(Object.getPrototypeOf(src));
|
|
}
|
|
|
|
Object.keys(src).forEach(function (key) {
|
|
dst[key] = src[key];
|
|
});
|
|
return dst;
|
|
}
|
|
else return src;
|
|
}
|
|
|
|
},{}],73:[function(require,module,exports){
|
|
(function (process,__filename){
|
|
|
|
/**
|
|
* Module dependencies.
|
|
*/
|
|
|
|
var fs = require('fs')
|
|
, path = require('path')
|
|
, join = path.join
|
|
, dirname = path.dirname
|
|
, exists = fs.existsSync || path.existsSync
|
|
, defaults = {
|
|
arrow: process.env.NODE_BINDINGS_ARROW || ' → '
|
|
, compiled: process.env.NODE_BINDINGS_COMPILED_DIR || 'compiled'
|
|
, platform: process.platform
|
|
, arch: process.arch
|
|
, version: process.versions.node
|
|
, bindings: 'bindings.node'
|
|
, try: [
|
|
// node-gyp's linked version in the "build" dir
|
|
[ 'module_root', 'build', 'bindings' ]
|
|
// node-waf and gyp_addon (a.k.a node-gyp)
|
|
, [ 'module_root', 'build', 'Debug', 'bindings' ]
|
|
, [ 'module_root', 'build', 'Release', 'bindings' ]
|
|
// Debug files, for development (legacy behavior, remove for node v0.9)
|
|
, [ 'module_root', 'out', 'Debug', 'bindings' ]
|
|
, [ 'module_root', 'Debug', 'bindings' ]
|
|
// Release files, but manually compiled (legacy behavior, remove for node v0.9)
|
|
, [ 'module_root', 'out', 'Release', 'bindings' ]
|
|
, [ 'module_root', 'Release', 'bindings' ]
|
|
// Legacy from node-waf, node <= 0.4.x
|
|
, [ 'module_root', 'build', 'default', 'bindings' ]
|
|
// Production "Release" buildtype binary (meh...)
|
|
, [ 'module_root', 'compiled', 'version', 'platform', 'arch', 'bindings' ]
|
|
]
|
|
}
|
|
|
|
/**
|
|
* The main `bindings()` function loads the compiled bindings for a given module.
|
|
* It uses V8's Error API to determine the parent filename that this function is
|
|
* being invoked from, which is then used to find the root directory.
|
|
*/
|
|
|
|
function bindings (opts) {
|
|
|
|
// Argument surgery
|
|
if (typeof opts == 'string') {
|
|
opts = { bindings: opts }
|
|
} else if (!opts) {
|
|
opts = {}
|
|
}
|
|
opts.__proto__ = defaults
|
|
|
|
// Get the module root
|
|
if (!opts.module_root) {
|
|
opts.module_root = exports.getRoot(exports.getFileName())
|
|
}
|
|
|
|
// Ensure the given bindings name ends with .node
|
|
if (path.extname(opts.bindings) != '.node') {
|
|
opts.bindings += '.node'
|
|
}
|
|
|
|
var tries = []
|
|
, i = 0
|
|
, l = opts.try.length
|
|
, n
|
|
, b
|
|
, err
|
|
|
|
for (; i<l; i++) {
|
|
n = join.apply(null, opts.try[i].map(function (p) {
|
|
return opts[p] || p
|
|
}))
|
|
tries.push(n)
|
|
try {
|
|
b = opts.path ? require.resolve(n) : require(n)
|
|
if (!opts.path) {
|
|
b.path = n
|
|
}
|
|
return b
|
|
} catch (e) {
|
|
if (!/not find/i.test(e.message)) {
|
|
throw e
|
|
}
|
|
}
|
|
}
|
|
|
|
err = new Error('Could not locate the bindings file. Tried:\n'
|
|
+ tries.map(function (a) { return opts.arrow + a }).join('\n'))
|
|
err.tries = tries
|
|
throw err
|
|
}
|
|
module.exports = exports = bindings
|
|
|
|
|
|
/**
|
|
* Gets the filename of the JavaScript file that invokes this function.
|
|
* Used to help find the root directory of a module.
|
|
*/
|
|
|
|
exports.getFileName = function getFileName () {
|
|
var origPST = Error.prepareStackTrace
|
|
, origSTL = Error.stackTraceLimit
|
|
, dummy = {}
|
|
, fileName
|
|
|
|
Error.stackTraceLimit = 10
|
|
|
|
Error.prepareStackTrace = function (e, st) {
|
|
for (var i=0, l=st.length; i<l; i++) {
|
|
fileName = st[i].getFileName()
|
|
if (fileName !== __filename) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// run the 'prepareStackTrace' function above
|
|
Error.captureStackTrace(dummy)
|
|
dummy.stack
|
|
|
|
// cleanup
|
|
Error.prepareStackTrace = origPST
|
|
Error.stackTraceLimit = origSTL
|
|
|
|
return fileName
|
|
}
|
|
|
|
/**
|
|
* Gets the root directory of a module, given an arbitrary filename
|
|
* somewhere in the module tree. The "root directory" is the directory
|
|
* containing the `package.json` file.
|
|
*
|
|
* In: /home/nate/node-native-module/lib/index.js
|
|
* Out: /home/nate/node-native-module
|
|
*/
|
|
|
|
exports.getRoot = function getRoot (file) {
|
|
var dir = dirname(file)
|
|
, prev
|
|
while (true) {
|
|
if (dir === '.') {
|
|
// Avoids an infinite loop in rare cases, like the REPL
|
|
dir = process.cwd()
|
|
}
|
|
if (exists(join(dir, 'package.json')) || exists(join(dir, 'node_modules'))) {
|
|
// Found the 'package.json' file or 'node_modules' dir; we're done
|
|
return dir
|
|
}
|
|
if (prev === dir) {
|
|
// Got to the top
|
|
throw new Error('Could not find module root given file: "' + file
|
|
+ '". Do you have a `package.json` file? ')
|
|
}
|
|
// Try the parent dir next
|
|
prev = dir
|
|
dir = join(dir, '..')
|
|
}
|
|
}
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"),"/node_modules/bindings/bindings.js")
|
|
},{"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"fs":76,"path":97}],"buffertools":[function(require,module,exports){
|
|
module.exports=require('fugeBw');
|
|
},{}],"fugeBw":[function(require,module,exports){
|
|
(function (Buffer){
|
|
'use strict';
|
|
|
|
|
|
// requires node 3.1
|
|
var events = require('events');
|
|
var util = require('util');
|
|
|
|
var buffertools = {};
|
|
module.exports.Buffer = Buffer;
|
|
|
|
|
|
var is_buffer = function(x) {
|
|
return x instanceof Buffer || x instanceof Uint8Array;
|
|
};
|
|
|
|
var unaryAction = function(f) {
|
|
return function() {
|
|
var target = this;
|
|
if (is_buffer(target)) {} else if (is_buffer(arguments[0])) {
|
|
target = arguments[0];
|
|
Array.prototype.shift.apply(arguments);
|
|
} else {
|
|
throw new Error('Argument should be a buffer object.');
|
|
}
|
|
return f.apply(target, arguments);
|
|
};
|
|
};
|
|
|
|
var binaryAction = function(f) {
|
|
return function() {
|
|
var target = this;
|
|
|
|
// first argument
|
|
if (is_buffer(target)) {} else if (is_buffer(arguments[0])) {
|
|
target = arguments[0];
|
|
Array.prototype.shift.apply(arguments);
|
|
} else {
|
|
throw Error('Argument should be a buffer object.');
|
|
}
|
|
|
|
// second argument
|
|
var next = arguments[0];
|
|
if (typeof next == 'string' || next instanceof String || is_buffer(next)) {
|
|
return f.apply(target, arguments);
|
|
}
|
|
throw new Error('Second argument must be a string or a buffer.');
|
|
};
|
|
};
|
|
|
|
buffertools.clear = unaryAction(function() {
|
|
for (var i = 0; i < this.length; i++) {
|
|
this[i] = 0;
|
|
}
|
|
return this;
|
|
});
|
|
|
|
buffertools.fill = unaryAction(function(data) {
|
|
var step = typeof data.length === 'undefined' ? 1 : data.length;
|
|
for (var i = 0; i < this.length; i += step) {
|
|
for (var k = 0; k < step; k++) {
|
|
this[i + k] = typeof data.length === 'undefined' ? data :
|
|
(typeof data[k] === 'string' ? data[k].charCodeAt(0) : data[k]);
|
|
}
|
|
}
|
|
return this;
|
|
});
|
|
|
|
buffertools.indexOf = unaryAction(function(data, startFrom) {
|
|
startFrom = startFrom || 0;
|
|
if (data.length === 0) return -1;
|
|
for (var i = startFrom; i < this.length - data.length + 1; i += 1) {
|
|
var found = true;
|
|
for (var j = 0; j < data.length; j++) {
|
|
var a = this[i + j];
|
|
var b = data[j];
|
|
if (typeof b === 'string') b = b.charCodeAt(0);
|
|
if (a !== b) {
|
|
found = false;
|
|
break;
|
|
}
|
|
}
|
|
if (found) return i;
|
|
}
|
|
return -1;
|
|
});
|
|
|
|
buffertools.equals = binaryAction(function(data) {
|
|
return buffertools.compare(this, data) === 0;
|
|
});
|
|
|
|
buffertools.compare = binaryAction(function(data) {
|
|
var buffer = this;
|
|
var l1 = buffer.length;
|
|
var l2 = data.length;
|
|
if (l1 !== l2) {
|
|
return l1 > l2 ? 1 : -1;
|
|
}
|
|
for (var i = 0; i < l1; i++) {
|
|
var a = buffer[i];
|
|
var b = data[i];
|
|
if (typeof b === 'string') b = b.charCodeAt(0);
|
|
if (a === b) continue;
|
|
return a > b ? 1 : -1;
|
|
}
|
|
return 0;
|
|
});
|
|
buffertools.concat = function() {
|
|
var len = 0;
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
if (arguments[i].length === undefined) throw Error('all arguments must be strings or Buffers');
|
|
len += arguments[i].length;
|
|
}
|
|
var ret = new Buffer(len);
|
|
var k = 0;
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
for (var j = 0; j < arguments[i].length; j++) {
|
|
ret[k++] = typeof arguments[i][j] === 'string' ?
|
|
arguments[i][j].charCodeAt(0) : arguments[i][j];
|
|
}
|
|
}
|
|
return ret;
|
|
};
|
|
buffertools.reverse = unaryAction(function() {
|
|
var ret = new Buffer(this.length);
|
|
for (var i = 0; i < this.length; i++) {
|
|
ret[i] = this[this.length - i - 1];
|
|
}
|
|
return ret;
|
|
});
|
|
|
|
buffertools.toHex = unaryAction(function() {
|
|
var s = '';
|
|
for (var i = 0; i < this.length; i++) {
|
|
var h = this[i].toString(16);
|
|
if (h.length == 1) h = '0' + h;
|
|
if (h.length > 2) h = h.substring(1,3);
|
|
s += h;
|
|
}
|
|
return s;
|
|
});
|
|
buffertools.fromHex = unaryAction(function() {
|
|
var l = this.length;
|
|
if (l % 2 !== 0) throw new Error('Invalid hex string length');
|
|
var ret = new Buffer(l / 2);
|
|
for (var i = 0; i < ret.length; i++) {
|
|
var c1 = String.fromCharCode(this[2 * i]);
|
|
var c2 = String.fromCharCode(this[2 * i + 1]);
|
|
ret[i] = parseInt(c1 + c2, 16);
|
|
}
|
|
return ret;
|
|
});
|
|
|
|
exports.extend = function() {
|
|
var receivers;
|
|
if (arguments.length > 0) {
|
|
receivers = Array.prototype.slice.call(arguments);
|
|
} else if (typeof Uint8Array === 'function') {
|
|
receivers = [Buffer.prototype, Uint8Array.prototype];
|
|
} else {
|
|
receivers = [Buffer.prototype];
|
|
}
|
|
for (var i = 0, n = receivers.length; i < n; i += 1) {
|
|
var receiver = receivers[i];
|
|
for (var key in buffertools) {
|
|
receiver[key] = buffertools[key];
|
|
}
|
|
if (receiver !== exports) {
|
|
receiver.concat = function() {
|
|
var args = [this].concat(Array.prototype.slice.call(arguments));
|
|
return buffertools.concat.apply(buffertools, args);
|
|
};
|
|
}
|
|
}
|
|
};
|
|
exports.extend(exports);
|
|
|
|
//
|
|
// WritableBufferStream
|
|
//
|
|
// - never emits 'error'
|
|
// - never emits 'drain'
|
|
//
|
|
function WritableBufferStream() {
|
|
this.writable = true;
|
|
this.buffer = null;
|
|
}
|
|
|
|
util.inherits(WritableBufferStream, events.EventEmitter);
|
|
|
|
WritableBufferStream.prototype._append = function(buffer, encoding) {
|
|
if (!this.writable) {
|
|
throw new Error('Stream is not writable.');
|
|
}
|
|
|
|
if (Buffer.isBuffer(buffer)) {
|
|
// no action required
|
|
} else if (typeof buffer == 'string') {
|
|
// TODO optimize
|
|
buffer = new Buffer(buffer, encoding || 'utf8');
|
|
} else {
|
|
throw new Error('Argument should be either a buffer or a string.');
|
|
}
|
|
|
|
// FIXME optimize!
|
|
if (this.buffer) {
|
|
this.buffer = buffertools.concat(this.buffer, buffer);
|
|
} else {
|
|
this.buffer = new Buffer(buffer.length);
|
|
buffer.copy(this.buffer);
|
|
}
|
|
};
|
|
|
|
WritableBufferStream.prototype.write = function(buffer, encoding) {
|
|
this._append(buffer, encoding);
|
|
|
|
// signal that it's safe to immediately write again
|
|
return true;
|
|
};
|
|
|
|
WritableBufferStream.prototype.end = function(buffer, encoding) {
|
|
if (buffer) {
|
|
this._append(buffer, encoding);
|
|
}
|
|
|
|
this.emit('close');
|
|
|
|
this.writable = false;
|
|
};
|
|
|
|
WritableBufferStream.prototype.getBuffer = function() {
|
|
if (this.buffer) {
|
|
return this.buffer;
|
|
}
|
|
return new Buffer(0);
|
|
};
|
|
|
|
WritableBufferStream.prototype.toString = function() {
|
|
return this.getBuffer().toString();
|
|
};
|
|
|
|
exports.WritableBufferStream = WritableBufferStream;
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80,"events":89,"util":112}],76:[function(require,module,exports){
|
|
|
|
},{}],77:[function(require,module,exports){
|
|
// http://wiki.commonjs.org/wiki/Unit_Testing/1.0
|
|
//
|
|
// THIS IS NOT TESTED NOR LIKELY TO WORK OUTSIDE V8!
|
|
//
|
|
// Originally from narwhal.js (http://narwhaljs.org)
|
|
// Copyright (c) 2009 Thomas Robinson <280north.com>
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the 'Software'), to
|
|
// deal in the Software without restriction, including without limitation the
|
|
// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
// sell copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// when used in node, this will actually load the util module we depend on
|
|
// versus loading the builtin util module as happens otherwise
|
|
// this is a bug in node module loading as far as I am concerned
|
|
var util = require('util/');
|
|
|
|
var pSlice = Array.prototype.slice;
|
|
var hasOwn = Object.prototype.hasOwnProperty;
|
|
|
|
// 1. The assert module provides functions that throw
|
|
// AssertionError's when particular conditions are not met. The
|
|
// assert module must conform to the following interface.
|
|
|
|
var assert = module.exports = ok;
|
|
|
|
// 2. The AssertionError is defined in assert.
|
|
// new assert.AssertionError({ message: message,
|
|
// actual: actual,
|
|
// expected: expected })
|
|
|
|
assert.AssertionError = function AssertionError(options) {
|
|
this.name = 'AssertionError';
|
|
this.actual = options.actual;
|
|
this.expected = options.expected;
|
|
this.operator = options.operator;
|
|
if (options.message) {
|
|
this.message = options.message;
|
|
this.generatedMessage = false;
|
|
} else {
|
|
this.message = getMessage(this);
|
|
this.generatedMessage = true;
|
|
}
|
|
var stackStartFunction = options.stackStartFunction || fail;
|
|
|
|
if (Error.captureStackTrace) {
|
|
Error.captureStackTrace(this, stackStartFunction);
|
|
}
|
|
else {
|
|
// non v8 browsers so we can have a stacktrace
|
|
var err = new Error();
|
|
if (err.stack) {
|
|
var out = err.stack;
|
|
|
|
// try to strip useless frames
|
|
var fn_name = stackStartFunction.name;
|
|
var idx = out.indexOf('\n' + fn_name);
|
|
if (idx >= 0) {
|
|
// once we have located the function frame
|
|
// we need to strip out everything before it (and its line)
|
|
var next_line = out.indexOf('\n', idx + 1);
|
|
out = out.substring(next_line + 1);
|
|
}
|
|
|
|
this.stack = out;
|
|
}
|
|
}
|
|
};
|
|
|
|
// assert.AssertionError instanceof Error
|
|
util.inherits(assert.AssertionError, Error);
|
|
|
|
function replacer(key, value) {
|
|
if (util.isUndefined(value)) {
|
|
return '' + value;
|
|
}
|
|
if (util.isNumber(value) && (isNaN(value) || !isFinite(value))) {
|
|
return value.toString();
|
|
}
|
|
if (util.isFunction(value) || util.isRegExp(value)) {
|
|
return value.toString();
|
|
}
|
|
return value;
|
|
}
|
|
|
|
function truncate(s, n) {
|
|
if (util.isString(s)) {
|
|
return s.length < n ? s : s.slice(0, n);
|
|
} else {
|
|
return s;
|
|
}
|
|
}
|
|
|
|
function getMessage(self) {
|
|
return truncate(JSON.stringify(self.actual, replacer), 128) + ' ' +
|
|
self.operator + ' ' +
|
|
truncate(JSON.stringify(self.expected, replacer), 128);
|
|
}
|
|
|
|
// At present only the three keys mentioned above are used and
|
|
// understood by the spec. Implementations or sub modules can pass
|
|
// other keys to the AssertionError's constructor - they will be
|
|
// ignored.
|
|
|
|
// 3. All of the following functions must throw an AssertionError
|
|
// when a corresponding condition is not met, with a message that
|
|
// may be undefined if not provided. All assertion methods provide
|
|
// both the actual and expected values to the assertion error for
|
|
// display purposes.
|
|
|
|
function fail(actual, expected, message, operator, stackStartFunction) {
|
|
throw new assert.AssertionError({
|
|
message: message,
|
|
actual: actual,
|
|
expected: expected,
|
|
operator: operator,
|
|
stackStartFunction: stackStartFunction
|
|
});
|
|
}
|
|
|
|
// EXTENSION! allows for well behaved errors defined elsewhere.
|
|
assert.fail = fail;
|
|
|
|
// 4. Pure assertion tests whether a value is truthy, as determined
|
|
// by !!guard.
|
|
// assert.ok(guard, message_opt);
|
|
// This statement is equivalent to assert.equal(true, !!guard,
|
|
// message_opt);. To test strictly for the value true, use
|
|
// assert.strictEqual(true, guard, message_opt);.
|
|
|
|
function ok(value, message) {
|
|
if (!value) fail(value, true, message, '==', assert.ok);
|
|
}
|
|
assert.ok = ok;
|
|
|
|
// 5. The equality assertion tests shallow, coercive equality with
|
|
// ==.
|
|
// assert.equal(actual, expected, message_opt);
|
|
|
|
assert.equal = function equal(actual, expected, message) {
|
|
if (actual != expected) fail(actual, expected, message, '==', assert.equal);
|
|
};
|
|
|
|
// 6. The non-equality assertion tests for whether two objects are not equal
|
|
// with != assert.notEqual(actual, expected, message_opt);
|
|
|
|
assert.notEqual = function notEqual(actual, expected, message) {
|
|
if (actual == expected) {
|
|
fail(actual, expected, message, '!=', assert.notEqual);
|
|
}
|
|
};
|
|
|
|
// 7. The equivalence assertion tests a deep equality relation.
|
|
// assert.deepEqual(actual, expected, message_opt);
|
|
|
|
assert.deepEqual = function deepEqual(actual, expected, message) {
|
|
if (!_deepEqual(actual, expected)) {
|
|
fail(actual, expected, message, 'deepEqual', assert.deepEqual);
|
|
}
|
|
};
|
|
|
|
function _deepEqual(actual, expected) {
|
|
// 7.1. All identical values are equivalent, as determined by ===.
|
|
if (actual === expected) {
|
|
return true;
|
|
|
|
} else if (util.isBuffer(actual) && util.isBuffer(expected)) {
|
|
if (actual.length != expected.length) return false;
|
|
|
|
for (var i = 0; i < actual.length; i++) {
|
|
if (actual[i] !== expected[i]) return false;
|
|
}
|
|
|
|
return true;
|
|
|
|
// 7.2. If the expected value is a Date object, the actual value is
|
|
// equivalent if it is also a Date object that refers to the same time.
|
|
} else if (util.isDate(actual) && util.isDate(expected)) {
|
|
return actual.getTime() === expected.getTime();
|
|
|
|
// 7.3 If the expected value is a RegExp object, the actual value is
|
|
// equivalent if it is also a RegExp object with the same source and
|
|
// properties (`global`, `multiline`, `lastIndex`, `ignoreCase`).
|
|
} else if (util.isRegExp(actual) && util.isRegExp(expected)) {
|
|
return actual.source === expected.source &&
|
|
actual.global === expected.global &&
|
|
actual.multiline === expected.multiline &&
|
|
actual.lastIndex === expected.lastIndex &&
|
|
actual.ignoreCase === expected.ignoreCase;
|
|
|
|
// 7.4. Other pairs that do not both pass typeof value == 'object',
|
|
// equivalence is determined by ==.
|
|
} else if (!util.isObject(actual) && !util.isObject(expected)) {
|
|
return actual == expected;
|
|
|
|
// 7.5 For all other Object pairs, including Array objects, equivalence is
|
|
// determined by having the same number of owned properties (as verified
|
|
// with Object.prototype.hasOwnProperty.call), the same set of keys
|
|
// (although not necessarily the same order), equivalent values for every
|
|
// corresponding key, and an identical 'prototype' property. Note: this
|
|
// accounts for both named and indexed properties on Arrays.
|
|
} else {
|
|
return objEquiv(actual, expected);
|
|
}
|
|
}
|
|
|
|
function isArguments(object) {
|
|
return Object.prototype.toString.call(object) == '[object Arguments]';
|
|
}
|
|
|
|
function objEquiv(a, b) {
|
|
if (util.isNullOrUndefined(a) || util.isNullOrUndefined(b))
|
|
return false;
|
|
// an identical 'prototype' property.
|
|
if (a.prototype !== b.prototype) return false;
|
|
//~~~I've managed to break Object.keys through screwy arguments passing.
|
|
// Converting to array solves the problem.
|
|
if (isArguments(a)) {
|
|
if (!isArguments(b)) {
|
|
return false;
|
|
}
|
|
a = pSlice.call(a);
|
|
b = pSlice.call(b);
|
|
return _deepEqual(a, b);
|
|
}
|
|
try {
|
|
var ka = objectKeys(a),
|
|
kb = objectKeys(b),
|
|
key, i;
|
|
} catch (e) {//happens when one is a string literal and the other isn't
|
|
return false;
|
|
}
|
|
// having the same number of owned properties (keys incorporates
|
|
// hasOwnProperty)
|
|
if (ka.length != kb.length)
|
|
return false;
|
|
//the same set of keys (although not necessarily the same order),
|
|
ka.sort();
|
|
kb.sort();
|
|
//~~~cheap key test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
if (ka[i] != kb[i])
|
|
return false;
|
|
}
|
|
//equivalent values for every corresponding key, and
|
|
//~~~possibly expensive deep test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
key = ka[i];
|
|
if (!_deepEqual(a[key], b[key])) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// 8. The non-equivalence assertion tests for any deep inequality.
|
|
// assert.notDeepEqual(actual, expected, message_opt);
|
|
|
|
assert.notDeepEqual = function notDeepEqual(actual, expected, message) {
|
|
if (_deepEqual(actual, expected)) {
|
|
fail(actual, expected, message, 'notDeepEqual', assert.notDeepEqual);
|
|
}
|
|
};
|
|
|
|
// 9. The strict equality assertion tests strict equality, as determined by ===.
|
|
// assert.strictEqual(actual, expected, message_opt);
|
|
|
|
assert.strictEqual = function strictEqual(actual, expected, message) {
|
|
if (actual !== expected) {
|
|
fail(actual, expected, message, '===', assert.strictEqual);
|
|
}
|
|
};
|
|
|
|
// 10. The strict non-equality assertion tests for strict inequality, as
|
|
// determined by !==. assert.notStrictEqual(actual, expected, message_opt);
|
|
|
|
assert.notStrictEqual = function notStrictEqual(actual, expected, message) {
|
|
if (actual === expected) {
|
|
fail(actual, expected, message, '!==', assert.notStrictEqual);
|
|
}
|
|
};
|
|
|
|
function expectedException(actual, expected) {
|
|
if (!actual || !expected) {
|
|
return false;
|
|
}
|
|
|
|
if (Object.prototype.toString.call(expected) == '[object RegExp]') {
|
|
return expected.test(actual);
|
|
} else if (actual instanceof expected) {
|
|
return true;
|
|
} else if (expected.call({}, actual) === true) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function _throws(shouldThrow, block, expected, message) {
|
|
var actual;
|
|
|
|
if (util.isString(expected)) {
|
|
message = expected;
|
|
expected = null;
|
|
}
|
|
|
|
try {
|
|
block();
|
|
} catch (e) {
|
|
actual = e;
|
|
}
|
|
|
|
message = (expected && expected.name ? ' (' + expected.name + ').' : '.') +
|
|
(message ? ' ' + message : '.');
|
|
|
|
if (shouldThrow && !actual) {
|
|
fail(actual, expected, 'Missing expected exception' + message);
|
|
}
|
|
|
|
if (!shouldThrow && expectedException(actual, expected)) {
|
|
fail(actual, expected, 'Got unwanted exception' + message);
|
|
}
|
|
|
|
if ((shouldThrow && actual && expected &&
|
|
!expectedException(actual, expected)) || (!shouldThrow && actual)) {
|
|
throw actual;
|
|
}
|
|
}
|
|
|
|
// 11. Expected to throw an error:
|
|
// assert.throws(block, Error_opt, message_opt);
|
|
|
|
assert.throws = function(block, /*optional*/error, /*optional*/message) {
|
|
_throws.apply(this, [true].concat(pSlice.call(arguments)));
|
|
};
|
|
|
|
// EXTENSION! This is annoying to write outside this module.
|
|
assert.doesNotThrow = function(block, /*optional*/message) {
|
|
_throws.apply(this, [false].concat(pSlice.call(arguments)));
|
|
};
|
|
|
|
assert.ifError = function(err) { if (err) {throw err;}};
|
|
|
|
var objectKeys = Object.keys || function (obj) {
|
|
var keys = [];
|
|
for (var key in obj) {
|
|
if (hasOwn.call(obj, key)) keys.push(key);
|
|
}
|
|
return keys;
|
|
};
|
|
|
|
},{"util/":79}],78:[function(require,module,exports){
|
|
module.exports = function isBuffer(arg) {
|
|
return arg && typeof arg === 'object'
|
|
&& typeof arg.copy === 'function'
|
|
&& typeof arg.fill === 'function'
|
|
&& typeof arg.readUInt8 === 'function';
|
|
}
|
|
},{}],79:[function(require,module,exports){
|
|
(function (process,global){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var formatRegExp = /%[sdj%]/g;
|
|
exports.format = function(f) {
|
|
if (!isString(f)) {
|
|
var objects = [];
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
objects.push(inspect(arguments[i]));
|
|
}
|
|
return objects.join(' ');
|
|
}
|
|
|
|
var i = 1;
|
|
var args = arguments;
|
|
var len = args.length;
|
|
var str = String(f).replace(formatRegExp, function(x) {
|
|
if (x === '%%') return '%';
|
|
if (i >= len) return x;
|
|
switch (x) {
|
|
case '%s': return String(args[i++]);
|
|
case '%d': return Number(args[i++]);
|
|
case '%j':
|
|
try {
|
|
return JSON.stringify(args[i++]);
|
|
} catch (_) {
|
|
return '[Circular]';
|
|
}
|
|
default:
|
|
return x;
|
|
}
|
|
});
|
|
for (var x = args[i]; i < len; x = args[++i]) {
|
|
if (isNull(x) || !isObject(x)) {
|
|
str += ' ' + x;
|
|
} else {
|
|
str += ' ' + inspect(x);
|
|
}
|
|
}
|
|
return str;
|
|
};
|
|
|
|
|
|
// Mark that a method should not be used.
|
|
// Returns a modified function which warns once by default.
|
|
// If --no-deprecation is set, then it is a no-op.
|
|
exports.deprecate = function(fn, msg) {
|
|
// Allow for deprecating things in the process of starting up.
|
|
if (isUndefined(global.process)) {
|
|
return function() {
|
|
return exports.deprecate(fn, msg).apply(this, arguments);
|
|
};
|
|
}
|
|
|
|
if (process.noDeprecation === true) {
|
|
return fn;
|
|
}
|
|
|
|
var warned = false;
|
|
function deprecated() {
|
|
if (!warned) {
|
|
if (process.throwDeprecation) {
|
|
throw new Error(msg);
|
|
} else if (process.traceDeprecation) {
|
|
console.trace(msg);
|
|
} else {
|
|
console.error(msg);
|
|
}
|
|
warned = true;
|
|
}
|
|
return fn.apply(this, arguments);
|
|
}
|
|
|
|
return deprecated;
|
|
};
|
|
|
|
|
|
var debugs = {};
|
|
var debugEnviron;
|
|
exports.debuglog = function(set) {
|
|
if (isUndefined(debugEnviron))
|
|
debugEnviron = process.env.NODE_DEBUG || '';
|
|
set = set.toUpperCase();
|
|
if (!debugs[set]) {
|
|
if (new RegExp('\\b' + set + '\\b', 'i').test(debugEnviron)) {
|
|
var pid = process.pid;
|
|
debugs[set] = function() {
|
|
var msg = exports.format.apply(exports, arguments);
|
|
console.error('%s %d: %s', set, pid, msg);
|
|
};
|
|
} else {
|
|
debugs[set] = function() {};
|
|
}
|
|
}
|
|
return debugs[set];
|
|
};
|
|
|
|
|
|
/**
|
|
* Echos the value of a value. Trys to print the value out
|
|
* in the best way possible given the different types.
|
|
*
|
|
* @param {Object} obj The object to print out.
|
|
* @param {Object} opts Optional options object that alters the output.
|
|
*/
|
|
/* legacy: obj, showHidden, depth, colors*/
|
|
function inspect(obj, opts) {
|
|
// default options
|
|
var ctx = {
|
|
seen: [],
|
|
stylize: stylizeNoColor
|
|
};
|
|
// legacy...
|
|
if (arguments.length >= 3) ctx.depth = arguments[2];
|
|
if (arguments.length >= 4) ctx.colors = arguments[3];
|
|
if (isBoolean(opts)) {
|
|
// legacy...
|
|
ctx.showHidden = opts;
|
|
} else if (opts) {
|
|
// got an "options" object
|
|
exports._extend(ctx, opts);
|
|
}
|
|
// set default options
|
|
if (isUndefined(ctx.showHidden)) ctx.showHidden = false;
|
|
if (isUndefined(ctx.depth)) ctx.depth = 2;
|
|
if (isUndefined(ctx.colors)) ctx.colors = false;
|
|
if (isUndefined(ctx.customInspect)) ctx.customInspect = true;
|
|
if (ctx.colors) ctx.stylize = stylizeWithColor;
|
|
return formatValue(ctx, obj, ctx.depth);
|
|
}
|
|
exports.inspect = inspect;
|
|
|
|
|
|
// http://en.wikipedia.org/wiki/ANSI_escape_code#graphics
|
|
inspect.colors = {
|
|
'bold' : [1, 22],
|
|
'italic' : [3, 23],
|
|
'underline' : [4, 24],
|
|
'inverse' : [7, 27],
|
|
'white' : [37, 39],
|
|
'grey' : [90, 39],
|
|
'black' : [30, 39],
|
|
'blue' : [34, 39],
|
|
'cyan' : [36, 39],
|
|
'green' : [32, 39],
|
|
'magenta' : [35, 39],
|
|
'red' : [31, 39],
|
|
'yellow' : [33, 39]
|
|
};
|
|
|
|
// Don't use 'blue' not visible on cmd.exe
|
|
inspect.styles = {
|
|
'special': 'cyan',
|
|
'number': 'yellow',
|
|
'boolean': 'yellow',
|
|
'undefined': 'grey',
|
|
'null': 'bold',
|
|
'string': 'green',
|
|
'date': 'magenta',
|
|
// "name": intentionally not styling
|
|
'regexp': 'red'
|
|
};
|
|
|
|
|
|
function stylizeWithColor(str, styleType) {
|
|
var style = inspect.styles[styleType];
|
|
|
|
if (style) {
|
|
return '\u001b[' + inspect.colors[style][0] + 'm' + str +
|
|
'\u001b[' + inspect.colors[style][1] + 'm';
|
|
} else {
|
|
return str;
|
|
}
|
|
}
|
|
|
|
|
|
function stylizeNoColor(str, styleType) {
|
|
return str;
|
|
}
|
|
|
|
|
|
function arrayToHash(array) {
|
|
var hash = {};
|
|
|
|
array.forEach(function(val, idx) {
|
|
hash[val] = true;
|
|
});
|
|
|
|
return hash;
|
|
}
|
|
|
|
|
|
function formatValue(ctx, value, recurseTimes) {
|
|
// Provide a hook for user-specified inspect functions.
|
|
// Check that value is an object with an inspect function on it
|
|
if (ctx.customInspect &&
|
|
value &&
|
|
isFunction(value.inspect) &&
|
|
// Filter out the util module, it's inspect function is special
|
|
value.inspect !== exports.inspect &&
|
|
// Also filter out any prototype objects using the circular check.
|
|
!(value.constructor && value.constructor.prototype === value)) {
|
|
var ret = value.inspect(recurseTimes, ctx);
|
|
if (!isString(ret)) {
|
|
ret = formatValue(ctx, ret, recurseTimes);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// Primitive types cannot have properties
|
|
var primitive = formatPrimitive(ctx, value);
|
|
if (primitive) {
|
|
return primitive;
|
|
}
|
|
|
|
// Look up the keys of the object.
|
|
var keys = Object.keys(value);
|
|
var visibleKeys = arrayToHash(keys);
|
|
|
|
if (ctx.showHidden) {
|
|
keys = Object.getOwnPropertyNames(value);
|
|
}
|
|
|
|
// IE doesn't make error fields non-enumerable
|
|
// http://msdn.microsoft.com/en-us/library/ie/dww52sbt(v=vs.94).aspx
|
|
if (isError(value)
|
|
&& (keys.indexOf('message') >= 0 || keys.indexOf('description') >= 0)) {
|
|
return formatError(value);
|
|
}
|
|
|
|
// Some type of object without properties can be shortcutted.
|
|
if (keys.length === 0) {
|
|
if (isFunction(value)) {
|
|
var name = value.name ? ': ' + value.name : '';
|
|
return ctx.stylize('[Function' + name + ']', 'special');
|
|
}
|
|
if (isRegExp(value)) {
|
|
return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp');
|
|
}
|
|
if (isDate(value)) {
|
|
return ctx.stylize(Date.prototype.toString.call(value), 'date');
|
|
}
|
|
if (isError(value)) {
|
|
return formatError(value);
|
|
}
|
|
}
|
|
|
|
var base = '', array = false, braces = ['{', '}'];
|
|
|
|
// Make Array say that they are Array
|
|
if (isArray(value)) {
|
|
array = true;
|
|
braces = ['[', ']'];
|
|
}
|
|
|
|
// Make functions say that they are functions
|
|
if (isFunction(value)) {
|
|
var n = value.name ? ': ' + value.name : '';
|
|
base = ' [Function' + n + ']';
|
|
}
|
|
|
|
// Make RegExps say that they are RegExps
|
|
if (isRegExp(value)) {
|
|
base = ' ' + RegExp.prototype.toString.call(value);
|
|
}
|
|
|
|
// Make dates with properties first say the date
|
|
if (isDate(value)) {
|
|
base = ' ' + Date.prototype.toUTCString.call(value);
|
|
}
|
|
|
|
// Make error with message first say the error
|
|
if (isError(value)) {
|
|
base = ' ' + formatError(value);
|
|
}
|
|
|
|
if (keys.length === 0 && (!array || value.length == 0)) {
|
|
return braces[0] + base + braces[1];
|
|
}
|
|
|
|
if (recurseTimes < 0) {
|
|
if (isRegExp(value)) {
|
|
return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp');
|
|
} else {
|
|
return ctx.stylize('[Object]', 'special');
|
|
}
|
|
}
|
|
|
|
ctx.seen.push(value);
|
|
|
|
var output;
|
|
if (array) {
|
|
output = formatArray(ctx, value, recurseTimes, visibleKeys, keys);
|
|
} else {
|
|
output = keys.map(function(key) {
|
|
return formatProperty(ctx, value, recurseTimes, visibleKeys, key, array);
|
|
});
|
|
}
|
|
|
|
ctx.seen.pop();
|
|
|
|
return reduceToSingleString(output, base, braces);
|
|
}
|
|
|
|
|
|
function formatPrimitive(ctx, value) {
|
|
if (isUndefined(value))
|
|
return ctx.stylize('undefined', 'undefined');
|
|
if (isString(value)) {
|
|
var simple = '\'' + JSON.stringify(value).replace(/^"|"$/g, '')
|
|
.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"') + '\'';
|
|
return ctx.stylize(simple, 'string');
|
|
}
|
|
if (isNumber(value))
|
|
return ctx.stylize('' + value, 'number');
|
|
if (isBoolean(value))
|
|
return ctx.stylize('' + value, 'boolean');
|
|
// For some reason typeof null is "object", so special case here.
|
|
if (isNull(value))
|
|
return ctx.stylize('null', 'null');
|
|
}
|
|
|
|
|
|
function formatError(value) {
|
|
return '[' + Error.prototype.toString.call(value) + ']';
|
|
}
|
|
|
|
|
|
function formatArray(ctx, value, recurseTimes, visibleKeys, keys) {
|
|
var output = [];
|
|
for (var i = 0, l = value.length; i < l; ++i) {
|
|
if (hasOwnProperty(value, String(i))) {
|
|
output.push(formatProperty(ctx, value, recurseTimes, visibleKeys,
|
|
String(i), true));
|
|
} else {
|
|
output.push('');
|
|
}
|
|
}
|
|
keys.forEach(function(key) {
|
|
if (!key.match(/^\d+$/)) {
|
|
output.push(formatProperty(ctx, value, recurseTimes, visibleKeys,
|
|
key, true));
|
|
}
|
|
});
|
|
return output;
|
|
}
|
|
|
|
|
|
function formatProperty(ctx, value, recurseTimes, visibleKeys, key, array) {
|
|
var name, str, desc;
|
|
desc = Object.getOwnPropertyDescriptor(value, key) || { value: value[key] };
|
|
if (desc.get) {
|
|
if (desc.set) {
|
|
str = ctx.stylize('[Getter/Setter]', 'special');
|
|
} else {
|
|
str = ctx.stylize('[Getter]', 'special');
|
|
}
|
|
} else {
|
|
if (desc.set) {
|
|
str = ctx.stylize('[Setter]', 'special');
|
|
}
|
|
}
|
|
if (!hasOwnProperty(visibleKeys, key)) {
|
|
name = '[' + key + ']';
|
|
}
|
|
if (!str) {
|
|
if (ctx.seen.indexOf(desc.value) < 0) {
|
|
if (isNull(recurseTimes)) {
|
|
str = formatValue(ctx, desc.value, null);
|
|
} else {
|
|
str = formatValue(ctx, desc.value, recurseTimes - 1);
|
|
}
|
|
if (str.indexOf('\n') > -1) {
|
|
if (array) {
|
|
str = str.split('\n').map(function(line) {
|
|
return ' ' + line;
|
|
}).join('\n').substr(2);
|
|
} else {
|
|
str = '\n' + str.split('\n').map(function(line) {
|
|
return ' ' + line;
|
|
}).join('\n');
|
|
}
|
|
}
|
|
} else {
|
|
str = ctx.stylize('[Circular]', 'special');
|
|
}
|
|
}
|
|
if (isUndefined(name)) {
|
|
if (array && key.match(/^\d+$/)) {
|
|
return str;
|
|
}
|
|
name = JSON.stringify('' + key);
|
|
if (name.match(/^"([a-zA-Z_][a-zA-Z_0-9]*)"$/)) {
|
|
name = name.substr(1, name.length - 2);
|
|
name = ctx.stylize(name, 'name');
|
|
} else {
|
|
name = name.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"')
|
|
.replace(/(^"|"$)/g, "'");
|
|
name = ctx.stylize(name, 'string');
|
|
}
|
|
}
|
|
|
|
return name + ': ' + str;
|
|
}
|
|
|
|
|
|
function reduceToSingleString(output, base, braces) {
|
|
var numLinesEst = 0;
|
|
var length = output.reduce(function(prev, cur) {
|
|
numLinesEst++;
|
|
if (cur.indexOf('\n') >= 0) numLinesEst++;
|
|
return prev + cur.replace(/\u001b\[\d\d?m/g, '').length + 1;
|
|
}, 0);
|
|
|
|
if (length > 60) {
|
|
return braces[0] +
|
|
(base === '' ? '' : base + '\n ') +
|
|
' ' +
|
|
output.join(',\n ') +
|
|
' ' +
|
|
braces[1];
|
|
}
|
|
|
|
return braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1];
|
|
}
|
|
|
|
|
|
// NOTE: These type checking functions intentionally don't use `instanceof`
|
|
// because it is fragile and can be easily faked with `Object.create()`.
|
|
function isArray(ar) {
|
|
return Array.isArray(ar);
|
|
}
|
|
exports.isArray = isArray;
|
|
|
|
function isBoolean(arg) {
|
|
return typeof arg === 'boolean';
|
|
}
|
|
exports.isBoolean = isBoolean;
|
|
|
|
function isNull(arg) {
|
|
return arg === null;
|
|
}
|
|
exports.isNull = isNull;
|
|
|
|
function isNullOrUndefined(arg) {
|
|
return arg == null;
|
|
}
|
|
exports.isNullOrUndefined = isNullOrUndefined;
|
|
|
|
function isNumber(arg) {
|
|
return typeof arg === 'number';
|
|
}
|
|
exports.isNumber = isNumber;
|
|
|
|
function isString(arg) {
|
|
return typeof arg === 'string';
|
|
}
|
|
exports.isString = isString;
|
|
|
|
function isSymbol(arg) {
|
|
return typeof arg === 'symbol';
|
|
}
|
|
exports.isSymbol = isSymbol;
|
|
|
|
function isUndefined(arg) {
|
|
return arg === void 0;
|
|
}
|
|
exports.isUndefined = isUndefined;
|
|
|
|
function isRegExp(re) {
|
|
return isObject(re) && objectToString(re) === '[object RegExp]';
|
|
}
|
|
exports.isRegExp = isRegExp;
|
|
|
|
function isObject(arg) {
|
|
return typeof arg === 'object' && arg !== null;
|
|
}
|
|
exports.isObject = isObject;
|
|
|
|
function isDate(d) {
|
|
return isObject(d) && objectToString(d) === '[object Date]';
|
|
}
|
|
exports.isDate = isDate;
|
|
|
|
function isError(e) {
|
|
return isObject(e) &&
|
|
(objectToString(e) === '[object Error]' || e instanceof Error);
|
|
}
|
|
exports.isError = isError;
|
|
|
|
function isFunction(arg) {
|
|
return typeof arg === 'function';
|
|
}
|
|
exports.isFunction = isFunction;
|
|
|
|
function isPrimitive(arg) {
|
|
return arg === null ||
|
|
typeof arg === 'boolean' ||
|
|
typeof arg === 'number' ||
|
|
typeof arg === 'string' ||
|
|
typeof arg === 'symbol' || // ES6 symbol
|
|
typeof arg === 'undefined';
|
|
}
|
|
exports.isPrimitive = isPrimitive;
|
|
|
|
exports.isBuffer = require('./support/isBuffer');
|
|
|
|
function objectToString(o) {
|
|
return Object.prototype.toString.call(o);
|
|
}
|
|
|
|
|
|
function pad(n) {
|
|
return n < 10 ? '0' + n.toString(10) : n.toString(10);
|
|
}
|
|
|
|
|
|
var months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep',
|
|
'Oct', 'Nov', 'Dec'];
|
|
|
|
// 26 Feb 16:19:34
|
|
function timestamp() {
|
|
var d = new Date();
|
|
var time = [pad(d.getHours()),
|
|
pad(d.getMinutes()),
|
|
pad(d.getSeconds())].join(':');
|
|
return [d.getDate(), months[d.getMonth()], time].join(' ');
|
|
}
|
|
|
|
|
|
// log is just a thin wrapper to console.log that prepends a timestamp
|
|
exports.log = function() {
|
|
console.log('%s - %s', timestamp(), exports.format.apply(exports, arguments));
|
|
};
|
|
|
|
|
|
/**
|
|
* Inherit the prototype methods from one constructor into another.
|
|
*
|
|
* The Function.prototype.inherits from lang.js rewritten as a standalone
|
|
* function (not on Function.prototype). NOTE: If this file is to be loaded
|
|
* during bootstrapping this function needs to be rewritten using some native
|
|
* functions as prototype setup using normal JavaScript does not work as
|
|
* expected during bootstrapping (see mirror.js in r114903).
|
|
*
|
|
* @param {function} ctor Constructor function which needs to inherit the
|
|
* prototype.
|
|
* @param {function} superCtor Constructor function to inherit prototype from.
|
|
*/
|
|
exports.inherits = require('inherits');
|
|
|
|
exports._extend = function(origin, add) {
|
|
// Don't do anything if add isn't an object
|
|
if (!add || !isObject(add)) return origin;
|
|
|
|
var keys = Object.keys(add);
|
|
var i = keys.length;
|
|
while (i--) {
|
|
origin[keys[i]] = add[keys[i]];
|
|
}
|
|
return origin;
|
|
};
|
|
|
|
function hasOwnProperty(obj, prop) {
|
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
|
}
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"),typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
|
|
},{"./support/isBuffer":78,"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"inherits":95}],80:[function(require,module,exports){
|
|
/*!
|
|
* The buffer module from node.js, for the browser.
|
|
*
|
|
* @author Feross Aboukhadijeh <feross@feross.org> <http://feross.org>
|
|
* @license MIT
|
|
*/
|
|
|
|
var base64 = require('base64-js')
|
|
var ieee754 = require('ieee754')
|
|
|
|
exports.Buffer = Buffer
|
|
exports.SlowBuffer = Buffer
|
|
exports.INSPECT_MAX_BYTES = 50
|
|
Buffer.poolSize = 8192
|
|
|
|
/**
|
|
* If `Buffer._useTypedArrays`:
|
|
* === true Use Uint8Array implementation (fastest)
|
|
* === false Use Object implementation (compatible down to IE6)
|
|
*/
|
|
Buffer._useTypedArrays = (function () {
|
|
// Detect if browser supports Typed Arrays. Supported browsers are IE 10+, Firefox 4+,
|
|
// Chrome 7+, Safari 5.1+, Opera 11.6+, iOS 4.2+. If the browser does not support adding
|
|
// properties to `Uint8Array` instances, then that's the same as no `Uint8Array` support
|
|
// because we need to be able to add all the node Buffer API methods. This is an issue
|
|
// in Firefox 4-29. Now fixed: https://bugzilla.mozilla.org/show_bug.cgi?id=695438
|
|
try {
|
|
var buf = new ArrayBuffer(0)
|
|
var arr = new Uint8Array(buf)
|
|
arr.foo = function () { return 42 }
|
|
return 42 === arr.foo() &&
|
|
typeof arr.subarray === 'function' // Chrome 9-10 lack `subarray`
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
})()
|
|
|
|
/**
|
|
* Class: Buffer
|
|
* =============
|
|
*
|
|
* The Buffer constructor returns instances of `Uint8Array` that are augmented
|
|
* with function properties for all the node `Buffer` API functions. We use
|
|
* `Uint8Array` so that square bracket notation works as expected -- it returns
|
|
* a single octet.
|
|
*
|
|
* By augmenting the instances, we can avoid modifying the `Uint8Array`
|
|
* prototype.
|
|
*/
|
|
function Buffer (subject, encoding, noZero) {
|
|
if (!(this instanceof Buffer))
|
|
return new Buffer(subject, encoding, noZero)
|
|
|
|
var type = typeof subject
|
|
|
|
// Workaround: node's base64 implementation allows for non-padded strings
|
|
// while base64-js does not.
|
|
if (encoding === 'base64' && type === 'string') {
|
|
subject = stringtrim(subject)
|
|
while (subject.length % 4 !== 0) {
|
|
subject = subject + '='
|
|
}
|
|
}
|
|
|
|
// Find the length
|
|
var length
|
|
if (type === 'number')
|
|
length = coerce(subject)
|
|
else if (type === 'string')
|
|
length = Buffer.byteLength(subject, encoding)
|
|
else if (type === 'object')
|
|
length = coerce(subject.length) // assume that object is array-like
|
|
else
|
|
throw new Error('First argument needs to be a number, array or string.')
|
|
|
|
var buf
|
|
if (Buffer._useTypedArrays) {
|
|
// Preferred: Return an augmented `Uint8Array` instance for best performance
|
|
buf = Buffer._augment(new Uint8Array(length))
|
|
} else {
|
|
// Fallback: Return THIS instance of Buffer (created by `new`)
|
|
buf = this
|
|
buf.length = length
|
|
buf._isBuffer = true
|
|
}
|
|
|
|
var i
|
|
if (Buffer._useTypedArrays && typeof subject.byteLength === 'number') {
|
|
// Speed optimization -- use set if we're copying from a typed array
|
|
buf._set(subject)
|
|
} else if (isArrayish(subject)) {
|
|
// Treat array-ish objects as a byte array
|
|
for (i = 0; i < length; i++) {
|
|
if (Buffer.isBuffer(subject))
|
|
buf[i] = subject.readUInt8(i)
|
|
else
|
|
buf[i] = subject[i]
|
|
}
|
|
} else if (type === 'string') {
|
|
buf.write(subject, 0, encoding)
|
|
} else if (type === 'number' && !Buffer._useTypedArrays && !noZero) {
|
|
for (i = 0; i < length; i++) {
|
|
buf[i] = 0
|
|
}
|
|
}
|
|
|
|
return buf
|
|
}
|
|
|
|
// STATIC METHODS
|
|
// ==============
|
|
|
|
Buffer.isEncoding = function (encoding) {
|
|
switch (String(encoding).toLowerCase()) {
|
|
case 'hex':
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
case 'ascii':
|
|
case 'binary':
|
|
case 'base64':
|
|
case 'raw':
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
case 'utf16le':
|
|
case 'utf-16le':
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
Buffer.isBuffer = function (b) {
|
|
return !!(b !== null && b !== undefined && b._isBuffer)
|
|
}
|
|
|
|
Buffer.byteLength = function (str, encoding) {
|
|
var ret
|
|
str = str + ''
|
|
switch (encoding || 'utf8') {
|
|
case 'hex':
|
|
ret = str.length / 2
|
|
break
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
ret = utf8ToBytes(str).length
|
|
break
|
|
case 'ascii':
|
|
case 'binary':
|
|
case 'raw':
|
|
ret = str.length
|
|
break
|
|
case 'base64':
|
|
ret = base64ToBytes(str).length
|
|
break
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
case 'utf16le':
|
|
case 'utf-16le':
|
|
ret = str.length * 2
|
|
break
|
|
default:
|
|
throw new Error('Unknown encoding')
|
|
}
|
|
return ret
|
|
}
|
|
|
|
Buffer.concat = function (list, totalLength) {
|
|
assert(isArray(list), 'Usage: Buffer.concat(list, [totalLength])\n' +
|
|
'list should be an Array.')
|
|
|
|
if (list.length === 0) {
|
|
return new Buffer(0)
|
|
} else if (list.length === 1) {
|
|
return list[0]
|
|
}
|
|
|
|
var i
|
|
if (typeof totalLength !== 'number') {
|
|
totalLength = 0
|
|
for (i = 0; i < list.length; i++) {
|
|
totalLength += list[i].length
|
|
}
|
|
}
|
|
|
|
var buf = new Buffer(totalLength)
|
|
var pos = 0
|
|
for (i = 0; i < list.length; i++) {
|
|
var item = list[i]
|
|
item.copy(buf, pos)
|
|
pos += item.length
|
|
}
|
|
return buf
|
|
}
|
|
|
|
// BUFFER INSTANCE METHODS
|
|
// =======================
|
|
|
|
function _hexWrite (buf, string, offset, length) {
|
|
offset = Number(offset) || 0
|
|
var remaining = buf.length - offset
|
|
if (!length) {
|
|
length = remaining
|
|
} else {
|
|
length = Number(length)
|
|
if (length > remaining) {
|
|
length = remaining
|
|
}
|
|
}
|
|
|
|
// must be an even number of digits
|
|
var strLen = string.length
|
|
assert(strLen % 2 === 0, 'Invalid hex string')
|
|
|
|
if (length > strLen / 2) {
|
|
length = strLen / 2
|
|
}
|
|
for (var i = 0; i < length; i++) {
|
|
var byte = parseInt(string.substr(i * 2, 2), 16)
|
|
assert(!isNaN(byte), 'Invalid hex string')
|
|
buf[offset + i] = byte
|
|
}
|
|
Buffer._charsWritten = i * 2
|
|
return i
|
|
}
|
|
|
|
function _utf8Write (buf, string, offset, length) {
|
|
var charsWritten = Buffer._charsWritten =
|
|
blitBuffer(utf8ToBytes(string), buf, offset, length)
|
|
return charsWritten
|
|
}
|
|
|
|
function _asciiWrite (buf, string, offset, length) {
|
|
var charsWritten = Buffer._charsWritten =
|
|
blitBuffer(asciiToBytes(string), buf, offset, length)
|
|
return charsWritten
|
|
}
|
|
|
|
function _binaryWrite (buf, string, offset, length) {
|
|
return _asciiWrite(buf, string, offset, length)
|
|
}
|
|
|
|
function _base64Write (buf, string, offset, length) {
|
|
var charsWritten = Buffer._charsWritten =
|
|
blitBuffer(base64ToBytes(string), buf, offset, length)
|
|
return charsWritten
|
|
}
|
|
|
|
function _utf16leWrite (buf, string, offset, length) {
|
|
var charsWritten = Buffer._charsWritten =
|
|
blitBuffer(utf16leToBytes(string), buf, offset, length)
|
|
return charsWritten
|
|
}
|
|
|
|
Buffer.prototype.write = function (string, offset, length, encoding) {
|
|
// Support both (string, offset, length, encoding)
|
|
// and the legacy (string, encoding, offset, length)
|
|
if (isFinite(offset)) {
|
|
if (!isFinite(length)) {
|
|
encoding = length
|
|
length = undefined
|
|
}
|
|
} else { // legacy
|
|
var swap = encoding
|
|
encoding = offset
|
|
offset = length
|
|
length = swap
|
|
}
|
|
|
|
offset = Number(offset) || 0
|
|
var remaining = this.length - offset
|
|
if (!length) {
|
|
length = remaining
|
|
} else {
|
|
length = Number(length)
|
|
if (length > remaining) {
|
|
length = remaining
|
|
}
|
|
}
|
|
encoding = String(encoding || 'utf8').toLowerCase()
|
|
|
|
var ret
|
|
switch (encoding) {
|
|
case 'hex':
|
|
ret = _hexWrite(this, string, offset, length)
|
|
break
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
ret = _utf8Write(this, string, offset, length)
|
|
break
|
|
case 'ascii':
|
|
ret = _asciiWrite(this, string, offset, length)
|
|
break
|
|
case 'binary':
|
|
ret = _binaryWrite(this, string, offset, length)
|
|
break
|
|
case 'base64':
|
|
ret = _base64Write(this, string, offset, length)
|
|
break
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
case 'utf16le':
|
|
case 'utf-16le':
|
|
ret = _utf16leWrite(this, string, offset, length)
|
|
break
|
|
default:
|
|
throw new Error('Unknown encoding')
|
|
}
|
|
return ret
|
|
}
|
|
|
|
Buffer.prototype.toString = function (encoding, start, end) {
|
|
var self = this
|
|
|
|
encoding = String(encoding || 'utf8').toLowerCase()
|
|
start = Number(start) || 0
|
|
end = (end !== undefined)
|
|
? Number(end)
|
|
: end = self.length
|
|
|
|
// Fastpath empty strings
|
|
if (end === start)
|
|
return ''
|
|
|
|
var ret
|
|
switch (encoding) {
|
|
case 'hex':
|
|
ret = _hexSlice(self, start, end)
|
|
break
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
ret = _utf8Slice(self, start, end)
|
|
break
|
|
case 'ascii':
|
|
ret = _asciiSlice(self, start, end)
|
|
break
|
|
case 'binary':
|
|
ret = _binarySlice(self, start, end)
|
|
break
|
|
case 'base64':
|
|
ret = _base64Slice(self, start, end)
|
|
break
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
case 'utf16le':
|
|
case 'utf-16le':
|
|
ret = _utf16leSlice(self, start, end)
|
|
break
|
|
default:
|
|
throw new Error('Unknown encoding')
|
|
}
|
|
return ret
|
|
}
|
|
|
|
Buffer.prototype.toJSON = function () {
|
|
return {
|
|
type: 'Buffer',
|
|
data: Array.prototype.slice.call(this._arr || this, 0)
|
|
}
|
|
}
|
|
|
|
// copy(targetBuffer, targetStart=0, sourceStart=0, sourceEnd=buffer.length)
|
|
Buffer.prototype.copy = function (target, target_start, start, end) {
|
|
var source = this
|
|
|
|
if (!start) start = 0
|
|
if (!end && end !== 0) end = this.length
|
|
if (!target_start) target_start = 0
|
|
|
|
// Copy 0 bytes; we're done
|
|
if (end === start) return
|
|
if (target.length === 0 || source.length === 0) return
|
|
|
|
// Fatal error conditions
|
|
assert(end >= start, 'sourceEnd < sourceStart')
|
|
assert(target_start >= 0 && target_start < target.length,
|
|
'targetStart out of bounds')
|
|
assert(start >= 0 && start < source.length, 'sourceStart out of bounds')
|
|
assert(end >= 0 && end <= source.length, 'sourceEnd out of bounds')
|
|
|
|
// Are we oob?
|
|
if (end > this.length)
|
|
end = this.length
|
|
if (target.length - target_start < end - start)
|
|
end = target.length - target_start + start
|
|
|
|
var len = end - start
|
|
|
|
if (len < 100 || !Buffer._useTypedArrays) {
|
|
for (var i = 0; i < len; i++)
|
|
target[i + target_start] = this[i + start]
|
|
} else {
|
|
target._set(this.subarray(start, start + len), target_start)
|
|
}
|
|
}
|
|
|
|
function _base64Slice (buf, start, end) {
|
|
if (start === 0 && end === buf.length) {
|
|
return base64.fromByteArray(buf)
|
|
} else {
|
|
return base64.fromByteArray(buf.slice(start, end))
|
|
}
|
|
}
|
|
|
|
function _utf8Slice (buf, start, end) {
|
|
var res = ''
|
|
var tmp = ''
|
|
end = Math.min(buf.length, end)
|
|
|
|
for (var i = start; i < end; i++) {
|
|
if (buf[i] <= 0x7F) {
|
|
res += decodeUtf8Char(tmp) + String.fromCharCode(buf[i])
|
|
tmp = ''
|
|
} else {
|
|
tmp += '%' + buf[i].toString(16)
|
|
}
|
|
}
|
|
|
|
return res + decodeUtf8Char(tmp)
|
|
}
|
|
|
|
function _asciiSlice (buf, start, end) {
|
|
var ret = ''
|
|
end = Math.min(buf.length, end)
|
|
|
|
for (var i = start; i < end; i++)
|
|
ret += String.fromCharCode(buf[i])
|
|
return ret
|
|
}
|
|
|
|
function _binarySlice (buf, start, end) {
|
|
return _asciiSlice(buf, start, end)
|
|
}
|
|
|
|
function _hexSlice (buf, start, end) {
|
|
var len = buf.length
|
|
|
|
if (!start || start < 0) start = 0
|
|
if (!end || end < 0 || end > len) end = len
|
|
|
|
var out = ''
|
|
for (var i = start; i < end; i++) {
|
|
out += toHex(buf[i])
|
|
}
|
|
return out
|
|
}
|
|
|
|
function _utf16leSlice (buf, start, end) {
|
|
var bytes = buf.slice(start, end)
|
|
var res = ''
|
|
for (var i = 0; i < bytes.length; i += 2) {
|
|
res += String.fromCharCode(bytes[i] + bytes[i+1] * 256)
|
|
}
|
|
return res
|
|
}
|
|
|
|
Buffer.prototype.slice = function (start, end) {
|
|
var len = this.length
|
|
start = clamp(start, len, 0)
|
|
end = clamp(end, len, len)
|
|
|
|
if (Buffer._useTypedArrays) {
|
|
return Buffer._augment(this.subarray(start, end))
|
|
} else {
|
|
var sliceLen = end - start
|
|
var newBuf = new Buffer(sliceLen, undefined, true)
|
|
for (var i = 0; i < sliceLen; i++) {
|
|
newBuf[i] = this[i + start]
|
|
}
|
|
return newBuf
|
|
}
|
|
}
|
|
|
|
// `get` will be removed in Node 0.13+
|
|
Buffer.prototype.get = function (offset) {
|
|
console.log('.get() is deprecated. Access using array indexes instead.')
|
|
return this.readUInt8(offset)
|
|
}
|
|
|
|
// `set` will be removed in Node 0.13+
|
|
Buffer.prototype.set = function (v, offset) {
|
|
console.log('.set() is deprecated. Access using array indexes instead.')
|
|
return this.writeUInt8(v, offset)
|
|
}
|
|
|
|
Buffer.prototype.readUInt8 = function (offset, noAssert) {
|
|
if (!noAssert) {
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset < this.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
if (offset >= this.length)
|
|
return
|
|
|
|
return this[offset]
|
|
}
|
|
|
|
function _readUInt16 (buf, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 1 < buf.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
var val
|
|
if (littleEndian) {
|
|
val = buf[offset]
|
|
if (offset + 1 < len)
|
|
val |= buf[offset + 1] << 8
|
|
} else {
|
|
val = buf[offset] << 8
|
|
if (offset + 1 < len)
|
|
val |= buf[offset + 1]
|
|
}
|
|
return val
|
|
}
|
|
|
|
Buffer.prototype.readUInt16LE = function (offset, noAssert) {
|
|
return _readUInt16(this, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readUInt16BE = function (offset, noAssert) {
|
|
return _readUInt16(this, offset, false, noAssert)
|
|
}
|
|
|
|
function _readUInt32 (buf, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 3 < buf.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
var val
|
|
if (littleEndian) {
|
|
if (offset + 2 < len)
|
|
val = buf[offset + 2] << 16
|
|
if (offset + 1 < len)
|
|
val |= buf[offset + 1] << 8
|
|
val |= buf[offset]
|
|
if (offset + 3 < len)
|
|
val = val + (buf[offset + 3] << 24 >>> 0)
|
|
} else {
|
|
if (offset + 1 < len)
|
|
val = buf[offset + 1] << 16
|
|
if (offset + 2 < len)
|
|
val |= buf[offset + 2] << 8
|
|
if (offset + 3 < len)
|
|
val |= buf[offset + 3]
|
|
val = val + (buf[offset] << 24 >>> 0)
|
|
}
|
|
return val
|
|
}
|
|
|
|
Buffer.prototype.readUInt32LE = function (offset, noAssert) {
|
|
return _readUInt32(this, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readUInt32BE = function (offset, noAssert) {
|
|
return _readUInt32(this, offset, false, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readInt8 = function (offset, noAssert) {
|
|
if (!noAssert) {
|
|
assert(offset !== undefined && offset !== null,
|
|
'missing offset')
|
|
assert(offset < this.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
if (offset >= this.length)
|
|
return
|
|
|
|
var neg = this[offset] & 0x80
|
|
if (neg)
|
|
return (0xff - this[offset] + 1) * -1
|
|
else
|
|
return this[offset]
|
|
}
|
|
|
|
function _readInt16 (buf, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 1 < buf.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
var val = _readUInt16(buf, offset, littleEndian, true)
|
|
var neg = val & 0x8000
|
|
if (neg)
|
|
return (0xffff - val + 1) * -1
|
|
else
|
|
return val
|
|
}
|
|
|
|
Buffer.prototype.readInt16LE = function (offset, noAssert) {
|
|
return _readInt16(this, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readInt16BE = function (offset, noAssert) {
|
|
return _readInt16(this, offset, false, noAssert)
|
|
}
|
|
|
|
function _readInt32 (buf, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 3 < buf.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
var val = _readUInt32(buf, offset, littleEndian, true)
|
|
var neg = val & 0x80000000
|
|
if (neg)
|
|
return (0xffffffff - val + 1) * -1
|
|
else
|
|
return val
|
|
}
|
|
|
|
Buffer.prototype.readInt32LE = function (offset, noAssert) {
|
|
return _readInt32(this, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readInt32BE = function (offset, noAssert) {
|
|
return _readInt32(this, offset, false, noAssert)
|
|
}
|
|
|
|
function _readFloat (buf, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset + 3 < buf.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
return ieee754.read(buf, offset, littleEndian, 23, 4)
|
|
}
|
|
|
|
Buffer.prototype.readFloatLE = function (offset, noAssert) {
|
|
return _readFloat(this, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readFloatBE = function (offset, noAssert) {
|
|
return _readFloat(this, offset, false, noAssert)
|
|
}
|
|
|
|
function _readDouble (buf, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset + 7 < buf.length, 'Trying to read beyond buffer length')
|
|
}
|
|
|
|
return ieee754.read(buf, offset, littleEndian, 52, 8)
|
|
}
|
|
|
|
Buffer.prototype.readDoubleLE = function (offset, noAssert) {
|
|
return _readDouble(this, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.readDoubleBE = function (offset, noAssert) {
|
|
return _readDouble(this, offset, false, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeUInt8 = function (value, offset, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset < this.length, 'trying to write beyond buffer length')
|
|
verifuint(value, 0xff)
|
|
}
|
|
|
|
if (offset >= this.length) return
|
|
|
|
this[offset] = value
|
|
}
|
|
|
|
function _writeUInt16 (buf, value, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 1 < buf.length, 'trying to write beyond buffer length')
|
|
verifuint(value, 0xffff)
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
for (var i = 0, j = Math.min(len - offset, 2); i < j; i++) {
|
|
buf[offset + i] =
|
|
(value & (0xff << (8 * (littleEndian ? i : 1 - i)))) >>>
|
|
(littleEndian ? i : 1 - i) * 8
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeUInt16LE = function (value, offset, noAssert) {
|
|
_writeUInt16(this, value, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeUInt16BE = function (value, offset, noAssert) {
|
|
_writeUInt16(this, value, offset, false, noAssert)
|
|
}
|
|
|
|
function _writeUInt32 (buf, value, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 3 < buf.length, 'trying to write beyond buffer length')
|
|
verifuint(value, 0xffffffff)
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
for (var i = 0, j = Math.min(len - offset, 4); i < j; i++) {
|
|
buf[offset + i] =
|
|
(value >>> (littleEndian ? i : 3 - i) * 8) & 0xff
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeUInt32LE = function (value, offset, noAssert) {
|
|
_writeUInt32(this, value, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeUInt32BE = function (value, offset, noAssert) {
|
|
_writeUInt32(this, value, offset, false, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeInt8 = function (value, offset, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset < this.length, 'Trying to write beyond buffer length')
|
|
verifsint(value, 0x7f, -0x80)
|
|
}
|
|
|
|
if (offset >= this.length)
|
|
return
|
|
|
|
if (value >= 0)
|
|
this.writeUInt8(value, offset, noAssert)
|
|
else
|
|
this.writeUInt8(0xff + value + 1, offset, noAssert)
|
|
}
|
|
|
|
function _writeInt16 (buf, value, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 1 < buf.length, 'Trying to write beyond buffer length')
|
|
verifsint(value, 0x7fff, -0x8000)
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
if (value >= 0)
|
|
_writeUInt16(buf, value, offset, littleEndian, noAssert)
|
|
else
|
|
_writeUInt16(buf, 0xffff + value + 1, offset, littleEndian, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeInt16LE = function (value, offset, noAssert) {
|
|
_writeInt16(this, value, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeInt16BE = function (value, offset, noAssert) {
|
|
_writeInt16(this, value, offset, false, noAssert)
|
|
}
|
|
|
|
function _writeInt32 (buf, value, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 3 < buf.length, 'Trying to write beyond buffer length')
|
|
verifsint(value, 0x7fffffff, -0x80000000)
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
if (value >= 0)
|
|
_writeUInt32(buf, value, offset, littleEndian, noAssert)
|
|
else
|
|
_writeUInt32(buf, 0xffffffff + value + 1, offset, littleEndian, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeInt32LE = function (value, offset, noAssert) {
|
|
_writeInt32(this, value, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeInt32BE = function (value, offset, noAssert) {
|
|
_writeInt32(this, value, offset, false, noAssert)
|
|
}
|
|
|
|
function _writeFloat (buf, value, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 3 < buf.length, 'Trying to write beyond buffer length')
|
|
verifIEEE754(value, 3.4028234663852886e+38, -3.4028234663852886e+38)
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
ieee754.write(buf, value, offset, littleEndian, 23, 4)
|
|
}
|
|
|
|
Buffer.prototype.writeFloatLE = function (value, offset, noAssert) {
|
|
_writeFloat(this, value, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeFloatBE = function (value, offset, noAssert) {
|
|
_writeFloat(this, value, offset, false, noAssert)
|
|
}
|
|
|
|
function _writeDouble (buf, value, offset, littleEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert(value !== undefined && value !== null, 'missing value')
|
|
assert(typeof littleEndian === 'boolean', 'missing or invalid endian')
|
|
assert(offset !== undefined && offset !== null, 'missing offset')
|
|
assert(offset + 7 < buf.length,
|
|
'Trying to write beyond buffer length')
|
|
verifIEEE754(value, 1.7976931348623157E+308, -1.7976931348623157E+308)
|
|
}
|
|
|
|
var len = buf.length
|
|
if (offset >= len)
|
|
return
|
|
|
|
ieee754.write(buf, value, offset, littleEndian, 52, 8)
|
|
}
|
|
|
|
Buffer.prototype.writeDoubleLE = function (value, offset, noAssert) {
|
|
_writeDouble(this, value, offset, true, noAssert)
|
|
}
|
|
|
|
Buffer.prototype.writeDoubleBE = function (value, offset, noAssert) {
|
|
_writeDouble(this, value, offset, false, noAssert)
|
|
}
|
|
|
|
// fill(value, start=0, end=buffer.length)
|
|
Buffer.prototype.fill = function (value, start, end) {
|
|
if (!value) value = 0
|
|
if (!start) start = 0
|
|
if (!end) end = this.length
|
|
|
|
if (typeof value === 'string') {
|
|
value = value.charCodeAt(0)
|
|
}
|
|
|
|
assert(typeof value === 'number' && !isNaN(value), 'value is not a number')
|
|
assert(end >= start, 'end < start')
|
|
|
|
// Fill 0 bytes; we're done
|
|
if (end === start) return
|
|
if (this.length === 0) return
|
|
|
|
assert(start >= 0 && start < this.length, 'start out of bounds')
|
|
assert(end >= 0 && end <= this.length, 'end out of bounds')
|
|
|
|
for (var i = start; i < end; i++) {
|
|
this[i] = value
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.inspect = function () {
|
|
var out = []
|
|
var len = this.length
|
|
for (var i = 0; i < len; i++) {
|
|
out[i] = toHex(this[i])
|
|
if (i === exports.INSPECT_MAX_BYTES) {
|
|
out[i + 1] = '...'
|
|
break
|
|
}
|
|
}
|
|
return '<Buffer ' + out.join(' ') + '>'
|
|
}
|
|
|
|
/**
|
|
* Creates a new `ArrayBuffer` with the *copied* memory of the buffer instance.
|
|
* Added in Node 0.12. Only available in browsers that support ArrayBuffer.
|
|
*/
|
|
Buffer.prototype.toArrayBuffer = function () {
|
|
if (typeof Uint8Array !== 'undefined') {
|
|
if (Buffer._useTypedArrays) {
|
|
return (new Buffer(this)).buffer
|
|
} else {
|
|
var buf = new Uint8Array(this.length)
|
|
for (var i = 0, len = buf.length; i < len; i += 1)
|
|
buf[i] = this[i]
|
|
return buf.buffer
|
|
}
|
|
} else {
|
|
throw new Error('Buffer.toArrayBuffer not supported in this browser')
|
|
}
|
|
}
|
|
|
|
// HELPER FUNCTIONS
|
|
// ================
|
|
|
|
function stringtrim (str) {
|
|
if (str.trim) return str.trim()
|
|
return str.replace(/^\s+|\s+$/g, '')
|
|
}
|
|
|
|
var BP = Buffer.prototype
|
|
|
|
/**
|
|
* Augment a Uint8Array *instance* (not the Uint8Array class!) with Buffer methods
|
|
*/
|
|
Buffer._augment = function (arr) {
|
|
arr._isBuffer = true
|
|
|
|
// save reference to original Uint8Array get/set methods before overwriting
|
|
arr._get = arr.get
|
|
arr._set = arr.set
|
|
|
|
// deprecated, will be removed in node 0.13+
|
|
arr.get = BP.get
|
|
arr.set = BP.set
|
|
|
|
arr.write = BP.write
|
|
arr.toString = BP.toString
|
|
arr.toLocaleString = BP.toString
|
|
arr.toJSON = BP.toJSON
|
|
arr.copy = BP.copy
|
|
arr.slice = BP.slice
|
|
arr.readUInt8 = BP.readUInt8
|
|
arr.readUInt16LE = BP.readUInt16LE
|
|
arr.readUInt16BE = BP.readUInt16BE
|
|
arr.readUInt32LE = BP.readUInt32LE
|
|
arr.readUInt32BE = BP.readUInt32BE
|
|
arr.readInt8 = BP.readInt8
|
|
arr.readInt16LE = BP.readInt16LE
|
|
arr.readInt16BE = BP.readInt16BE
|
|
arr.readInt32LE = BP.readInt32LE
|
|
arr.readInt32BE = BP.readInt32BE
|
|
arr.readFloatLE = BP.readFloatLE
|
|
arr.readFloatBE = BP.readFloatBE
|
|
arr.readDoubleLE = BP.readDoubleLE
|
|
arr.readDoubleBE = BP.readDoubleBE
|
|
arr.writeUInt8 = BP.writeUInt8
|
|
arr.writeUInt16LE = BP.writeUInt16LE
|
|
arr.writeUInt16BE = BP.writeUInt16BE
|
|
arr.writeUInt32LE = BP.writeUInt32LE
|
|
arr.writeUInt32BE = BP.writeUInt32BE
|
|
arr.writeInt8 = BP.writeInt8
|
|
arr.writeInt16LE = BP.writeInt16LE
|
|
arr.writeInt16BE = BP.writeInt16BE
|
|
arr.writeInt32LE = BP.writeInt32LE
|
|
arr.writeInt32BE = BP.writeInt32BE
|
|
arr.writeFloatLE = BP.writeFloatLE
|
|
arr.writeFloatBE = BP.writeFloatBE
|
|
arr.writeDoubleLE = BP.writeDoubleLE
|
|
arr.writeDoubleBE = BP.writeDoubleBE
|
|
arr.fill = BP.fill
|
|
arr.inspect = BP.inspect
|
|
arr.toArrayBuffer = BP.toArrayBuffer
|
|
|
|
return arr
|
|
}
|
|
|
|
// slice(start, end)
|
|
function clamp (index, len, defaultValue) {
|
|
if (typeof index !== 'number') return defaultValue
|
|
index = ~~index; // Coerce to integer.
|
|
if (index >= len) return len
|
|
if (index >= 0) return index
|
|
index += len
|
|
if (index >= 0) return index
|
|
return 0
|
|
}
|
|
|
|
function coerce (length) {
|
|
// Coerce length to a number (possibly NaN), round up
|
|
// in case it's fractional (e.g. 123.456) then do a
|
|
// double negate to coerce a NaN to 0. Easy, right?
|
|
length = ~~Math.ceil(+length)
|
|
return length < 0 ? 0 : length
|
|
}
|
|
|
|
function isArray (subject) {
|
|
return (Array.isArray || function (subject) {
|
|
return Object.prototype.toString.call(subject) === '[object Array]'
|
|
})(subject)
|
|
}
|
|
|
|
function isArrayish (subject) {
|
|
return isArray(subject) || Buffer.isBuffer(subject) ||
|
|
subject && typeof subject === 'object' &&
|
|
typeof subject.length === 'number'
|
|
}
|
|
|
|
function toHex (n) {
|
|
if (n < 16) return '0' + n.toString(16)
|
|
return n.toString(16)
|
|
}
|
|
|
|
function utf8ToBytes (str) {
|
|
var byteArray = []
|
|
for (var i = 0; i < str.length; i++) {
|
|
var b = str.charCodeAt(i)
|
|
if (b <= 0x7F)
|
|
byteArray.push(str.charCodeAt(i))
|
|
else {
|
|
var start = i
|
|
if (b >= 0xD800 && b <= 0xDFFF) i++
|
|
var h = encodeURIComponent(str.slice(start, i+1)).substr(1).split('%')
|
|
for (var j = 0; j < h.length; j++)
|
|
byteArray.push(parseInt(h[j], 16))
|
|
}
|
|
}
|
|
return byteArray
|
|
}
|
|
|
|
function asciiToBytes (str) {
|
|
var byteArray = []
|
|
for (var i = 0; i < str.length; i++) {
|
|
// Node's code seems to be doing this and not & 0x7F..
|
|
byteArray.push(str.charCodeAt(i) & 0xFF)
|
|
}
|
|
return byteArray
|
|
}
|
|
|
|
function utf16leToBytes (str) {
|
|
var c, hi, lo
|
|
var byteArray = []
|
|
for (var i = 0; i < str.length; i++) {
|
|
c = str.charCodeAt(i)
|
|
hi = c >> 8
|
|
lo = c % 256
|
|
byteArray.push(lo)
|
|
byteArray.push(hi)
|
|
}
|
|
|
|
return byteArray
|
|
}
|
|
|
|
function base64ToBytes (str) {
|
|
return base64.toByteArray(str)
|
|
}
|
|
|
|
function blitBuffer (src, dst, offset, length) {
|
|
var pos
|
|
for (var i = 0; i < length; i++) {
|
|
if ((i + offset >= dst.length) || (i >= src.length))
|
|
break
|
|
dst[i + offset] = src[i]
|
|
}
|
|
return i
|
|
}
|
|
|
|
function decodeUtf8Char (str) {
|
|
try {
|
|
return decodeURIComponent(str)
|
|
} catch (err) {
|
|
return String.fromCharCode(0xFFFD) // UTF 8 invalid char
|
|
}
|
|
}
|
|
|
|
/*
|
|
* We have to make sure that the value is a valid integer. This means that it
|
|
* is non-negative. It has no fractional component and that it does not
|
|
* exceed the maximum allowed value.
|
|
*/
|
|
function verifuint (value, max) {
|
|
assert(typeof value === 'number', 'cannot write a non-number as a number')
|
|
assert(value >= 0, 'specified a negative value for writing an unsigned value')
|
|
assert(value <= max, 'value is larger than maximum value for type')
|
|
assert(Math.floor(value) === value, 'value has a fractional component')
|
|
}
|
|
|
|
function verifsint (value, max, min) {
|
|
assert(typeof value === 'number', 'cannot write a non-number as a number')
|
|
assert(value <= max, 'value larger than maximum allowed value')
|
|
assert(value >= min, 'value smaller than minimum allowed value')
|
|
assert(Math.floor(value) === value, 'value has a fractional component')
|
|
}
|
|
|
|
function verifIEEE754 (value, max, min) {
|
|
assert(typeof value === 'number', 'cannot write a non-number as a number')
|
|
assert(value <= max, 'value larger than maximum allowed value')
|
|
assert(value >= min, 'value smaller than minimum allowed value')
|
|
}
|
|
|
|
function assert (test, message) {
|
|
if (!test) throw new Error(message || 'Failed assertion')
|
|
}
|
|
|
|
},{"base64-js":81,"ieee754":82}],81:[function(require,module,exports){
|
|
var lookup = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
|
|
|
|
;(function (exports) {
|
|
'use strict';
|
|
|
|
var Arr = (typeof Uint8Array !== 'undefined')
|
|
? Uint8Array
|
|
: Array
|
|
|
|
var ZERO = '0'.charCodeAt(0)
|
|
var PLUS = '+'.charCodeAt(0)
|
|
var SLASH = '/'.charCodeAt(0)
|
|
var NUMBER = '0'.charCodeAt(0)
|
|
var LOWER = 'a'.charCodeAt(0)
|
|
var UPPER = 'A'.charCodeAt(0)
|
|
|
|
function decode (elt) {
|
|
var code = elt.charCodeAt(0)
|
|
if (code === PLUS)
|
|
return 62 // '+'
|
|
if (code === SLASH)
|
|
return 63 // '/'
|
|
if (code < NUMBER)
|
|
return -1 //no match
|
|
if (code < NUMBER + 10)
|
|
return code - NUMBER + 26 + 26
|
|
if (code < UPPER + 26)
|
|
return code - UPPER
|
|
if (code < LOWER + 26)
|
|
return code - LOWER + 26
|
|
}
|
|
|
|
function b64ToByteArray (b64) {
|
|
var i, j, l, tmp, placeHolders, arr
|
|
|
|
if (b64.length % 4 > 0) {
|
|
throw new Error('Invalid string. Length must be a multiple of 4')
|
|
}
|
|
|
|
// the number of equal signs (place holders)
|
|
// if there are two placeholders, than the two characters before it
|
|
// represent one byte
|
|
// if there is only one, then the three characters before it represent 2 bytes
|
|
// this is just a cheap hack to not do indexOf twice
|
|
var len = b64.length
|
|
placeHolders = '=' === b64.charAt(len - 2) ? 2 : '=' === b64.charAt(len - 1) ? 1 : 0
|
|
|
|
// base64 is 4/3 + up to two characters of the original data
|
|
arr = new Arr(b64.length * 3 / 4 - placeHolders)
|
|
|
|
// if there are placeholders, only get up to the last complete 4 chars
|
|
l = placeHolders > 0 ? b64.length - 4 : b64.length
|
|
|
|
var L = 0
|
|
|
|
function push (v) {
|
|
arr[L++] = v
|
|
}
|
|
|
|
for (i = 0, j = 0; i < l; i += 4, j += 3) {
|
|
tmp = (decode(b64.charAt(i)) << 18) | (decode(b64.charAt(i + 1)) << 12) | (decode(b64.charAt(i + 2)) << 6) | decode(b64.charAt(i + 3))
|
|
push((tmp & 0xFF0000) >> 16)
|
|
push((tmp & 0xFF00) >> 8)
|
|
push(tmp & 0xFF)
|
|
}
|
|
|
|
if (placeHolders === 2) {
|
|
tmp = (decode(b64.charAt(i)) << 2) | (decode(b64.charAt(i + 1)) >> 4)
|
|
push(tmp & 0xFF)
|
|
} else if (placeHolders === 1) {
|
|
tmp = (decode(b64.charAt(i)) << 10) | (decode(b64.charAt(i + 1)) << 4) | (decode(b64.charAt(i + 2)) >> 2)
|
|
push((tmp >> 8) & 0xFF)
|
|
push(tmp & 0xFF)
|
|
}
|
|
|
|
return arr
|
|
}
|
|
|
|
function uint8ToBase64 (uint8) {
|
|
var i,
|
|
extraBytes = uint8.length % 3, // if we have 1 byte left, pad 2 bytes
|
|
output = "",
|
|
temp, length
|
|
|
|
function encode (num) {
|
|
return lookup.charAt(num)
|
|
}
|
|
|
|
function tripletToBase64 (num) {
|
|
return encode(num >> 18 & 0x3F) + encode(num >> 12 & 0x3F) + encode(num >> 6 & 0x3F) + encode(num & 0x3F)
|
|
}
|
|
|
|
// go through the array every three bytes, we'll deal with trailing stuff later
|
|
for (i = 0, length = uint8.length - extraBytes; i < length; i += 3) {
|
|
temp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2])
|
|
output += tripletToBase64(temp)
|
|
}
|
|
|
|
// pad the end with zeros, but make sure to not forget the extra bytes
|
|
switch (extraBytes) {
|
|
case 1:
|
|
temp = uint8[uint8.length - 1]
|
|
output += encode(temp >> 2)
|
|
output += encode((temp << 4) & 0x3F)
|
|
output += '=='
|
|
break
|
|
case 2:
|
|
temp = (uint8[uint8.length - 2] << 8) + (uint8[uint8.length - 1])
|
|
output += encode(temp >> 10)
|
|
output += encode((temp >> 4) & 0x3F)
|
|
output += encode((temp << 2) & 0x3F)
|
|
output += '='
|
|
break
|
|
}
|
|
|
|
return output
|
|
}
|
|
|
|
module.exports.toByteArray = b64ToByteArray
|
|
module.exports.fromByteArray = uint8ToBase64
|
|
}())
|
|
|
|
},{}],82:[function(require,module,exports){
|
|
exports.read = function(buffer, offset, isLE, mLen, nBytes) {
|
|
var e, m,
|
|
eLen = nBytes * 8 - mLen - 1,
|
|
eMax = (1 << eLen) - 1,
|
|
eBias = eMax >> 1,
|
|
nBits = -7,
|
|
i = isLE ? (nBytes - 1) : 0,
|
|
d = isLE ? -1 : 1,
|
|
s = buffer[offset + i];
|
|
|
|
i += d;
|
|
|
|
e = s & ((1 << (-nBits)) - 1);
|
|
s >>= (-nBits);
|
|
nBits += eLen;
|
|
for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8);
|
|
|
|
m = e & ((1 << (-nBits)) - 1);
|
|
e >>= (-nBits);
|
|
nBits += mLen;
|
|
for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8);
|
|
|
|
if (e === 0) {
|
|
e = 1 - eBias;
|
|
} else if (e === eMax) {
|
|
return m ? NaN : ((s ? -1 : 1) * Infinity);
|
|
} else {
|
|
m = m + Math.pow(2, mLen);
|
|
e = e - eBias;
|
|
}
|
|
return (s ? -1 : 1) * m * Math.pow(2, e - mLen);
|
|
};
|
|
|
|
exports.write = function(buffer, value, offset, isLE, mLen, nBytes) {
|
|
var e, m, c,
|
|
eLen = nBytes * 8 - mLen - 1,
|
|
eMax = (1 << eLen) - 1,
|
|
eBias = eMax >> 1,
|
|
rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0),
|
|
i = isLE ? 0 : (nBytes - 1),
|
|
d = isLE ? 1 : -1,
|
|
s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0;
|
|
|
|
value = Math.abs(value);
|
|
|
|
if (isNaN(value) || value === Infinity) {
|
|
m = isNaN(value) ? 1 : 0;
|
|
e = eMax;
|
|
} else {
|
|
e = Math.floor(Math.log(value) / Math.LN2);
|
|
if (value * (c = Math.pow(2, -e)) < 1) {
|
|
e--;
|
|
c *= 2;
|
|
}
|
|
if (e + eBias >= 1) {
|
|
value += rt / c;
|
|
} else {
|
|
value += rt * Math.pow(2, 1 - eBias);
|
|
}
|
|
if (value * c >= 2) {
|
|
e++;
|
|
c /= 2;
|
|
}
|
|
|
|
if (e + eBias >= eMax) {
|
|
m = 0;
|
|
e = eMax;
|
|
} else if (e + eBias >= 1) {
|
|
m = (value * c - 1) * Math.pow(2, mLen);
|
|
e = e + eBias;
|
|
} else {
|
|
m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);
|
|
e = 0;
|
|
}
|
|
}
|
|
|
|
for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8);
|
|
|
|
e = (e << mLen) | m;
|
|
eLen += mLen;
|
|
for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8);
|
|
|
|
buffer[offset + i - d] |= s * 128;
|
|
};
|
|
|
|
},{}],83:[function(require,module,exports){
|
|
var Buffer = require('buffer').Buffer;
|
|
var intSize = 4;
|
|
var zeroBuffer = new Buffer(intSize); zeroBuffer.fill(0);
|
|
var chrsz = 8;
|
|
|
|
function toArray(buf, bigEndian) {
|
|
if ((buf.length % intSize) !== 0) {
|
|
var len = buf.length + (intSize - (buf.length % intSize));
|
|
buf = Buffer.concat([buf, zeroBuffer], len);
|
|
}
|
|
|
|
var arr = [];
|
|
var fn = bigEndian ? buf.readInt32BE : buf.readInt32LE;
|
|
for (var i = 0; i < buf.length; i += intSize) {
|
|
arr.push(fn.call(buf, i));
|
|
}
|
|
return arr;
|
|
}
|
|
|
|
function toBuffer(arr, size, bigEndian) {
|
|
var buf = new Buffer(size);
|
|
var fn = bigEndian ? buf.writeInt32BE : buf.writeInt32LE;
|
|
for (var i = 0; i < arr.length; i++) {
|
|
fn.call(buf, arr[i], i * 4, true);
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
function hash(buf, fn, hashSize, bigEndian) {
|
|
if (!Buffer.isBuffer(buf)) buf = new Buffer(buf);
|
|
var arr = fn(toArray(buf, bigEndian), buf.length * chrsz);
|
|
return toBuffer(arr, hashSize, bigEndian);
|
|
}
|
|
|
|
module.exports = { hash: hash };
|
|
|
|
},{"buffer":80}],84:[function(require,module,exports){
|
|
var Buffer = require('buffer').Buffer
|
|
var sha = require('./sha')
|
|
var sha256 = require('./sha256')
|
|
var rng = require('./rng')
|
|
var md5 = require('./md5')
|
|
|
|
var algorithms = {
|
|
sha1: sha,
|
|
sha256: sha256,
|
|
md5: md5
|
|
}
|
|
|
|
var blocksize = 64
|
|
var zeroBuffer = new Buffer(blocksize); zeroBuffer.fill(0)
|
|
function hmac(fn, key, data) {
|
|
if(!Buffer.isBuffer(key)) key = new Buffer(key)
|
|
if(!Buffer.isBuffer(data)) data = new Buffer(data)
|
|
|
|
if(key.length > blocksize) {
|
|
key = fn(key)
|
|
} else if(key.length < blocksize) {
|
|
key = Buffer.concat([key, zeroBuffer], blocksize)
|
|
}
|
|
|
|
var ipad = new Buffer(blocksize), opad = new Buffer(blocksize)
|
|
for(var i = 0; i < blocksize; i++) {
|
|
ipad[i] = key[i] ^ 0x36
|
|
opad[i] = key[i] ^ 0x5C
|
|
}
|
|
|
|
var hash = fn(Buffer.concat([ipad, data]))
|
|
return fn(Buffer.concat([opad, hash]))
|
|
}
|
|
|
|
function hash(alg, key) {
|
|
alg = alg || 'sha1'
|
|
var fn = algorithms[alg]
|
|
var bufs = []
|
|
var length = 0
|
|
if(!fn) error('algorithm:', alg, 'is not yet supported')
|
|
return {
|
|
update: function (data) {
|
|
if(!Buffer.isBuffer(data)) data = new Buffer(data)
|
|
|
|
bufs.push(data)
|
|
length += data.length
|
|
return this
|
|
},
|
|
digest: function (enc) {
|
|
var buf = Buffer.concat(bufs)
|
|
var r = key ? hmac(fn, key, buf) : fn(buf)
|
|
bufs = null
|
|
return enc ? r.toString(enc) : r
|
|
}
|
|
}
|
|
}
|
|
|
|
function error () {
|
|
var m = [].slice.call(arguments).join(' ')
|
|
throw new Error([
|
|
m,
|
|
'we accept pull requests',
|
|
'http://github.com/dominictarr/crypto-browserify'
|
|
].join('\n'))
|
|
}
|
|
|
|
exports.createHash = function (alg) { return hash(alg) }
|
|
exports.createHmac = function (alg, key) { return hash(alg, key) }
|
|
exports.randomBytes = function(size, callback) {
|
|
if (callback && callback.call) {
|
|
try {
|
|
callback.call(this, undefined, new Buffer(rng(size)))
|
|
} catch (err) { callback(err) }
|
|
} else {
|
|
return new Buffer(rng(size))
|
|
}
|
|
}
|
|
|
|
function each(a, f) {
|
|
for(var i in a)
|
|
f(a[i], i)
|
|
}
|
|
|
|
// the least I can do is make error messages for the rest of the node.js/crypto api.
|
|
each(['createCredentials'
|
|
, 'createCipher'
|
|
, 'createCipheriv'
|
|
, 'createDecipher'
|
|
, 'createDecipheriv'
|
|
, 'createSign'
|
|
, 'createVerify'
|
|
, 'createDiffieHellman'
|
|
, 'pbkdf2'], function (name) {
|
|
exports[name] = function () {
|
|
error('sorry,', name, 'is not implemented yet')
|
|
}
|
|
})
|
|
|
|
},{"./md5":85,"./rng":86,"./sha":87,"./sha256":88,"buffer":80}],85:[function(require,module,exports){
|
|
/*
|
|
* A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
|
|
* Digest Algorithm, as defined in RFC 1321.
|
|
* Version 2.1 Copyright (C) Paul Johnston 1999 - 2002.
|
|
* Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
|
|
* Distributed under the BSD License
|
|
* See http://pajhome.org.uk/crypt/md5 for more info.
|
|
*/
|
|
|
|
var helpers = require('./helpers');
|
|
|
|
/*
|
|
* Perform a simple self-test to see if the VM is working
|
|
*/
|
|
function md5_vm_test()
|
|
{
|
|
return hex_md5("abc") == "900150983cd24fb0d6963f7d28e17f72";
|
|
}
|
|
|
|
/*
|
|
* Calculate the MD5 of an array of little-endian words, and a bit length
|
|
*/
|
|
function core_md5(x, len)
|
|
{
|
|
/* append padding */
|
|
x[len >> 5] |= 0x80 << ((len) % 32);
|
|
x[(((len + 64) >>> 9) << 4) + 14] = len;
|
|
|
|
var a = 1732584193;
|
|
var b = -271733879;
|
|
var c = -1732584194;
|
|
var d = 271733878;
|
|
|
|
for(var i = 0; i < x.length; i += 16)
|
|
{
|
|
var olda = a;
|
|
var oldb = b;
|
|
var oldc = c;
|
|
var oldd = d;
|
|
|
|
a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
|
|
d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
|
|
c = md5_ff(c, d, a, b, x[i+ 2], 17, 606105819);
|
|
b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
|
|
a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
|
|
d = md5_ff(d, a, b, c, x[i+ 5], 12, 1200080426);
|
|
c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
|
|
b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
|
|
a = md5_ff(a, b, c, d, x[i+ 8], 7 , 1770035416);
|
|
d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
|
|
c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
|
|
b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
|
|
a = md5_ff(a, b, c, d, x[i+12], 7 , 1804603682);
|
|
d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
|
|
c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
|
|
b = md5_ff(b, c, d, a, x[i+15], 22, 1236535329);
|
|
|
|
a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
|
|
d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
|
|
c = md5_gg(c, d, a, b, x[i+11], 14, 643717713);
|
|
b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
|
|
a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
|
|
d = md5_gg(d, a, b, c, x[i+10], 9 , 38016083);
|
|
c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
|
|
b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
|
|
a = md5_gg(a, b, c, d, x[i+ 9], 5 , 568446438);
|
|
d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
|
|
c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
|
|
b = md5_gg(b, c, d, a, x[i+ 8], 20, 1163531501);
|
|
a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
|
|
d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
|
|
c = md5_gg(c, d, a, b, x[i+ 7], 14, 1735328473);
|
|
b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);
|
|
|
|
a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
|
|
d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
|
|
c = md5_hh(c, d, a, b, x[i+11], 16, 1839030562);
|
|
b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
|
|
a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
|
|
d = md5_hh(d, a, b, c, x[i+ 4], 11, 1272893353);
|
|
c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
|
|
b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
|
|
a = md5_hh(a, b, c, d, x[i+13], 4 , 681279174);
|
|
d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
|
|
c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
|
|
b = md5_hh(b, c, d, a, x[i+ 6], 23, 76029189);
|
|
a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
|
|
d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
|
|
c = md5_hh(c, d, a, b, x[i+15], 16, 530742520);
|
|
b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);
|
|
|
|
a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
|
|
d = md5_ii(d, a, b, c, x[i+ 7], 10, 1126891415);
|
|
c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
|
|
b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
|
|
a = md5_ii(a, b, c, d, x[i+12], 6 , 1700485571);
|
|
d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
|
|
c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
|
|
b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
|
|
a = md5_ii(a, b, c, d, x[i+ 8], 6 , 1873313359);
|
|
d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
|
|
c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
|
|
b = md5_ii(b, c, d, a, x[i+13], 21, 1309151649);
|
|
a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
|
|
d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
|
|
c = md5_ii(c, d, a, b, x[i+ 2], 15, 718787259);
|
|
b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);
|
|
|
|
a = safe_add(a, olda);
|
|
b = safe_add(b, oldb);
|
|
c = safe_add(c, oldc);
|
|
d = safe_add(d, oldd);
|
|
}
|
|
return Array(a, b, c, d);
|
|
|
|
}
|
|
|
|
/*
|
|
* These functions implement the four basic operations the algorithm uses.
|
|
*/
|
|
function md5_cmn(q, a, b, x, s, t)
|
|
{
|
|
return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s),b);
|
|
}
|
|
function md5_ff(a, b, c, d, x, s, t)
|
|
{
|
|
return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
|
|
}
|
|
function md5_gg(a, b, c, d, x, s, t)
|
|
{
|
|
return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
|
|
}
|
|
function md5_hh(a, b, c, d, x, s, t)
|
|
{
|
|
return md5_cmn(b ^ c ^ d, a, b, x, s, t);
|
|
}
|
|
function md5_ii(a, b, c, d, x, s, t)
|
|
{
|
|
return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
|
|
}
|
|
|
|
/*
|
|
* Add integers, wrapping at 2^32. This uses 16-bit operations internally
|
|
* to work around bugs in some JS interpreters.
|
|
*/
|
|
function safe_add(x, y)
|
|
{
|
|
var lsw = (x & 0xFFFF) + (y & 0xFFFF);
|
|
var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
|
|
return (msw << 16) | (lsw & 0xFFFF);
|
|
}
|
|
|
|
/*
|
|
* Bitwise rotate a 32-bit number to the left.
|
|
*/
|
|
function bit_rol(num, cnt)
|
|
{
|
|
return (num << cnt) | (num >>> (32 - cnt));
|
|
}
|
|
|
|
module.exports = function md5(buf) {
|
|
return helpers.hash(buf, core_md5, 16);
|
|
};
|
|
|
|
},{"./helpers":83}],86:[function(require,module,exports){
|
|
// Original code adapted from Robert Kieffer.
|
|
// details at https://github.com/broofa/node-uuid
|
|
(function() {
|
|
var _global = this;
|
|
|
|
var mathRNG, whatwgRNG;
|
|
|
|
// NOTE: Math.random() does not guarantee "cryptographic quality"
|
|
mathRNG = function(size) {
|
|
var bytes = new Array(size);
|
|
var r;
|
|
|
|
for (var i = 0, r; i < size; i++) {
|
|
if ((i & 0x03) == 0) r = Math.random() * 0x100000000;
|
|
bytes[i] = r >>> ((i & 0x03) << 3) & 0xff;
|
|
}
|
|
|
|
return bytes;
|
|
}
|
|
|
|
if (_global.crypto && crypto.getRandomValues) {
|
|
whatwgRNG = function(size) {
|
|
var bytes = new Uint8Array(size);
|
|
crypto.getRandomValues(bytes);
|
|
return bytes;
|
|
}
|
|
}
|
|
|
|
module.exports = whatwgRNG || mathRNG;
|
|
|
|
}())
|
|
|
|
},{}],87:[function(require,module,exports){
|
|
/*
|
|
* A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined
|
|
* in FIPS PUB 180-1
|
|
* Version 2.1a Copyright Paul Johnston 2000 - 2002.
|
|
* Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
|
|
* Distributed under the BSD License
|
|
* See http://pajhome.org.uk/crypt/md5 for details.
|
|
*/
|
|
|
|
var helpers = require('./helpers');
|
|
|
|
/*
|
|
* Calculate the SHA-1 of an array of big-endian words, and a bit length
|
|
*/
|
|
function core_sha1(x, len)
|
|
{
|
|
/* append padding */
|
|
x[len >> 5] |= 0x80 << (24 - len % 32);
|
|
x[((len + 64 >> 9) << 4) + 15] = len;
|
|
|
|
var w = Array(80);
|
|
var a = 1732584193;
|
|
var b = -271733879;
|
|
var c = -1732584194;
|
|
var d = 271733878;
|
|
var e = -1009589776;
|
|
|
|
for(var i = 0; i < x.length; i += 16)
|
|
{
|
|
var olda = a;
|
|
var oldb = b;
|
|
var oldc = c;
|
|
var oldd = d;
|
|
var olde = e;
|
|
|
|
for(var j = 0; j < 80; j++)
|
|
{
|
|
if(j < 16) w[j] = x[i + j];
|
|
else w[j] = rol(w[j-3] ^ w[j-8] ^ w[j-14] ^ w[j-16], 1);
|
|
var t = safe_add(safe_add(rol(a, 5), sha1_ft(j, b, c, d)),
|
|
safe_add(safe_add(e, w[j]), sha1_kt(j)));
|
|
e = d;
|
|
d = c;
|
|
c = rol(b, 30);
|
|
b = a;
|
|
a = t;
|
|
}
|
|
|
|
a = safe_add(a, olda);
|
|
b = safe_add(b, oldb);
|
|
c = safe_add(c, oldc);
|
|
d = safe_add(d, oldd);
|
|
e = safe_add(e, olde);
|
|
}
|
|
return Array(a, b, c, d, e);
|
|
|
|
}
|
|
|
|
/*
|
|
* Perform the appropriate triplet combination function for the current
|
|
* iteration
|
|
*/
|
|
function sha1_ft(t, b, c, d)
|
|
{
|
|
if(t < 20) return (b & c) | ((~b) & d);
|
|
if(t < 40) return b ^ c ^ d;
|
|
if(t < 60) return (b & c) | (b & d) | (c & d);
|
|
return b ^ c ^ d;
|
|
}
|
|
|
|
/*
|
|
* Determine the appropriate additive constant for the current iteration
|
|
*/
|
|
function sha1_kt(t)
|
|
{
|
|
return (t < 20) ? 1518500249 : (t < 40) ? 1859775393 :
|
|
(t < 60) ? -1894007588 : -899497514;
|
|
}
|
|
|
|
/*
|
|
* Add integers, wrapping at 2^32. This uses 16-bit operations internally
|
|
* to work around bugs in some JS interpreters.
|
|
*/
|
|
function safe_add(x, y)
|
|
{
|
|
var lsw = (x & 0xFFFF) + (y & 0xFFFF);
|
|
var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
|
|
return (msw << 16) | (lsw & 0xFFFF);
|
|
}
|
|
|
|
/*
|
|
* Bitwise rotate a 32-bit number to the left.
|
|
*/
|
|
function rol(num, cnt)
|
|
{
|
|
return (num << cnt) | (num >>> (32 - cnt));
|
|
}
|
|
|
|
module.exports = function sha1(buf) {
|
|
return helpers.hash(buf, core_sha1, 20, true);
|
|
};
|
|
|
|
},{"./helpers":83}],88:[function(require,module,exports){
|
|
|
|
/**
|
|
* A JavaScript implementation of the Secure Hash Algorithm, SHA-256, as defined
|
|
* in FIPS 180-2
|
|
* Version 2.2-beta Copyright Angel Marin, Paul Johnston 2000 - 2009.
|
|
* Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
|
|
*
|
|
*/
|
|
|
|
var helpers = require('./helpers');
|
|
|
|
var safe_add = function(x, y) {
|
|
var lsw = (x & 0xFFFF) + (y & 0xFFFF);
|
|
var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
|
|
return (msw << 16) | (lsw & 0xFFFF);
|
|
};
|
|
|
|
var S = function(X, n) {
|
|
return (X >>> n) | (X << (32 - n));
|
|
};
|
|
|
|
var R = function(X, n) {
|
|
return (X >>> n);
|
|
};
|
|
|
|
var Ch = function(x, y, z) {
|
|
return ((x & y) ^ ((~x) & z));
|
|
};
|
|
|
|
var Maj = function(x, y, z) {
|
|
return ((x & y) ^ (x & z) ^ (y & z));
|
|
};
|
|
|
|
var Sigma0256 = function(x) {
|
|
return (S(x, 2) ^ S(x, 13) ^ S(x, 22));
|
|
};
|
|
|
|
var Sigma1256 = function(x) {
|
|
return (S(x, 6) ^ S(x, 11) ^ S(x, 25));
|
|
};
|
|
|
|
var Gamma0256 = function(x) {
|
|
return (S(x, 7) ^ S(x, 18) ^ R(x, 3));
|
|
};
|
|
|
|
var Gamma1256 = function(x) {
|
|
return (S(x, 17) ^ S(x, 19) ^ R(x, 10));
|
|
};
|
|
|
|
var core_sha256 = function(m, l) {
|
|
var K = new Array(0x428A2F98,0x71374491,0xB5C0FBCF,0xE9B5DBA5,0x3956C25B,0x59F111F1,0x923F82A4,0xAB1C5ED5,0xD807AA98,0x12835B01,0x243185BE,0x550C7DC3,0x72BE5D74,0x80DEB1FE,0x9BDC06A7,0xC19BF174,0xE49B69C1,0xEFBE4786,0xFC19DC6,0x240CA1CC,0x2DE92C6F,0x4A7484AA,0x5CB0A9DC,0x76F988DA,0x983E5152,0xA831C66D,0xB00327C8,0xBF597FC7,0xC6E00BF3,0xD5A79147,0x6CA6351,0x14292967,0x27B70A85,0x2E1B2138,0x4D2C6DFC,0x53380D13,0x650A7354,0x766A0ABB,0x81C2C92E,0x92722C85,0xA2BFE8A1,0xA81A664B,0xC24B8B70,0xC76C51A3,0xD192E819,0xD6990624,0xF40E3585,0x106AA070,0x19A4C116,0x1E376C08,0x2748774C,0x34B0BCB5,0x391C0CB3,0x4ED8AA4A,0x5B9CCA4F,0x682E6FF3,0x748F82EE,0x78A5636F,0x84C87814,0x8CC70208,0x90BEFFFA,0xA4506CEB,0xBEF9A3F7,0xC67178F2);
|
|
var HASH = new Array(0x6A09E667, 0xBB67AE85, 0x3C6EF372, 0xA54FF53A, 0x510E527F, 0x9B05688C, 0x1F83D9AB, 0x5BE0CD19);
|
|
var W = new Array(64);
|
|
var a, b, c, d, e, f, g, h, i, j;
|
|
var T1, T2;
|
|
/* append padding */
|
|
m[l >> 5] |= 0x80 << (24 - l % 32);
|
|
m[((l + 64 >> 9) << 4) + 15] = l;
|
|
for (var i = 0; i < m.length; i += 16) {
|
|
a = HASH[0]; b = HASH[1]; c = HASH[2]; d = HASH[3]; e = HASH[4]; f = HASH[5]; g = HASH[6]; h = HASH[7];
|
|
for (var j = 0; j < 64; j++) {
|
|
if (j < 16) {
|
|
W[j] = m[j + i];
|
|
} else {
|
|
W[j] = safe_add(safe_add(safe_add(Gamma1256(W[j - 2]), W[j - 7]), Gamma0256(W[j - 15])), W[j - 16]);
|
|
}
|
|
T1 = safe_add(safe_add(safe_add(safe_add(h, Sigma1256(e)), Ch(e, f, g)), K[j]), W[j]);
|
|
T2 = safe_add(Sigma0256(a), Maj(a, b, c));
|
|
h = g; g = f; f = e; e = safe_add(d, T1); d = c; c = b; b = a; a = safe_add(T1, T2);
|
|
}
|
|
HASH[0] = safe_add(a, HASH[0]); HASH[1] = safe_add(b, HASH[1]); HASH[2] = safe_add(c, HASH[2]); HASH[3] = safe_add(d, HASH[3]);
|
|
HASH[4] = safe_add(e, HASH[4]); HASH[5] = safe_add(f, HASH[5]); HASH[6] = safe_add(g, HASH[6]); HASH[7] = safe_add(h, HASH[7]);
|
|
}
|
|
return HASH;
|
|
};
|
|
|
|
module.exports = function sha256(buf) {
|
|
return helpers.hash(buf, core_sha256, 32, true);
|
|
};
|
|
|
|
},{"./helpers":83}],89:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
function EventEmitter() {
|
|
this._events = this._events || {};
|
|
this._maxListeners = this._maxListeners || undefined;
|
|
}
|
|
module.exports = EventEmitter;
|
|
|
|
// Backwards-compat with node 0.10.x
|
|
EventEmitter.EventEmitter = EventEmitter;
|
|
|
|
EventEmitter.prototype._events = undefined;
|
|
EventEmitter.prototype._maxListeners = undefined;
|
|
|
|
// By default EventEmitters will print a warning if more than 10 listeners are
|
|
// added to it. This is a useful default which helps finding memory leaks.
|
|
EventEmitter.defaultMaxListeners = 10;
|
|
|
|
// Obviously not all Emitters should be limited to 10. This function allows
|
|
// that to be increased. Set to zero for unlimited.
|
|
EventEmitter.prototype.setMaxListeners = function(n) {
|
|
if (!isNumber(n) || n < 0 || isNaN(n))
|
|
throw TypeError('n must be a positive number');
|
|
this._maxListeners = n;
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.emit = function(type) {
|
|
var er, handler, len, args, i, listeners;
|
|
|
|
if (!this._events)
|
|
this._events = {};
|
|
|
|
// If there is no 'error' event listener then throw.
|
|
if (type === 'error') {
|
|
if (!this._events.error ||
|
|
(isObject(this._events.error) && !this._events.error.length)) {
|
|
er = arguments[1];
|
|
if (er instanceof Error) {
|
|
throw er; // Unhandled 'error' event
|
|
} else {
|
|
throw TypeError('Uncaught, unspecified "error" event.');
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
handler = this._events[type];
|
|
|
|
if (isUndefined(handler))
|
|
return false;
|
|
|
|
if (isFunction(handler)) {
|
|
switch (arguments.length) {
|
|
// fast cases
|
|
case 1:
|
|
handler.call(this);
|
|
break;
|
|
case 2:
|
|
handler.call(this, arguments[1]);
|
|
break;
|
|
case 3:
|
|
handler.call(this, arguments[1], arguments[2]);
|
|
break;
|
|
// slower
|
|
default:
|
|
len = arguments.length;
|
|
args = new Array(len - 1);
|
|
for (i = 1; i < len; i++)
|
|
args[i - 1] = arguments[i];
|
|
handler.apply(this, args);
|
|
}
|
|
} else if (isObject(handler)) {
|
|
len = arguments.length;
|
|
args = new Array(len - 1);
|
|
for (i = 1; i < len; i++)
|
|
args[i - 1] = arguments[i];
|
|
|
|
listeners = handler.slice();
|
|
len = listeners.length;
|
|
for (i = 0; i < len; i++)
|
|
listeners[i].apply(this, args);
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
EventEmitter.prototype.addListener = function(type, listener) {
|
|
var m;
|
|
|
|
if (!isFunction(listener))
|
|
throw TypeError('listener must be a function');
|
|
|
|
if (!this._events)
|
|
this._events = {};
|
|
|
|
// To avoid recursion in the case that type === "newListener"! Before
|
|
// adding it to the listeners, first emit "newListener".
|
|
if (this._events.newListener)
|
|
this.emit('newListener', type,
|
|
isFunction(listener.listener) ?
|
|
listener.listener : listener);
|
|
|
|
if (!this._events[type])
|
|
// Optimize the case of one listener. Don't need the extra array object.
|
|
this._events[type] = listener;
|
|
else if (isObject(this._events[type]))
|
|
// If we've already got an array, just append.
|
|
this._events[type].push(listener);
|
|
else
|
|
// Adding the second element, need to change to array.
|
|
this._events[type] = [this._events[type], listener];
|
|
|
|
// Check for listener leak
|
|
if (isObject(this._events[type]) && !this._events[type].warned) {
|
|
var m;
|
|
if (!isUndefined(this._maxListeners)) {
|
|
m = this._maxListeners;
|
|
} else {
|
|
m = EventEmitter.defaultMaxListeners;
|
|
}
|
|
|
|
if (m && m > 0 && this._events[type].length > m) {
|
|
this._events[type].warned = true;
|
|
console.error('(node) warning: possible EventEmitter memory ' +
|
|
'leak detected. %d listeners added. ' +
|
|
'Use emitter.setMaxListeners() to increase limit.',
|
|
this._events[type].length);
|
|
if (typeof console.trace === 'function') {
|
|
// not supported in IE 10
|
|
console.trace();
|
|
}
|
|
}
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.on = EventEmitter.prototype.addListener;
|
|
|
|
EventEmitter.prototype.once = function(type, listener) {
|
|
if (!isFunction(listener))
|
|
throw TypeError('listener must be a function');
|
|
|
|
var fired = false;
|
|
|
|
function g() {
|
|
this.removeListener(type, g);
|
|
|
|
if (!fired) {
|
|
fired = true;
|
|
listener.apply(this, arguments);
|
|
}
|
|
}
|
|
|
|
g.listener = listener;
|
|
this.on(type, g);
|
|
|
|
return this;
|
|
};
|
|
|
|
// emits a 'removeListener' event iff the listener was removed
|
|
EventEmitter.prototype.removeListener = function(type, listener) {
|
|
var list, position, length, i;
|
|
|
|
if (!isFunction(listener))
|
|
throw TypeError('listener must be a function');
|
|
|
|
if (!this._events || !this._events[type])
|
|
return this;
|
|
|
|
list = this._events[type];
|
|
length = list.length;
|
|
position = -1;
|
|
|
|
if (list === listener ||
|
|
(isFunction(list.listener) && list.listener === listener)) {
|
|
delete this._events[type];
|
|
if (this._events.removeListener)
|
|
this.emit('removeListener', type, listener);
|
|
|
|
} else if (isObject(list)) {
|
|
for (i = length; i-- > 0;) {
|
|
if (list[i] === listener ||
|
|
(list[i].listener && list[i].listener === listener)) {
|
|
position = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (position < 0)
|
|
return this;
|
|
|
|
if (list.length === 1) {
|
|
list.length = 0;
|
|
delete this._events[type];
|
|
} else {
|
|
list.splice(position, 1);
|
|
}
|
|
|
|
if (this._events.removeListener)
|
|
this.emit('removeListener', type, listener);
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.removeAllListeners = function(type) {
|
|
var key, listeners;
|
|
|
|
if (!this._events)
|
|
return this;
|
|
|
|
// not listening for removeListener, no need to emit
|
|
if (!this._events.removeListener) {
|
|
if (arguments.length === 0)
|
|
this._events = {};
|
|
else if (this._events[type])
|
|
delete this._events[type];
|
|
return this;
|
|
}
|
|
|
|
// emit removeListener for all listeners on all events
|
|
if (arguments.length === 0) {
|
|
for (key in this._events) {
|
|
if (key === 'removeListener') continue;
|
|
this.removeAllListeners(key);
|
|
}
|
|
this.removeAllListeners('removeListener');
|
|
this._events = {};
|
|
return this;
|
|
}
|
|
|
|
listeners = this._events[type];
|
|
|
|
if (isFunction(listeners)) {
|
|
this.removeListener(type, listeners);
|
|
} else {
|
|
// LIFO order
|
|
while (listeners.length)
|
|
this.removeListener(type, listeners[listeners.length - 1]);
|
|
}
|
|
delete this._events[type];
|
|
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.listeners = function(type) {
|
|
var ret;
|
|
if (!this._events || !this._events[type])
|
|
ret = [];
|
|
else if (isFunction(this._events[type]))
|
|
ret = [this._events[type]];
|
|
else
|
|
ret = this._events[type].slice();
|
|
return ret;
|
|
};
|
|
|
|
EventEmitter.listenerCount = function(emitter, type) {
|
|
var ret;
|
|
if (!emitter._events || !emitter._events[type])
|
|
ret = 0;
|
|
else if (isFunction(emitter._events[type]))
|
|
ret = 1;
|
|
else
|
|
ret = emitter._events[type].length;
|
|
return ret;
|
|
};
|
|
|
|
function isFunction(arg) {
|
|
return typeof arg === 'function';
|
|
}
|
|
|
|
function isNumber(arg) {
|
|
return typeof arg === 'number';
|
|
}
|
|
|
|
function isObject(arg) {
|
|
return typeof arg === 'object' && arg !== null;
|
|
}
|
|
|
|
function isUndefined(arg) {
|
|
return arg === void 0;
|
|
}
|
|
|
|
},{}],90:[function(require,module,exports){
|
|
var http = module.exports;
|
|
var EventEmitter = require('events').EventEmitter;
|
|
var Request = require('./lib/request');
|
|
var url = require('url')
|
|
|
|
http.request = function (params, cb) {
|
|
if (typeof params === 'string') {
|
|
params = url.parse(params)
|
|
}
|
|
if (!params) params = {};
|
|
if (!params.host && !params.port) {
|
|
params.port = parseInt(window.location.port, 10);
|
|
}
|
|
if (!params.host && params.hostname) {
|
|
params.host = params.hostname;
|
|
}
|
|
|
|
if (!params.scheme) params.scheme = window.location.protocol.split(':')[0];
|
|
if (!params.host) {
|
|
params.host = window.location.hostname || window.location.host;
|
|
}
|
|
if (/:/.test(params.host)) {
|
|
if (!params.port) {
|
|
params.port = params.host.split(':')[1];
|
|
}
|
|
params.host = params.host.split(':')[0];
|
|
}
|
|
if (!params.port) params.port = params.scheme == 'https' ? 443 : 80;
|
|
|
|
var req = new Request(new xhrHttp, params);
|
|
if (cb) req.on('response', cb);
|
|
return req;
|
|
};
|
|
|
|
http.get = function (params, cb) {
|
|
params.method = 'GET';
|
|
var req = http.request(params, cb);
|
|
req.end();
|
|
return req;
|
|
};
|
|
|
|
http.Agent = function () {};
|
|
http.Agent.defaultMaxSockets = 4;
|
|
|
|
var xhrHttp = (function () {
|
|
if (typeof window === 'undefined') {
|
|
throw new Error('no window object present');
|
|
}
|
|
else if (window.XMLHttpRequest) {
|
|
return window.XMLHttpRequest;
|
|
}
|
|
else if (window.ActiveXObject) {
|
|
var axs = [
|
|
'Msxml2.XMLHTTP.6.0',
|
|
'Msxml2.XMLHTTP.3.0',
|
|
'Microsoft.XMLHTTP'
|
|
];
|
|
for (var i = 0; i < axs.length; i++) {
|
|
try {
|
|
var ax = new(window.ActiveXObject)(axs[i]);
|
|
return function () {
|
|
if (ax) {
|
|
var ax_ = ax;
|
|
ax = null;
|
|
return ax_;
|
|
}
|
|
else {
|
|
return new(window.ActiveXObject)(axs[i]);
|
|
}
|
|
};
|
|
}
|
|
catch (e) {}
|
|
}
|
|
throw new Error('ajax not supported in this browser')
|
|
}
|
|
else {
|
|
throw new Error('ajax not supported in this browser');
|
|
}
|
|
})();
|
|
|
|
http.STATUS_CODES = {
|
|
100 : 'Continue',
|
|
101 : 'Switching Protocols',
|
|
102 : 'Processing', // RFC 2518, obsoleted by RFC 4918
|
|
200 : 'OK',
|
|
201 : 'Created',
|
|
202 : 'Accepted',
|
|
203 : 'Non-Authoritative Information',
|
|
204 : 'No Content',
|
|
205 : 'Reset Content',
|
|
206 : 'Partial Content',
|
|
207 : 'Multi-Status', // RFC 4918
|
|
300 : 'Multiple Choices',
|
|
301 : 'Moved Permanently',
|
|
302 : 'Moved Temporarily',
|
|
303 : 'See Other',
|
|
304 : 'Not Modified',
|
|
305 : 'Use Proxy',
|
|
307 : 'Temporary Redirect',
|
|
400 : 'Bad Request',
|
|
401 : 'Unauthorized',
|
|
402 : 'Payment Required',
|
|
403 : 'Forbidden',
|
|
404 : 'Not Found',
|
|
405 : 'Method Not Allowed',
|
|
406 : 'Not Acceptable',
|
|
407 : 'Proxy Authentication Required',
|
|
408 : 'Request Time-out',
|
|
409 : 'Conflict',
|
|
410 : 'Gone',
|
|
411 : 'Length Required',
|
|
412 : 'Precondition Failed',
|
|
413 : 'Request Entity Too Large',
|
|
414 : 'Request-URI Too Large',
|
|
415 : 'Unsupported Media Type',
|
|
416 : 'Requested Range Not Satisfiable',
|
|
417 : 'Expectation Failed',
|
|
418 : 'I\'m a teapot', // RFC 2324
|
|
422 : 'Unprocessable Entity', // RFC 4918
|
|
423 : 'Locked', // RFC 4918
|
|
424 : 'Failed Dependency', // RFC 4918
|
|
425 : 'Unordered Collection', // RFC 4918
|
|
426 : 'Upgrade Required', // RFC 2817
|
|
428 : 'Precondition Required', // RFC 6585
|
|
429 : 'Too Many Requests', // RFC 6585
|
|
431 : 'Request Header Fields Too Large',// RFC 6585
|
|
500 : 'Internal Server Error',
|
|
501 : 'Not Implemented',
|
|
502 : 'Bad Gateway',
|
|
503 : 'Service Unavailable',
|
|
504 : 'Gateway Time-out',
|
|
505 : 'HTTP Version Not Supported',
|
|
506 : 'Variant Also Negotiates', // RFC 2295
|
|
507 : 'Insufficient Storage', // RFC 4918
|
|
509 : 'Bandwidth Limit Exceeded',
|
|
510 : 'Not Extended', // RFC 2774
|
|
511 : 'Network Authentication Required' // RFC 6585
|
|
};
|
|
},{"./lib/request":91,"events":89,"url":110}],91:[function(require,module,exports){
|
|
var Stream = require('stream');
|
|
var Response = require('./response');
|
|
var Base64 = require('Base64');
|
|
var inherits = require('inherits');
|
|
|
|
var Request = module.exports = function (xhr, params) {
|
|
var self = this;
|
|
self.writable = true;
|
|
self.xhr = xhr;
|
|
self.body = [];
|
|
|
|
self.uri = (params.scheme || 'http') + '://'
|
|
+ params.host
|
|
+ (params.port ? ':' + params.port : '')
|
|
+ (params.path || '/')
|
|
;
|
|
|
|
if (typeof params.withCredentials === 'undefined') {
|
|
params.withCredentials = true;
|
|
}
|
|
|
|
try { xhr.withCredentials = params.withCredentials }
|
|
catch (e) {}
|
|
|
|
xhr.open(
|
|
params.method || 'GET',
|
|
self.uri,
|
|
true
|
|
);
|
|
|
|
self._headers = {};
|
|
|
|
if (params.headers) {
|
|
var keys = objectKeys(params.headers);
|
|
for (var i = 0; i < keys.length; i++) {
|
|
var key = keys[i];
|
|
if (!self.isSafeRequestHeader(key)) continue;
|
|
var value = params.headers[key];
|
|
self.setHeader(key, value);
|
|
}
|
|
}
|
|
|
|
if (params.auth) {
|
|
//basic auth
|
|
this.setHeader('Authorization', 'Basic ' + Base64.btoa(params.auth));
|
|
}
|
|
|
|
var res = new Response;
|
|
res.on('close', function () {
|
|
self.emit('close');
|
|
});
|
|
|
|
res.on('ready', function () {
|
|
self.emit('response', res);
|
|
});
|
|
|
|
xhr.onreadystatechange = function () {
|
|
// Fix for IE9 bug
|
|
// SCRIPT575: Could not complete the operation due to error c00c023f
|
|
// It happens when a request is aborted, calling the success callback anyway with readyState === 4
|
|
if (xhr.__aborted) return;
|
|
res.handle(xhr);
|
|
};
|
|
};
|
|
|
|
inherits(Request, Stream);
|
|
|
|
Request.prototype.setHeader = function (key, value) {
|
|
this._headers[key.toLowerCase()] = value
|
|
};
|
|
|
|
Request.prototype.getHeader = function (key) {
|
|
return this._headers[key.toLowerCase()]
|
|
};
|
|
|
|
Request.prototype.removeHeader = function (key) {
|
|
delete this._headers[key.toLowerCase()]
|
|
};
|
|
|
|
Request.prototype.write = function (s) {
|
|
this.body.push(s);
|
|
};
|
|
|
|
Request.prototype.destroy = function (s) {
|
|
this.xhr.__aborted = true;
|
|
this.xhr.abort();
|
|
this.emit('close');
|
|
};
|
|
|
|
Request.prototype.end = function (s) {
|
|
if (s !== undefined) this.body.push(s);
|
|
|
|
var keys = objectKeys(this._headers);
|
|
for (var i = 0; i < keys.length; i++) {
|
|
var key = keys[i];
|
|
var value = this._headers[key];
|
|
if (isArray(value)) {
|
|
for (var j = 0; j < value.length; j++) {
|
|
this.xhr.setRequestHeader(key, value[j]);
|
|
}
|
|
}
|
|
else this.xhr.setRequestHeader(key, value)
|
|
}
|
|
|
|
if (this.body.length === 0) {
|
|
this.xhr.send('');
|
|
}
|
|
else if (typeof this.body[0] === 'string') {
|
|
this.xhr.send(this.body.join(''));
|
|
}
|
|
else if (isArray(this.body[0])) {
|
|
var body = [];
|
|
for (var i = 0; i < this.body.length; i++) {
|
|
body.push.apply(body, this.body[i]);
|
|
}
|
|
this.xhr.send(body);
|
|
}
|
|
else if (/Array/.test(Object.prototype.toString.call(this.body[0]))) {
|
|
var len = 0;
|
|
for (var i = 0; i < this.body.length; i++) {
|
|
len += this.body[i].length;
|
|
}
|
|
var body = new(this.body[0].constructor)(len);
|
|
var k = 0;
|
|
|
|
for (var i = 0; i < this.body.length; i++) {
|
|
var b = this.body[i];
|
|
for (var j = 0; j < b.length; j++) {
|
|
body[k++] = b[j];
|
|
}
|
|
}
|
|
this.xhr.send(body);
|
|
}
|
|
else {
|
|
var body = '';
|
|
for (var i = 0; i < this.body.length; i++) {
|
|
body += this.body[i].toString();
|
|
}
|
|
this.xhr.send(body);
|
|
}
|
|
};
|
|
|
|
// Taken from http://dxr.mozilla.org/mozilla/mozilla-central/content/base/src/nsXMLHttpRequest.cpp.html
|
|
Request.unsafeHeaders = [
|
|
"accept-charset",
|
|
"accept-encoding",
|
|
"access-control-request-headers",
|
|
"access-control-request-method",
|
|
"connection",
|
|
"content-length",
|
|
"cookie",
|
|
"cookie2",
|
|
"content-transfer-encoding",
|
|
"date",
|
|
"expect",
|
|
"host",
|
|
"keep-alive",
|
|
"origin",
|
|
"referer",
|
|
"te",
|
|
"trailer",
|
|
"transfer-encoding",
|
|
"upgrade",
|
|
"user-agent",
|
|
"via"
|
|
];
|
|
|
|
Request.prototype.isSafeRequestHeader = function (headerName) {
|
|
if (!headerName) return false;
|
|
return indexOf(Request.unsafeHeaders, headerName.toLowerCase()) === -1;
|
|
};
|
|
|
|
var objectKeys = Object.keys || function (obj) {
|
|
var keys = [];
|
|
for (var key in obj) keys.push(key);
|
|
return keys;
|
|
};
|
|
|
|
var isArray = Array.isArray || function (xs) {
|
|
return Object.prototype.toString.call(xs) === '[object Array]';
|
|
};
|
|
|
|
var indexOf = function (xs, x) {
|
|
if (xs.indexOf) return xs.indexOf(x);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (xs[i] === x) return i;
|
|
}
|
|
return -1;
|
|
};
|
|
|
|
},{"./response":92,"Base64":93,"inherits":95,"stream":103}],92:[function(require,module,exports){
|
|
var Stream = require('stream');
|
|
var util = require('util');
|
|
|
|
var Response = module.exports = function (res) {
|
|
this.offset = 0;
|
|
this.readable = true;
|
|
};
|
|
|
|
util.inherits(Response, Stream);
|
|
|
|
var capable = {
|
|
streaming : true,
|
|
status2 : true
|
|
};
|
|
|
|
function parseHeaders (res) {
|
|
var lines = res.getAllResponseHeaders().split(/\r?\n/);
|
|
var headers = {};
|
|
for (var i = 0; i < lines.length; i++) {
|
|
var line = lines[i];
|
|
if (line === '') continue;
|
|
|
|
var m = line.match(/^([^:]+):\s*(.*)/);
|
|
if (m) {
|
|
var key = m[1].toLowerCase(), value = m[2];
|
|
|
|
if (headers[key] !== undefined) {
|
|
|
|
if (isArray(headers[key])) {
|
|
headers[key].push(value);
|
|
}
|
|
else {
|
|
headers[key] = [ headers[key], value ];
|
|
}
|
|
}
|
|
else {
|
|
headers[key] = value;
|
|
}
|
|
}
|
|
else {
|
|
headers[line] = true;
|
|
}
|
|
}
|
|
return headers;
|
|
}
|
|
|
|
Response.prototype.getResponse = function (xhr) {
|
|
var respType = String(xhr.responseType).toLowerCase();
|
|
if (respType === 'blob') return xhr.responseBlob || xhr.response;
|
|
if (respType === 'arraybuffer') return xhr.response;
|
|
return xhr.responseText;
|
|
}
|
|
|
|
Response.prototype.getHeader = function (key) {
|
|
return this.headers[key.toLowerCase()];
|
|
};
|
|
|
|
Response.prototype.handle = function (res) {
|
|
if (res.readyState === 2 && capable.status2) {
|
|
try {
|
|
this.statusCode = res.status;
|
|
this.headers = parseHeaders(res);
|
|
}
|
|
catch (err) {
|
|
capable.status2 = false;
|
|
}
|
|
|
|
if (capable.status2) {
|
|
this.emit('ready');
|
|
}
|
|
}
|
|
else if (capable.streaming && res.readyState === 3) {
|
|
try {
|
|
if (!this.statusCode) {
|
|
this.statusCode = res.status;
|
|
this.headers = parseHeaders(res);
|
|
this.emit('ready');
|
|
}
|
|
}
|
|
catch (err) {}
|
|
|
|
try {
|
|
this._emitData(res);
|
|
}
|
|
catch (err) {
|
|
capable.streaming = false;
|
|
}
|
|
}
|
|
else if (res.readyState === 4) {
|
|
if (!this.statusCode) {
|
|
this.statusCode = res.status;
|
|
this.emit('ready');
|
|
}
|
|
this._emitData(res);
|
|
|
|
if (res.error) {
|
|
this.emit('error', this.getResponse(res));
|
|
}
|
|
else this.emit('end');
|
|
|
|
this.emit('close');
|
|
}
|
|
};
|
|
|
|
Response.prototype._emitData = function (res) {
|
|
var respBody = this.getResponse(res);
|
|
if (respBody.toString().match(/ArrayBuffer/)) {
|
|
this.emit('data', new Uint8Array(respBody, this.offset));
|
|
this.offset = respBody.byteLength;
|
|
return;
|
|
}
|
|
if (respBody.length > this.offset) {
|
|
this.emit('data', respBody.slice(this.offset));
|
|
this.offset = respBody.length;
|
|
}
|
|
};
|
|
|
|
var isArray = Array.isArray || function (xs) {
|
|
return Object.prototype.toString.call(xs) === '[object Array]';
|
|
};
|
|
|
|
},{"stream":103,"util":112}],93:[function(require,module,exports){
|
|
;(function () {
|
|
|
|
var object = typeof exports != 'undefined' ? exports : this; // #8: web workers
|
|
var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
|
|
|
|
function InvalidCharacterError(message) {
|
|
this.message = message;
|
|
}
|
|
InvalidCharacterError.prototype = new Error;
|
|
InvalidCharacterError.prototype.name = 'InvalidCharacterError';
|
|
|
|
// encoder
|
|
// [https://gist.github.com/999166] by [https://github.com/nignag]
|
|
object.btoa || (
|
|
object.btoa = function (input) {
|
|
for (
|
|
// initialize result and counter
|
|
var block, charCode, idx = 0, map = chars, output = '';
|
|
// if the next input index does not exist:
|
|
// change the mapping table to "="
|
|
// check if d has no fractional digits
|
|
input.charAt(idx | 0) || (map = '=', idx % 1);
|
|
// "8 - idx % 1 * 8" generates the sequence 2, 4, 6, 8
|
|
output += map.charAt(63 & block >> 8 - idx % 1 * 8)
|
|
) {
|
|
charCode = input.charCodeAt(idx += 3/4);
|
|
if (charCode > 0xFF) {
|
|
throw new InvalidCharacterError("'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.");
|
|
}
|
|
block = block << 8 | charCode;
|
|
}
|
|
return output;
|
|
});
|
|
|
|
// decoder
|
|
// [https://gist.github.com/1020396] by [https://github.com/atk]
|
|
object.atob || (
|
|
object.atob = function (input) {
|
|
input = input.replace(/=+$/, '');
|
|
if (input.length % 4 == 1) {
|
|
throw new InvalidCharacterError("'atob' failed: The string to be decoded is not correctly encoded.");
|
|
}
|
|
for (
|
|
// initialize result and counters
|
|
var bc = 0, bs, buffer, idx = 0, output = '';
|
|
// get next character
|
|
buffer = input.charAt(idx++);
|
|
// character found in table? initialize bit storage and add its ascii value;
|
|
~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,
|
|
// and if not first of each 4 characters,
|
|
// convert the first 8 bits to one ascii character
|
|
bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0
|
|
) {
|
|
// try to find character in table (0-63, not found => -1)
|
|
buffer = chars.indexOf(buffer);
|
|
}
|
|
return output;
|
|
});
|
|
|
|
}());
|
|
|
|
},{}],94:[function(require,module,exports){
|
|
var http = require('http');
|
|
|
|
var https = module.exports;
|
|
|
|
for (var key in http) {
|
|
if (http.hasOwnProperty(key)) https[key] = http[key];
|
|
};
|
|
|
|
https.request = function (params, cb) {
|
|
if (!params) params = {};
|
|
params.scheme = 'https';
|
|
return http.request.call(this, params, cb);
|
|
}
|
|
|
|
},{"http":90}],95:[function(require,module,exports){
|
|
if (typeof Object.create === 'function') {
|
|
// implementation from standard node.js 'util' module
|
|
module.exports = function inherits(ctor, superCtor) {
|
|
ctor.super_ = superCtor
|
|
ctor.prototype = Object.create(superCtor.prototype, {
|
|
constructor: {
|
|
value: ctor,
|
|
enumerable: false,
|
|
writable: true,
|
|
configurable: true
|
|
}
|
|
});
|
|
};
|
|
} else {
|
|
// old school shim for old browsers
|
|
module.exports = function inherits(ctor, superCtor) {
|
|
ctor.super_ = superCtor
|
|
var TempCtor = function () {}
|
|
TempCtor.prototype = superCtor.prototype
|
|
ctor.prototype = new TempCtor()
|
|
ctor.prototype.constructor = ctor
|
|
}
|
|
}
|
|
|
|
},{}],96:[function(require,module,exports){
|
|
// shim for using process in browser
|
|
|
|
var process = module.exports = {};
|
|
|
|
process.nextTick = (function () {
|
|
var canSetImmediate = typeof window !== 'undefined'
|
|
&& window.setImmediate;
|
|
var canPost = typeof window !== 'undefined'
|
|
&& window.postMessage && window.addEventListener
|
|
;
|
|
|
|
if (canSetImmediate) {
|
|
return function (f) { return window.setImmediate(f) };
|
|
}
|
|
|
|
if (canPost) {
|
|
var queue = [];
|
|
window.addEventListener('message', function (ev) {
|
|
var source = ev.source;
|
|
if ((source === window || source === null) && ev.data === 'process-tick') {
|
|
ev.stopPropagation();
|
|
if (queue.length > 0) {
|
|
var fn = queue.shift();
|
|
fn();
|
|
}
|
|
}
|
|
}, true);
|
|
|
|
return function nextTick(fn) {
|
|
queue.push(fn);
|
|
window.postMessage('process-tick', '*');
|
|
};
|
|
}
|
|
|
|
return function nextTick(fn) {
|
|
setTimeout(fn, 0);
|
|
};
|
|
})();
|
|
|
|
process.title = 'browser';
|
|
process.browser = true;
|
|
process.env = {};
|
|
process.argv = [];
|
|
|
|
function noop() {}
|
|
|
|
process.on = noop;
|
|
process.once = noop;
|
|
process.off = noop;
|
|
process.emit = noop;
|
|
|
|
process.binding = function (name) {
|
|
throw new Error('process.binding is not supported');
|
|
}
|
|
|
|
// TODO(shtylman)
|
|
process.cwd = function () { return '/' };
|
|
process.chdir = function (dir) {
|
|
throw new Error('process.chdir is not supported');
|
|
};
|
|
|
|
},{}],97:[function(require,module,exports){
|
|
(function (process){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// resolves . and .. elements in a path array with directory names there
|
|
// must be no slashes, empty elements, or device names (c:\) in the array
|
|
// (so also no leading and trailing slashes - it does not distinguish
|
|
// relative and absolute paths)
|
|
function normalizeArray(parts, allowAboveRoot) {
|
|
// if the path tries to go above the root, `up` ends up > 0
|
|
var up = 0;
|
|
for (var i = parts.length - 1; i >= 0; i--) {
|
|
var last = parts[i];
|
|
if (last === '.') {
|
|
parts.splice(i, 1);
|
|
} else if (last === '..') {
|
|
parts.splice(i, 1);
|
|
up++;
|
|
} else if (up) {
|
|
parts.splice(i, 1);
|
|
up--;
|
|
}
|
|
}
|
|
|
|
// if the path is allowed to go above the root, restore leading ..s
|
|
if (allowAboveRoot) {
|
|
for (; up--; up) {
|
|
parts.unshift('..');
|
|
}
|
|
}
|
|
|
|
return parts;
|
|
}
|
|
|
|
// Split a filename into [root, dir, basename, ext], unix version
|
|
// 'root' is just a slash, or nothing.
|
|
var splitPathRe =
|
|
/^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/;
|
|
var splitPath = function(filename) {
|
|
return splitPathRe.exec(filename).slice(1);
|
|
};
|
|
|
|
// path.resolve([from ...], to)
|
|
// posix version
|
|
exports.resolve = function() {
|
|
var resolvedPath = '',
|
|
resolvedAbsolute = false;
|
|
|
|
for (var i = arguments.length - 1; i >= -1 && !resolvedAbsolute; i--) {
|
|
var path = (i >= 0) ? arguments[i] : process.cwd();
|
|
|
|
// Skip empty and invalid entries
|
|
if (typeof path !== 'string') {
|
|
throw new TypeError('Arguments to path.resolve must be strings');
|
|
} else if (!path) {
|
|
continue;
|
|
}
|
|
|
|
resolvedPath = path + '/' + resolvedPath;
|
|
resolvedAbsolute = path.charAt(0) === '/';
|
|
}
|
|
|
|
// At this point the path should be resolved to a full absolute path, but
|
|
// handle relative paths to be safe (might happen when process.cwd() fails)
|
|
|
|
// Normalize the path
|
|
resolvedPath = normalizeArray(filter(resolvedPath.split('/'), function(p) {
|
|
return !!p;
|
|
}), !resolvedAbsolute).join('/');
|
|
|
|
return ((resolvedAbsolute ? '/' : '') + resolvedPath) || '.';
|
|
};
|
|
|
|
// path.normalize(path)
|
|
// posix version
|
|
exports.normalize = function(path) {
|
|
var isAbsolute = exports.isAbsolute(path),
|
|
trailingSlash = substr(path, -1) === '/';
|
|
|
|
// Normalize the path
|
|
path = normalizeArray(filter(path.split('/'), function(p) {
|
|
return !!p;
|
|
}), !isAbsolute).join('/');
|
|
|
|
if (!path && !isAbsolute) {
|
|
path = '.';
|
|
}
|
|
if (path && trailingSlash) {
|
|
path += '/';
|
|
}
|
|
|
|
return (isAbsolute ? '/' : '') + path;
|
|
};
|
|
|
|
// posix version
|
|
exports.isAbsolute = function(path) {
|
|
return path.charAt(0) === '/';
|
|
};
|
|
|
|
// posix version
|
|
exports.join = function() {
|
|
var paths = Array.prototype.slice.call(arguments, 0);
|
|
return exports.normalize(filter(paths, function(p, index) {
|
|
if (typeof p !== 'string') {
|
|
throw new TypeError('Arguments to path.join must be strings');
|
|
}
|
|
return p;
|
|
}).join('/'));
|
|
};
|
|
|
|
|
|
// path.relative(from, to)
|
|
// posix version
|
|
exports.relative = function(from, to) {
|
|
from = exports.resolve(from).substr(1);
|
|
to = exports.resolve(to).substr(1);
|
|
|
|
function trim(arr) {
|
|
var start = 0;
|
|
for (; start < arr.length; start++) {
|
|
if (arr[start] !== '') break;
|
|
}
|
|
|
|
var end = arr.length - 1;
|
|
for (; end >= 0; end--) {
|
|
if (arr[end] !== '') break;
|
|
}
|
|
|
|
if (start > end) return [];
|
|
return arr.slice(start, end - start + 1);
|
|
}
|
|
|
|
var fromParts = trim(from.split('/'));
|
|
var toParts = trim(to.split('/'));
|
|
|
|
var length = Math.min(fromParts.length, toParts.length);
|
|
var samePartsLength = length;
|
|
for (var i = 0; i < length; i++) {
|
|
if (fromParts[i] !== toParts[i]) {
|
|
samePartsLength = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
var outputParts = [];
|
|
for (var i = samePartsLength; i < fromParts.length; i++) {
|
|
outputParts.push('..');
|
|
}
|
|
|
|
outputParts = outputParts.concat(toParts.slice(samePartsLength));
|
|
|
|
return outputParts.join('/');
|
|
};
|
|
|
|
exports.sep = '/';
|
|
exports.delimiter = ':';
|
|
|
|
exports.dirname = function(path) {
|
|
var result = splitPath(path),
|
|
root = result[0],
|
|
dir = result[1];
|
|
|
|
if (!root && !dir) {
|
|
// No dirname whatsoever
|
|
return '.';
|
|
}
|
|
|
|
if (dir) {
|
|
// It has a dirname, strip trailing slash
|
|
dir = dir.substr(0, dir.length - 1);
|
|
}
|
|
|
|
return root + dir;
|
|
};
|
|
|
|
|
|
exports.basename = function(path, ext) {
|
|
var f = splitPath(path)[2];
|
|
// TODO: make this comparison case-insensitive on windows?
|
|
if (ext && f.substr(-1 * ext.length) === ext) {
|
|
f = f.substr(0, f.length - ext.length);
|
|
}
|
|
return f;
|
|
};
|
|
|
|
|
|
exports.extname = function(path) {
|
|
return splitPath(path)[3];
|
|
};
|
|
|
|
function filter (xs, f) {
|
|
if (xs.filter) return xs.filter(f);
|
|
var res = [];
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (f(xs[i], i, xs)) res.push(xs[i]);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
// String.prototype.substr - negative index don't work in IE8
|
|
var substr = 'ab'.substr(-1) === 'b'
|
|
? function (str, start, len) { return str.substr(start, len) }
|
|
: function (str, start, len) {
|
|
if (start < 0) start = str.length + start;
|
|
return str.substr(start, len);
|
|
}
|
|
;
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"))
|
|
},{"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96}],98:[function(require,module,exports){
|
|
(function (global){
|
|
/*! http://mths.be/punycode v1.2.4 by @mathias */
|
|
;(function(root) {
|
|
|
|
/** Detect free variables */
|
|
var freeExports = typeof exports == 'object' && exports;
|
|
var freeModule = typeof module == 'object' && module &&
|
|
module.exports == freeExports && module;
|
|
var freeGlobal = typeof global == 'object' && global;
|
|
if (freeGlobal.global === freeGlobal || freeGlobal.window === freeGlobal) {
|
|
root = freeGlobal;
|
|
}
|
|
|
|
/**
|
|
* The `punycode` object.
|
|
* @name punycode
|
|
* @type Object
|
|
*/
|
|
var punycode,
|
|
|
|
/** Highest positive signed 32-bit float value */
|
|
maxInt = 2147483647, // aka. 0x7FFFFFFF or 2^31-1
|
|
|
|
/** Bootstring parameters */
|
|
base = 36,
|
|
tMin = 1,
|
|
tMax = 26,
|
|
skew = 38,
|
|
damp = 700,
|
|
initialBias = 72,
|
|
initialN = 128, // 0x80
|
|
delimiter = '-', // '\x2D'
|
|
|
|
/** Regular expressions */
|
|
regexPunycode = /^xn--/,
|
|
regexNonASCII = /[^ -~]/, // unprintable ASCII chars + non-ASCII chars
|
|
regexSeparators = /\x2E|\u3002|\uFF0E|\uFF61/g, // RFC 3490 separators
|
|
|
|
/** Error messages */
|
|
errors = {
|
|
'overflow': 'Overflow: input needs wider integers to process',
|
|
'not-basic': 'Illegal input >= 0x80 (not a basic code point)',
|
|
'invalid-input': 'Invalid input'
|
|
},
|
|
|
|
/** Convenience shortcuts */
|
|
baseMinusTMin = base - tMin,
|
|
floor = Math.floor,
|
|
stringFromCharCode = String.fromCharCode,
|
|
|
|
/** Temporary variable */
|
|
key;
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* A generic error utility function.
|
|
* @private
|
|
* @param {String} type The error type.
|
|
* @returns {Error} Throws a `RangeError` with the applicable error message.
|
|
*/
|
|
function error(type) {
|
|
throw RangeError(errors[type]);
|
|
}
|
|
|
|
/**
|
|
* A generic `Array#map` utility function.
|
|
* @private
|
|
* @param {Array} array The array to iterate over.
|
|
* @param {Function} callback The function that gets called for every array
|
|
* item.
|
|
* @returns {Array} A new array of values returned by the callback function.
|
|
*/
|
|
function map(array, fn) {
|
|
var length = array.length;
|
|
while (length--) {
|
|
array[length] = fn(array[length]);
|
|
}
|
|
return array;
|
|
}
|
|
|
|
/**
|
|
* A simple `Array#map`-like wrapper to work with domain name strings.
|
|
* @private
|
|
* @param {String} domain The domain name.
|
|
* @param {Function} callback The function that gets called for every
|
|
* character.
|
|
* @returns {Array} A new string of characters returned by the callback
|
|
* function.
|
|
*/
|
|
function mapDomain(string, fn) {
|
|
return map(string.split(regexSeparators), fn).join('.');
|
|
}
|
|
|
|
/**
|
|
* Creates an array containing the numeric code points of each Unicode
|
|
* character in the string. While JavaScript uses UCS-2 internally,
|
|
* this function will convert a pair of surrogate halves (each of which
|
|
* UCS-2 exposes as separate characters) into a single code point,
|
|
* matching UTF-16.
|
|
* @see `punycode.ucs2.encode`
|
|
* @see <http://mathiasbynens.be/notes/javascript-encoding>
|
|
* @memberOf punycode.ucs2
|
|
* @name decode
|
|
* @param {String} string The Unicode input string (UCS-2).
|
|
* @returns {Array} The new array of code points.
|
|
*/
|
|
function ucs2decode(string) {
|
|
var output = [],
|
|
counter = 0,
|
|
length = string.length,
|
|
value,
|
|
extra;
|
|
while (counter < length) {
|
|
value = string.charCodeAt(counter++);
|
|
if (value >= 0xD800 && value <= 0xDBFF && counter < length) {
|
|
// high surrogate, and there is a next character
|
|
extra = string.charCodeAt(counter++);
|
|
if ((extra & 0xFC00) == 0xDC00) { // low surrogate
|
|
output.push(((value & 0x3FF) << 10) + (extra & 0x3FF) + 0x10000);
|
|
} else {
|
|
// unmatched surrogate; only append this code unit, in case the next
|
|
// code unit is the high surrogate of a surrogate pair
|
|
output.push(value);
|
|
counter--;
|
|
}
|
|
} else {
|
|
output.push(value);
|
|
}
|
|
}
|
|
return output;
|
|
}
|
|
|
|
/**
|
|
* Creates a string based on an array of numeric code points.
|
|
* @see `punycode.ucs2.decode`
|
|
* @memberOf punycode.ucs2
|
|
* @name encode
|
|
* @param {Array} codePoints The array of numeric code points.
|
|
* @returns {String} The new Unicode string (UCS-2).
|
|
*/
|
|
function ucs2encode(array) {
|
|
return map(array, function(value) {
|
|
var output = '';
|
|
if (value > 0xFFFF) {
|
|
value -= 0x10000;
|
|
output += stringFromCharCode(value >>> 10 & 0x3FF | 0xD800);
|
|
value = 0xDC00 | value & 0x3FF;
|
|
}
|
|
output += stringFromCharCode(value);
|
|
return output;
|
|
}).join('');
|
|
}
|
|
|
|
/**
|
|
* Converts a basic code point into a digit/integer.
|
|
* @see `digitToBasic()`
|
|
* @private
|
|
* @param {Number} codePoint The basic numeric code point value.
|
|
* @returns {Number} The numeric value of a basic code point (for use in
|
|
* representing integers) in the range `0` to `base - 1`, or `base` if
|
|
* the code point does not represent a value.
|
|
*/
|
|
function basicToDigit(codePoint) {
|
|
if (codePoint - 48 < 10) {
|
|
return codePoint - 22;
|
|
}
|
|
if (codePoint - 65 < 26) {
|
|
return codePoint - 65;
|
|
}
|
|
if (codePoint - 97 < 26) {
|
|
return codePoint - 97;
|
|
}
|
|
return base;
|
|
}
|
|
|
|
/**
|
|
* Converts a digit/integer into a basic code point.
|
|
* @see `basicToDigit()`
|
|
* @private
|
|
* @param {Number} digit The numeric value of a basic code point.
|
|
* @returns {Number} The basic code point whose value (when used for
|
|
* representing integers) is `digit`, which needs to be in the range
|
|
* `0` to `base - 1`. If `flag` is non-zero, the uppercase form is
|
|
* used; else, the lowercase form is used. The behavior is undefined
|
|
* if `flag` is non-zero and `digit` has no uppercase form.
|
|
*/
|
|
function digitToBasic(digit, flag) {
|
|
// 0..25 map to ASCII a..z or A..Z
|
|
// 26..35 map to ASCII 0..9
|
|
return digit + 22 + 75 * (digit < 26) - ((flag != 0) << 5);
|
|
}
|
|
|
|
/**
|
|
* Bias adaptation function as per section 3.4 of RFC 3492.
|
|
* http://tools.ietf.org/html/rfc3492#section-3.4
|
|
* @private
|
|
*/
|
|
function adapt(delta, numPoints, firstTime) {
|
|
var k = 0;
|
|
delta = firstTime ? floor(delta / damp) : delta >> 1;
|
|
delta += floor(delta / numPoints);
|
|
for (/* no initialization */; delta > baseMinusTMin * tMax >> 1; k += base) {
|
|
delta = floor(delta / baseMinusTMin);
|
|
}
|
|
return floor(k + (baseMinusTMin + 1) * delta / (delta + skew));
|
|
}
|
|
|
|
/**
|
|
* Converts a Punycode string of ASCII-only symbols to a string of Unicode
|
|
* symbols.
|
|
* @memberOf punycode
|
|
* @param {String} input The Punycode string of ASCII-only symbols.
|
|
* @returns {String} The resulting string of Unicode symbols.
|
|
*/
|
|
function decode(input) {
|
|
// Don't use UCS-2
|
|
var output = [],
|
|
inputLength = input.length,
|
|
out,
|
|
i = 0,
|
|
n = initialN,
|
|
bias = initialBias,
|
|
basic,
|
|
j,
|
|
index,
|
|
oldi,
|
|
w,
|
|
k,
|
|
digit,
|
|
t,
|
|
/** Cached calculation results */
|
|
baseMinusT;
|
|
|
|
// Handle the basic code points: let `basic` be the number of input code
|
|
// points before the last delimiter, or `0` if there is none, then copy
|
|
// the first basic code points to the output.
|
|
|
|
basic = input.lastIndexOf(delimiter);
|
|
if (basic < 0) {
|
|
basic = 0;
|
|
}
|
|
|
|
for (j = 0; j < basic; ++j) {
|
|
// if it's not a basic code point
|
|
if (input.charCodeAt(j) >= 0x80) {
|
|
error('not-basic');
|
|
}
|
|
output.push(input.charCodeAt(j));
|
|
}
|
|
|
|
// Main decoding loop: start just after the last delimiter if any basic code
|
|
// points were copied; start at the beginning otherwise.
|
|
|
|
for (index = basic > 0 ? basic + 1 : 0; index < inputLength; /* no final expression */) {
|
|
|
|
// `index` is the index of the next character to be consumed.
|
|
// Decode a generalized variable-length integer into `delta`,
|
|
// which gets added to `i`. The overflow checking is easier
|
|
// if we increase `i` as we go, then subtract off its starting
|
|
// value at the end to obtain `delta`.
|
|
for (oldi = i, w = 1, k = base; /* no condition */; k += base) {
|
|
|
|
if (index >= inputLength) {
|
|
error('invalid-input');
|
|
}
|
|
|
|
digit = basicToDigit(input.charCodeAt(index++));
|
|
|
|
if (digit >= base || digit > floor((maxInt - i) / w)) {
|
|
error('overflow');
|
|
}
|
|
|
|
i += digit * w;
|
|
t = k <= bias ? tMin : (k >= bias + tMax ? tMax : k - bias);
|
|
|
|
if (digit < t) {
|
|
break;
|
|
}
|
|
|
|
baseMinusT = base - t;
|
|
if (w > floor(maxInt / baseMinusT)) {
|
|
error('overflow');
|
|
}
|
|
|
|
w *= baseMinusT;
|
|
|
|
}
|
|
|
|
out = output.length + 1;
|
|
bias = adapt(i - oldi, out, oldi == 0);
|
|
|
|
// `i` was supposed to wrap around from `out` to `0`,
|
|
// incrementing `n` each time, so we'll fix that now:
|
|
if (floor(i / out) > maxInt - n) {
|
|
error('overflow');
|
|
}
|
|
|
|
n += floor(i / out);
|
|
i %= out;
|
|
|
|
// Insert `n` at position `i` of the output
|
|
output.splice(i++, 0, n);
|
|
|
|
}
|
|
|
|
return ucs2encode(output);
|
|
}
|
|
|
|
/**
|
|
* Converts a string of Unicode symbols to a Punycode string of ASCII-only
|
|
* symbols.
|
|
* @memberOf punycode
|
|
* @param {String} input The string of Unicode symbols.
|
|
* @returns {String} The resulting Punycode string of ASCII-only symbols.
|
|
*/
|
|
function encode(input) {
|
|
var n,
|
|
delta,
|
|
handledCPCount,
|
|
basicLength,
|
|
bias,
|
|
j,
|
|
m,
|
|
q,
|
|
k,
|
|
t,
|
|
currentValue,
|
|
output = [],
|
|
/** `inputLength` will hold the number of code points in `input`. */
|
|
inputLength,
|
|
/** Cached calculation results */
|
|
handledCPCountPlusOne,
|
|
baseMinusT,
|
|
qMinusT;
|
|
|
|
// Convert the input in UCS-2 to Unicode
|
|
input = ucs2decode(input);
|
|
|
|
// Cache the length
|
|
inputLength = input.length;
|
|
|
|
// Initialize the state
|
|
n = initialN;
|
|
delta = 0;
|
|
bias = initialBias;
|
|
|
|
// Handle the basic code points
|
|
for (j = 0; j < inputLength; ++j) {
|
|
currentValue = input[j];
|
|
if (currentValue < 0x80) {
|
|
output.push(stringFromCharCode(currentValue));
|
|
}
|
|
}
|
|
|
|
handledCPCount = basicLength = output.length;
|
|
|
|
// `handledCPCount` is the number of code points that have been handled;
|
|
// `basicLength` is the number of basic code points.
|
|
|
|
// Finish the basic string - if it is not empty - with a delimiter
|
|
if (basicLength) {
|
|
output.push(delimiter);
|
|
}
|
|
|
|
// Main encoding loop:
|
|
while (handledCPCount < inputLength) {
|
|
|
|
// All non-basic code points < n have been handled already. Find the next
|
|
// larger one:
|
|
for (m = maxInt, j = 0; j < inputLength; ++j) {
|
|
currentValue = input[j];
|
|
if (currentValue >= n && currentValue < m) {
|
|
m = currentValue;
|
|
}
|
|
}
|
|
|
|
// Increase `delta` enough to advance the decoder's <n,i> state to <m,0>,
|
|
// but guard against overflow
|
|
handledCPCountPlusOne = handledCPCount + 1;
|
|
if (m - n > floor((maxInt - delta) / handledCPCountPlusOne)) {
|
|
error('overflow');
|
|
}
|
|
|
|
delta += (m - n) * handledCPCountPlusOne;
|
|
n = m;
|
|
|
|
for (j = 0; j < inputLength; ++j) {
|
|
currentValue = input[j];
|
|
|
|
if (currentValue < n && ++delta > maxInt) {
|
|
error('overflow');
|
|
}
|
|
|
|
if (currentValue == n) {
|
|
// Represent delta as a generalized variable-length integer
|
|
for (q = delta, k = base; /* no condition */; k += base) {
|
|
t = k <= bias ? tMin : (k >= bias + tMax ? tMax : k - bias);
|
|
if (q < t) {
|
|
break;
|
|
}
|
|
qMinusT = q - t;
|
|
baseMinusT = base - t;
|
|
output.push(
|
|
stringFromCharCode(digitToBasic(t + qMinusT % baseMinusT, 0))
|
|
);
|
|
q = floor(qMinusT / baseMinusT);
|
|
}
|
|
|
|
output.push(stringFromCharCode(digitToBasic(q, 0)));
|
|
bias = adapt(delta, handledCPCountPlusOne, handledCPCount == basicLength);
|
|
delta = 0;
|
|
++handledCPCount;
|
|
}
|
|
}
|
|
|
|
++delta;
|
|
++n;
|
|
|
|
}
|
|
return output.join('');
|
|
}
|
|
|
|
/**
|
|
* Converts a Punycode string representing a domain name to Unicode. Only the
|
|
* Punycoded parts of the domain name will be converted, i.e. it doesn't
|
|
* matter if you call it on a string that has already been converted to
|
|
* Unicode.
|
|
* @memberOf punycode
|
|
* @param {String} domain The Punycode domain name to convert to Unicode.
|
|
* @returns {String} The Unicode representation of the given Punycode
|
|
* string.
|
|
*/
|
|
function toUnicode(domain) {
|
|
return mapDomain(domain, function(string) {
|
|
return regexPunycode.test(string)
|
|
? decode(string.slice(4).toLowerCase())
|
|
: string;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Converts a Unicode string representing a domain name to Punycode. Only the
|
|
* non-ASCII parts of the domain name will be converted, i.e. it doesn't
|
|
* matter if you call it with a domain that's already in ASCII.
|
|
* @memberOf punycode
|
|
* @param {String} domain The domain name to convert, as a Unicode string.
|
|
* @returns {String} The Punycode representation of the given domain name.
|
|
*/
|
|
function toASCII(domain) {
|
|
return mapDomain(domain, function(string) {
|
|
return regexNonASCII.test(string)
|
|
? 'xn--' + encode(string)
|
|
: string;
|
|
});
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/** Define the public API */
|
|
punycode = {
|
|
/**
|
|
* A string representing the current Punycode.js version number.
|
|
* @memberOf punycode
|
|
* @type String
|
|
*/
|
|
'version': '1.2.4',
|
|
/**
|
|
* An object of methods to convert from JavaScript's internal character
|
|
* representation (UCS-2) to Unicode code points, and back.
|
|
* @see <http://mathiasbynens.be/notes/javascript-encoding>
|
|
* @memberOf punycode
|
|
* @type Object
|
|
*/
|
|
'ucs2': {
|
|
'decode': ucs2decode,
|
|
'encode': ucs2encode
|
|
},
|
|
'decode': decode,
|
|
'encode': encode,
|
|
'toASCII': toASCII,
|
|
'toUnicode': toUnicode
|
|
};
|
|
|
|
/** Expose `punycode` */
|
|
// Some AMD build optimizers, like r.js, check for specific condition patterns
|
|
// like the following:
|
|
if (
|
|
typeof define == 'function' &&
|
|
typeof define.amd == 'object' &&
|
|
define.amd
|
|
) {
|
|
define('punycode', function() {
|
|
return punycode;
|
|
});
|
|
} else if (freeExports && !freeExports.nodeType) {
|
|
if (freeModule) { // in Node.js or RingoJS v0.8.0+
|
|
freeModule.exports = punycode;
|
|
} else { // in Narwhal or RingoJS v0.7.0-
|
|
for (key in punycode) {
|
|
punycode.hasOwnProperty(key) && (freeExports[key] = punycode[key]);
|
|
}
|
|
}
|
|
} else { // in Rhino or a web browser
|
|
root.punycode = punycode;
|
|
}
|
|
|
|
}(this));
|
|
|
|
}).call(this,typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
|
|
},{}],99:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
'use strict';
|
|
|
|
// If obj.hasOwnProperty has been overridden, then calling
|
|
// obj.hasOwnProperty(prop) will break.
|
|
// See: https://github.com/joyent/node/issues/1707
|
|
function hasOwnProperty(obj, prop) {
|
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
|
}
|
|
|
|
module.exports = function(qs, sep, eq, options) {
|
|
sep = sep || '&';
|
|
eq = eq || '=';
|
|
var obj = {};
|
|
|
|
if (typeof qs !== 'string' || qs.length === 0) {
|
|
return obj;
|
|
}
|
|
|
|
var regexp = /\+/g;
|
|
qs = qs.split(sep);
|
|
|
|
var maxKeys = 1000;
|
|
if (options && typeof options.maxKeys === 'number') {
|
|
maxKeys = options.maxKeys;
|
|
}
|
|
|
|
var len = qs.length;
|
|
// maxKeys <= 0 means that we should not limit keys count
|
|
if (maxKeys > 0 && len > maxKeys) {
|
|
len = maxKeys;
|
|
}
|
|
|
|
for (var i = 0; i < len; ++i) {
|
|
var x = qs[i].replace(regexp, '%20'),
|
|
idx = x.indexOf(eq),
|
|
kstr, vstr, k, v;
|
|
|
|
if (idx >= 0) {
|
|
kstr = x.substr(0, idx);
|
|
vstr = x.substr(idx + 1);
|
|
} else {
|
|
kstr = x;
|
|
vstr = '';
|
|
}
|
|
|
|
k = decodeURIComponent(kstr);
|
|
v = decodeURIComponent(vstr);
|
|
|
|
if (!hasOwnProperty(obj, k)) {
|
|
obj[k] = v;
|
|
} else if (isArray(obj[k])) {
|
|
obj[k].push(v);
|
|
} else {
|
|
obj[k] = [obj[k], v];
|
|
}
|
|
}
|
|
|
|
return obj;
|
|
};
|
|
|
|
var isArray = Array.isArray || function (xs) {
|
|
return Object.prototype.toString.call(xs) === '[object Array]';
|
|
};
|
|
|
|
},{}],100:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
'use strict';
|
|
|
|
var stringifyPrimitive = function(v) {
|
|
switch (typeof v) {
|
|
case 'string':
|
|
return v;
|
|
|
|
case 'boolean':
|
|
return v ? 'true' : 'false';
|
|
|
|
case 'number':
|
|
return isFinite(v) ? v : '';
|
|
|
|
default:
|
|
return '';
|
|
}
|
|
};
|
|
|
|
module.exports = function(obj, sep, eq, name) {
|
|
sep = sep || '&';
|
|
eq = eq || '=';
|
|
if (obj === null) {
|
|
obj = undefined;
|
|
}
|
|
|
|
if (typeof obj === 'object') {
|
|
return map(objectKeys(obj), function(k) {
|
|
var ks = encodeURIComponent(stringifyPrimitive(k)) + eq;
|
|
if (isArray(obj[k])) {
|
|
return obj[k].map(function(v) {
|
|
return ks + encodeURIComponent(stringifyPrimitive(v));
|
|
}).join(sep);
|
|
} else {
|
|
return ks + encodeURIComponent(stringifyPrimitive(obj[k]));
|
|
}
|
|
}).join(sep);
|
|
|
|
}
|
|
|
|
if (!name) return '';
|
|
return encodeURIComponent(stringifyPrimitive(name)) + eq +
|
|
encodeURIComponent(stringifyPrimitive(obj));
|
|
};
|
|
|
|
var isArray = Array.isArray || function (xs) {
|
|
return Object.prototype.toString.call(xs) === '[object Array]';
|
|
};
|
|
|
|
function map (xs, f) {
|
|
if (xs.map) return xs.map(f);
|
|
var res = [];
|
|
for (var i = 0; i < xs.length; i++) {
|
|
res.push(f(xs[i], i));
|
|
}
|
|
return res;
|
|
}
|
|
|
|
var objectKeys = Object.keys || function (obj) {
|
|
var res = [];
|
|
for (var key in obj) {
|
|
if (Object.prototype.hasOwnProperty.call(obj, key)) res.push(key);
|
|
}
|
|
return res;
|
|
};
|
|
|
|
},{}],101:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
exports.decode = exports.parse = require('./decode');
|
|
exports.encode = exports.stringify = require('./encode');
|
|
|
|
},{"./decode":99,"./encode":100}],102:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// a duplex stream is just a stream that is both readable and writable.
|
|
// Since JS doesn't have multiple prototypal inheritance, this class
|
|
// prototypally inherits from Readable, and then parasitically from
|
|
// Writable.
|
|
|
|
module.exports = Duplex;
|
|
var inherits = require('inherits');
|
|
var setImmediate = require('process/browser.js').nextTick;
|
|
var Readable = require('./readable.js');
|
|
var Writable = require('./writable.js');
|
|
|
|
inherits(Duplex, Readable);
|
|
|
|
Duplex.prototype.write = Writable.prototype.write;
|
|
Duplex.prototype.end = Writable.prototype.end;
|
|
Duplex.prototype._write = Writable.prototype._write;
|
|
|
|
function Duplex(options) {
|
|
if (!(this instanceof Duplex))
|
|
return new Duplex(options);
|
|
|
|
Readable.call(this, options);
|
|
Writable.call(this, options);
|
|
|
|
if (options && options.readable === false)
|
|
this.readable = false;
|
|
|
|
if (options && options.writable === false)
|
|
this.writable = false;
|
|
|
|
this.allowHalfOpen = true;
|
|
if (options && options.allowHalfOpen === false)
|
|
this.allowHalfOpen = false;
|
|
|
|
this.once('end', onend);
|
|
}
|
|
|
|
// the no-half-open enforcer
|
|
function onend() {
|
|
// if we allow half-open state, or if the writable side ended,
|
|
// then we're ok.
|
|
if (this.allowHalfOpen || this._writableState.ended)
|
|
return;
|
|
|
|
// no more data can be written.
|
|
// But allow more writes to happen in this tick.
|
|
var self = this;
|
|
setImmediate(function () {
|
|
self.end();
|
|
});
|
|
}
|
|
|
|
},{"./readable.js":106,"./writable.js":108,"inherits":95,"process/browser.js":104}],103:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
module.exports = Stream;
|
|
|
|
var EE = require('events').EventEmitter;
|
|
var inherits = require('inherits');
|
|
|
|
inherits(Stream, EE);
|
|
Stream.Readable = require('./readable.js');
|
|
Stream.Writable = require('./writable.js');
|
|
Stream.Duplex = require('./duplex.js');
|
|
Stream.Transform = require('./transform.js');
|
|
Stream.PassThrough = require('./passthrough.js');
|
|
|
|
// Backwards-compat with node 0.4.x
|
|
Stream.Stream = Stream;
|
|
|
|
|
|
|
|
// old-style streams. Note that the pipe method (the only relevant
|
|
// part of this class) is overridden in the Readable class.
|
|
|
|
function Stream() {
|
|
EE.call(this);
|
|
}
|
|
|
|
Stream.prototype.pipe = function(dest, options) {
|
|
var source = this;
|
|
|
|
function ondata(chunk) {
|
|
if (dest.writable) {
|
|
if (false === dest.write(chunk) && source.pause) {
|
|
source.pause();
|
|
}
|
|
}
|
|
}
|
|
|
|
source.on('data', ondata);
|
|
|
|
function ondrain() {
|
|
if (source.readable && source.resume) {
|
|
source.resume();
|
|
}
|
|
}
|
|
|
|
dest.on('drain', ondrain);
|
|
|
|
// If the 'end' option is not supplied, dest.end() will be called when
|
|
// source gets the 'end' or 'close' events. Only dest.end() once.
|
|
if (!dest._isStdio && (!options || options.end !== false)) {
|
|
source.on('end', onend);
|
|
source.on('close', onclose);
|
|
}
|
|
|
|
var didOnEnd = false;
|
|
function onend() {
|
|
if (didOnEnd) return;
|
|
didOnEnd = true;
|
|
|
|
dest.end();
|
|
}
|
|
|
|
|
|
function onclose() {
|
|
if (didOnEnd) return;
|
|
didOnEnd = true;
|
|
|
|
if (typeof dest.destroy === 'function') dest.destroy();
|
|
}
|
|
|
|
// don't leave dangling pipes when there are errors.
|
|
function onerror(er) {
|
|
cleanup();
|
|
if (EE.listenerCount(this, 'error') === 0) {
|
|
throw er; // Unhandled stream error in pipe.
|
|
}
|
|
}
|
|
|
|
source.on('error', onerror);
|
|
dest.on('error', onerror);
|
|
|
|
// remove all the event listeners that were added.
|
|
function cleanup() {
|
|
source.removeListener('data', ondata);
|
|
dest.removeListener('drain', ondrain);
|
|
|
|
source.removeListener('end', onend);
|
|
source.removeListener('close', onclose);
|
|
|
|
source.removeListener('error', onerror);
|
|
dest.removeListener('error', onerror);
|
|
|
|
source.removeListener('end', cleanup);
|
|
source.removeListener('close', cleanup);
|
|
|
|
dest.removeListener('close', cleanup);
|
|
}
|
|
|
|
source.on('end', cleanup);
|
|
source.on('close', cleanup);
|
|
|
|
dest.on('close', cleanup);
|
|
|
|
dest.emit('pipe', source);
|
|
|
|
// Allow for unix-like usage: A.pipe(B).pipe(C)
|
|
return dest;
|
|
};
|
|
|
|
},{"./duplex.js":102,"./passthrough.js":105,"./readable.js":106,"./transform.js":107,"./writable.js":108,"events":89,"inherits":95}],104:[function(require,module,exports){
|
|
// shim for using process in browser
|
|
|
|
var process = module.exports = {};
|
|
|
|
process.nextTick = (function () {
|
|
var canSetImmediate = typeof window !== 'undefined'
|
|
&& window.setImmediate;
|
|
var canPost = typeof window !== 'undefined'
|
|
&& window.postMessage && window.addEventListener
|
|
;
|
|
|
|
if (canSetImmediate) {
|
|
return function (f) { return window.setImmediate(f) };
|
|
}
|
|
|
|
if (canPost) {
|
|
var queue = [];
|
|
window.addEventListener('message', function (ev) {
|
|
var source = ev.source;
|
|
if ((source === window || source === null) && ev.data === 'process-tick') {
|
|
ev.stopPropagation();
|
|
if (queue.length > 0) {
|
|
var fn = queue.shift();
|
|
fn();
|
|
}
|
|
}
|
|
}, true);
|
|
|
|
return function nextTick(fn) {
|
|
queue.push(fn);
|
|
window.postMessage('process-tick', '*');
|
|
};
|
|
}
|
|
|
|
return function nextTick(fn) {
|
|
setTimeout(fn, 0);
|
|
};
|
|
})();
|
|
|
|
process.title = 'browser';
|
|
process.browser = true;
|
|
process.env = {};
|
|
process.argv = [];
|
|
|
|
process.binding = function (name) {
|
|
throw new Error('process.binding is not supported');
|
|
}
|
|
|
|
// TODO(shtylman)
|
|
process.cwd = function () { return '/' };
|
|
process.chdir = function (dir) {
|
|
throw new Error('process.chdir is not supported');
|
|
};
|
|
|
|
},{}],105:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// a passthrough stream.
|
|
// basically just the most minimal sort of Transform stream.
|
|
// Every written chunk gets output as-is.
|
|
|
|
module.exports = PassThrough;
|
|
|
|
var Transform = require('./transform.js');
|
|
var inherits = require('inherits');
|
|
inherits(PassThrough, Transform);
|
|
|
|
function PassThrough(options) {
|
|
if (!(this instanceof PassThrough))
|
|
return new PassThrough(options);
|
|
|
|
Transform.call(this, options);
|
|
}
|
|
|
|
PassThrough.prototype._transform = function(chunk, encoding, cb) {
|
|
cb(null, chunk);
|
|
};
|
|
|
|
},{"./transform.js":107,"inherits":95}],106:[function(require,module,exports){
|
|
(function (process){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
module.exports = Readable;
|
|
Readable.ReadableState = ReadableState;
|
|
|
|
var EE = require('events').EventEmitter;
|
|
var Stream = require('./index.js');
|
|
var Buffer = require('buffer').Buffer;
|
|
var setImmediate = require('process/browser.js').nextTick;
|
|
var StringDecoder;
|
|
|
|
var inherits = require('inherits');
|
|
inherits(Readable, Stream);
|
|
|
|
function ReadableState(options, stream) {
|
|
options = options || {};
|
|
|
|
// the point at which it stops calling _read() to fill the buffer
|
|
// Note: 0 is a valid value, means "don't call _read preemptively ever"
|
|
var hwm = options.highWaterMark;
|
|
this.highWaterMark = (hwm || hwm === 0) ? hwm : 16 * 1024;
|
|
|
|
// cast to ints.
|
|
this.highWaterMark = ~~this.highWaterMark;
|
|
|
|
this.buffer = [];
|
|
this.length = 0;
|
|
this.pipes = null;
|
|
this.pipesCount = 0;
|
|
this.flowing = false;
|
|
this.ended = false;
|
|
this.endEmitted = false;
|
|
this.reading = false;
|
|
|
|
// In streams that never have any data, and do push(null) right away,
|
|
// the consumer can miss the 'end' event if they do some I/O before
|
|
// consuming the stream. So, we don't emit('end') until some reading
|
|
// happens.
|
|
this.calledRead = false;
|
|
|
|
// a flag to be able to tell if the onwrite cb is called immediately,
|
|
// or on a later tick. We set this to true at first, becuase any
|
|
// actions that shouldn't happen until "later" should generally also
|
|
// not happen before the first write call.
|
|
this.sync = true;
|
|
|
|
// whenever we return null, then we set a flag to say
|
|
// that we're awaiting a 'readable' event emission.
|
|
this.needReadable = false;
|
|
this.emittedReadable = false;
|
|
this.readableListening = false;
|
|
|
|
|
|
// object stream flag. Used to make read(n) ignore n and to
|
|
// make all the buffer merging and length checks go away
|
|
this.objectMode = !!options.objectMode;
|
|
|
|
// Crypto is kind of old and crusty. Historically, its default string
|
|
// encoding is 'binary' so we have to make this configurable.
|
|
// Everything else in the universe uses 'utf8', though.
|
|
this.defaultEncoding = options.defaultEncoding || 'utf8';
|
|
|
|
// when piping, we only care about 'readable' events that happen
|
|
// after read()ing all the bytes and not getting any pushback.
|
|
this.ranOut = false;
|
|
|
|
// the number of writers that are awaiting a drain event in .pipe()s
|
|
this.awaitDrain = 0;
|
|
|
|
// if true, a maybeReadMore has been scheduled
|
|
this.readingMore = false;
|
|
|
|
this.decoder = null;
|
|
this.encoding = null;
|
|
if (options.encoding) {
|
|
if (!StringDecoder)
|
|
StringDecoder = require('string_decoder').StringDecoder;
|
|
this.decoder = new StringDecoder(options.encoding);
|
|
this.encoding = options.encoding;
|
|
}
|
|
}
|
|
|
|
function Readable(options) {
|
|
if (!(this instanceof Readable))
|
|
return new Readable(options);
|
|
|
|
this._readableState = new ReadableState(options, this);
|
|
|
|
// legacy
|
|
this.readable = true;
|
|
|
|
Stream.call(this);
|
|
}
|
|
|
|
// Manually shove something into the read() buffer.
|
|
// This returns true if the highWaterMark has not been hit yet,
|
|
// similar to how Writable.write() returns true if you should
|
|
// write() some more.
|
|
Readable.prototype.push = function(chunk, encoding) {
|
|
var state = this._readableState;
|
|
|
|
if (typeof chunk === 'string' && !state.objectMode) {
|
|
encoding = encoding || state.defaultEncoding;
|
|
if (encoding !== state.encoding) {
|
|
chunk = new Buffer(chunk, encoding);
|
|
encoding = '';
|
|
}
|
|
}
|
|
|
|
return readableAddChunk(this, state, chunk, encoding, false);
|
|
};
|
|
|
|
// Unshift should *always* be something directly out of read()
|
|
Readable.prototype.unshift = function(chunk) {
|
|
var state = this._readableState;
|
|
return readableAddChunk(this, state, chunk, '', true);
|
|
};
|
|
|
|
function readableAddChunk(stream, state, chunk, encoding, addToFront) {
|
|
var er = chunkInvalid(state, chunk);
|
|
if (er) {
|
|
stream.emit('error', er);
|
|
} else if (chunk === null || chunk === undefined) {
|
|
state.reading = false;
|
|
if (!state.ended)
|
|
onEofChunk(stream, state);
|
|
} else if (state.objectMode || chunk && chunk.length > 0) {
|
|
if (state.ended && !addToFront) {
|
|
var e = new Error('stream.push() after EOF');
|
|
stream.emit('error', e);
|
|
} else if (state.endEmitted && addToFront) {
|
|
var e = new Error('stream.unshift() after end event');
|
|
stream.emit('error', e);
|
|
} else {
|
|
if (state.decoder && !addToFront && !encoding)
|
|
chunk = state.decoder.write(chunk);
|
|
|
|
// update the buffer info.
|
|
state.length += state.objectMode ? 1 : chunk.length;
|
|
if (addToFront) {
|
|
state.buffer.unshift(chunk);
|
|
} else {
|
|
state.reading = false;
|
|
state.buffer.push(chunk);
|
|
}
|
|
|
|
if (state.needReadable)
|
|
emitReadable(stream);
|
|
|
|
maybeReadMore(stream, state);
|
|
}
|
|
} else if (!addToFront) {
|
|
state.reading = false;
|
|
}
|
|
|
|
return needMoreData(state);
|
|
}
|
|
|
|
|
|
|
|
// if it's past the high water mark, we can push in some more.
|
|
// Also, if we have no data yet, we can stand some
|
|
// more bytes. This is to work around cases where hwm=0,
|
|
// such as the repl. Also, if the push() triggered a
|
|
// readable event, and the user called read(largeNumber) such that
|
|
// needReadable was set, then we ought to push more, so that another
|
|
// 'readable' event will be triggered.
|
|
function needMoreData(state) {
|
|
return !state.ended &&
|
|
(state.needReadable ||
|
|
state.length < state.highWaterMark ||
|
|
state.length === 0);
|
|
}
|
|
|
|
// backwards compatibility.
|
|
Readable.prototype.setEncoding = function(enc) {
|
|
if (!StringDecoder)
|
|
StringDecoder = require('string_decoder').StringDecoder;
|
|
this._readableState.decoder = new StringDecoder(enc);
|
|
this._readableState.encoding = enc;
|
|
};
|
|
|
|
// Don't raise the hwm > 128MB
|
|
var MAX_HWM = 0x800000;
|
|
function roundUpToNextPowerOf2(n) {
|
|
if (n >= MAX_HWM) {
|
|
n = MAX_HWM;
|
|
} else {
|
|
// Get the next highest power of 2
|
|
n--;
|
|
for (var p = 1; p < 32; p <<= 1) n |= n >> p;
|
|
n++;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
function howMuchToRead(n, state) {
|
|
if (state.length === 0 && state.ended)
|
|
return 0;
|
|
|
|
if (state.objectMode)
|
|
return n === 0 ? 0 : 1;
|
|
|
|
if (isNaN(n) || n === null) {
|
|
// only flow one buffer at a time
|
|
if (state.flowing && state.buffer.length)
|
|
return state.buffer[0].length;
|
|
else
|
|
return state.length;
|
|
}
|
|
|
|
if (n <= 0)
|
|
return 0;
|
|
|
|
// If we're asking for more than the target buffer level,
|
|
// then raise the water mark. Bump up to the next highest
|
|
// power of 2, to prevent increasing it excessively in tiny
|
|
// amounts.
|
|
if (n > state.highWaterMark)
|
|
state.highWaterMark = roundUpToNextPowerOf2(n);
|
|
|
|
// don't have that much. return null, unless we've ended.
|
|
if (n > state.length) {
|
|
if (!state.ended) {
|
|
state.needReadable = true;
|
|
return 0;
|
|
} else
|
|
return state.length;
|
|
}
|
|
|
|
return n;
|
|
}
|
|
|
|
// you can override either this method, or the async _read(n) below.
|
|
Readable.prototype.read = function(n) {
|
|
var state = this._readableState;
|
|
state.calledRead = true;
|
|
var nOrig = n;
|
|
|
|
if (typeof n !== 'number' || n > 0)
|
|
state.emittedReadable = false;
|
|
|
|
// if we're doing read(0) to trigger a readable event, but we
|
|
// already have a bunch of data in the buffer, then just trigger
|
|
// the 'readable' event and move on.
|
|
if (n === 0 &&
|
|
state.needReadable &&
|
|
(state.length >= state.highWaterMark || state.ended)) {
|
|
emitReadable(this);
|
|
return null;
|
|
}
|
|
|
|
n = howMuchToRead(n, state);
|
|
|
|
// if we've ended, and we're now clear, then finish it up.
|
|
if (n === 0 && state.ended) {
|
|
if (state.length === 0)
|
|
endReadable(this);
|
|
return null;
|
|
}
|
|
|
|
// All the actual chunk generation logic needs to be
|
|
// *below* the call to _read. The reason is that in certain
|
|
// synthetic stream cases, such as passthrough streams, _read
|
|
// may be a completely synchronous operation which may change
|
|
// the state of the read buffer, providing enough data when
|
|
// before there was *not* enough.
|
|
//
|
|
// So, the steps are:
|
|
// 1. Figure out what the state of things will be after we do
|
|
// a read from the buffer.
|
|
//
|
|
// 2. If that resulting state will trigger a _read, then call _read.
|
|
// Note that this may be asynchronous, or synchronous. Yes, it is
|
|
// deeply ugly to write APIs this way, but that still doesn't mean
|
|
// that the Readable class should behave improperly, as streams are
|
|
// designed to be sync/async agnostic.
|
|
// Take note if the _read call is sync or async (ie, if the read call
|
|
// has returned yet), so that we know whether or not it's safe to emit
|
|
// 'readable' etc.
|
|
//
|
|
// 3. Actually pull the requested chunks out of the buffer and return.
|
|
|
|
// if we need a readable event, then we need to do some reading.
|
|
var doRead = state.needReadable;
|
|
|
|
// if we currently have less than the highWaterMark, then also read some
|
|
if (state.length - n <= state.highWaterMark)
|
|
doRead = true;
|
|
|
|
// however, if we've ended, then there's no point, and if we're already
|
|
// reading, then it's unnecessary.
|
|
if (state.ended || state.reading)
|
|
doRead = false;
|
|
|
|
if (doRead) {
|
|
state.reading = true;
|
|
state.sync = true;
|
|
// if the length is currently zero, then we *need* a readable event.
|
|
if (state.length === 0)
|
|
state.needReadable = true;
|
|
// call internal read method
|
|
this._read(state.highWaterMark);
|
|
state.sync = false;
|
|
}
|
|
|
|
// If _read called its callback synchronously, then `reading`
|
|
// will be false, and we need to re-evaluate how much data we
|
|
// can return to the user.
|
|
if (doRead && !state.reading)
|
|
n = howMuchToRead(nOrig, state);
|
|
|
|
var ret;
|
|
if (n > 0)
|
|
ret = fromList(n, state);
|
|
else
|
|
ret = null;
|
|
|
|
if (ret === null) {
|
|
state.needReadable = true;
|
|
n = 0;
|
|
}
|
|
|
|
state.length -= n;
|
|
|
|
// If we have nothing in the buffer, then we want to know
|
|
// as soon as we *do* get something into the buffer.
|
|
if (state.length === 0 && !state.ended)
|
|
state.needReadable = true;
|
|
|
|
// If we happened to read() exactly the remaining amount in the
|
|
// buffer, and the EOF has been seen at this point, then make sure
|
|
// that we emit 'end' on the very next tick.
|
|
if (state.ended && !state.endEmitted && state.length === 0)
|
|
endReadable(this);
|
|
|
|
return ret;
|
|
};
|
|
|
|
function chunkInvalid(state, chunk) {
|
|
var er = null;
|
|
if (!Buffer.isBuffer(chunk) &&
|
|
'string' !== typeof chunk &&
|
|
chunk !== null &&
|
|
chunk !== undefined &&
|
|
!state.objectMode &&
|
|
!er) {
|
|
er = new TypeError('Invalid non-string/buffer chunk');
|
|
}
|
|
return er;
|
|
}
|
|
|
|
|
|
function onEofChunk(stream, state) {
|
|
if (state.decoder && !state.ended) {
|
|
var chunk = state.decoder.end();
|
|
if (chunk && chunk.length) {
|
|
state.buffer.push(chunk);
|
|
state.length += state.objectMode ? 1 : chunk.length;
|
|
}
|
|
}
|
|
state.ended = true;
|
|
|
|
// if we've ended and we have some data left, then emit
|
|
// 'readable' now to make sure it gets picked up.
|
|
if (state.length > 0)
|
|
emitReadable(stream);
|
|
else
|
|
endReadable(stream);
|
|
}
|
|
|
|
// Don't emit readable right away in sync mode, because this can trigger
|
|
// another read() call => stack overflow. This way, it might trigger
|
|
// a nextTick recursion warning, but that's not so bad.
|
|
function emitReadable(stream) {
|
|
var state = stream._readableState;
|
|
state.needReadable = false;
|
|
if (state.emittedReadable)
|
|
return;
|
|
|
|
state.emittedReadable = true;
|
|
if (state.sync)
|
|
setImmediate(function() {
|
|
emitReadable_(stream);
|
|
});
|
|
else
|
|
emitReadable_(stream);
|
|
}
|
|
|
|
function emitReadable_(stream) {
|
|
stream.emit('readable');
|
|
}
|
|
|
|
|
|
// at this point, the user has presumably seen the 'readable' event,
|
|
// and called read() to consume some data. that may have triggered
|
|
// in turn another _read(n) call, in which case reading = true if
|
|
// it's in progress.
|
|
// However, if we're not ended, or reading, and the length < hwm,
|
|
// then go ahead and try to read some more preemptively.
|
|
function maybeReadMore(stream, state) {
|
|
if (!state.readingMore) {
|
|
state.readingMore = true;
|
|
setImmediate(function() {
|
|
maybeReadMore_(stream, state);
|
|
});
|
|
}
|
|
}
|
|
|
|
function maybeReadMore_(stream, state) {
|
|
var len = state.length;
|
|
while (!state.reading && !state.flowing && !state.ended &&
|
|
state.length < state.highWaterMark) {
|
|
stream.read(0);
|
|
if (len === state.length)
|
|
// didn't get any data, stop spinning.
|
|
break;
|
|
else
|
|
len = state.length;
|
|
}
|
|
state.readingMore = false;
|
|
}
|
|
|
|
// abstract method. to be overridden in specific implementation classes.
|
|
// call cb(er, data) where data is <= n in length.
|
|
// for virtual (non-string, non-buffer) streams, "length" is somewhat
|
|
// arbitrary, and perhaps not very meaningful.
|
|
Readable.prototype._read = function(n) {
|
|
this.emit('error', new Error('not implemented'));
|
|
};
|
|
|
|
Readable.prototype.pipe = function(dest, pipeOpts) {
|
|
var src = this;
|
|
var state = this._readableState;
|
|
|
|
switch (state.pipesCount) {
|
|
case 0:
|
|
state.pipes = dest;
|
|
break;
|
|
case 1:
|
|
state.pipes = [state.pipes, dest];
|
|
break;
|
|
default:
|
|
state.pipes.push(dest);
|
|
break;
|
|
}
|
|
state.pipesCount += 1;
|
|
|
|
var doEnd = (!pipeOpts || pipeOpts.end !== false) &&
|
|
dest !== process.stdout &&
|
|
dest !== process.stderr;
|
|
|
|
var endFn = doEnd ? onend : cleanup;
|
|
if (state.endEmitted)
|
|
setImmediate(endFn);
|
|
else
|
|
src.once('end', endFn);
|
|
|
|
dest.on('unpipe', onunpipe);
|
|
function onunpipe(readable) {
|
|
if (readable !== src) return;
|
|
cleanup();
|
|
}
|
|
|
|
function onend() {
|
|
dest.end();
|
|
}
|
|
|
|
// when the dest drains, it reduces the awaitDrain counter
|
|
// on the source. This would be more elegant with a .once()
|
|
// handler in flow(), but adding and removing repeatedly is
|
|
// too slow.
|
|
var ondrain = pipeOnDrain(src);
|
|
dest.on('drain', ondrain);
|
|
|
|
function cleanup() {
|
|
// cleanup event handlers once the pipe is broken
|
|
dest.removeListener('close', onclose);
|
|
dest.removeListener('finish', onfinish);
|
|
dest.removeListener('drain', ondrain);
|
|
dest.removeListener('error', onerror);
|
|
dest.removeListener('unpipe', onunpipe);
|
|
src.removeListener('end', onend);
|
|
src.removeListener('end', cleanup);
|
|
|
|
// if the reader is waiting for a drain event from this
|
|
// specific writer, then it would cause it to never start
|
|
// flowing again.
|
|
// So, if this is awaiting a drain, then we just call it now.
|
|
// If we don't know, then assume that we are waiting for one.
|
|
if (!dest._writableState || dest._writableState.needDrain)
|
|
ondrain();
|
|
}
|
|
|
|
// if the dest has an error, then stop piping into it.
|
|
// however, don't suppress the throwing behavior for this.
|
|
// check for listeners before emit removes one-time listeners.
|
|
var errListeners = EE.listenerCount(dest, 'error');
|
|
function onerror(er) {
|
|
unpipe();
|
|
if (errListeners === 0 && EE.listenerCount(dest, 'error') === 0)
|
|
dest.emit('error', er);
|
|
}
|
|
dest.once('error', onerror);
|
|
|
|
// Both close and finish should trigger unpipe, but only once.
|
|
function onclose() {
|
|
dest.removeListener('finish', onfinish);
|
|
unpipe();
|
|
}
|
|
dest.once('close', onclose);
|
|
function onfinish() {
|
|
dest.removeListener('close', onclose);
|
|
unpipe();
|
|
}
|
|
dest.once('finish', onfinish);
|
|
|
|
function unpipe() {
|
|
src.unpipe(dest);
|
|
}
|
|
|
|
// tell the dest that it's being piped to
|
|
dest.emit('pipe', src);
|
|
|
|
// start the flow if it hasn't been started already.
|
|
if (!state.flowing) {
|
|
// the handler that waits for readable events after all
|
|
// the data gets sucked out in flow.
|
|
// This would be easier to follow with a .once() handler
|
|
// in flow(), but that is too slow.
|
|
this.on('readable', pipeOnReadable);
|
|
|
|
state.flowing = true;
|
|
setImmediate(function() {
|
|
flow(src);
|
|
});
|
|
}
|
|
|
|
return dest;
|
|
};
|
|
|
|
function pipeOnDrain(src) {
|
|
return function() {
|
|
var dest = this;
|
|
var state = src._readableState;
|
|
state.awaitDrain--;
|
|
if (state.awaitDrain === 0)
|
|
flow(src);
|
|
};
|
|
}
|
|
|
|
function flow(src) {
|
|
var state = src._readableState;
|
|
var chunk;
|
|
state.awaitDrain = 0;
|
|
|
|
function write(dest, i, list) {
|
|
var written = dest.write(chunk);
|
|
if (false === written) {
|
|
state.awaitDrain++;
|
|
}
|
|
}
|
|
|
|
while (state.pipesCount && null !== (chunk = src.read())) {
|
|
|
|
if (state.pipesCount === 1)
|
|
write(state.pipes, 0, null);
|
|
else
|
|
forEach(state.pipes, write);
|
|
|
|
src.emit('data', chunk);
|
|
|
|
// if anyone needs a drain, then we have to wait for that.
|
|
if (state.awaitDrain > 0)
|
|
return;
|
|
}
|
|
|
|
// if every destination was unpiped, either before entering this
|
|
// function, or in the while loop, then stop flowing.
|
|
//
|
|
// NB: This is a pretty rare edge case.
|
|
if (state.pipesCount === 0) {
|
|
state.flowing = false;
|
|
|
|
// if there were data event listeners added, then switch to old mode.
|
|
if (EE.listenerCount(src, 'data') > 0)
|
|
emitDataEvents(src);
|
|
return;
|
|
}
|
|
|
|
// at this point, no one needed a drain, so we just ran out of data
|
|
// on the next readable event, start it over again.
|
|
state.ranOut = true;
|
|
}
|
|
|
|
function pipeOnReadable() {
|
|
if (this._readableState.ranOut) {
|
|
this._readableState.ranOut = false;
|
|
flow(this);
|
|
}
|
|
}
|
|
|
|
|
|
Readable.prototype.unpipe = function(dest) {
|
|
var state = this._readableState;
|
|
|
|
// if we're not piping anywhere, then do nothing.
|
|
if (state.pipesCount === 0)
|
|
return this;
|
|
|
|
// just one destination. most common case.
|
|
if (state.pipesCount === 1) {
|
|
// passed in one, but it's not the right one.
|
|
if (dest && dest !== state.pipes)
|
|
return this;
|
|
|
|
if (!dest)
|
|
dest = state.pipes;
|
|
|
|
// got a match.
|
|
state.pipes = null;
|
|
state.pipesCount = 0;
|
|
this.removeListener('readable', pipeOnReadable);
|
|
state.flowing = false;
|
|
if (dest)
|
|
dest.emit('unpipe', this);
|
|
return this;
|
|
}
|
|
|
|
// slow case. multiple pipe destinations.
|
|
|
|
if (!dest) {
|
|
// remove all.
|
|
var dests = state.pipes;
|
|
var len = state.pipesCount;
|
|
state.pipes = null;
|
|
state.pipesCount = 0;
|
|
this.removeListener('readable', pipeOnReadable);
|
|
state.flowing = false;
|
|
|
|
for (var i = 0; i < len; i++)
|
|
dests[i].emit('unpipe', this);
|
|
return this;
|
|
}
|
|
|
|
// try to find the right one.
|
|
var i = indexOf(state.pipes, dest);
|
|
if (i === -1)
|
|
return this;
|
|
|
|
state.pipes.splice(i, 1);
|
|
state.pipesCount -= 1;
|
|
if (state.pipesCount === 1)
|
|
state.pipes = state.pipes[0];
|
|
|
|
dest.emit('unpipe', this);
|
|
|
|
return this;
|
|
};
|
|
|
|
// set up data events if they are asked for
|
|
// Ensure readable listeners eventually get something
|
|
Readable.prototype.on = function(ev, fn) {
|
|
var res = Stream.prototype.on.call(this, ev, fn);
|
|
|
|
if (ev === 'data' && !this._readableState.flowing)
|
|
emitDataEvents(this);
|
|
|
|
if (ev === 'readable' && this.readable) {
|
|
var state = this._readableState;
|
|
if (!state.readableListening) {
|
|
state.readableListening = true;
|
|
state.emittedReadable = false;
|
|
state.needReadable = true;
|
|
if (!state.reading) {
|
|
this.read(0);
|
|
} else if (state.length) {
|
|
emitReadable(this, state);
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
};
|
|
Readable.prototype.addListener = Readable.prototype.on;
|
|
|
|
// pause() and resume() are remnants of the legacy readable stream API
|
|
// If the user uses them, then switch into old mode.
|
|
Readable.prototype.resume = function() {
|
|
emitDataEvents(this);
|
|
this.read(0);
|
|
this.emit('resume');
|
|
};
|
|
|
|
Readable.prototype.pause = function() {
|
|
emitDataEvents(this, true);
|
|
this.emit('pause');
|
|
};
|
|
|
|
function emitDataEvents(stream, startPaused) {
|
|
var state = stream._readableState;
|
|
|
|
if (state.flowing) {
|
|
// https://github.com/isaacs/readable-stream/issues/16
|
|
throw new Error('Cannot switch to old mode now.');
|
|
}
|
|
|
|
var paused = startPaused || false;
|
|
var readable = false;
|
|
|
|
// convert to an old-style stream.
|
|
stream.readable = true;
|
|
stream.pipe = Stream.prototype.pipe;
|
|
stream.on = stream.addListener = Stream.prototype.on;
|
|
|
|
stream.on('readable', function() {
|
|
readable = true;
|
|
|
|
var c;
|
|
while (!paused && (null !== (c = stream.read())))
|
|
stream.emit('data', c);
|
|
|
|
if (c === null) {
|
|
readable = false;
|
|
stream._readableState.needReadable = true;
|
|
}
|
|
});
|
|
|
|
stream.pause = function() {
|
|
paused = true;
|
|
this.emit('pause');
|
|
};
|
|
|
|
stream.resume = function() {
|
|
paused = false;
|
|
if (readable)
|
|
setImmediate(function() {
|
|
stream.emit('readable');
|
|
});
|
|
else
|
|
this.read(0);
|
|
this.emit('resume');
|
|
};
|
|
|
|
// now make it start, just in case it hadn't already.
|
|
stream.emit('readable');
|
|
}
|
|
|
|
// wrap an old-style stream as the async data source.
|
|
// This is *not* part of the readable stream interface.
|
|
// It is an ugly unfortunate mess of history.
|
|
Readable.prototype.wrap = function(stream) {
|
|
var state = this._readableState;
|
|
var paused = false;
|
|
|
|
var self = this;
|
|
stream.on('end', function() {
|
|
if (state.decoder && !state.ended) {
|
|
var chunk = state.decoder.end();
|
|
if (chunk && chunk.length)
|
|
self.push(chunk);
|
|
}
|
|
|
|
self.push(null);
|
|
});
|
|
|
|
stream.on('data', function(chunk) {
|
|
if (state.decoder)
|
|
chunk = state.decoder.write(chunk);
|
|
if (!chunk || !state.objectMode && !chunk.length)
|
|
return;
|
|
|
|
var ret = self.push(chunk);
|
|
if (!ret) {
|
|
paused = true;
|
|
stream.pause();
|
|
}
|
|
});
|
|
|
|
// proxy all the other methods.
|
|
// important when wrapping filters and duplexes.
|
|
for (var i in stream) {
|
|
if (typeof stream[i] === 'function' &&
|
|
typeof this[i] === 'undefined') {
|
|
this[i] = function(method) { return function() {
|
|
return stream[method].apply(stream, arguments);
|
|
}}(i);
|
|
}
|
|
}
|
|
|
|
// proxy certain important events.
|
|
var events = ['error', 'close', 'destroy', 'pause', 'resume'];
|
|
forEach(events, function(ev) {
|
|
stream.on(ev, function (x) {
|
|
return self.emit.apply(self, ev, x);
|
|
});
|
|
});
|
|
|
|
// when we try to consume some more bytes, simply unpause the
|
|
// underlying stream.
|
|
self._read = function(n) {
|
|
if (paused) {
|
|
paused = false;
|
|
stream.resume();
|
|
}
|
|
};
|
|
|
|
return self;
|
|
};
|
|
|
|
|
|
|
|
// exposed for testing purposes only.
|
|
Readable._fromList = fromList;
|
|
|
|
// Pluck off n bytes from an array of buffers.
|
|
// Length is the combined lengths of all the buffers in the list.
|
|
function fromList(n, state) {
|
|
var list = state.buffer;
|
|
var length = state.length;
|
|
var stringMode = !!state.decoder;
|
|
var objectMode = !!state.objectMode;
|
|
var ret;
|
|
|
|
// nothing in the list, definitely empty.
|
|
if (list.length === 0)
|
|
return null;
|
|
|
|
if (length === 0)
|
|
ret = null;
|
|
else if (objectMode)
|
|
ret = list.shift();
|
|
else if (!n || n >= length) {
|
|
// read it all, truncate the array.
|
|
if (stringMode)
|
|
ret = list.join('');
|
|
else
|
|
ret = Buffer.concat(list, length);
|
|
list.length = 0;
|
|
} else {
|
|
// read just some of it.
|
|
if (n < list[0].length) {
|
|
// just take a part of the first list item.
|
|
// slice is the same for buffers and strings.
|
|
var buf = list[0];
|
|
ret = buf.slice(0, n);
|
|
list[0] = buf.slice(n);
|
|
} else if (n === list[0].length) {
|
|
// first list is a perfect match
|
|
ret = list.shift();
|
|
} else {
|
|
// complex case.
|
|
// we have enough to cover it, but it spans past the first buffer.
|
|
if (stringMode)
|
|
ret = '';
|
|
else
|
|
ret = new Buffer(n);
|
|
|
|
var c = 0;
|
|
for (var i = 0, l = list.length; i < l && c < n; i++) {
|
|
var buf = list[0];
|
|
var cpy = Math.min(n - c, buf.length);
|
|
|
|
if (stringMode)
|
|
ret += buf.slice(0, cpy);
|
|
else
|
|
buf.copy(ret, c, 0, cpy);
|
|
|
|
if (cpy < buf.length)
|
|
list[0] = buf.slice(cpy);
|
|
else
|
|
list.shift();
|
|
|
|
c += cpy;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
function endReadable(stream) {
|
|
var state = stream._readableState;
|
|
|
|
// If we get here before consuming all the bytes, then that is a
|
|
// bug in node. Should never happen.
|
|
if (state.length > 0)
|
|
throw new Error('endReadable called on non-empty stream');
|
|
|
|
if (!state.endEmitted && state.calledRead) {
|
|
state.ended = true;
|
|
setImmediate(function() {
|
|
// Check that we didn't get one last unshift.
|
|
if (!state.endEmitted && state.length === 0) {
|
|
state.endEmitted = true;
|
|
stream.readable = false;
|
|
stream.emit('end');
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
function forEach (xs, f) {
|
|
for (var i = 0, l = xs.length; i < l; i++) {
|
|
f(xs[i], i);
|
|
}
|
|
}
|
|
|
|
function indexOf (xs, x) {
|
|
for (var i = 0, l = xs.length; i < l; i++) {
|
|
if (xs[i] === x) return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"))
|
|
},{"./index.js":103,"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"buffer":80,"events":89,"inherits":95,"process/browser.js":104,"string_decoder":109}],107:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// a transform stream is a readable/writable stream where you do
|
|
// something with the data. Sometimes it's called a "filter",
|
|
// but that's not a great name for it, since that implies a thing where
|
|
// some bits pass through, and others are simply ignored. (That would
|
|
// be a valid example of a transform, of course.)
|
|
//
|
|
// While the output is causally related to the input, it's not a
|
|
// necessarily symmetric or synchronous transformation. For example,
|
|
// a zlib stream might take multiple plain-text writes(), and then
|
|
// emit a single compressed chunk some time in the future.
|
|
//
|
|
// Here's how this works:
|
|
//
|
|
// The Transform stream has all the aspects of the readable and writable
|
|
// stream classes. When you write(chunk), that calls _write(chunk,cb)
|
|
// internally, and returns false if there's a lot of pending writes
|
|
// buffered up. When you call read(), that calls _read(n) until
|
|
// there's enough pending readable data buffered up.
|
|
//
|
|
// In a transform stream, the written data is placed in a buffer. When
|
|
// _read(n) is called, it transforms the queued up data, calling the
|
|
// buffered _write cb's as it consumes chunks. If consuming a single
|
|
// written chunk would result in multiple output chunks, then the first
|
|
// outputted bit calls the readcb, and subsequent chunks just go into
|
|
// the read buffer, and will cause it to emit 'readable' if necessary.
|
|
//
|
|
// This way, back-pressure is actually determined by the reading side,
|
|
// since _read has to be called to start processing a new chunk. However,
|
|
// a pathological inflate type of transform can cause excessive buffering
|
|
// here. For example, imagine a stream where every byte of input is
|
|
// interpreted as an integer from 0-255, and then results in that many
|
|
// bytes of output. Writing the 4 bytes {ff,ff,ff,ff} would result in
|
|
// 1kb of data being output. In this case, you could write a very small
|
|
// amount of input, and end up with a very large amount of output. In
|
|
// such a pathological inflating mechanism, there'd be no way to tell
|
|
// the system to stop doing the transform. A single 4MB write could
|
|
// cause the system to run out of memory.
|
|
//
|
|
// However, even in such a pathological case, only a single written chunk
|
|
// would be consumed, and then the rest would wait (un-transformed) until
|
|
// the results of the previous transformed chunk were consumed.
|
|
|
|
module.exports = Transform;
|
|
|
|
var Duplex = require('./duplex.js');
|
|
var inherits = require('inherits');
|
|
inherits(Transform, Duplex);
|
|
|
|
|
|
function TransformState(options, stream) {
|
|
this.afterTransform = function(er, data) {
|
|
return afterTransform(stream, er, data);
|
|
};
|
|
|
|
this.needTransform = false;
|
|
this.transforming = false;
|
|
this.writecb = null;
|
|
this.writechunk = null;
|
|
}
|
|
|
|
function afterTransform(stream, er, data) {
|
|
var ts = stream._transformState;
|
|
ts.transforming = false;
|
|
|
|
var cb = ts.writecb;
|
|
|
|
if (!cb)
|
|
return stream.emit('error', new Error('no writecb in Transform class'));
|
|
|
|
ts.writechunk = null;
|
|
ts.writecb = null;
|
|
|
|
if (data !== null && data !== undefined)
|
|
stream.push(data);
|
|
|
|
if (cb)
|
|
cb(er);
|
|
|
|
var rs = stream._readableState;
|
|
rs.reading = false;
|
|
if (rs.needReadable || rs.length < rs.highWaterMark) {
|
|
stream._read(rs.highWaterMark);
|
|
}
|
|
}
|
|
|
|
|
|
function Transform(options) {
|
|
if (!(this instanceof Transform))
|
|
return new Transform(options);
|
|
|
|
Duplex.call(this, options);
|
|
|
|
var ts = this._transformState = new TransformState(options, this);
|
|
|
|
// when the writable side finishes, then flush out anything remaining.
|
|
var stream = this;
|
|
|
|
// start out asking for a readable event once data is transformed.
|
|
this._readableState.needReadable = true;
|
|
|
|
// we have implemented the _read method, and done the other things
|
|
// that Readable wants before the first _read call, so unset the
|
|
// sync guard flag.
|
|
this._readableState.sync = false;
|
|
|
|
this.once('finish', function() {
|
|
if ('function' === typeof this._flush)
|
|
this._flush(function(er) {
|
|
done(stream, er);
|
|
});
|
|
else
|
|
done(stream);
|
|
});
|
|
}
|
|
|
|
Transform.prototype.push = function(chunk, encoding) {
|
|
this._transformState.needTransform = false;
|
|
return Duplex.prototype.push.call(this, chunk, encoding);
|
|
};
|
|
|
|
// This is the part where you do stuff!
|
|
// override this function in implementation classes.
|
|
// 'chunk' is an input chunk.
|
|
//
|
|
// Call `push(newChunk)` to pass along transformed output
|
|
// to the readable side. You may call 'push' zero or more times.
|
|
//
|
|
// Call `cb(err)` when you are done with this chunk. If you pass
|
|
// an error, then that'll put the hurt on the whole operation. If you
|
|
// never call cb(), then you'll never get another chunk.
|
|
Transform.prototype._transform = function(chunk, encoding, cb) {
|
|
throw new Error('not implemented');
|
|
};
|
|
|
|
Transform.prototype._write = function(chunk, encoding, cb) {
|
|
var ts = this._transformState;
|
|
ts.writecb = cb;
|
|
ts.writechunk = chunk;
|
|
ts.writeencoding = encoding;
|
|
if (!ts.transforming) {
|
|
var rs = this._readableState;
|
|
if (ts.needTransform ||
|
|
rs.needReadable ||
|
|
rs.length < rs.highWaterMark)
|
|
this._read(rs.highWaterMark);
|
|
}
|
|
};
|
|
|
|
// Doesn't matter what the args are here.
|
|
// _transform does all the work.
|
|
// That we got here means that the readable side wants more data.
|
|
Transform.prototype._read = function(n) {
|
|
var ts = this._transformState;
|
|
|
|
if (ts.writechunk && ts.writecb && !ts.transforming) {
|
|
ts.transforming = true;
|
|
this._transform(ts.writechunk, ts.writeencoding, ts.afterTransform);
|
|
} else {
|
|
// mark that we need a transform, so that any data that comes in
|
|
// will get processed, now that we've asked for it.
|
|
ts.needTransform = true;
|
|
}
|
|
};
|
|
|
|
|
|
function done(stream, er) {
|
|
if (er)
|
|
return stream.emit('error', er);
|
|
|
|
// if there's nothing in the write buffer, then that means
|
|
// that nothing more will ever be provided
|
|
var ws = stream._writableState;
|
|
var rs = stream._readableState;
|
|
var ts = stream._transformState;
|
|
|
|
if (ws.length)
|
|
throw new Error('calling transform done when ws.length != 0');
|
|
|
|
if (ts.transforming)
|
|
throw new Error('calling transform done when still transforming');
|
|
|
|
return stream.push(null);
|
|
}
|
|
|
|
},{"./duplex.js":102,"inherits":95}],108:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// A bit simpler than readable streams.
|
|
// Implement an async ._write(chunk, cb), and it'll handle all
|
|
// the drain event emission and buffering.
|
|
|
|
module.exports = Writable;
|
|
Writable.WritableState = WritableState;
|
|
|
|
var isUint8Array = typeof Uint8Array !== 'undefined'
|
|
? function (x) { return x instanceof Uint8Array }
|
|
: function (x) {
|
|
return x && x.constructor && x.constructor.name === 'Uint8Array'
|
|
}
|
|
;
|
|
var isArrayBuffer = typeof ArrayBuffer !== 'undefined'
|
|
? function (x) { return x instanceof ArrayBuffer }
|
|
: function (x) {
|
|
return x && x.constructor && x.constructor.name === 'ArrayBuffer'
|
|
}
|
|
;
|
|
|
|
var inherits = require('inherits');
|
|
var Stream = require('./index.js');
|
|
var setImmediate = require('process/browser.js').nextTick;
|
|
var Buffer = require('buffer').Buffer;
|
|
|
|
inherits(Writable, Stream);
|
|
|
|
function WriteReq(chunk, encoding, cb) {
|
|
this.chunk = chunk;
|
|
this.encoding = encoding;
|
|
this.callback = cb;
|
|
}
|
|
|
|
function WritableState(options, stream) {
|
|
options = options || {};
|
|
|
|
// the point at which write() starts returning false
|
|
// Note: 0 is a valid value, means that we always return false if
|
|
// the entire buffer is not flushed immediately on write()
|
|
var hwm = options.highWaterMark;
|
|
this.highWaterMark = (hwm || hwm === 0) ? hwm : 16 * 1024;
|
|
|
|
// object stream flag to indicate whether or not this stream
|
|
// contains buffers or objects.
|
|
this.objectMode = !!options.objectMode;
|
|
|
|
// cast to ints.
|
|
this.highWaterMark = ~~this.highWaterMark;
|
|
|
|
this.needDrain = false;
|
|
// at the start of calling end()
|
|
this.ending = false;
|
|
// when end() has been called, and returned
|
|
this.ended = false;
|
|
// when 'finish' is emitted
|
|
this.finished = false;
|
|
|
|
// should we decode strings into buffers before passing to _write?
|
|
// this is here so that some node-core streams can optimize string
|
|
// handling at a lower level.
|
|
var noDecode = options.decodeStrings === false;
|
|
this.decodeStrings = !noDecode;
|
|
|
|
// Crypto is kind of old and crusty. Historically, its default string
|
|
// encoding is 'binary' so we have to make this configurable.
|
|
// Everything else in the universe uses 'utf8', though.
|
|
this.defaultEncoding = options.defaultEncoding || 'utf8';
|
|
|
|
// not an actual buffer we keep track of, but a measurement
|
|
// of how much we're waiting to get pushed to some underlying
|
|
// socket or file.
|
|
this.length = 0;
|
|
|
|
// a flag to see when we're in the middle of a write.
|
|
this.writing = false;
|
|
|
|
// a flag to be able to tell if the onwrite cb is called immediately,
|
|
// or on a later tick. We set this to true at first, becuase any
|
|
// actions that shouldn't happen until "later" should generally also
|
|
// not happen before the first write call.
|
|
this.sync = true;
|
|
|
|
// a flag to know if we're processing previously buffered items, which
|
|
// may call the _write() callback in the same tick, so that we don't
|
|
// end up in an overlapped onwrite situation.
|
|
this.bufferProcessing = false;
|
|
|
|
// the callback that's passed to _write(chunk,cb)
|
|
this.onwrite = function(er) {
|
|
onwrite(stream, er);
|
|
};
|
|
|
|
// the callback that the user supplies to write(chunk,encoding,cb)
|
|
this.writecb = null;
|
|
|
|
// the amount that is being written when _write is called.
|
|
this.writelen = 0;
|
|
|
|
this.buffer = [];
|
|
}
|
|
|
|
function Writable(options) {
|
|
// Writable ctor is applied to Duplexes, though they're not
|
|
// instanceof Writable, they're instanceof Readable.
|
|
if (!(this instanceof Writable) && !(this instanceof Stream.Duplex))
|
|
return new Writable(options);
|
|
|
|
this._writableState = new WritableState(options, this);
|
|
|
|
// legacy.
|
|
this.writable = true;
|
|
|
|
Stream.call(this);
|
|
}
|
|
|
|
// Otherwise people can pipe Writable streams, which is just wrong.
|
|
Writable.prototype.pipe = function() {
|
|
this.emit('error', new Error('Cannot pipe. Not readable.'));
|
|
};
|
|
|
|
|
|
function writeAfterEnd(stream, state, cb) {
|
|
var er = new Error('write after end');
|
|
// TODO: defer error events consistently everywhere, not just the cb
|
|
stream.emit('error', er);
|
|
setImmediate(function() {
|
|
cb(er);
|
|
});
|
|
}
|
|
|
|
// If we get something that is not a buffer, string, null, or undefined,
|
|
// and we're not in objectMode, then that's an error.
|
|
// Otherwise stream chunks are all considered to be of length=1, and the
|
|
// watermarks determine how many objects to keep in the buffer, rather than
|
|
// how many bytes or characters.
|
|
function validChunk(stream, state, chunk, cb) {
|
|
var valid = true;
|
|
if (!Buffer.isBuffer(chunk) &&
|
|
'string' !== typeof chunk &&
|
|
chunk !== null &&
|
|
chunk !== undefined &&
|
|
!state.objectMode) {
|
|
var er = new TypeError('Invalid non-string/buffer chunk');
|
|
stream.emit('error', er);
|
|
setImmediate(function() {
|
|
cb(er);
|
|
});
|
|
valid = false;
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
Writable.prototype.write = function(chunk, encoding, cb) {
|
|
var state = this._writableState;
|
|
var ret = false;
|
|
|
|
if (typeof encoding === 'function') {
|
|
cb = encoding;
|
|
encoding = null;
|
|
}
|
|
|
|
if (!Buffer.isBuffer(chunk) && isUint8Array(chunk))
|
|
chunk = new Buffer(chunk);
|
|
if (isArrayBuffer(chunk) && typeof Uint8Array !== 'undefined')
|
|
chunk = new Buffer(new Uint8Array(chunk));
|
|
|
|
if (Buffer.isBuffer(chunk))
|
|
encoding = 'buffer';
|
|
else if (!encoding)
|
|
encoding = state.defaultEncoding;
|
|
|
|
if (typeof cb !== 'function')
|
|
cb = function() {};
|
|
|
|
if (state.ended)
|
|
writeAfterEnd(this, state, cb);
|
|
else if (validChunk(this, state, chunk, cb))
|
|
ret = writeOrBuffer(this, state, chunk, encoding, cb);
|
|
|
|
return ret;
|
|
};
|
|
|
|
function decodeChunk(state, chunk, encoding) {
|
|
if (!state.objectMode &&
|
|
state.decodeStrings !== false &&
|
|
typeof chunk === 'string') {
|
|
chunk = new Buffer(chunk, encoding);
|
|
}
|
|
return chunk;
|
|
}
|
|
|
|
// if we're already writing something, then just put this
|
|
// in the queue, and wait our turn. Otherwise, call _write
|
|
// If we return false, then we need a drain event, so set that flag.
|
|
function writeOrBuffer(stream, state, chunk, encoding, cb) {
|
|
chunk = decodeChunk(state, chunk, encoding);
|
|
var len = state.objectMode ? 1 : chunk.length;
|
|
|
|
state.length += len;
|
|
|
|
var ret = state.length < state.highWaterMark;
|
|
state.needDrain = !ret;
|
|
|
|
if (state.writing)
|
|
state.buffer.push(new WriteReq(chunk, encoding, cb));
|
|
else
|
|
doWrite(stream, state, len, chunk, encoding, cb);
|
|
|
|
return ret;
|
|
}
|
|
|
|
function doWrite(stream, state, len, chunk, encoding, cb) {
|
|
state.writelen = len;
|
|
state.writecb = cb;
|
|
state.writing = true;
|
|
state.sync = true;
|
|
stream._write(chunk, encoding, state.onwrite);
|
|
state.sync = false;
|
|
}
|
|
|
|
function onwriteError(stream, state, sync, er, cb) {
|
|
if (sync)
|
|
setImmediate(function() {
|
|
cb(er);
|
|
});
|
|
else
|
|
cb(er);
|
|
|
|
stream.emit('error', er);
|
|
}
|
|
|
|
function onwriteStateUpdate(state) {
|
|
state.writing = false;
|
|
state.writecb = null;
|
|
state.length -= state.writelen;
|
|
state.writelen = 0;
|
|
}
|
|
|
|
function onwrite(stream, er) {
|
|
var state = stream._writableState;
|
|
var sync = state.sync;
|
|
var cb = state.writecb;
|
|
|
|
onwriteStateUpdate(state);
|
|
|
|
if (er)
|
|
onwriteError(stream, state, sync, er, cb);
|
|
else {
|
|
// Check if we're actually ready to finish, but don't emit yet
|
|
var finished = needFinish(stream, state);
|
|
|
|
if (!finished && !state.bufferProcessing && state.buffer.length)
|
|
clearBuffer(stream, state);
|
|
|
|
if (sync) {
|
|
setImmediate(function() {
|
|
afterWrite(stream, state, finished, cb);
|
|
});
|
|
} else {
|
|
afterWrite(stream, state, finished, cb);
|
|
}
|
|
}
|
|
}
|
|
|
|
function afterWrite(stream, state, finished, cb) {
|
|
if (!finished)
|
|
onwriteDrain(stream, state);
|
|
cb();
|
|
if (finished)
|
|
finishMaybe(stream, state);
|
|
}
|
|
|
|
// Must force callback to be called on nextTick, so that we don't
|
|
// emit 'drain' before the write() consumer gets the 'false' return
|
|
// value, and has a chance to attach a 'drain' listener.
|
|
function onwriteDrain(stream, state) {
|
|
if (state.length === 0 && state.needDrain) {
|
|
state.needDrain = false;
|
|
stream.emit('drain');
|
|
}
|
|
}
|
|
|
|
|
|
// if there's something in the buffer waiting, then process it
|
|
function clearBuffer(stream, state) {
|
|
state.bufferProcessing = true;
|
|
|
|
for (var c = 0; c < state.buffer.length; c++) {
|
|
var entry = state.buffer[c];
|
|
var chunk = entry.chunk;
|
|
var encoding = entry.encoding;
|
|
var cb = entry.callback;
|
|
var len = state.objectMode ? 1 : chunk.length;
|
|
|
|
doWrite(stream, state, len, chunk, encoding, cb);
|
|
|
|
// if we didn't call the onwrite immediately, then
|
|
// it means that we need to wait until it does.
|
|
// also, that means that the chunk and cb are currently
|
|
// being processed, so move the buffer counter past them.
|
|
if (state.writing) {
|
|
c++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
state.bufferProcessing = false;
|
|
if (c < state.buffer.length)
|
|
state.buffer = state.buffer.slice(c);
|
|
else
|
|
state.buffer.length = 0;
|
|
}
|
|
|
|
Writable.prototype._write = function(chunk, encoding, cb) {
|
|
cb(new Error('not implemented'));
|
|
};
|
|
|
|
Writable.prototype.end = function(chunk, encoding, cb) {
|
|
var state = this._writableState;
|
|
|
|
if (typeof chunk === 'function') {
|
|
cb = chunk;
|
|
chunk = null;
|
|
encoding = null;
|
|
} else if (typeof encoding === 'function') {
|
|
cb = encoding;
|
|
encoding = null;
|
|
}
|
|
|
|
if (typeof chunk !== 'undefined' && chunk !== null)
|
|
this.write(chunk, encoding);
|
|
|
|
// ignore unnecessary end() calls.
|
|
if (!state.ending && !state.finished)
|
|
endWritable(this, state, cb);
|
|
};
|
|
|
|
|
|
function needFinish(stream, state) {
|
|
return (state.ending &&
|
|
state.length === 0 &&
|
|
!state.finished &&
|
|
!state.writing);
|
|
}
|
|
|
|
function finishMaybe(stream, state) {
|
|
var need = needFinish(stream, state);
|
|
if (need) {
|
|
state.finished = true;
|
|
stream.emit('finish');
|
|
}
|
|
return need;
|
|
}
|
|
|
|
function endWritable(stream, state, cb) {
|
|
state.ending = true;
|
|
finishMaybe(stream, state);
|
|
if (cb) {
|
|
if (state.finished)
|
|
setImmediate(cb);
|
|
else
|
|
stream.once('finish', cb);
|
|
}
|
|
state.ended = true;
|
|
}
|
|
|
|
},{"./index.js":103,"buffer":80,"inherits":95,"process/browser.js":104}],109:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var Buffer = require('buffer').Buffer;
|
|
|
|
function assertEncoding(encoding) {
|
|
if (encoding && !Buffer.isEncoding(encoding)) {
|
|
throw new Error('Unknown encoding: ' + encoding);
|
|
}
|
|
}
|
|
|
|
var StringDecoder = exports.StringDecoder = function(encoding) {
|
|
this.encoding = (encoding || 'utf8').toLowerCase().replace(/[-_]/, '');
|
|
assertEncoding(encoding);
|
|
switch (this.encoding) {
|
|
case 'utf8':
|
|
// CESU-8 represents each of Surrogate Pair by 3-bytes
|
|
this.surrogateSize = 3;
|
|
break;
|
|
case 'ucs2':
|
|
case 'utf16le':
|
|
// UTF-16 represents each of Surrogate Pair by 2-bytes
|
|
this.surrogateSize = 2;
|
|
this.detectIncompleteChar = utf16DetectIncompleteChar;
|
|
break;
|
|
case 'base64':
|
|
// Base-64 stores 3 bytes in 4 chars, and pads the remainder.
|
|
this.surrogateSize = 3;
|
|
this.detectIncompleteChar = base64DetectIncompleteChar;
|
|
break;
|
|
default:
|
|
this.write = passThroughWrite;
|
|
return;
|
|
}
|
|
|
|
this.charBuffer = new Buffer(6);
|
|
this.charReceived = 0;
|
|
this.charLength = 0;
|
|
};
|
|
|
|
|
|
StringDecoder.prototype.write = function(buffer) {
|
|
var charStr = '';
|
|
var offset = 0;
|
|
|
|
// if our last write ended with an incomplete multibyte character
|
|
while (this.charLength) {
|
|
// determine how many remaining bytes this buffer has to offer for this char
|
|
var i = (buffer.length >= this.charLength - this.charReceived) ?
|
|
this.charLength - this.charReceived :
|
|
buffer.length;
|
|
|
|
// add the new bytes to the char buffer
|
|
buffer.copy(this.charBuffer, this.charReceived, offset, i);
|
|
this.charReceived += (i - offset);
|
|
offset = i;
|
|
|
|
if (this.charReceived < this.charLength) {
|
|
// still not enough chars in this buffer? wait for more ...
|
|
return '';
|
|
}
|
|
|
|
// get the character that was split
|
|
charStr = this.charBuffer.slice(0, this.charLength).toString(this.encoding);
|
|
|
|
// lead surrogate (D800-DBFF) is also the incomplete character
|
|
var charCode = charStr.charCodeAt(charStr.length - 1);
|
|
if (charCode >= 0xD800 && charCode <= 0xDBFF) {
|
|
this.charLength += this.surrogateSize;
|
|
charStr = '';
|
|
continue;
|
|
}
|
|
this.charReceived = this.charLength = 0;
|
|
|
|
// if there are no more bytes in this buffer, just emit our char
|
|
if (i == buffer.length) return charStr;
|
|
|
|
// otherwise cut off the characters end from the beginning of this buffer
|
|
buffer = buffer.slice(i, buffer.length);
|
|
break;
|
|
}
|
|
|
|
var lenIncomplete = this.detectIncompleteChar(buffer);
|
|
|
|
var end = buffer.length;
|
|
if (this.charLength) {
|
|
// buffer the incomplete character bytes we got
|
|
buffer.copy(this.charBuffer, 0, buffer.length - lenIncomplete, end);
|
|
this.charReceived = lenIncomplete;
|
|
end -= lenIncomplete;
|
|
}
|
|
|
|
charStr += buffer.toString(this.encoding, 0, end);
|
|
|
|
var end = charStr.length - 1;
|
|
var charCode = charStr.charCodeAt(end);
|
|
// lead surrogate (D800-DBFF) is also the incomplete character
|
|
if (charCode >= 0xD800 && charCode <= 0xDBFF) {
|
|
var size = this.surrogateSize;
|
|
this.charLength += size;
|
|
this.charReceived += size;
|
|
this.charBuffer.copy(this.charBuffer, size, 0, size);
|
|
this.charBuffer.write(charStr.charAt(charStr.length - 1), this.encoding);
|
|
return charStr.substring(0, end);
|
|
}
|
|
|
|
// or just emit the charStr
|
|
return charStr;
|
|
};
|
|
|
|
StringDecoder.prototype.detectIncompleteChar = function(buffer) {
|
|
// determine how many bytes we have to check at the end of this buffer
|
|
var i = (buffer.length >= 3) ? 3 : buffer.length;
|
|
|
|
// Figure out if one of the last i bytes of our buffer announces an
|
|
// incomplete char.
|
|
for (; i > 0; i--) {
|
|
var c = buffer[buffer.length - i];
|
|
|
|
// See http://en.wikipedia.org/wiki/UTF-8#Description
|
|
|
|
// 110XXXXX
|
|
if (i == 1 && c >> 5 == 0x06) {
|
|
this.charLength = 2;
|
|
break;
|
|
}
|
|
|
|
// 1110XXXX
|
|
if (i <= 2 && c >> 4 == 0x0E) {
|
|
this.charLength = 3;
|
|
break;
|
|
}
|
|
|
|
// 11110XXX
|
|
if (i <= 3 && c >> 3 == 0x1E) {
|
|
this.charLength = 4;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return i;
|
|
};
|
|
|
|
StringDecoder.prototype.end = function(buffer) {
|
|
var res = '';
|
|
if (buffer && buffer.length)
|
|
res = this.write(buffer);
|
|
|
|
if (this.charReceived) {
|
|
var cr = this.charReceived;
|
|
var buf = this.charBuffer;
|
|
var enc = this.encoding;
|
|
res += buf.slice(0, cr).toString(enc);
|
|
}
|
|
|
|
return res;
|
|
};
|
|
|
|
function passThroughWrite(buffer) {
|
|
return buffer.toString(this.encoding);
|
|
}
|
|
|
|
function utf16DetectIncompleteChar(buffer) {
|
|
var incomplete = this.charReceived = buffer.length % 2;
|
|
this.charLength = incomplete ? 2 : 0;
|
|
return incomplete;
|
|
}
|
|
|
|
function base64DetectIncompleteChar(buffer) {
|
|
var incomplete = this.charReceived = buffer.length % 3;
|
|
this.charLength = incomplete ? 3 : 0;
|
|
return incomplete;
|
|
}
|
|
|
|
},{"buffer":80}],110:[function(require,module,exports){
|
|
/*jshint strict:true node:true es5:true onevar:true laxcomma:true laxbreak:true eqeqeq:true immed:true latedef:true*/
|
|
(function () {
|
|
"use strict";
|
|
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var punycode = require('punycode');
|
|
|
|
exports.parse = urlParse;
|
|
exports.resolve = urlResolve;
|
|
exports.resolveObject = urlResolveObject;
|
|
exports.format = urlFormat;
|
|
|
|
// Reference: RFC 3986, RFC 1808, RFC 2396
|
|
|
|
// define these here so at least they only have to be
|
|
// compiled once on the first module load.
|
|
var protocolPattern = /^([a-z0-9.+-]+:)/i,
|
|
portPattern = /:[0-9]*$/,
|
|
|
|
// RFC 2396: characters reserved for delimiting URLs.
|
|
// We actually just auto-escape these.
|
|
delims = ['<', '>', '"', '`', ' ', '\r', '\n', '\t'],
|
|
|
|
// RFC 2396: characters not allowed for various reasons.
|
|
unwise = ['{', '}', '|', '\\', '^', '~', '`'].concat(delims),
|
|
|
|
// Allowed by RFCs, but cause of XSS attacks. Always escape these.
|
|
autoEscape = ['\''].concat(delims),
|
|
// Characters that are never ever allowed in a hostname.
|
|
// Note that any invalid chars are also handled, but these
|
|
// are the ones that are *expected* to be seen, so we fast-path
|
|
// them.
|
|
nonHostChars = ['%', '/', '?', ';', '#']
|
|
.concat(unwise).concat(autoEscape),
|
|
nonAuthChars = ['/', '@', '?', '#'].concat(delims),
|
|
hostnameMaxLen = 255,
|
|
hostnamePartPattern = /^[a-zA-Z0-9][a-z0-9A-Z_-]{0,62}$/,
|
|
hostnamePartStart = /^([a-zA-Z0-9][a-z0-9A-Z_-]{0,62})(.*)$/,
|
|
// protocols that can allow "unsafe" and "unwise" chars.
|
|
unsafeProtocol = {
|
|
'javascript': true,
|
|
'javascript:': true
|
|
},
|
|
// protocols that never have a hostname.
|
|
hostlessProtocol = {
|
|
'javascript': true,
|
|
'javascript:': true
|
|
},
|
|
// protocols that always have a path component.
|
|
pathedProtocol = {
|
|
'http': true,
|
|
'https': true,
|
|
'ftp': true,
|
|
'gopher': true,
|
|
'file': true,
|
|
'http:': true,
|
|
'ftp:': true,
|
|
'gopher:': true,
|
|
'file:': true
|
|
},
|
|
// protocols that always contain a // bit.
|
|
slashedProtocol = {
|
|
'http': true,
|
|
'https': true,
|
|
'ftp': true,
|
|
'gopher': true,
|
|
'file': true,
|
|
'http:': true,
|
|
'https:': true,
|
|
'ftp:': true,
|
|
'gopher:': true,
|
|
'file:': true
|
|
},
|
|
querystring = require('querystring');
|
|
|
|
function urlParse(url, parseQueryString, slashesDenoteHost) {
|
|
if (url && typeof(url) === 'object' && url.href) return url;
|
|
|
|
if (typeof url !== 'string') {
|
|
throw new TypeError("Parameter 'url' must be a string, not " + typeof url);
|
|
}
|
|
|
|
var out = {},
|
|
rest = url;
|
|
|
|
// trim before proceeding.
|
|
// This is to support parse stuff like " http://foo.com \n"
|
|
rest = rest.trim();
|
|
|
|
var proto = protocolPattern.exec(rest);
|
|
if (proto) {
|
|
proto = proto[0];
|
|
var lowerProto = proto.toLowerCase();
|
|
out.protocol = lowerProto;
|
|
rest = rest.substr(proto.length);
|
|
}
|
|
|
|
// figure out if it's got a host
|
|
// user@server is *always* interpreted as a hostname, and url
|
|
// resolution will treat //foo/bar as host=foo,path=bar because that's
|
|
// how the browser resolves relative URLs.
|
|
if (slashesDenoteHost || proto || rest.match(/^\/\/[^@\/]+@[^@\/]+/)) {
|
|
var slashes = rest.substr(0, 2) === '//';
|
|
if (slashes && !(proto && hostlessProtocol[proto])) {
|
|
rest = rest.substr(2);
|
|
out.slashes = true;
|
|
}
|
|
}
|
|
|
|
if (!hostlessProtocol[proto] &&
|
|
(slashes || (proto && !slashedProtocol[proto]))) {
|
|
// there's a hostname.
|
|
// the first instance of /, ?, ;, or # ends the host.
|
|
// don't enforce full RFC correctness, just be unstupid about it.
|
|
|
|
// If there is an @ in the hostname, then non-host chars *are* allowed
|
|
// to the left of the first @ sign, unless some non-auth character
|
|
// comes *before* the @-sign.
|
|
// URLs are obnoxious.
|
|
var atSign = rest.indexOf('@');
|
|
if (atSign !== -1) {
|
|
var auth = rest.slice(0, atSign);
|
|
|
|
// there *may be* an auth
|
|
var hasAuth = true;
|
|
for (var i = 0, l = nonAuthChars.length; i < l; i++) {
|
|
if (auth.indexOf(nonAuthChars[i]) !== -1) {
|
|
// not a valid auth. Something like http://foo.com/bar@baz/
|
|
hasAuth = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (hasAuth) {
|
|
// pluck off the auth portion.
|
|
out.auth = decodeURIComponent(auth);
|
|
rest = rest.substr(atSign + 1);
|
|
}
|
|
}
|
|
|
|
var firstNonHost = -1;
|
|
for (var i = 0, l = nonHostChars.length; i < l; i++) {
|
|
var index = rest.indexOf(nonHostChars[i]);
|
|
if (index !== -1 &&
|
|
(firstNonHost < 0 || index < firstNonHost)) firstNonHost = index;
|
|
}
|
|
|
|
if (firstNonHost !== -1) {
|
|
out.host = rest.substr(0, firstNonHost);
|
|
rest = rest.substr(firstNonHost);
|
|
} else {
|
|
out.host = rest;
|
|
rest = '';
|
|
}
|
|
|
|
// pull out port.
|
|
var p = parseHost(out.host);
|
|
var keys = Object.keys(p);
|
|
for (var i = 0, l = keys.length; i < l; i++) {
|
|
var key = keys[i];
|
|
out[key] = p[key];
|
|
}
|
|
|
|
// we've indicated that there is a hostname,
|
|
// so even if it's empty, it has to be present.
|
|
out.hostname = out.hostname || '';
|
|
|
|
// if hostname begins with [ and ends with ]
|
|
// assume that it's an IPv6 address.
|
|
var ipv6Hostname = out.hostname[0] === '[' &&
|
|
out.hostname[out.hostname.length - 1] === ']';
|
|
|
|
// validate a little.
|
|
if (out.hostname.length > hostnameMaxLen) {
|
|
out.hostname = '';
|
|
} else if (!ipv6Hostname) {
|
|
var hostparts = out.hostname.split(/\./);
|
|
for (var i = 0, l = hostparts.length; i < l; i++) {
|
|
var part = hostparts[i];
|
|
if (!part) continue;
|
|
if (!part.match(hostnamePartPattern)) {
|
|
var newpart = '';
|
|
for (var j = 0, k = part.length; j < k; j++) {
|
|
if (part.charCodeAt(j) > 127) {
|
|
// we replace non-ASCII char with a temporary placeholder
|
|
// we need this to make sure size of hostname is not
|
|
// broken by replacing non-ASCII by nothing
|
|
newpart += 'x';
|
|
} else {
|
|
newpart += part[j];
|
|
}
|
|
}
|
|
// we test again with ASCII char only
|
|
if (!newpart.match(hostnamePartPattern)) {
|
|
var validParts = hostparts.slice(0, i);
|
|
var notHost = hostparts.slice(i + 1);
|
|
var bit = part.match(hostnamePartStart);
|
|
if (bit) {
|
|
validParts.push(bit[1]);
|
|
notHost.unshift(bit[2]);
|
|
}
|
|
if (notHost.length) {
|
|
rest = '/' + notHost.join('.') + rest;
|
|
}
|
|
out.hostname = validParts.join('.');
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// hostnames are always lower case.
|
|
out.hostname = out.hostname.toLowerCase();
|
|
|
|
if (!ipv6Hostname) {
|
|
// IDNA Support: Returns a puny coded representation of "domain".
|
|
// It only converts the part of the domain name that
|
|
// has non ASCII characters. I.e. it dosent matter if
|
|
// you call it with a domain that already is in ASCII.
|
|
var domainArray = out.hostname.split('.');
|
|
var newOut = [];
|
|
for (var i = 0; i < domainArray.length; ++i) {
|
|
var s = domainArray[i];
|
|
newOut.push(s.match(/[^A-Za-z0-9_-]/) ?
|
|
'xn--' + punycode.encode(s) : s);
|
|
}
|
|
out.hostname = newOut.join('.');
|
|
}
|
|
|
|
out.host = (out.hostname || '') +
|
|
((out.port) ? ':' + out.port : '');
|
|
out.href += out.host;
|
|
|
|
// strip [ and ] from the hostname
|
|
if (ipv6Hostname) {
|
|
out.hostname = out.hostname.substr(1, out.hostname.length - 2);
|
|
if (rest[0] !== '/') {
|
|
rest = '/' + rest;
|
|
}
|
|
}
|
|
}
|
|
|
|
// now rest is set to the post-host stuff.
|
|
// chop off any delim chars.
|
|
if (!unsafeProtocol[lowerProto]) {
|
|
|
|
// First, make 100% sure that any "autoEscape" chars get
|
|
// escaped, even if encodeURIComponent doesn't think they
|
|
// need to be.
|
|
for (var i = 0, l = autoEscape.length; i < l; i++) {
|
|
var ae = autoEscape[i];
|
|
var esc = encodeURIComponent(ae);
|
|
if (esc === ae) {
|
|
esc = escape(ae);
|
|
}
|
|
rest = rest.split(ae).join(esc);
|
|
}
|
|
}
|
|
|
|
|
|
// chop off from the tail first.
|
|
var hash = rest.indexOf('#');
|
|
if (hash !== -1) {
|
|
// got a fragment string.
|
|
out.hash = rest.substr(hash);
|
|
rest = rest.slice(0, hash);
|
|
}
|
|
var qm = rest.indexOf('?');
|
|
if (qm !== -1) {
|
|
out.search = rest.substr(qm);
|
|
out.query = rest.substr(qm + 1);
|
|
if (parseQueryString) {
|
|
out.query = querystring.parse(out.query);
|
|
}
|
|
rest = rest.slice(0, qm);
|
|
} else if (parseQueryString) {
|
|
// no query string, but parseQueryString still requested
|
|
out.search = '';
|
|
out.query = {};
|
|
}
|
|
if (rest) out.pathname = rest;
|
|
if (slashedProtocol[proto] &&
|
|
out.hostname && !out.pathname) {
|
|
out.pathname = '/';
|
|
}
|
|
|
|
//to support http.request
|
|
if (out.pathname || out.search) {
|
|
out.path = (out.pathname ? out.pathname : '') +
|
|
(out.search ? out.search : '');
|
|
}
|
|
|
|
// finally, reconstruct the href based on what has been validated.
|
|
out.href = urlFormat(out);
|
|
return out;
|
|
}
|
|
|
|
// format a parsed object into a url string
|
|
function urlFormat(obj) {
|
|
// ensure it's an object, and not a string url.
|
|
// If it's an obj, this is a no-op.
|
|
// this way, you can call url_format() on strings
|
|
// to clean up potentially wonky urls.
|
|
if (typeof(obj) === 'string') obj = urlParse(obj);
|
|
|
|
var auth = obj.auth || '';
|
|
if (auth) {
|
|
auth = encodeURIComponent(auth);
|
|
auth = auth.replace(/%3A/i, ':');
|
|
auth += '@';
|
|
}
|
|
|
|
var protocol = obj.protocol || '',
|
|
pathname = obj.pathname || '',
|
|
hash = obj.hash || '',
|
|
host = false,
|
|
query = '';
|
|
|
|
if (obj.host !== undefined) {
|
|
host = auth + obj.host;
|
|
} else if (obj.hostname !== undefined) {
|
|
host = auth + (obj.hostname.indexOf(':') === -1 ?
|
|
obj.hostname :
|
|
'[' + obj.hostname + ']');
|
|
if (obj.port) {
|
|
host += ':' + obj.port;
|
|
}
|
|
}
|
|
|
|
if (obj.query && typeof obj.query === 'object' &&
|
|
Object.keys(obj.query).length) {
|
|
query = querystring.stringify(obj.query);
|
|
}
|
|
|
|
var search = obj.search || (query && ('?' + query)) || '';
|
|
|
|
if (protocol && protocol.substr(-1) !== ':') protocol += ':';
|
|
|
|
// only the slashedProtocols get the //. Not mailto:, xmpp:, etc.
|
|
// unless they had them to begin with.
|
|
if (obj.slashes ||
|
|
(!protocol || slashedProtocol[protocol]) && host !== false) {
|
|
host = '//' + (host || '');
|
|
if (pathname && pathname.charAt(0) !== '/') pathname = '/' + pathname;
|
|
} else if (!host) {
|
|
host = '';
|
|
}
|
|
|
|
if (hash && hash.charAt(0) !== '#') hash = '#' + hash;
|
|
if (search && search.charAt(0) !== '?') search = '?' + search;
|
|
|
|
return protocol + host + pathname + search + hash;
|
|
}
|
|
|
|
function urlResolve(source, relative) {
|
|
return urlFormat(urlResolveObject(source, relative));
|
|
}
|
|
|
|
function urlResolveObject(source, relative) {
|
|
if (!source) return relative;
|
|
|
|
source = urlParse(urlFormat(source), false, true);
|
|
relative = urlParse(urlFormat(relative), false, true);
|
|
|
|
// hash is always overridden, no matter what.
|
|
source.hash = relative.hash;
|
|
|
|
if (relative.href === '') {
|
|
source.href = urlFormat(source);
|
|
return source;
|
|
}
|
|
|
|
// hrefs like //foo/bar always cut to the protocol.
|
|
if (relative.slashes && !relative.protocol) {
|
|
relative.protocol = source.protocol;
|
|
//urlParse appends trailing / to urls like http://www.example.com
|
|
if (slashedProtocol[relative.protocol] &&
|
|
relative.hostname && !relative.pathname) {
|
|
relative.path = relative.pathname = '/';
|
|
}
|
|
relative.href = urlFormat(relative);
|
|
return relative;
|
|
}
|
|
|
|
if (relative.protocol && relative.protocol !== source.protocol) {
|
|
// if it's a known url protocol, then changing
|
|
// the protocol does weird things
|
|
// first, if it's not file:, then we MUST have a host,
|
|
// and if there was a path
|
|
// to begin with, then we MUST have a path.
|
|
// if it is file:, then the host is dropped,
|
|
// because that's known to be hostless.
|
|
// anything else is assumed to be absolute.
|
|
if (!slashedProtocol[relative.protocol]) {
|
|
relative.href = urlFormat(relative);
|
|
return relative;
|
|
}
|
|
source.protocol = relative.protocol;
|
|
if (!relative.host && !hostlessProtocol[relative.protocol]) {
|
|
var relPath = (relative.pathname || '').split('/');
|
|
while (relPath.length && !(relative.host = relPath.shift()));
|
|
if (!relative.host) relative.host = '';
|
|
if (!relative.hostname) relative.hostname = '';
|
|
if (relPath[0] !== '') relPath.unshift('');
|
|
if (relPath.length < 2) relPath.unshift('');
|
|
relative.pathname = relPath.join('/');
|
|
}
|
|
source.pathname = relative.pathname;
|
|
source.search = relative.search;
|
|
source.query = relative.query;
|
|
source.host = relative.host || '';
|
|
source.auth = relative.auth;
|
|
source.hostname = relative.hostname || relative.host;
|
|
source.port = relative.port;
|
|
//to support http.request
|
|
if (source.pathname !== undefined || source.search !== undefined) {
|
|
source.path = (source.pathname ? source.pathname : '') +
|
|
(source.search ? source.search : '');
|
|
}
|
|
source.slashes = source.slashes || relative.slashes;
|
|
source.href = urlFormat(source);
|
|
return source;
|
|
}
|
|
|
|
var isSourceAbs = (source.pathname && source.pathname.charAt(0) === '/'),
|
|
isRelAbs = (
|
|
relative.host !== undefined ||
|
|
relative.pathname && relative.pathname.charAt(0) === '/'
|
|
),
|
|
mustEndAbs = (isRelAbs || isSourceAbs ||
|
|
(source.host && relative.pathname)),
|
|
removeAllDots = mustEndAbs,
|
|
srcPath = source.pathname && source.pathname.split('/') || [],
|
|
relPath = relative.pathname && relative.pathname.split('/') || [],
|
|
psychotic = source.protocol &&
|
|
!slashedProtocol[source.protocol];
|
|
|
|
// if the url is a non-slashed url, then relative
|
|
// links like ../.. should be able
|
|
// to crawl up to the hostname, as well. This is strange.
|
|
// source.protocol has already been set by now.
|
|
// Later on, put the first path part into the host field.
|
|
if (psychotic) {
|
|
|
|
delete source.hostname;
|
|
delete source.port;
|
|
if (source.host) {
|
|
if (srcPath[0] === '') srcPath[0] = source.host;
|
|
else srcPath.unshift(source.host);
|
|
}
|
|
delete source.host;
|
|
if (relative.protocol) {
|
|
delete relative.hostname;
|
|
delete relative.port;
|
|
if (relative.host) {
|
|
if (relPath[0] === '') relPath[0] = relative.host;
|
|
else relPath.unshift(relative.host);
|
|
}
|
|
delete relative.host;
|
|
}
|
|
mustEndAbs = mustEndAbs && (relPath[0] === '' || srcPath[0] === '');
|
|
}
|
|
|
|
if (isRelAbs) {
|
|
// it's absolute.
|
|
source.host = (relative.host || relative.host === '') ?
|
|
relative.host : source.host;
|
|
source.hostname = (relative.hostname || relative.hostname === '') ?
|
|
relative.hostname : source.hostname;
|
|
source.search = relative.search;
|
|
source.query = relative.query;
|
|
srcPath = relPath;
|
|
// fall through to the dot-handling below.
|
|
} else if (relPath.length) {
|
|
// it's relative
|
|
// throw away the existing file, and take the new path instead.
|
|
if (!srcPath) srcPath = [];
|
|
srcPath.pop();
|
|
srcPath = srcPath.concat(relPath);
|
|
source.search = relative.search;
|
|
source.query = relative.query;
|
|
} else if ('search' in relative) {
|
|
// just pull out the search.
|
|
// like href='?foo'.
|
|
// Put this after the other two cases because it simplifies the booleans
|
|
if (psychotic) {
|
|
source.hostname = source.host = srcPath.shift();
|
|
//occationaly the auth can get stuck only in host
|
|
//this especialy happens in cases like
|
|
//url.resolveObject('mailto:local1@domain1', 'local2@domain2')
|
|
var authInHost = source.host && source.host.indexOf('@') > 0 ?
|
|
source.host.split('@') : false;
|
|
if (authInHost) {
|
|
source.auth = authInHost.shift();
|
|
source.host = source.hostname = authInHost.shift();
|
|
}
|
|
}
|
|
source.search = relative.search;
|
|
source.query = relative.query;
|
|
//to support http.request
|
|
if (source.pathname !== undefined || source.search !== undefined) {
|
|
source.path = (source.pathname ? source.pathname : '') +
|
|
(source.search ? source.search : '');
|
|
}
|
|
source.href = urlFormat(source);
|
|
return source;
|
|
}
|
|
if (!srcPath.length) {
|
|
// no path at all. easy.
|
|
// we've already handled the other stuff above.
|
|
delete source.pathname;
|
|
//to support http.request
|
|
if (!source.search) {
|
|
source.path = '/' + source.search;
|
|
} else {
|
|
delete source.path;
|
|
}
|
|
source.href = urlFormat(source);
|
|
return source;
|
|
}
|
|
// if a url ENDs in . or .., then it must get a trailing slash.
|
|
// however, if it ends in anything else non-slashy,
|
|
// then it must NOT get a trailing slash.
|
|
var last = srcPath.slice(-1)[0];
|
|
var hasTrailingSlash = (
|
|
(source.host || relative.host) && (last === '.' || last === '..') ||
|
|
last === '');
|
|
|
|
// strip single dots, resolve double dots to parent dir
|
|
// if the path tries to go above the root, `up` ends up > 0
|
|
var up = 0;
|
|
for (var i = srcPath.length; i >= 0; i--) {
|
|
last = srcPath[i];
|
|
if (last == '.') {
|
|
srcPath.splice(i, 1);
|
|
} else if (last === '..') {
|
|
srcPath.splice(i, 1);
|
|
up++;
|
|
} else if (up) {
|
|
srcPath.splice(i, 1);
|
|
up--;
|
|
}
|
|
}
|
|
|
|
// if the path is allowed to go above the root, restore leading ..s
|
|
if (!mustEndAbs && !removeAllDots) {
|
|
for (; up--; up) {
|
|
srcPath.unshift('..');
|
|
}
|
|
}
|
|
|
|
if (mustEndAbs && srcPath[0] !== '' &&
|
|
(!srcPath[0] || srcPath[0].charAt(0) !== '/')) {
|
|
srcPath.unshift('');
|
|
}
|
|
|
|
if (hasTrailingSlash && (srcPath.join('/').substr(-1) !== '/')) {
|
|
srcPath.push('');
|
|
}
|
|
|
|
var isAbsolute = srcPath[0] === '' ||
|
|
(srcPath[0] && srcPath[0].charAt(0) === '/');
|
|
|
|
// put the host back
|
|
if (psychotic) {
|
|
source.hostname = source.host = isAbsolute ? '' :
|
|
srcPath.length ? srcPath.shift() : '';
|
|
//occationaly the auth can get stuck only in host
|
|
//this especialy happens in cases like
|
|
//url.resolveObject('mailto:local1@domain1', 'local2@domain2')
|
|
var authInHost = source.host && source.host.indexOf('@') > 0 ?
|
|
source.host.split('@') : false;
|
|
if (authInHost) {
|
|
source.auth = authInHost.shift();
|
|
source.host = source.hostname = authInHost.shift();
|
|
}
|
|
}
|
|
|
|
mustEndAbs = mustEndAbs || (source.host && srcPath.length);
|
|
|
|
if (mustEndAbs && !isAbsolute) {
|
|
srcPath.unshift('');
|
|
}
|
|
|
|
source.pathname = srcPath.join('/');
|
|
//to support request.http
|
|
if (source.pathname !== undefined || source.search !== undefined) {
|
|
source.path = (source.pathname ? source.pathname : '') +
|
|
(source.search ? source.search : '');
|
|
}
|
|
source.auth = relative.auth || source.auth;
|
|
source.slashes = source.slashes || relative.slashes;
|
|
source.href = urlFormat(source);
|
|
return source;
|
|
}
|
|
|
|
function parseHost(host) {
|
|
var out = {};
|
|
var port = portPattern.exec(host);
|
|
if (port) {
|
|
port = port[0];
|
|
if (port !== ':') {
|
|
out.port = port.substr(1);
|
|
}
|
|
host = host.substr(0, host.length - port.length);
|
|
}
|
|
if (host) out.hostname = host;
|
|
return out;
|
|
}
|
|
|
|
}());
|
|
|
|
},{"punycode":98,"querystring":101}],111:[function(require,module,exports){
|
|
module.exports=require(78)
|
|
},{}],112:[function(require,module,exports){
|
|
module.exports=require(79)
|
|
},{"./support/isBuffer":111,"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"inherits":95}],"aXRuS6":[function(require,module,exports){
|
|
(function (Buffer){
|
|
function BufferPut () {
|
|
this.words = [];
|
|
this.len = 0;
|
|
};
|
|
module.exports = BufferPut;
|
|
|
|
BufferPut.prototype.put = function(buf) {
|
|
this.words.push({buffer: buf});
|
|
this.len += buf.length;
|
|
return this;
|
|
};
|
|
|
|
BufferPut.prototype.word8 = function(x) {
|
|
this.words.push({bytes: 1, value: x});
|
|
this.len += 1;
|
|
return this;
|
|
};
|
|
|
|
BufferPut.prototype.floatle = function(x) {
|
|
this.words.push({bytes: 'float', endian: 'little', value: x});
|
|
this.len += 4;
|
|
return this;
|
|
};
|
|
|
|
BufferPut.prototype.varint = function(i) {
|
|
if(i < 0xFD) {
|
|
this.word8(i);
|
|
} else if(i <= 1<<16) {
|
|
this.word8(0xFD);
|
|
this.word16le(i);
|
|
} else if(i <= 1<<32) {
|
|
this.word8(0xFE);
|
|
this.word32le(i);
|
|
} else {
|
|
this.word8(0xFF);
|
|
this.word64le(i);
|
|
}
|
|
};
|
|
|
|
[8, 16, 24, 32, 64].forEach(function(bits) {
|
|
BufferPut.prototype['word'+bits+'be'] = function(x) {
|
|
this.words.push({endian: 'big', bytes: bits / 8, value: x});
|
|
this.len += bits / 8;
|
|
return this;
|
|
};
|
|
|
|
BufferPut.prototype['word'+bits+'le'] = function(x) {
|
|
this.words.push({endian: 'little', bytes: bits / 8, value: x});
|
|
this.len += bits / 8;
|
|
return this;
|
|
};
|
|
});
|
|
|
|
BufferPut.prototype.pad = function(bytes) {
|
|
this.words.push({endian: 'big', bytes: bytes, value: 0});
|
|
this.len += bytes;
|
|
return this;
|
|
};
|
|
|
|
BufferPut.prototype.length = function() {
|
|
return this.len;
|
|
};
|
|
|
|
BufferPut.prototype.buffer = function () {
|
|
var buf = new Buffer(this.len);
|
|
var offset = 0;
|
|
this.words.forEach(function(word) {
|
|
if(word.buffer) {
|
|
word.buffer.copy(buf, offset, 0);
|
|
offset += word.buffer.length;
|
|
} else if(word.bytes == 'float') {
|
|
// s * f * 2^e
|
|
var v = Math.abs(word.value);
|
|
var s = (word.value >= 0) * 1;
|
|
var e = Math.ceil(Math.log(v) / Math.LN2);
|
|
var f = v / (1 << e);
|
|
|
|
// s:1, e:7, f:23
|
|
// [seeeeeee][efffffff][ffffffff][ffffffff]
|
|
buf[offset++] = (s << 7) & ~~(e / 2);
|
|
buf[offset++] = ((e & 1) << 7) & ~~(f / (1 << 16));
|
|
buf[offset++] = 0;
|
|
buf[offset++] = 0;
|
|
offset += 4;
|
|
} else {
|
|
var big = word.endian === 'big';
|
|
var ix = big ? [ (word.bytes - 1) * 8, -8 ] : [ 0, 8 ];
|
|
for(var i=ix[0]; big ? i >= 0 : i < word.bytes * 8; i += ix[1]) {
|
|
if(i >= 32) {
|
|
buf[offset++] = Math.floor(word.value / Math.pow(2, i)) & 0xff;
|
|
} else {
|
|
buf[offset++] = (word.value >> i) & 0xff;
|
|
}
|
|
}
|
|
}
|
|
});
|
|
return buf;
|
|
};
|
|
|
|
BufferPut.prototype.write = function(stream) {
|
|
stream.write(this.buffer());
|
|
};
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80}],"bufferput":[function(require,module,exports){
|
|
module.exports=require('aXRuS6');
|
|
},{}],"buffers":[function(require,module,exports){
|
|
module.exports=require('OBo3aV');
|
|
},{}],"OBo3aV":[function(require,module,exports){
|
|
(function (Buffer){
|
|
module.exports = Buffers;
|
|
|
|
function Buffers (bufs) {
|
|
if (!(this instanceof Buffers)) return new Buffers(bufs);
|
|
this.buffers = bufs || [];
|
|
this.length = this.buffers.reduce(function (size, buf) {
|
|
return size + buf.length
|
|
}, 0);
|
|
}
|
|
|
|
Buffers.prototype.push = function () {
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
if (!Buffer.isBuffer(arguments[i])) {
|
|
throw new TypeError('Tried to push a non-buffer');
|
|
}
|
|
}
|
|
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
var buf = arguments[i];
|
|
this.buffers.push(buf);
|
|
this.length += buf.length;
|
|
}
|
|
return this.length;
|
|
};
|
|
|
|
Buffers.prototype.unshift = function () {
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
if (!Buffer.isBuffer(arguments[i])) {
|
|
throw new TypeError('Tried to unshift a non-buffer');
|
|
}
|
|
}
|
|
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
var buf = arguments[i];
|
|
this.buffers.unshift(buf);
|
|
this.length += buf.length;
|
|
}
|
|
return this.length;
|
|
};
|
|
|
|
Buffers.prototype.copy = function (dst, dStart, start, end) {
|
|
return this.slice(start, end).copy(dst, dStart, 0, end - start);
|
|
};
|
|
|
|
Buffers.prototype.splice = function (i, howMany) {
|
|
var buffers = this.buffers;
|
|
var index = i >= 0 ? i : this.length - i;
|
|
var reps = [].slice.call(arguments, 2);
|
|
|
|
if (howMany === undefined) {
|
|
howMany = this.length - index;
|
|
}
|
|
else if (howMany > this.length - index) {
|
|
howMany = this.length - index;
|
|
}
|
|
|
|
for (var i = 0; i < reps.length; i++) {
|
|
this.length += reps[i].length;
|
|
}
|
|
|
|
var removed = new Buffers();
|
|
var bytes = 0;
|
|
|
|
var startBytes = 0;
|
|
for (
|
|
var ii = 0;
|
|
ii < buffers.length && startBytes + buffers[ii].length < index;
|
|
ii ++
|
|
) { startBytes += buffers[ii].length }
|
|
|
|
if (index - startBytes > 0) {
|
|
var start = index - startBytes;
|
|
|
|
if (start + howMany < buffers[ii].length) {
|
|
removed.push(buffers[ii].slice(start, start + howMany));
|
|
|
|
var orig = buffers[ii];
|
|
//var buf = new Buffer(orig.length - howMany);
|
|
var buf0 = new Buffer(start);
|
|
for (var i = 0; i < start; i++) {
|
|
buf0[i] = orig[i];
|
|
}
|
|
|
|
var buf1 = new Buffer(orig.length - start - howMany);
|
|
for (var i = start + howMany; i < orig.length; i++) {
|
|
buf1[ i - howMany - start ] = orig[i]
|
|
}
|
|
|
|
if (reps.length > 0) {
|
|
var reps_ = reps.slice();
|
|
reps_.unshift(buf0);
|
|
reps_.push(buf1);
|
|
buffers.splice.apply(buffers, [ ii, 1 ].concat(reps_));
|
|
ii += reps_.length;
|
|
reps = [];
|
|
}
|
|
else {
|
|
buffers.splice(ii, 1, buf0, buf1);
|
|
//buffers[ii] = buf;
|
|
ii += 2;
|
|
}
|
|
}
|
|
else {
|
|
removed.push(buffers[ii].slice(start));
|
|
buffers[ii] = buffers[ii].slice(0, start);
|
|
ii ++;
|
|
}
|
|
}
|
|
|
|
if (reps.length > 0) {
|
|
buffers.splice.apply(buffers, [ ii, 0 ].concat(reps));
|
|
ii += reps.length;
|
|
}
|
|
|
|
while (removed.length < howMany) {
|
|
var buf = buffers[ii];
|
|
var len = buf.length;
|
|
var take = Math.min(len, howMany - removed.length);
|
|
|
|
if (take === len) {
|
|
removed.push(buf);
|
|
buffers.splice(ii, 1);
|
|
}
|
|
else {
|
|
removed.push(buf.slice(0, take));
|
|
buffers[ii] = buffers[ii].slice(take);
|
|
}
|
|
}
|
|
|
|
this.length -= removed.length;
|
|
|
|
return removed;
|
|
};
|
|
|
|
Buffers.prototype.slice = function (i, j) {
|
|
var buffers = this.buffers;
|
|
if (j === undefined) j = this.length;
|
|
if (i === undefined) i = 0;
|
|
|
|
if (j > this.length) j = this.length;
|
|
|
|
var startBytes = 0;
|
|
for (
|
|
var si = 0;
|
|
si < buffers.length && startBytes + buffers[si].length <= i;
|
|
si ++
|
|
) { startBytes += buffers[si].length }
|
|
|
|
var target = new Buffer(j - i);
|
|
|
|
var ti = 0;
|
|
for (var ii = si; ti < j - i && ii < buffers.length; ii++) {
|
|
var len = buffers[ii].length;
|
|
|
|
var start = ti === 0 ? i - startBytes : 0;
|
|
var end = ti + len >= j - i
|
|
? Math.min(start + (j - i) - ti, len)
|
|
: len
|
|
;
|
|
|
|
buffers[ii].copy(target, ti, start, end);
|
|
ti += end - start;
|
|
}
|
|
|
|
return target;
|
|
};
|
|
|
|
Buffers.prototype.pos = function (i) {
|
|
if (i < 0 || i >= this.length) throw new Error('oob');
|
|
var l = i, bi = 0, bu = null;
|
|
for (;;) {
|
|
bu = this.buffers[bi];
|
|
if (l < bu.length) {
|
|
return {buf: bi, offset: l};
|
|
} else {
|
|
l -= bu.length;
|
|
}
|
|
bi++;
|
|
}
|
|
};
|
|
|
|
Buffers.prototype.get = function get (i) {
|
|
var pos = this.pos(i);
|
|
|
|
return this.buffers[pos.buf].get(pos.offset);
|
|
};
|
|
|
|
Buffers.prototype.set = function set (i, b) {
|
|
var pos = this.pos(i);
|
|
|
|
return this.buffers[pos.buf].set(pos.offset, b);
|
|
};
|
|
|
|
Buffers.prototype.indexOf = function (needle, offset) {
|
|
if ("string" === typeof needle) {
|
|
needle = new Buffer(needle);
|
|
} else if (needle instanceof Buffer) {
|
|
// already a buffer
|
|
} else {
|
|
throw new Error('Invalid type for a search string');
|
|
}
|
|
|
|
if (!needle.length) {
|
|
return 0;
|
|
}
|
|
|
|
if (!this.length) {
|
|
return -1;
|
|
}
|
|
|
|
var i = 0, j = 0, match = 0, mstart, pos = 0;
|
|
|
|
// start search from a particular point in the virtual buffer
|
|
if (offset) {
|
|
var p = this.pos(offset);
|
|
i = p.buf;
|
|
j = p.offset;
|
|
pos = offset;
|
|
}
|
|
|
|
// for each character in virtual buffer
|
|
for (;;) {
|
|
while (j >= this.buffers[i].length) {
|
|
j = 0;
|
|
i++;
|
|
|
|
if (i >= this.buffers.length) {
|
|
// search string not found
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
var char = this.buffers[i][j];
|
|
|
|
if (char == needle[match]) {
|
|
// keep track where match started
|
|
if (match == 0) {
|
|
mstart = {
|
|
i: i,
|
|
j: j,
|
|
pos: pos
|
|
};
|
|
}
|
|
match++;
|
|
if (match == needle.length) {
|
|
// full match
|
|
return mstart.pos;
|
|
}
|
|
} else if (match != 0) {
|
|
// a partial match ended, go back to match starting position
|
|
// this will continue the search at the next character
|
|
i = mstart.i;
|
|
j = mstart.j;
|
|
pos = mstart.pos;
|
|
match = 0;
|
|
}
|
|
|
|
j++;
|
|
pos++;
|
|
}
|
|
};
|
|
|
|
Buffers.prototype.toBuffer = function() {
|
|
return this.slice();
|
|
}
|
|
|
|
Buffers.prototype.toString = function(encoding, start, end) {
|
|
return this.slice(start, end).toString(encoding);
|
|
}
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80}],117:[function(require,module,exports){
|
|
/*
|
|
A JavaScript implementation of the SHA family of hashes, as
|
|
defined in FIPS PUB 180-2 as well as the corresponding HMAC implementation
|
|
as defined in FIPS PUB 198a
|
|
|
|
Copyright Brian Turek 2008-2013
|
|
Distributed under the BSD License
|
|
See http://caligatio.github.com/jsSHA/ for more information
|
|
|
|
Several functions taken from Paul Johnston
|
|
*/
|
|
(function(T){function z(a,c,b){var g=0,f=[0],h="",l=null,h=b||"UTF8";if("UTF8"!==h&&"UTF16"!==h)throw"encoding must be UTF8 or UTF16";if("HEX"===c){if(0!==a.length%2)throw"srcString of HEX type must be in byte increments";l=B(a);g=l.binLen;f=l.value}else if("ASCII"===c||"TEXT"===c)l=J(a,h),g=l.binLen,f=l.value;else if("B64"===c)l=K(a),g=l.binLen,f=l.value;else throw"inputFormat must be HEX, TEXT, ASCII, or B64";this.getHash=function(a,c,b,h){var l=null,d=f.slice(),n=g,p;3===arguments.length?"number"!==
|
|
typeof b&&(h=b,b=1):2===arguments.length&&(b=1);if(b!==parseInt(b,10)||1>b)throw"numRounds must a integer >= 1";switch(c){case "HEX":l=L;break;case "B64":l=M;break;default:throw"format must be HEX or B64";}if("SHA-1"===a)for(p=0;p<b;p++)d=y(d,n),n=160;else if("SHA-224"===a)for(p=0;p<b;p++)d=v(d,n,a),n=224;else if("SHA-256"===a)for(p=0;p<b;p++)d=v(d,n,a),n=256;else if("SHA-384"===a)for(p=0;p<b;p++)d=v(d,n,a),n=384;else if("SHA-512"===a)for(p=0;p<b;p++)d=v(d,n,a),n=512;else throw"Chosen SHA variant is not supported";
|
|
return l(d,N(h))};this.getHMAC=function(a,b,c,l,s){var d,n,p,m,w=[],x=[];d=null;switch(l){case "HEX":l=L;break;case "B64":l=M;break;default:throw"outputFormat must be HEX or B64";}if("SHA-1"===c)n=64,m=160;else if("SHA-224"===c)n=64,m=224;else if("SHA-256"===c)n=64,m=256;else if("SHA-384"===c)n=128,m=384;else if("SHA-512"===c)n=128,m=512;else throw"Chosen SHA variant is not supported";if("HEX"===b)d=B(a),p=d.binLen,d=d.value;else if("ASCII"===b||"TEXT"===b)d=J(a,h),p=d.binLen,d=d.value;else if("B64"===
|
|
b)d=K(a),p=d.binLen,d=d.value;else throw"inputFormat must be HEX, TEXT, ASCII, or B64";a=8*n;b=n/4-1;n<p/8?(d="SHA-1"===c?y(d,p):v(d,p,c),d[b]&=4294967040):n>p/8&&(d[b]&=4294967040);for(n=0;n<=b;n+=1)w[n]=d[n]^909522486,x[n]=d[n]^1549556828;c="SHA-1"===c?y(x.concat(y(w.concat(f),a+g)),a+m):v(x.concat(v(w.concat(f),a+g,c)),a+m,c);return l(c,N(s))}}function s(a,c){this.a=a;this.b=c}function J(a,c){var b=[],g,f=[],h=0,l;if("UTF8"===c)for(l=0;l<a.length;l+=1)for(g=a.charCodeAt(l),f=[],2048<g?(f[0]=224|
|
|
(g&61440)>>>12,f[1]=128|(g&4032)>>>6,f[2]=128|g&63):128<g?(f[0]=192|(g&1984)>>>6,f[1]=128|g&63):f[0]=g,g=0;g<f.length;g+=1)b[h>>>2]|=f[g]<<24-h%4*8,h+=1;else if("UTF16"===c)for(l=0;l<a.length;l+=1)b[h>>>2]|=a.charCodeAt(l)<<16-h%4*8,h+=2;return{value:b,binLen:8*h}}function B(a){var c=[],b=a.length,g,f;if(0!==b%2)throw"String of HEX type must be in byte increments";for(g=0;g<b;g+=2){f=parseInt(a.substr(g,2),16);if(isNaN(f))throw"String of HEX type contains invalid characters";c[g>>>3]|=f<<24-g%8*4}return{value:c,
|
|
binLen:4*b}}function K(a){var c=[],b=0,g,f,h,l,r;if(-1===a.search(/^[a-zA-Z0-9=+\/]+$/))throw"Invalid character in base-64 string";g=a.indexOf("=");a=a.replace(/\=/g,"");if(-1!==g&&g<a.length)throw"Invalid '=' found in base-64 string";for(f=0;f<a.length;f+=4){r=a.substr(f,4);for(h=l=0;h<r.length;h+=1)g="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".indexOf(r[h]),l|=g<<18-6*h;for(h=0;h<r.length-1;h+=1)c[b>>2]|=(l>>>16-8*h&255)<<24-b%4*8,b+=1}return{value:c,binLen:8*b}}function L(a,
|
|
c){var b="",g=4*a.length,f,h;for(f=0;f<g;f+=1)h=a[f>>>2]>>>8*(3-f%4),b+="0123456789abcdef".charAt(h>>>4&15)+"0123456789abcdef".charAt(h&15);return c.outputUpper?b.toUpperCase():b}function M(a,c){var b="",g=4*a.length,f,h,l;for(f=0;f<g;f+=3)for(l=(a[f>>>2]>>>8*(3-f%4)&255)<<16|(a[f+1>>>2]>>>8*(3-(f+1)%4)&255)<<8|a[f+2>>>2]>>>8*(3-(f+2)%4)&255,h=0;4>h;h+=1)b=8*f+6*h<=32*a.length?b+"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".charAt(l>>>6*(3-h)&63):b+c.b64Pad;return b}function N(a){var c=
|
|
{outputUpper:!1,b64Pad:"="};try{a.hasOwnProperty("outputUpper")&&(c.outputUpper=a.outputUpper),a.hasOwnProperty("b64Pad")&&(c.b64Pad=a.b64Pad)}catch(b){}if("boolean"!==typeof c.outputUpper)throw"Invalid outputUpper formatting option";if("string"!==typeof c.b64Pad)throw"Invalid b64Pad formatting option";return c}function U(a,c){return a<<c|a>>>32-c}function u(a,c){return a>>>c|a<<32-c}function t(a,c){var b=null,b=new s(a.a,a.b);return b=32>=c?new s(b.a>>>c|b.b<<32-c&4294967295,b.b>>>c|b.a<<32-c&4294967295):
|
|
new s(b.b>>>c-32|b.a<<64-c&4294967295,b.a>>>c-32|b.b<<64-c&4294967295)}function O(a,c){var b=null;return b=32>=c?new s(a.a>>>c,a.b>>>c|a.a<<32-c&4294967295):new s(0,a.a>>>c-32)}function V(a,c,b){return a^c^b}function P(a,c,b){return a&c^~a&b}function W(a,c,b){return new s(a.a&c.a^~a.a&b.a,a.b&c.b^~a.b&b.b)}function Q(a,c,b){return a&c^a&b^c&b}function X(a,c,b){return new s(a.a&c.a^a.a&b.a^c.a&b.a,a.b&c.b^a.b&b.b^c.b&b.b)}function Y(a){return u(a,2)^u(a,13)^u(a,22)}function Z(a){var c=t(a,28),b=t(a,
|
|
34);a=t(a,39);return new s(c.a^b.a^a.a,c.b^b.b^a.b)}function $(a){return u(a,6)^u(a,11)^u(a,25)}function aa(a){var c=t(a,14),b=t(a,18);a=t(a,41);return new s(c.a^b.a^a.a,c.b^b.b^a.b)}function ba(a){return u(a,7)^u(a,18)^a>>>3}function ca(a){var c=t(a,1),b=t(a,8);a=O(a,7);return new s(c.a^b.a^a.a,c.b^b.b^a.b)}function da(a){return u(a,17)^u(a,19)^a>>>10}function ea(a){var c=t(a,19),b=t(a,61);a=O(a,6);return new s(c.a^b.a^a.a,c.b^b.b^a.b)}function R(a,c){var b=(a&65535)+(c&65535);return((a>>>16)+(c>>>
|
|
16)+(b>>>16)&65535)<<16|b&65535}function fa(a,c,b,g){var f=(a&65535)+(c&65535)+(b&65535)+(g&65535);return((a>>>16)+(c>>>16)+(b>>>16)+(g>>>16)+(f>>>16)&65535)<<16|f&65535}function S(a,c,b,g,f){var h=(a&65535)+(c&65535)+(b&65535)+(g&65535)+(f&65535);return((a>>>16)+(c>>>16)+(b>>>16)+(g>>>16)+(f>>>16)+(h>>>16)&65535)<<16|h&65535}function ga(a,c){var b,g,f;b=(a.b&65535)+(c.b&65535);g=(a.b>>>16)+(c.b>>>16)+(b>>>16);f=(g&65535)<<16|b&65535;b=(a.a&65535)+(c.a&65535)+(g>>>16);g=(a.a>>>16)+(c.a>>>16)+(b>>>
|
|
16);return new s((g&65535)<<16|b&65535,f)}function ha(a,c,b,g){var f,h,l;f=(a.b&65535)+(c.b&65535)+(b.b&65535)+(g.b&65535);h=(a.b>>>16)+(c.b>>>16)+(b.b>>>16)+(g.b>>>16)+(f>>>16);l=(h&65535)<<16|f&65535;f=(a.a&65535)+(c.a&65535)+(b.a&65535)+(g.a&65535)+(h>>>16);h=(a.a>>>16)+(c.a>>>16)+(b.a>>>16)+(g.a>>>16)+(f>>>16);return new s((h&65535)<<16|f&65535,l)}function ia(a,c,b,g,f){var h,l,r;h=(a.b&65535)+(c.b&65535)+(b.b&65535)+(g.b&65535)+(f.b&65535);l=(a.b>>>16)+(c.b>>>16)+(b.b>>>16)+(g.b>>>16)+(f.b>>>
|
|
16)+(h>>>16);r=(l&65535)<<16|h&65535;h=(a.a&65535)+(c.a&65535)+(b.a&65535)+(g.a&65535)+(f.a&65535)+(l>>>16);l=(a.a>>>16)+(c.a>>>16)+(b.a>>>16)+(g.a>>>16)+(f.a>>>16)+(h>>>16);return new s((l&65535)<<16|h&65535,r)}function y(a,c){var b=[],g,f,h,l,r,s,u=P,t=V,v=Q,d=U,n=R,p,m,w=S,x,q=[1732584193,4023233417,2562383102,271733878,3285377520];a[c>>>5]|=128<<24-c%32;a[(c+65>>>9<<4)+15]=c;x=a.length;for(p=0;p<x;p+=16){g=q[0];f=q[1];h=q[2];l=q[3];r=q[4];for(m=0;80>m;m+=1)b[m]=16>m?a[m+p]:d(b[m-3]^b[m-8]^b[m-
|
|
14]^b[m-16],1),s=20>m?w(d(g,5),u(f,h,l),r,1518500249,b[m]):40>m?w(d(g,5),t(f,h,l),r,1859775393,b[m]):60>m?w(d(g,5),v(f,h,l),r,2400959708,b[m]):w(d(g,5),t(f,h,l),r,3395469782,b[m]),r=l,l=h,h=d(f,30),f=g,g=s;q[0]=n(g,q[0]);q[1]=n(f,q[1]);q[2]=n(h,q[2]);q[3]=n(l,q[3]);q[4]=n(r,q[4])}return q}function v(a,c,b){var g,f,h,l,r,t,u,v,z,d,n,p,m,w,x,q,y,C,D,E,F,G,H,I,e,A=[],B,k=[1116352408,1899447441,3049323471,3921009573,961987163,1508970993,2453635748,2870763221,3624381080,310598401,607225278,1426881987,
|
|
1925078388,2162078206,2614888103,3248222580,3835390401,4022224774,264347078,604807628,770255983,1249150122,1555081692,1996064986,2554220882,2821834349,2952996808,3210313671,3336571891,3584528711,113926993,338241895,666307205,773529912,1294757372,1396182291,1695183700,1986661051,2177026350,2456956037,2730485921,2820302411,3259730800,3345764771,3516065817,3600352804,4094571909,275423344,430227734,506948616,659060556,883997877,958139571,1322822218,1537002063,1747873779,1955562222,2024104815,2227730452,
|
|
2361852424,2428436474,2756734187,3204031479,3329325298];d=[3238371032,914150663,812702999,4144912697,4290775857,1750603025,1694076839,3204075428];f=[1779033703,3144134277,1013904242,2773480762,1359893119,2600822924,528734635,1541459225];if("SHA-224"===b||"SHA-256"===b)n=64,g=(c+65>>>9<<4)+15,w=16,x=1,e=Number,q=R,y=fa,C=S,D=ba,E=da,F=Y,G=$,I=Q,H=P,d="SHA-224"===b?d:f;else if("SHA-384"===b||"SHA-512"===b)n=80,g=(c+128>>>10<<5)+31,w=32,x=2,e=s,q=ga,y=ha,C=ia,D=ca,E=ea,F=Z,G=aa,I=X,H=W,k=[new e(k[0],
|
|
3609767458),new e(k[1],602891725),new e(k[2],3964484399),new e(k[3],2173295548),new e(k[4],4081628472),new e(k[5],3053834265),new e(k[6],2937671579),new e(k[7],3664609560),new e(k[8],2734883394),new e(k[9],1164996542),new e(k[10],1323610764),new e(k[11],3590304994),new e(k[12],4068182383),new e(k[13],991336113),new e(k[14],633803317),new e(k[15],3479774868),new e(k[16],2666613458),new e(k[17],944711139),new e(k[18],2341262773),new e(k[19],2007800933),new e(k[20],1495990901),new e(k[21],1856431235),
|
|
new e(k[22],3175218132),new e(k[23],2198950837),new e(k[24],3999719339),new e(k[25],766784016),new e(k[26],2566594879),new e(k[27],3203337956),new e(k[28],1034457026),new e(k[29],2466948901),new e(k[30],3758326383),new e(k[31],168717936),new e(k[32],1188179964),new e(k[33],1546045734),new e(k[34],1522805485),new e(k[35],2643833823),new e(k[36],2343527390),new e(k[37],1014477480),new e(k[38],1206759142),new e(k[39],344077627),new e(k[40],1290863460),new e(k[41],3158454273),new e(k[42],3505952657),
|
|
new e(k[43],106217008),new e(k[44],3606008344),new e(k[45],1432725776),new e(k[46],1467031594),new e(k[47],851169720),new e(k[48],3100823752),new e(k[49],1363258195),new e(k[50],3750685593),new e(k[51],3785050280),new e(k[52],3318307427),new e(k[53],3812723403),new e(k[54],2003034995),new e(k[55],3602036899),new e(k[56],1575990012),new e(k[57],1125592928),new e(k[58],2716904306),new e(k[59],442776044),new e(k[60],593698344),new e(k[61],3733110249),new e(k[62],2999351573),new e(k[63],3815920427),new e(3391569614,
|
|
3928383900),new e(3515267271,566280711),new e(3940187606,3454069534),new e(4118630271,4000239992),new e(116418474,1914138554),new e(174292421,2731055270),new e(289380356,3203993006),new e(460393269,320620315),new e(685471733,587496836),new e(852142971,1086792851),new e(1017036298,365543100),new e(1126000580,2618297676),new e(1288033470,3409855158),new e(1501505948,4234509866),new e(1607167915,987167468),new e(1816402316,1246189591)],d="SHA-384"===b?[new e(3418070365,d[0]),new e(1654270250,d[1]),new e(2438529370,
|
|
d[2]),new e(355462360,d[3]),new e(1731405415,d[4]),new e(41048885895,d[5]),new e(3675008525,d[6]),new e(1203062813,d[7])]:[new e(f[0],4089235720),new e(f[1],2227873595),new e(f[2],4271175723),new e(f[3],1595750129),new e(f[4],2917565137),new e(f[5],725511199),new e(f[6],4215389547),new e(f[7],327033209)];else throw"Unexpected error in SHA-2 implementation";a[c>>>5]|=128<<24-c%32;a[g]=c;B=a.length;for(p=0;p<B;p+=w){c=d[0];g=d[1];f=d[2];h=d[3];l=d[4];r=d[5];t=d[6];u=d[7];for(m=0;m<n;m+=1)A[m]=16>m?
|
|
new e(a[m*x+p],a[m*x+p+1]):y(E(A[m-2]),A[m-7],D(A[m-15]),A[m-16]),v=C(u,G(l),H(l,r,t),k[m],A[m]),z=q(F(c),I(c,g,f)),u=t,t=r,r=l,l=q(h,v),h=f,f=g,g=c,c=q(v,z);d[0]=q(c,d[0]);d[1]=q(g,d[1]);d[2]=q(f,d[2]);d[3]=q(h,d[3]);d[4]=q(l,d[4]);d[5]=q(r,d[5]);d[6]=q(t,d[6]);d[7]=q(u,d[7])}if("SHA-224"===b)a=[d[0],d[1],d[2],d[3],d[4],d[5],d[6]];else if("SHA-256"===b)a=d;else if("SHA-384"===b)a=[d[0].a,d[0].b,d[1].a,d[1].b,d[2].a,d[2].b,d[3].a,d[3].b,d[4].a,d[4].b,d[5].a,d[5].b];else if("SHA-512"===b)a=[d[0].a,
|
|
d[0].b,d[1].a,d[1].b,d[2].a,d[2].b,d[3].a,d[3].b,d[4].a,d[4].b,d[5].a,d[5].b,d[6].a,d[6].b,d[7].a,d[7].b];else throw"Unexpected error in SHA-2 implementation";return a}"function"===typeof define&&typeof define.amd?define(function(){return z}):"undefined"!==typeof exports?"undefined"!==typeof module&&module.exports?module.exports=exports=z:exports=z:T.jsSHA=z})(this);
|
|
|
|
},{}],118:[function(require,module,exports){
|
|
(function (Buffer){
|
|
/**
|
|
* @author Matthew Caruana Galizia <m@m.cg>
|
|
* @license MIT
|
|
* @copyright Copyright (c) 2013, Matthew Caruana Galizia
|
|
* @preserve
|
|
*
|
|
* Portions of this code are copyright (c) 2011 Valentin Háloiu, redistributed and modified under the following license (MIT).
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
/*jshint node:true*/
|
|
|
|
var net = require('net');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
var inherits = require('util').inherits;
|
|
|
|
var htons = require('network-byte-order').htons;
|
|
var ipv6 = require('ipv6').v6;
|
|
|
|
module.exports = exports = Socks5ClientSocket;
|
|
|
|
exports.createConnection = function(options) {
|
|
var socksSocket, socksHost, socksPort;
|
|
|
|
socksHost = options.socksHost || 'localhost';
|
|
socksPort = options.socksPort || 1080;
|
|
socksSocket = new Socks5ClientSocket(socksHost, socksPort);
|
|
|
|
return socksSocket.connect(options.port, options.host);
|
|
};
|
|
|
|
function Socks5ClientSocket(socksHost, socksPort) {
|
|
var self = this;
|
|
|
|
EventEmitter.call(self);
|
|
|
|
self.socket = new net.Socket();
|
|
self.socksHost = socksHost;
|
|
self.socksPort = socksPort;
|
|
|
|
self.socket.on('error', function(err) {
|
|
self.emit('error', err);
|
|
});
|
|
|
|
self.on('error', function(err) {
|
|
if (!self.socket.destroyed) {
|
|
self.socket.destroy();
|
|
}
|
|
});
|
|
}
|
|
|
|
inherits(Socks5ClientSocket, EventEmitter);
|
|
|
|
Socks5ClientSocket.prototype.setTimeout = function(msecs, callback) {
|
|
return this.socket.setTimeout(msecs, callback);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.setNoDelay = function() {
|
|
return this.socket.setNoDelay();
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.setKeepAlive = function(setting, msecs) {
|
|
return this.socket.setKeepAlive(setting, msecs);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.address = function() {
|
|
return this.socket.address();
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.pause = function() {
|
|
return this.socket.pause();
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.resume = function() {
|
|
return this.socket.resume();
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.end = function(data, encoding) {
|
|
return this.socket.end(data, encoding);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.destroy = function(exception) {
|
|
return this.socket.destroy(exception);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.destroySoon = function() {
|
|
var ret = this.socket.destroySoon();
|
|
|
|
this.writable = false; // node's http library asserts writable to be false after destroySoon
|
|
|
|
return ret;
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.setEncoding = function(encoding) {
|
|
return this.socket.setEncoding(encoding);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.write = function(data, arg1, arg2) {
|
|
return this.socket.write(data, arg1, arg2);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.connect = function(port, host) {
|
|
var self = this;
|
|
|
|
self.socket.connect(self.socksPort, self.socksHost, function() {
|
|
self.establishSocksConnection(host, port);
|
|
});
|
|
|
|
return self;
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.handleSocksConnectToHost = function() {
|
|
var self = this;
|
|
|
|
self.socket.on('close', function(hadError) {
|
|
self.emit('close', hadError);
|
|
});
|
|
|
|
self.socket.on('end', function() {
|
|
self.emit('end');
|
|
});
|
|
|
|
self.socket.on('data', function(data) {
|
|
self.emit('data', data);
|
|
});
|
|
|
|
self.socket._httpMessage = self._httpMessage;
|
|
self.socket.parser = self.parser;
|
|
self.socket.ondata = self.ondata;
|
|
self.writable = true;
|
|
self.readable = true;
|
|
self.emit('connect');
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.establishSocksConnection = function(host, port) {
|
|
var self = this;
|
|
|
|
self.authenticateWithSocks(function() {
|
|
self.connectSocksToHost(host, port, function() {
|
|
self.handleSocksConnectToHost();
|
|
});
|
|
});
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.authenticateWithSocks = function(cb) {
|
|
var request, self = this;
|
|
|
|
self.socket.ondata = function(d, start, end) {
|
|
var error;
|
|
|
|
if (end - start !== 2) {
|
|
error = new Error('SOCKS authentication failed. Unexpected number of bytes received.');
|
|
} else if (d[start] !== 0x05) {
|
|
error = new Error('SOCKS authentication failed. Unexpected SOCKS version number: ' + d[start] + '.');
|
|
} else if (d[start + 1] !== 0x00) {
|
|
error = new Error('SOCKS authentication failed. Unexpected SOCKS authentication method: ' + d[start+1] + '.');
|
|
}
|
|
|
|
if (error) {
|
|
self.emit('error', error);
|
|
return;
|
|
}
|
|
|
|
if (cb) {
|
|
cb();
|
|
}
|
|
};
|
|
|
|
request = new Buffer(3);
|
|
request[0] = 0x05; // SOCKS version
|
|
request[1] = 0x01; // number of authentication methods
|
|
request[2] = 0x00; // no authentication
|
|
self.socket.write(request);
|
|
};
|
|
|
|
Socks5ClientSocket.prototype.connectSocksToHost = function(host, port, cb) {
|
|
var buffer, request, self = this;
|
|
|
|
this.socket.ondata = function(d, start, end) {
|
|
var i, address, addressLength, error;
|
|
|
|
if (d[start] !== 0x05) {
|
|
error = new Error('SOCKS connection failed. Unexpected SOCKS version number: ' + d[start] + '.');
|
|
} else if (d[start + 1] !== 0x00) {
|
|
error = new Error('SOCKS connection failed. ' + getErrorMessage(d[start + 1]) + '.');
|
|
} else if (d[start + 2] !== 0x00) {
|
|
error = new Error('SOCKS connection failed. The reserved byte must be 0x00.');
|
|
}
|
|
|
|
if (error) {
|
|
self.emit('error', error);
|
|
return;
|
|
}
|
|
|
|
address = '';
|
|
addressLength = 0;
|
|
|
|
switch (d[start + 3]) {
|
|
case 1:
|
|
address = d[start + 4] + '.' + d[start + 5] + '.' + d[start + 6] + '.' + d[start + 7];
|
|
addressLength = 4;
|
|
break;
|
|
case 3:
|
|
addressLength = d[start + 4] + 1;
|
|
for (i = start + 5; i < start + addressLength; i++) {
|
|
address += String.fromCharCode(d[i]);
|
|
}
|
|
break;
|
|
case 4:
|
|
addressLength = 16;
|
|
break;
|
|
default:
|
|
self.emit('error', new Error('SOCKS connection failed. Unknown addres type: ' + d[start + 3] + '.'));
|
|
return;
|
|
}
|
|
|
|
if (cb) {
|
|
cb();
|
|
}
|
|
};
|
|
|
|
buffer = [];
|
|
buffer.push(0x05); // SOCKS version
|
|
buffer.push(0x01); // Command code: establish a TCP/IP stream connection
|
|
buffer.push(0x00); // Reserved - myst be 0x00
|
|
|
|
switch (net.isIP(host)) {
|
|
case 0:
|
|
buffer.push(0x03);
|
|
parseDomainName(host, buffer);
|
|
break;
|
|
case 4:
|
|
buffer.push(0x01);
|
|
parseIPv4(host, buffer);
|
|
break;
|
|
case 6:
|
|
buffer.push(0x04);
|
|
if (parseIPv6(host, buffer) === false) {
|
|
self.emit('error', new Error('IPv6 host parsing failed. Invalid address.'));
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
|
|
parsePort(port, buffer);
|
|
|
|
request = new Buffer(buffer);
|
|
this.socket.write(request);
|
|
};
|
|
|
|
function parseIPv4(host, buffer) {
|
|
var i, ip, groups = host.split('.');
|
|
|
|
for (i = 0; i < groups.length; i++) {
|
|
ip = parseInt(groups[i], 10);
|
|
buffer.push(ip);
|
|
}
|
|
}
|
|
|
|
function parseIPv6(host, buffer) {
|
|
var i, b1, b2, part1, part2, address, groups;
|
|
|
|
address = new ipv6.Address(host).canonicalForm();
|
|
if (!address) {
|
|
return false;
|
|
}
|
|
|
|
groups = address.split(':');
|
|
|
|
for (i = 0; i < groups.length; i++) {
|
|
part1 = groups[i].substr(0,2);
|
|
part2 = groups[i].substr(2,2);
|
|
|
|
b1 = parseInt(part1, 16);
|
|
b2 = parseInt(part2, 16);
|
|
|
|
buffer.push(b1);
|
|
buffer.push(b2);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function parseDomainName(host, buffer) {
|
|
var i, c;
|
|
|
|
buffer.push(host.length);
|
|
for (i = 0; i < host.length; i++) {
|
|
c = host.charCodeAt(i);
|
|
buffer.push(c);
|
|
}
|
|
}
|
|
|
|
function parsePort(port, buffer) {
|
|
htons(buffer, buffer.length, port);
|
|
}
|
|
|
|
function getErrorMessage(code) {
|
|
switch (code) {
|
|
case 1:
|
|
return 'General SOCKS server failure';
|
|
case 2:
|
|
return 'Connection not allowed by ruleset';
|
|
case 3:
|
|
return 'Network unreachable';
|
|
case 4:
|
|
return 'Host unreachable';
|
|
case 5:
|
|
return 'Connection refused';
|
|
case 6:
|
|
return 'TTL expired';
|
|
case 7:
|
|
return 'Command not supported';
|
|
case 8:
|
|
return 'Address type not supported';
|
|
default:
|
|
return 'Unknown status code ' + code;
|
|
}
|
|
}
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80,"events":89,"ipv6":119,"net":76,"network-byte-order":123,"util":112}],119:[function(require,module,exports){
|
|
exports = module.exports = require('./ipv6.js');
|
|
},{"./ipv6.js":120}],120:[function(require,module,exports){
|
|
if (typeof exports !== 'undefined') {
|
|
var sprintf = require('sprintf').sprintf;
|
|
var BigInteger = require('./lib/node/bigint').BigInteger;
|
|
}
|
|
|
|
var v4 = this.v4 = {};
|
|
var v6 = this.v6 = {};
|
|
|
|
v4.GROUPS = 4;
|
|
v6.GROUPS = 8;
|
|
|
|
v4.BITS = 32;
|
|
v6.BITS = 128;
|
|
|
|
v6.SCOPES = {
|
|
0: 'Reserved',
|
|
1: 'Interface local',
|
|
2: 'Link local',
|
|
4: 'Admin local',
|
|
5: 'Site local',
|
|
8: 'Organization local',
|
|
15: 'Global',
|
|
16: 'Reserved'
|
|
};
|
|
|
|
v4.RE_ADDRESS = /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/g;
|
|
v4.RE_SUBNET_STRING = /\/\d{1,2}$/;
|
|
|
|
v6.RE_BAD_CHARACTERS = /([^0-9a-f:\/%])/ig;
|
|
v6.RE_BAD_ADDRESS = /([0-9a-f]{5,}|:{3,}|[^:]:$|^:[^:]|\/$)/ig;
|
|
|
|
v6.RE_SUBNET_STRING = /\/\d{1,3}(?=%|$)/;
|
|
v6.RE_ZONE_STRING = /%.*$/;
|
|
|
|
v6.RE_URL = new RegExp(/([0-9a-f:]+)/);
|
|
v6.RE_URL_WITH_PORT = new RegExp(/\[([0-9a-f:]+)\]:([0-9]{1,5})/);
|
|
|
|
// Convenience functions
|
|
function map(array, fn) {
|
|
var results = [];
|
|
var i;
|
|
|
|
for (i = 0; i < array.length; i++) {
|
|
results.push(fn(array[i], i));
|
|
}
|
|
|
|
return results;
|
|
}
|
|
|
|
function repeatString(s, n) {
|
|
var result = '';
|
|
var i;
|
|
|
|
for (i = 0; i < n; i++) {
|
|
result += s;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
function addCommas(number) {
|
|
var r = /(\d+)(\d{3})/;
|
|
|
|
while (r.test(number)) {
|
|
number = number.replace(r, '$1,$2');
|
|
}
|
|
|
|
return number;
|
|
}
|
|
|
|
function spanLeadingZeroesSimple(group) {
|
|
return group.replace(/^(0+)/, '<span class="zero">$1</span>');
|
|
}
|
|
|
|
function spanLeadingZeroes4(n) {
|
|
n = n.replace(/^(0{1,})([1-9]+)$/, '<span class="parse-error">$1</span>$2');
|
|
n = n.replace(/^(0{1,})(0)$/, '<span class="parse-error">$1</span>$2');
|
|
|
|
return n;
|
|
}
|
|
|
|
function simpleRegularExpression(addressArray) {
|
|
var output = [];
|
|
var i;
|
|
|
|
for (i = 0; i < addressArray.length; i++) {
|
|
var segment = addressArray[i];
|
|
|
|
if (segment.length < 4) {
|
|
output.push(sprintf('0{0,%d}%s', 4 - segment.length, segment));
|
|
} else {
|
|
output.push(segment);
|
|
}
|
|
}
|
|
|
|
return output.join(':');
|
|
}
|
|
|
|
function zeroPad(s, n) {
|
|
return String(repeatString(0, n) + s).slice(n * -1);
|
|
}
|
|
|
|
function isInSubnet(address) {
|
|
// XXX: This is a hunch
|
|
if (this.subnetMask < address.subnetMask) {
|
|
return false;
|
|
}
|
|
|
|
if (this.mask(address.subnetMask) === address.mask()) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* Instantiates an IPv4 address
|
|
*/
|
|
v4.Address = function (address) {
|
|
this.valid = false;
|
|
this.address = address;
|
|
this.groups = v4.GROUPS;
|
|
|
|
this.v4 = true;
|
|
|
|
this.subnet = '/32';
|
|
this.subnetMask = 32;
|
|
|
|
var subnet = v4.RE_SUBNET_STRING.exec(address);
|
|
|
|
if (subnet) {
|
|
this.parsedSubnet = subnet[0].replace('/', '');
|
|
this.subnetMask = parseInt(this.parsedSubnet, 10);
|
|
this.subnet = '/' + this.subnetMask;
|
|
|
|
if (this.subnetMask < 0 || this.subnetMask > v4.BITS) {
|
|
this.valid = false;
|
|
this.error = "Invalid subnet mask.";
|
|
|
|
return;
|
|
}
|
|
|
|
address = address.replace(v4.RE_SUBNET_STRING, '');
|
|
}
|
|
|
|
this.addressMinusSuffix = address;
|
|
|
|
this.parsedAddress = this.parse(address);
|
|
};
|
|
|
|
/*
|
|
* Parses a v4 address
|
|
*/
|
|
v4.Address.prototype.parse = function (address) {
|
|
var groups = address.split('.');
|
|
|
|
if (address.match(v4.RE_ADDRESS)) {
|
|
this.valid = true;
|
|
} else {
|
|
this.error = 'Invalid IPv4 address.';
|
|
}
|
|
|
|
return groups;
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is valid
|
|
*/
|
|
v4.Address.prototype.isValid = function () {
|
|
return this.valid;
|
|
};
|
|
|
|
/*
|
|
* Returns the correct form of an address
|
|
*/
|
|
v4.Address.prototype.correctForm = function () {
|
|
return this.parsedAddress.map(function (part) {
|
|
return parseInt(part, 10);
|
|
}).join('.');
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is correct, false otherwise
|
|
*/
|
|
v4.Address.prototype.isCorrect = function () {
|
|
return this.addressMinusSuffix === this.correctForm() &&
|
|
(this.subnetMask === 32 ||
|
|
this.parsedSubnet === String(this.subnet.replace('/')));
|
|
};
|
|
|
|
/*
|
|
* Converts a hex string to an IPv4 address object
|
|
*/
|
|
v4.Address.fromHex = function (hex) {
|
|
var padded = zeroPad(hex.replace(/:/g, ''), 8);
|
|
var groups = [];
|
|
var i;
|
|
|
|
for (i = 0; i < 8; i += 2) {
|
|
var h = padded.slice(i, i + 2);
|
|
|
|
groups.push(parseInt(h, 16));
|
|
}
|
|
|
|
return new v4.Address(groups.join('.'));
|
|
};
|
|
|
|
/*
|
|
* Converts an integer into a IPv4 address object
|
|
*/
|
|
v4.Address.fromInteger = function (integer) {
|
|
return v4.Address.fromHex(integer.toString(16));
|
|
};
|
|
|
|
/*
|
|
* Converts an IPv4 address object to a hex string
|
|
*/
|
|
v4.Address.prototype.toHex = function () {
|
|
return this.parsedAddress.map(function (part) {
|
|
return sprintf('%02x', parseInt(part, 10));
|
|
}).join(':');
|
|
};
|
|
|
|
/*
|
|
* Converts an IPv4 address object to an array of bytes
|
|
*/
|
|
v4.Address.prototype.toArray = function () {
|
|
return this.parsedAddress.map(function (part) {
|
|
return parseInt(part, 10);
|
|
});
|
|
};
|
|
|
|
/*
|
|
* Converts an IPv4 address object to an IPv6 address group
|
|
*/
|
|
v4.Address.prototype.toV6Group = function () {
|
|
var output = [];
|
|
var i;
|
|
|
|
for (i = 0; i < v4.GROUPS; i += 2) {
|
|
var hex = sprintf('%02x%02x',
|
|
parseInt(this.parsedAddress[i], 10),
|
|
parseInt(this.parsedAddress[i + 1], 10));
|
|
|
|
output.push(sprintf('%x', parseInt(hex, 16)));
|
|
}
|
|
|
|
return output.join(':');
|
|
};
|
|
|
|
/*
|
|
* Returns the address as a BigInteger
|
|
*/
|
|
v4.Address.prototype.bigInteger = function () {
|
|
if (!this.valid) {
|
|
return;
|
|
}
|
|
|
|
return new BigInteger(map(this.parsedAddress, function (n) {
|
|
return sprintf("%02x", parseInt(n, 10));
|
|
}).join(''), 16);
|
|
};
|
|
|
|
/*
|
|
* The first address in the range given by this address' subnet.
|
|
* Often referred to as the Network Address.
|
|
*/
|
|
v4.Address.prototype.startAddress = function () {
|
|
var startAddress = new BigInteger(this.mask() + repeatString(0,
|
|
v4.BITS - this.subnetMask), 2);
|
|
|
|
return v4.Address.fromBigInteger(startAddress);
|
|
};
|
|
|
|
/*
|
|
* The last address in the range given by this address' subnet
|
|
* Often referred to as the Broadcast
|
|
*/
|
|
v4.Address.prototype.endAddress = function () {
|
|
var endAddress = new BigInteger(this.mask() + repeatString(1,
|
|
v4.BITS - this.subnetMask), 2);
|
|
|
|
return v4.Address.fromBigInteger(endAddress);
|
|
};
|
|
|
|
/*
|
|
* Converts a BigInteger to a v4 address object
|
|
*/
|
|
v4.Address.fromBigInteger = function (bigInteger) {
|
|
return v4.Address.fromInteger(parseInt(bigInteger.toString(), 10));
|
|
};
|
|
|
|
/*
|
|
* Returns the first n bits of the address, defaulting to the
|
|
* subnet mask
|
|
*/
|
|
v4.Address.prototype.mask = function (opt_mask) {
|
|
if (opt_mask === undefined) {
|
|
opt_mask = this.subnetMask;
|
|
}
|
|
|
|
return this.getBitsBase2(0, opt_mask);
|
|
};
|
|
|
|
/*
|
|
* Returns the bits in the given range as a base-2 string
|
|
*/
|
|
v4.Address.prototype.getBitsBase2 = function (start, end) {
|
|
return this.binaryZeroPad().slice(start, end);
|
|
};
|
|
|
|
/*
|
|
* Returns true if the given address is in the subnet of the current address
|
|
*/
|
|
v4.Address.prototype.isInSubnet = isInSubnet;
|
|
|
|
/*
|
|
* Returns a zero-padded base-2 string representation of the address
|
|
*/
|
|
v4.Address.prototype.binaryZeroPad = function () {
|
|
return zeroPad(this.bigInteger().toString(2), v4.BITS);
|
|
};
|
|
|
|
/*
|
|
* Instantiates an IPv6 address
|
|
*/
|
|
v6.Address = function (address, opt_groups) {
|
|
if (opt_groups === undefined) {
|
|
this.groups = v6.GROUPS;
|
|
} else {
|
|
this.groups = opt_groups;
|
|
}
|
|
|
|
this.v4 = false;
|
|
|
|
this.subnet = '/128';
|
|
this.subnetMask = 128;
|
|
|
|
this.zone = '';
|
|
|
|
this.address = address;
|
|
|
|
var subnet = v6.RE_SUBNET_STRING.exec(address);
|
|
|
|
if (subnet) {
|
|
this.parsedSubnet = subnet[0].replace('/', '');
|
|
this.subnetMask = parseInt(this.parsedSubnet, 10);
|
|
this.subnet = '/' + this.subnetMask;
|
|
|
|
if (isNaN(this.subnetMask) ||
|
|
this.subnetMask < 0 ||
|
|
this.subnetMask > v6.BITS) {
|
|
this.valid = false;
|
|
this.error = "Invalid subnet mask.";
|
|
|
|
return;
|
|
}
|
|
|
|
address = address.replace(v6.RE_SUBNET_STRING, '');
|
|
} else if (/\//.test(address)) {
|
|
this.valid = false;
|
|
this.error = "Invalid subnet mask.";
|
|
|
|
return;
|
|
}
|
|
|
|
var zone = v6.RE_ZONE_STRING.exec(address);
|
|
|
|
if (zone) {
|
|
this.zone = zone[0];
|
|
|
|
address = address.replace(v6.RE_ZONE_STRING, '');
|
|
}
|
|
|
|
this.addressMinusSuffix = address;
|
|
|
|
this.parsedAddress = this.parse(this.addressMinusSuffix);
|
|
};
|
|
|
|
/*
|
|
* Converts a BigInteger to a v6 address object
|
|
*/
|
|
v6.Address.fromBigInteger = function (bigInteger) {
|
|
var hex = zeroPad(bigInteger.toString(16), 32);
|
|
var groups = [];
|
|
var i;
|
|
|
|
for (i = 0; i < 8; i++) {
|
|
groups.push(hex.slice(i * 4, (i + 1) * 4));
|
|
}
|
|
|
|
return new v6.Address(groups.join(':'));
|
|
};
|
|
|
|
/*
|
|
* Converts a URL (optional port number) to an address object
|
|
*/
|
|
v6.Address.fromURL = function (url) {
|
|
var host;
|
|
var port;
|
|
var result;
|
|
|
|
// If we have brackets parse them and find a port
|
|
if (url.indexOf('[') !== -1 && url.indexOf(']') !== -1) {
|
|
result = v6.RE_URL_WITH_PORT.exec(url);
|
|
|
|
if (result === null) {
|
|
return {
|
|
error: 'failed to parse address with port',
|
|
address: null,
|
|
port: null
|
|
};
|
|
}
|
|
|
|
host = result[1];
|
|
port = result[2];
|
|
// If there's a URL extract the address
|
|
} else if (url.indexOf('/') !== -1) {
|
|
// Remove the protocol prefix
|
|
url = url.replace(/^[a-z0-9]+:\/\//, '');
|
|
|
|
// Parse the address
|
|
result = v6.RE_URL.exec(url);
|
|
|
|
if (result === null) {
|
|
return {
|
|
error: 'failed to parse address from URL',
|
|
address: null,
|
|
port: null
|
|
};
|
|
}
|
|
|
|
host = result[1];
|
|
// Otherwise just assign the URL to the host and let the library parse it
|
|
} else {
|
|
host = url;
|
|
}
|
|
|
|
// If there's a port convert it to an integer
|
|
if (port) {
|
|
port = parseInt(port, 10);
|
|
|
|
//squelch out of range ports
|
|
if (port < 0 || port > 65536) {
|
|
port = null;
|
|
}
|
|
} else {
|
|
// Standardize `undefined` to `null`
|
|
port = null;
|
|
}
|
|
|
|
return {
|
|
address: new v6.Address(host),
|
|
port: port
|
|
};
|
|
};
|
|
|
|
/*
|
|
* A helper function to compact an array
|
|
*/
|
|
v6.Address.compact = function (address, slice) {
|
|
var s1 = [];
|
|
var s2 = [];
|
|
var i;
|
|
|
|
for (i = 0; i < address.length; i++) {
|
|
if (i < slice[0]) {
|
|
s1.push(address[i]);
|
|
} else if (i > slice[1]) {
|
|
s2.push(address[i]);
|
|
}
|
|
}
|
|
|
|
return s1.concat(['compact']).concat(s2);
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is valid, false otherwise
|
|
*/
|
|
v6.Address.prototype.isValid = function () {
|
|
return this.valid;
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is correct, false otherwise
|
|
*/
|
|
v6.Address.prototype.isCorrect = function () {
|
|
return this.addressMinusSuffix === this.correctForm() &&
|
|
(this.subnetMask === 128 ||
|
|
this.parsedSubnet === String(this.subnet.replace('/')));
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is a link local address, false otherwise
|
|
*/
|
|
v6.Address.prototype.isLinkLocal = function () {
|
|
// Zeroes are required, i.e. we can't check isInSubnet with 'fe80::/10'
|
|
if (this.getBitsBase2(0, 64) ===
|
|
"1111111010000000000000000000000000000000000000000000000000000000") {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is in the canonical form, false otherwise
|
|
*/
|
|
v6.Address.prototype.isCanonical = function () {
|
|
return this.addressMinusSuffix === this.canonicalForm();
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is a multicast address, false otherwise
|
|
*/
|
|
v6.Address.prototype.isMulticast = function () {
|
|
return this.getType() === 'Multicast';
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is a v4-in-v6 address, false otherwise
|
|
*/
|
|
v6.Address.prototype.is4 = function () {
|
|
return this.v4;
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is a Teredo address, false otherwise
|
|
*/
|
|
v6.Address.prototype.isTeredo = function () {
|
|
if (this.isInSubnet(new v6.Address('2001::/32'))) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is a 6to4 address, false otherwise
|
|
*/
|
|
v6.Address.prototype.is6to4 = function () {
|
|
if (this.isInSubnet(new v6.Address('2002::/16'))) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
/*
|
|
* Returns true if the address is a loopback address, false otherwise
|
|
*/
|
|
v6.Address.prototype.isLoopback = function () {
|
|
return this.getType() === 'Loopback';
|
|
};
|
|
|
|
/*
|
|
* Returns the Microsoft UNC transcription of the address
|
|
*/
|
|
v6.Address.prototype.microsoftTranscription = function () {
|
|
return sprintf('%s.ipv6-literal.net',
|
|
this.correctForm().replace(/:/g, '-'));
|
|
};
|
|
|
|
/*
|
|
* Returns the address in link form with a default port of 80
|
|
*/
|
|
v6.Address.prototype.href = function (opt_port) {
|
|
if (opt_port === undefined) {
|
|
opt_port = '';
|
|
} else {
|
|
opt_port = sprintf(':%s', opt_port);
|
|
}
|
|
|
|
return sprintf('http://[%s]%s/', this.correctForm(), opt_port);
|
|
};
|
|
|
|
/*
|
|
* Returns the first n bits of the address, defaulting to the
|
|
* subnet mask
|
|
*/
|
|
v6.Address.prototype.mask = function (opt_mask) {
|
|
if (opt_mask === undefined) {
|
|
opt_mask = this.subnetMask;
|
|
}
|
|
|
|
return this.getBitsBase2(0, opt_mask);
|
|
};
|
|
|
|
/*
|
|
* Returns a link suitable for conveying the address via a URL hash
|
|
*/
|
|
v6.Address.prototype.link = function (options) {
|
|
if (!options) {
|
|
options = {};
|
|
}
|
|
|
|
if (options.className === undefined) {
|
|
options.className = '';
|
|
}
|
|
|
|
if (options.prefix === undefined) {
|
|
options.prefix = '/#address=';
|
|
}
|
|
|
|
if (options.v4 === undefined) {
|
|
options.v4 = false;
|
|
}
|
|
|
|
var formFunction = this.correctForm;
|
|
|
|
if (options.v4) {
|
|
formFunction = this.v4inv6;
|
|
}
|
|
|
|
if (options.className) {
|
|
return sprintf('<a href="%1$s%2$s" class="%3$s">%2$s</a>',
|
|
options.prefix, formFunction.call(this), options.className);
|
|
}
|
|
|
|
return sprintf('<a href="%1$s%2$s">%2$s</a>', options.prefix,
|
|
formFunction.call(this));
|
|
};
|
|
|
|
/*
|
|
* Returns the number of possible subnets of a given size in the address
|
|
*/
|
|
v6.Address.prototype.possibleAddresses = function (opt_subnetSize) {
|
|
if (opt_subnetSize === undefined) {
|
|
opt_subnetSize = 0;
|
|
}
|
|
|
|
return addCommas(new BigInteger('2', 10).pow((v6.BITS - this.subnetMask) -
|
|
(v6.BITS - opt_subnetSize)).toString(10));
|
|
};
|
|
|
|
/*
|
|
* Returns true if the given address is in the subnet of the current address
|
|
*/
|
|
v6.Address.prototype.isInSubnet = isInSubnet;
|
|
|
|
/*
|
|
* Create an IPv6-mapped address given an IPv4 address
|
|
*/
|
|
v6.Address.fromAddress4 = function (address4) {
|
|
return new v6.Address('::ffff:' + address4);
|
|
};
|
|
|
|
/*
|
|
* The first address in the range given by this address' subnet
|
|
*/
|
|
v6.Address.prototype.startAddress = function () {
|
|
var startAddress = new BigInteger(this.mask() + repeatString(0,
|
|
v6.BITS - this.subnetMask), 2);
|
|
|
|
return v6.Address.fromBigInteger(startAddress);
|
|
};
|
|
|
|
/*
|
|
* The last address in the range given by this address' subnet
|
|
*/
|
|
v6.Address.prototype.endAddress = function () {
|
|
var endAddress = new BigInteger(this.mask() + repeatString(1,
|
|
v6.BITS - this.subnetMask), 2);
|
|
|
|
return v6.Address.fromBigInteger(endAddress);
|
|
};
|
|
|
|
/*
|
|
* Returns the scope of the address
|
|
*/
|
|
v6.Address.prototype.getScope = function () {
|
|
var scope = v6.SCOPES[this.getBits(12, 16)];
|
|
|
|
if (this.getType() === "Global unicast") {
|
|
if (scope !== "Link local") {
|
|
scope = "Global";
|
|
}
|
|
}
|
|
|
|
return scope;
|
|
};
|
|
|
|
/*
|
|
* Returns the type of the address
|
|
*/
|
|
v6.Address.prototype.getType = function () {
|
|
// TODO: Refactor this
|
|
// TODO: Add ff0x::fb, etc. for multicast DNS
|
|
var TYPES = {
|
|
'ff01::1/128': 'Multicast (All nodes on this interface)',
|
|
'ff01::2/128': 'Multicast (All routers on this interface)',
|
|
'ff02::1/128': 'Multicast (All nodes on this link)',
|
|
'ff02::2/128': 'Multicast (All routers on this link)',
|
|
'ff05::2/128': 'Multicast (All routers in this site)',
|
|
'ff02::5/128': 'Multicast (OSPFv3 AllSPF routers)',
|
|
'ff02::6/128': 'Multicast (OSPFv3 AllDR routers)',
|
|
'ff02::9/128': 'Multicast (RIP routers)',
|
|
'ff02::a/128': 'Multicast (EIGRP routers)',
|
|
'ff02::d/128': 'Multicast (PIM routers)',
|
|
'ff02::16/128': 'Multicast (MLDv2 reports)',
|
|
'ff01::fb/128': 'Multicast (mDNSv6)',
|
|
'ff02::fb/128': 'Multicast (mDNSv6)',
|
|
'ff05::fb/128': 'Multicast (mDNSv6)',
|
|
'ff02::1:2/128': 'Multicast (All DHCP servers and relay agents on this link)',
|
|
'ff05::1:2/128': 'Multicast (All DHCP servers and relay agents in this site)',
|
|
'ff02::1:3/128': 'Multicast (All DHCP servers on this link)',
|
|
'ff05::1:3/128': 'Multicast (All DHCP servers in this site)',
|
|
'::/128': 'Unspecified',
|
|
'::1/128': 'Loopback',
|
|
'ff00::/8': 'Multicast',
|
|
'fe80::/10': 'Link-local unicast'
|
|
};
|
|
|
|
var type = 'Global unicast';
|
|
var p;
|
|
|
|
for (p in TYPES) {
|
|
if (TYPES.hasOwnProperty(p)) {
|
|
if (this.isInSubnet(new v6.Address(p))) {
|
|
type = TYPES[p];
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return type;
|
|
};
|
|
|
|
/*
|
|
* Returns the bits in the given range as a BigInteger
|
|
*/
|
|
v6.Address.prototype.getBits = function (start, end) {
|
|
return new BigInteger(this.getBitsBase2(start, end), 2);
|
|
};
|
|
|
|
/*
|
|
* Returns the bits in the given range as a base-2 string
|
|
*/
|
|
v6.Address.prototype.getBitsBase2 = function (start, end) {
|
|
return this.binaryZeroPad().slice(start, end);
|
|
};
|
|
|
|
/*
|
|
* Returns the bits in the given range as a base-16 string
|
|
*/
|
|
v6.Address.prototype.getBitsBase16 = function (start, end) {
|
|
var length = end - start;
|
|
|
|
if (length % 4 !== 0) {
|
|
return;
|
|
}
|
|
|
|
return zeroPad(this.getBits(start, end).toString(16), length / 4);
|
|
};
|
|
|
|
/*
|
|
* Returns the bits that are set past the subnet mask length
|
|
*/
|
|
v6.Address.prototype.getBitsPastSubnet = function () {
|
|
return this.getBitsBase2(this.subnetMask, v6.BITS);
|
|
};
|
|
|
|
/*
|
|
* Returns the string with each character contained in a <span>
|
|
*/
|
|
v6.Address.spanAll = function (s, opt_offset) {
|
|
if (opt_offset === undefined) {
|
|
opt_offset = 0;
|
|
}
|
|
|
|
var letters = s.split('');
|
|
|
|
return map(letters, function (n, i) {
|
|
return sprintf('<span class="digit value-%s position-%d">%s</span>', n,
|
|
i + opt_offset,
|
|
v6.Address.spanAllZeroes(n)); // XXX Use #base-2 .value-0 instead?
|
|
}).join('');
|
|
};
|
|
|
|
/*
|
|
* Returns the string with all zeroes contained in a <span>
|
|
*/
|
|
v6.Address.spanAllZeroes = function (s) {
|
|
return s.replace(/(0+)/g, '<span class="zero">$1</span>');
|
|
};
|
|
|
|
/*
|
|
* Returns the string with leading zeroes contained in a <span>
|
|
*/
|
|
v6.Address.spanLeadingZeroes = function (address) {
|
|
var groups = address.split(':');
|
|
|
|
groups = map(groups, function (g) {
|
|
return spanLeadingZeroesSimple(g);
|
|
});
|
|
|
|
return groups.join(':');
|
|
};
|
|
|
|
/*
|
|
* Groups an address
|
|
*/
|
|
v6.Address.simpleGroup = function (addressString, offset) {
|
|
var groups = addressString.split(':');
|
|
|
|
if (!offset) {
|
|
offset = 0;
|
|
}
|
|
|
|
groups = map(groups, function (g, i) {
|
|
if (/group-v4/.test(g)) {
|
|
return g;
|
|
}
|
|
|
|
return sprintf('<span class="hover-group group-%d">%s</span>',
|
|
i + offset,
|
|
spanLeadingZeroesSimple(g));
|
|
});
|
|
|
|
return groups.join(':');
|
|
};
|
|
|
|
/*
|
|
* Groups an address
|
|
*/
|
|
v6.Address.group = function (addressString) {
|
|
var address6 = new v6.Address(addressString);
|
|
var address4 = address6.address.match(v4.RE_ADDRESS);
|
|
var i;
|
|
|
|
if (address4) {
|
|
// The IPv4 case
|
|
var segments = address4[0].split('.');
|
|
|
|
address6.address = address6.address.replace(v4.RE_ADDRESS,
|
|
sprintf('<span class="hover-group group-v4 group-6">%s</span>' +
|
|
'.' +
|
|
'<span class="hover-group group-v4 group-7">%s</span>',
|
|
segments.slice(0, 2).join('.'),
|
|
segments.slice(2, 4).join('.')));
|
|
}
|
|
|
|
if (address6.elidedGroups === 0) {
|
|
// The simple case
|
|
return v6.Address.simpleGroup(address6.address);
|
|
}
|
|
|
|
// The elided case
|
|
var output = [];
|
|
|
|
var halves = address6.address.split('::');
|
|
|
|
if (halves[0].length) {
|
|
output.push(v6.Address.simpleGroup(halves[0]));
|
|
} else {
|
|
output.push('');
|
|
}
|
|
|
|
var classes = ['hover-group'];
|
|
|
|
for (i = address6.elisionBegin; i < address6.elisionBegin +
|
|
address6.elidedGroups; i++) {
|
|
classes.push(sprintf('group-%d', i));
|
|
}
|
|
|
|
output.push(sprintf('<span class="%s"></span>', classes.join(' ')));
|
|
|
|
if (halves[1].length) {
|
|
output.push(v6.Address.simpleGroup(halves[1], address6.elisionEnd));
|
|
} else {
|
|
output.push('');
|
|
}
|
|
|
|
return output.join(':');
|
|
};
|
|
|
|
/*
|
|
* Returns the reversed ip6.arpa form of the address
|
|
*/
|
|
v6.Address.prototype.reverseForm = function () {
|
|
var characters = Math.floor(this.subnetMask / 4);
|
|
|
|
var reversed = this.canonicalForm()
|
|
.replace(/:/g, '')
|
|
.split('')
|
|
.slice(0, characters)
|
|
.reverse()
|
|
.join('.');
|
|
|
|
if (characters > 0) {
|
|
return sprintf("%s.ip6.arpa.", reversed);
|
|
}
|
|
|
|
return 'ip6.arpa.';
|
|
};
|
|
|
|
/*
|
|
* Returns the correct form of the address
|
|
*/
|
|
v6.Address.prototype.correctForm = function () {
|
|
if (!this.parsedAddress) {
|
|
return;
|
|
}
|
|
|
|
var i;
|
|
var groups = [];
|
|
|
|
var zeroCounter = 0;
|
|
var zeroes = [];
|
|
|
|
for (i = 0; i < this.parsedAddress.length; i++) {
|
|
var value = parseInt(this.parsedAddress[i], 16);
|
|
|
|
if (value === 0) {
|
|
zeroCounter++;
|
|
}
|
|
|
|
if (value !== 0 && zeroCounter > 0) {
|
|
if (zeroCounter > 1) {
|
|
zeroes.push([i - zeroCounter, i - 1]);
|
|
}
|
|
|
|
zeroCounter = 0;
|
|
}
|
|
}
|
|
|
|
// Do we end with a string of zeroes?
|
|
if (zeroCounter > 1) {
|
|
zeroes.push([this.parsedAddress.length - zeroCounter,
|
|
this.parsedAddress.length - 1]);
|
|
}
|
|
|
|
var zeroLengths = map(zeroes, function (n) {
|
|
return (n[1] - n[0]) + 1;
|
|
});
|
|
|
|
if (zeroes.length > 0) {
|
|
var max = Math.max.apply(Math, zeroLengths);
|
|
var index = zeroLengths.indexOf(max);
|
|
|
|
groups = v6.Address.compact(this.parsedAddress, zeroes[index]);
|
|
} else {
|
|
groups = this.parsedAddress;
|
|
}
|
|
|
|
for (i = 0; i < groups.length; i++) {
|
|
if (groups[i] !== 'compact') {
|
|
groups[i] = parseInt(groups[i], 16).toString(16);
|
|
}
|
|
}
|
|
|
|
var correct = groups.join(':');
|
|
|
|
correct = correct.replace(/^compact$/, '::');
|
|
correct = correct.replace(/^compact|compact$/, ':');
|
|
correct = correct.replace(/compact/, '');
|
|
|
|
return correct;
|
|
};
|
|
|
|
/*
|
|
* Returns a zero-padded base-2 string representation of the address
|
|
*/
|
|
v6.Address.prototype.binaryZeroPad = function () {
|
|
return zeroPad(this.bigInteger().toString(2), v6.BITS);
|
|
};
|
|
|
|
// TODO: Improve the semantics of this helper function
|
|
v6.Address.prototype.parse4in6 = function (address) {
|
|
var groups = address.split(':');
|
|
var lastGroup = groups.slice(-1)[0];
|
|
|
|
var address4 = lastGroup.match(v4.RE_ADDRESS);
|
|
|
|
if (address4) {
|
|
var temp4 = new v4.Address(address4[0]);
|
|
|
|
for (var i = 0; i < temp4.groups; i++) {
|
|
if (/^0[0-9]+/.test(temp4.parsedAddress[i])) {
|
|
this.valid = false;
|
|
this.error = 'IPv4 addresses can not have leading zeroes.';
|
|
|
|
this.parseError = address.replace(v4.RE_ADDRESS,
|
|
map(temp4.parsedAddress, spanLeadingZeroes4).join('.'));
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
this.v4 = true;
|
|
|
|
groups[groups.length - 1] = temp4.toV6Group();
|
|
|
|
address = groups.join(':');
|
|
}
|
|
|
|
return address;
|
|
};
|
|
|
|
// TODO: Make private?
|
|
v6.Address.prototype.parse = function (address) {
|
|
address = this.parse4in6(address);
|
|
|
|
if (this.error) {
|
|
return;
|
|
}
|
|
|
|
var badCharacters = address.match(v6.RE_BAD_CHARACTERS);
|
|
|
|
if (badCharacters) {
|
|
this.valid = false;
|
|
this.error = sprintf("Bad character%s detected in address: %s",
|
|
badCharacters.length > 1 ? 's' : '', badCharacters.join(''));
|
|
|
|
this.parseError = address.replace(v6.RE_BAD_CHARACTERS,
|
|
'<span class="parse-error">$1</span>');
|
|
|
|
return;
|
|
}
|
|
|
|
var badAddress = address.match(v6.RE_BAD_ADDRESS);
|
|
|
|
if (badAddress) {
|
|
this.valid = false;
|
|
this.error = sprintf("Address failed regex: %s", badAddress.join(''));
|
|
|
|
this.parseError = address.replace(v6.RE_BAD_ADDRESS,
|
|
'<span class="parse-error">$1</span>');
|
|
|
|
return;
|
|
}
|
|
|
|
var groups = [];
|
|
|
|
var halves = address.split('::');
|
|
|
|
if (halves.length === 2) {
|
|
var first = halves[0].split(':');
|
|
var last = halves[1].split(':');
|
|
|
|
if (first.length === 1 &&
|
|
first[0] === '') {
|
|
first = [];
|
|
}
|
|
|
|
if (last.length === 1 &&
|
|
last[0] === '') {
|
|
last = [];
|
|
}
|
|
|
|
var remaining = this.groups - (first.length + last.length);
|
|
|
|
if (!remaining) {
|
|
this.valid = false;
|
|
this.error = "Error parsing groups";
|
|
|
|
return;
|
|
}
|
|
|
|
this.elidedGroups = remaining;
|
|
|
|
this.elisionBegin = first.length;
|
|
this.elisionEnd = first.length + this.elidedGroups;
|
|
|
|
first.forEach(function (group) {
|
|
groups.push(group);
|
|
});
|
|
|
|
for (var i = 0; i < remaining; i++) {
|
|
groups.push(0);
|
|
}
|
|
|
|
last.forEach(function (group) {
|
|
groups.push(group);
|
|
});
|
|
} else if (halves.length === 1) {
|
|
groups = address.split(':');
|
|
|
|
this.elidedGroups = 0;
|
|
} else {
|
|
this.valid = false;
|
|
this.error = "Too many :: groups found";
|
|
|
|
return;
|
|
}
|
|
|
|
groups = map(groups, function (g) {
|
|
return sprintf('%x', parseInt(g, 16));
|
|
});
|
|
|
|
if (groups.length !== this.groups) {
|
|
this.valid = false;
|
|
this.error = "Incorrect number of groups found";
|
|
|
|
return;
|
|
}
|
|
|
|
groups.forEach(function (group, i) {
|
|
if (groups.length > 4 && !this.v4) {
|
|
this.valid = false;
|
|
this.error = sprintf("Group %d is too long", i + 1);
|
|
|
|
return;
|
|
}
|
|
});
|
|
|
|
this.valid = true;
|
|
|
|
return groups;
|
|
};
|
|
|
|
/*
|
|
* Generate a regular expression string that can be used to find or validate all
|
|
* variations of this address.
|
|
*/
|
|
v6.Address.prototype.regularExpressionString = function (opt_subString) {
|
|
if (opt_subString === undefined) {
|
|
opt_subString = false;
|
|
}
|
|
|
|
var i;
|
|
var output = [];
|
|
|
|
var address6 = new v6.Address(this.correctForm());
|
|
|
|
if (address6.elidedGroups === 0) {
|
|
// The simple case
|
|
output = simpleRegularExpression(address6.parsedAddress);
|
|
} else if (address6.elidedGroups === 8) {
|
|
output.push('::|');
|
|
|
|
// TODO: Validate this
|
|
for (i = 0; i < address6.elidedGroups; i++) {
|
|
var pipe = '|';
|
|
|
|
if (i === address6.elidedGroups - 1) {
|
|
pipe = '';
|
|
}
|
|
|
|
output.push(sprintf('(0{1,4}:){%d}:%s', address6.elidedGroups, pipe));
|
|
}
|
|
} else {
|
|
// The elided case
|
|
|
|
// TODO: Allow sloppy elision
|
|
// TODO: Compute all possible elisions
|
|
var halves = address6.address.split('::');
|
|
|
|
if (halves[0].length) {
|
|
output = output.concat(simpleRegularExpression(halves[0].split(':')));
|
|
output.push(':');
|
|
}
|
|
|
|
output.push(sprintf('((0{1,4}:){%d}|:)', address6.elidedGroups));
|
|
|
|
if (halves[1].length) {
|
|
output = output.concat(simpleRegularExpression(halves[1].split(':')));
|
|
}
|
|
}
|
|
|
|
if (!opt_subString) {
|
|
output = [].concat('\\b', output, '\\b');
|
|
}
|
|
|
|
return output.join('');
|
|
};
|
|
|
|
/*
|
|
* Generate a regular expression that can be used to find or validate all
|
|
* variations of this address.
|
|
*/
|
|
v6.Address.prototype.regularExpression = function () {
|
|
return new RegExp(this.regularExpressionString(), 'i');
|
|
};
|
|
|
|
/*
|
|
* Returns the canonical form of the address
|
|
*/
|
|
v6.Address.prototype.canonicalForm = function () {
|
|
if (!this.valid) {
|
|
return;
|
|
}
|
|
|
|
return map(this.parsedAddress, function (n) {
|
|
return sprintf("%04x", parseInt(n, 16));
|
|
}).join(':');
|
|
};
|
|
|
|
/*
|
|
* Returns the decimal form of the address
|
|
*/
|
|
v6.Address.prototype.decimal = function () {
|
|
if (!this.valid) {
|
|
return;
|
|
}
|
|
|
|
return map(this.parsedAddress, function (n) {
|
|
return sprintf("%05d", parseInt(n, 16));
|
|
}).join(':');
|
|
};
|
|
|
|
/*
|
|
* Returns the address as a BigInteger
|
|
*/
|
|
v6.Address.prototype.bigInteger = function () {
|
|
if (!this.valid) {
|
|
return;
|
|
}
|
|
|
|
return new BigInteger(map(this.parsedAddress, function (n) {
|
|
return sprintf("%04x", parseInt(n, 16));
|
|
}).join(''), 16);
|
|
};
|
|
|
|
/*
|
|
* Returns the v4-in-v6 form of the address
|
|
*/
|
|
v6.Address.prototype.v4inv6 = function () {
|
|
var binary = this.binaryZeroPad().split('');
|
|
|
|
var address4 = v4.Address.fromHex(new BigInteger(binary.slice(96, 128)
|
|
.join(''), 2).toString(16));
|
|
var address6 = new v6.Address(this.parsedAddress.slice(0, 6).join(':'), 6);
|
|
|
|
var correct = address6.correctForm();
|
|
|
|
var infix = '';
|
|
|
|
if (!/:$/.test(correct)) {
|
|
infix = ':';
|
|
}
|
|
|
|
return address6.correctForm() + infix + address4.address;
|
|
};
|
|
|
|
/*
|
|
* Returns an object containing the Teredo properties of the address
|
|
*/
|
|
v6.Address.prototype.teredo = function () {
|
|
/*
|
|
- Bits 0 to 31 are set to the Teredo prefix (normally 2001:0000::/32).
|
|
- Bits 32 to 63 embed the primary IPv4 address of the Teredo server that
|
|
is used.
|
|
- Bits 64 to 79 can be used to define some flags. Currently only the
|
|
higher order bit is used; it is set to 1 if the Teredo client is
|
|
located behind a cone NAT, 0 otherwise. For Microsoft's Windows Vista
|
|
and Windows Server 2008 implementations, more bits are used. In those
|
|
implementations, the format for these 16 bits is "CRAAAAUG AAAAAAAA",
|
|
where "C" remains the "Cone" flag. The "R" bit is reserved for future
|
|
use. The "U" bit is for the Universal/Local flag (set to 0). The "G" bit
|
|
is Individual/Group flag (set to 0). The A bits are set to a 12-bit
|
|
randomly generated number chosen by the Teredo client to introduce
|
|
additional protection for the Teredo node against IPv6-based scanning
|
|
attacks.
|
|
- Bits 80 to 95 contains the obfuscated UDP port number. This is the
|
|
port number that is mapped by the NAT to the Teredo client with all
|
|
bits inverted.
|
|
- Bits 96 to 127 contains the obfuscated IPv4 address. This is the
|
|
public IPv4 address of the NAT with all bits inverted.
|
|
*/
|
|
|
|
var prefix = this.getBitsBase16(0, 32);
|
|
|
|
var udpPort = this.getBits(80, 96).xor(new BigInteger('ffff', 16)).toString();
|
|
|
|
var server4 = v4.Address.fromHex(this.getBitsBase16(32, 64));
|
|
var client4 = v4.Address.fromHex(this.getBits(96, 128)
|
|
.xor(new BigInteger('ffffffff', 16)).toString(16));
|
|
|
|
var flags = this.getBits(64, 80);
|
|
var flagsBase2 = this.getBitsBase2(64, 80);
|
|
|
|
var coneNat = flags.testBit(15);
|
|
var reserved = flags.testBit(14);
|
|
var groupIndividual = flags.testBit(8);
|
|
var universalLocal = flags.testBit(9);
|
|
var nonce = new BigInteger(flagsBase2.slice(2, 6) +
|
|
flagsBase2.slice(8, 16), 2).toString(10);
|
|
|
|
return {
|
|
prefix: sprintf('%s:%s', prefix.slice(0, 4), prefix.slice(4, 8)),
|
|
server4: server4.address,
|
|
client4: client4.address,
|
|
flags: flagsBase2,
|
|
coneNat: coneNat,
|
|
microsoft: {
|
|
reserved: reserved,
|
|
universalLocal: universalLocal,
|
|
groupIndividual: groupIndividual,
|
|
nonce: nonce
|
|
},
|
|
udpPort: udpPort
|
|
};
|
|
};
|
|
|
|
/*
|
|
* Returns an object containing the 6to4 properties of the address
|
|
*/
|
|
v6.Address.prototype.six2four = function () {
|
|
/*
|
|
- Bits 0 to 15 are set to the 6to4 prefix (2002::/16).
|
|
- Bits 16 to 48 embed the IPv4 address of the 6to4 gateway that is used.
|
|
*/
|
|
|
|
var prefix = this.getBitsBase16(0, 16);
|
|
|
|
var gateway = v4.Address.fromHex(this.getBitsBase16(16, 48));
|
|
|
|
return {
|
|
prefix: sprintf('%s', prefix.slice(0, 4)),
|
|
gateway: gateway.address
|
|
};
|
|
};
|
|
|
|
},{"./lib/node/bigint":121,"sprintf":122}],121:[function(require,module,exports){
|
|
/**
|
|
* copped from https://github.com/joyent/node/blob/master/deps/v8/benchmarks/crypto.js (under same license).
|
|
*
|
|
* Copyright (c) 2003-2005 Tom Wu
|
|
* All Rights Reserved.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
|
|
* WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
|
|
*
|
|
* IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,
|
|
* INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER
|
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF
|
|
* THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT
|
|
* OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*
|
|
* In addition, the following condition applies:
|
|
*
|
|
* All redistributions must retain an intact copy of this copyright notice
|
|
* and disclaimer.
|
|
*/
|
|
|
|
// V8 optimized constants.
|
|
var dbits = 26
|
|
var BI_DB = dbits;
|
|
var BI_DM = ((1<<dbits)-1);
|
|
var BI_DV = (1<<dbits);
|
|
|
|
var BI_FP = 52;
|
|
var BI_FV = Math.pow(2,BI_FP);
|
|
var BI_F1 = BI_FP-dbits;
|
|
var BI_F2 = 2*dbits-BI_FP;
|
|
|
|
BigInteger = module.exports.BigInteger = function(a,b,c) {
|
|
this.array = new Array();
|
|
if(a != null)
|
|
if("number" == typeof a) this.fromNumber(a,b,c);
|
|
else if(b == null && "string" != typeof a) this.fromString(a,256);
|
|
else this.fromString(a,b);
|
|
}
|
|
|
|
// return new, unset BigInteger
|
|
function nbi() { return new BigInteger(null); }
|
|
|
|
// am: Compute w_j += (x*this_i), propagate carries,
|
|
// c is initial carry, returns final carry.
|
|
// c < 3*dvalue, x < 2*dvalue, this_i < dvalue
|
|
// We need to select the fastest one that works in this environment.
|
|
|
|
// This is tailored to VMs with 2-bit tagging. It makes sure
|
|
// that all the computations stay within the 29 bits available.
|
|
function am4(i,x,w,j,c,n) {
|
|
var this_array = this.array;
|
|
var w_array = w.array;
|
|
|
|
var xl = x&0x1fff, xh = x>>13;
|
|
while(--n >= 0) {
|
|
var l = this_array[i]&0x1fff;
|
|
var h = this_array[i++]>>13;
|
|
var m = xh*l+h*xl;
|
|
l = xl*l+((m&0x1fff)<<13)+w_array[j]+c;
|
|
c = (l>>26)+(m>>13)+xh*h;
|
|
w_array[j++] = l&0x3ffffff;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
// Digit conversions
|
|
var BI_RM = "0123456789abcdefghijklmnopqrstuvwxyz";
|
|
var BI_RC = new Array();
|
|
var rr,vv;
|
|
rr = "0".charCodeAt(0);
|
|
for(vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;
|
|
rr = "a".charCodeAt(0);
|
|
for(vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
|
|
rr = "A".charCodeAt(0);
|
|
for(vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
|
|
|
|
function int2char(n) { return BI_RM.charAt(n); }
|
|
function intAt(s,i) {
|
|
var c = BI_RC[s.charCodeAt(i)];
|
|
return (c==null)?-1:c;
|
|
}
|
|
|
|
// (protected) copy this to r
|
|
function bnpCopyTo(r) {
|
|
var this_array = this.array;
|
|
var r_array = r.array;
|
|
|
|
for(var i = this.t-1; i >= 0; --i) r_array[i] = this_array[i];
|
|
r.t = this.t;
|
|
r.s = this.s;
|
|
}
|
|
|
|
// (protected) set from integer value x, -DV <= x < DV
|
|
function bnpFromInt(x) {
|
|
var this_array = this.array;
|
|
this.t = 1;
|
|
this.s = (x<0)?-1:0;
|
|
if(x > 0) this_array[0] = x;
|
|
else if(x < -1) this_array[0] = x+DV; // WTH is DV? BI_DV maybe?
|
|
else this.t = 0;
|
|
}
|
|
|
|
// return bigint initialized to value
|
|
function nbv(i) { var r = nbi(); r.fromInt(i); return r; }
|
|
|
|
// (protected) set from string and radix
|
|
function bnpFromString(s,b) {
|
|
var this_array = this.array;
|
|
var k;
|
|
if(b == 16) k = 4;
|
|
else if(b == 8) k = 3;
|
|
else if(b == 256) k = 8; // byte array
|
|
else if(b == 2) k = 1;
|
|
else if(b == 32) k = 5;
|
|
else if(b == 4) k = 2;
|
|
else { this.fromRadix(s,b); return; }
|
|
this.t = 0;
|
|
this.s = 0;
|
|
var i = s.length, mi = false, sh = 0;
|
|
while(--i >= 0) {
|
|
var x = (k==8)?s[i]&0xff:intAt(s,i);
|
|
if(x < 0) {
|
|
if(s.charAt(i) == "-") mi = true;
|
|
continue;
|
|
}
|
|
mi = false;
|
|
if(sh == 0)
|
|
this_array[this.t++] = x;
|
|
else if(sh+k > BI_DB) {
|
|
this_array[this.t-1] |= (x&((1<<(BI_DB-sh))-1))<<sh;
|
|
this_array[this.t++] = (x>>(BI_DB-sh));
|
|
}
|
|
else
|
|
this_array[this.t-1] |= x<<sh;
|
|
sh += k;
|
|
if(sh >= BI_DB) sh -= BI_DB;
|
|
}
|
|
if(k == 8 && (s[0]&0x80) != 0) {
|
|
this.s = -1;
|
|
if(sh > 0) this_array[this.t-1] |= ((1<<(BI_DB-sh))-1)<<sh;
|
|
}
|
|
this.clamp();
|
|
if(mi) BigInteger.ZERO.subTo(this,this);
|
|
}
|
|
|
|
// (protected) clamp off excess high words
|
|
function bnpClamp() {
|
|
var this_array = this.array;
|
|
var c = this.s&BI_DM;
|
|
while(this.t > 0 && this_array[this.t-1] == c) --this.t;
|
|
}
|
|
|
|
// (public) return string representation in given radix
|
|
function bnToString(b) {
|
|
var this_array = this.array;
|
|
if(this.s < 0) return "-"+this.negate().toString(b);
|
|
var k;
|
|
if(b == 16) k = 4;
|
|
else if(b == 8) k = 3;
|
|
else if(b == 2) k = 1;
|
|
else if(b == 32) k = 5;
|
|
else if(b == 4) k = 2;
|
|
else return this.toRadix(b);
|
|
var km = (1<<k)-1, d, m = false, r = "", i = this.t;
|
|
var p = BI_DB-(i*BI_DB)%k;
|
|
if(i-- > 0) {
|
|
if(p < BI_DB && (d = this_array[i]>>p) > 0) { m = true; r = int2char(d); }
|
|
while(i >= 0) {
|
|
if(p < k) {
|
|
d = (this_array[i]&((1<<p)-1))<<(k-p);
|
|
d |= this_array[--i]>>(p+=BI_DB-k);
|
|
}
|
|
else {
|
|
d = (this_array[i]>>(p-=k))&km;
|
|
if(p <= 0) { p += BI_DB; --i; }
|
|
}
|
|
if(d > 0) m = true;
|
|
if(m) r += int2char(d);
|
|
}
|
|
}
|
|
return m?r:"0";
|
|
}
|
|
|
|
// (public) -this
|
|
function bnNegate() { var r = nbi(); BigInteger.ZERO.subTo(this,r); return r; }
|
|
|
|
// (public) |this|
|
|
function bnAbs() { return (this.s<0)?this.negate():this; }
|
|
|
|
// (public) return + if this > a, - if this < a, 0 if equal
|
|
function bnCompareTo(a) {
|
|
var this_array = this.array;
|
|
var a_array = a.array;
|
|
|
|
var r = this.s-a.s;
|
|
if(r != 0) return r;
|
|
var i = this.t;
|
|
r = i-a.t;
|
|
if(r != 0) return r;
|
|
while(--i >= 0) if((r=this_array[i]-a_array[i]) != 0) return r;
|
|
return 0;
|
|
}
|
|
|
|
// returns bit length of the integer x
|
|
function nbits(x) {
|
|
var r = 1, t;
|
|
if((t=x>>>16) != 0) { x = t; r += 16; }
|
|
if((t=x>>8) != 0) { x = t; r += 8; }
|
|
if((t=x>>4) != 0) { x = t; r += 4; }
|
|
if((t=x>>2) != 0) { x = t; r += 2; }
|
|
if((t=x>>1) != 0) { x = t; r += 1; }
|
|
return r;
|
|
}
|
|
|
|
// (public) return the number of bits in "this"
|
|
function bnBitLength() {
|
|
var this_array = this.array;
|
|
if(this.t <= 0) return 0;
|
|
return BI_DB*(this.t-1)+nbits(this_array[this.t-1]^(this.s&BI_DM));
|
|
}
|
|
|
|
// (protected) r = this << n*DB
|
|
function bnpDLShiftTo(n,r) {
|
|
var this_array = this.array;
|
|
var r_array = r.array;
|
|
var i;
|
|
for(i = this.t-1; i >= 0; --i) r_array[i+n] = this_array[i];
|
|
for(i = n-1; i >= 0; --i) r_array[i] = 0;
|
|
r.t = this.t+n;
|
|
r.s = this.s;
|
|
}
|
|
|
|
// (protected) r = this >> n*DB
|
|
function bnpDRShiftTo(n,r) {
|
|
var this_array = this.array;
|
|
var r_array = r.array;
|
|
for(var i = n; i < this.t; ++i) r_array[i-n] = this_array[i];
|
|
r.t = Math.max(this.t-n,0);
|
|
r.s = this.s;
|
|
}
|
|
|
|
// (protected) r = this << n
|
|
function bnpLShiftTo(n,r) {
|
|
var this_array = this.array;
|
|
var r_array = r.array;
|
|
var bs = n%BI_DB;
|
|
var cbs = BI_DB-bs;
|
|
var bm = (1<<cbs)-1;
|
|
var ds = Math.floor(n/BI_DB), c = (this.s<<bs)&BI_DM, i;
|
|
for(i = this.t-1; i >= 0; --i) {
|
|
r_array[i+ds+1] = (this_array[i]>>cbs)|c;
|
|
c = (this_array[i]&bm)<<bs;
|
|
}
|
|
for(i = ds-1; i >= 0; --i) r_array[i] = 0;
|
|
r_array[ds] = c;
|
|
r.t = this.t+ds+1;
|
|
r.s = this.s;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = this >> n
|
|
function bnpRShiftTo(n,r) {
|
|
var this_array = this.array;
|
|
var r_array = r.array;
|
|
r.s = this.s;
|
|
var ds = Math.floor(n/BI_DB);
|
|
if(ds >= this.t) { r.t = 0; return; }
|
|
var bs = n%BI_DB;
|
|
var cbs = BI_DB-bs;
|
|
var bm = (1<<bs)-1;
|
|
r_array[0] = this_array[ds]>>bs;
|
|
for(var i = ds+1; i < this.t; ++i) {
|
|
r_array[i-ds-1] |= (this_array[i]&bm)<<cbs;
|
|
r_array[i-ds] = this_array[i]>>bs;
|
|
}
|
|
if(bs > 0) r_array[this.t-ds-1] |= (this.s&bm)<<cbs;
|
|
r.t = this.t-ds;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = this - a
|
|
function bnpSubTo(a,r) {
|
|
var this_array = this.array;
|
|
var r_array = r.array;
|
|
var a_array = a.array;
|
|
var i = 0, c = 0, m = Math.min(a.t,this.t);
|
|
while(i < m) {
|
|
c += this_array[i]-a_array[i];
|
|
r_array[i++] = c&BI_DM;
|
|
c >>= BI_DB;
|
|
}
|
|
if(a.t < this.t) {
|
|
c -= a.s;
|
|
while(i < this.t) {
|
|
c += this_array[i];
|
|
r_array[i++] = c&BI_DM;
|
|
c >>= BI_DB;
|
|
}
|
|
c += this.s;
|
|
}
|
|
else {
|
|
c += this.s;
|
|
while(i < a.t) {
|
|
c -= a_array[i];
|
|
r_array[i++] = c&BI_DM;
|
|
c >>= BI_DB;
|
|
}
|
|
c -= a.s;
|
|
}
|
|
r.s = (c<0)?-1:0;
|
|
if(c < -1) r_array[i++] = BI_DV+c;
|
|
else if(c > 0) r_array[i++] = c;
|
|
r.t = i;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = this * a, r != this,a (HAC 14.12)
|
|
// "this" should be the larger one if appropriate.
|
|
function bnpMultiplyTo(a,r) {
|
|
var this_array = this.array; // Unused local var?
|
|
var r_array = r.array;
|
|
var x = this.abs(), y = a.abs();
|
|
var y_array = y.array;
|
|
|
|
var i = x.t;
|
|
r.t = i+y.t;
|
|
while(--i >= 0) r_array[i] = 0;
|
|
for(i = 0; i < y.t; ++i) r_array[i+x.t] = x.am(0,y_array[i],r,i,0,x.t);
|
|
r.s = 0;
|
|
r.clamp();
|
|
if(this.s != a.s) BigInteger.ZERO.subTo(r,r);
|
|
}
|
|
|
|
// (protected) r = this^2, r != this (HAC 14.16)
|
|
function bnpSquareTo(r) {
|
|
var x = this.abs();
|
|
var x_array = x.array;
|
|
var r_array = r.array;
|
|
|
|
var i = r.t = 2*x.t;
|
|
while(--i >= 0) r_array[i] = 0;
|
|
for(i = 0; i < x.t-1; ++i) {
|
|
var c = x.am(i,x_array[i],r,2*i,0,1);
|
|
if((r_array[i+x.t]+=x.am(i+1,2*x_array[i],r,2*i+1,c,x.t-i-1)) >= BI_DV) {
|
|
r_array[i+x.t] -= BI_DV;
|
|
r_array[i+x.t+1] = 1;
|
|
}
|
|
}
|
|
if(r.t > 0) r_array[r.t-1] += x.am(i,x_array[i],r,2*i,0,1);
|
|
r.s = 0;
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)
|
|
// r != q, this != m. q or r may be null.
|
|
function bnpDivRemTo(m,q,r) {
|
|
var pm = m.abs();
|
|
if(pm.t <= 0) return;
|
|
var pt = this.abs();
|
|
if(pt.t < pm.t) {
|
|
if(q != null) q.fromInt(0);
|
|
if(r != null) this.copyTo(r);
|
|
return;
|
|
}
|
|
if(r == null) r = nbi();
|
|
var y = nbi(), ts = this.s, ms = m.s;
|
|
var pm_array = pm.array;
|
|
var nsh = BI_DB-nbits(pm_array[pm.t-1]); // normalize modulus
|
|
if(nsh > 0) { pm.lShiftTo(nsh,y); pt.lShiftTo(nsh,r); }
|
|
else { pm.copyTo(y); pt.copyTo(r); }
|
|
var ys = y.t;
|
|
|
|
var y_array = y.array;
|
|
var y0 = y_array[ys-1];
|
|
if(y0 == 0) return;
|
|
var yt = y0*(1<<BI_F1)+((ys>1)?y_array[ys-2]>>BI_F2:0);
|
|
var d1 = BI_FV/yt, d2 = (1<<BI_F1)/yt, e = 1<<BI_F2;
|
|
var i = r.t, j = i-ys, t = (q==null)?nbi():q;
|
|
y.dlShiftTo(j,t);
|
|
|
|
var r_array = r.array;
|
|
if(r.compareTo(t) >= 0) {
|
|
r_array[r.t++] = 1;
|
|
r.subTo(t,r);
|
|
}
|
|
BigInteger.ONE.dlShiftTo(ys,t);
|
|
t.subTo(y,y); // "negative" y so we can replace sub with am later
|
|
while(y.t < ys) y_array[y.t++] = 0;
|
|
while(--j >= 0) {
|
|
// Estimate quotient digit
|
|
var qd = (r_array[--i]==y0)?BI_DM:Math.floor(r_array[i]*d1+(r_array[i-1]+e)*d2);
|
|
if((r_array[i]+=y.am(0,qd,r,j,0,ys)) < qd) { // Try it out
|
|
y.dlShiftTo(j,t);
|
|
r.subTo(t,r);
|
|
while(r_array[i] < --qd) r.subTo(t,r);
|
|
}
|
|
}
|
|
if(q != null) {
|
|
r.drShiftTo(ys,q);
|
|
if(ts != ms) BigInteger.ZERO.subTo(q,q);
|
|
}
|
|
r.t = ys;
|
|
r.clamp();
|
|
if(nsh > 0) r.rShiftTo(nsh,r); // Denormalize remainder
|
|
if(ts < 0) BigInteger.ZERO.subTo(r,r);
|
|
}
|
|
|
|
// (public) this mod a
|
|
function bnMod(a) {
|
|
var r = nbi();
|
|
this.abs().divRemTo(a,null,r);
|
|
if(this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r,r);
|
|
return r;
|
|
}
|
|
|
|
// Modular reduction using "classic" algorithm
|
|
function Classic(m) { this.m = m; }
|
|
function cConvert(x) {
|
|
if(x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);
|
|
else return x;
|
|
}
|
|
function cRevert(x) { return x; }
|
|
function cReduce(x) { x.divRemTo(this.m,null,x); }
|
|
function cMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
|
|
function cSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
|
|
|
|
Classic.prototype.convert = cConvert;
|
|
Classic.prototype.revert = cRevert;
|
|
Classic.prototype.reduce = cReduce;
|
|
Classic.prototype.mulTo = cMulTo;
|
|
Classic.prototype.sqrTo = cSqrTo;
|
|
|
|
// (protected) return "-1/this % 2^DB"; useful for Mont. reduction
|
|
// justification:
|
|
// xy == 1 (mod m)
|
|
// xy = 1+km
|
|
// xy(2-xy) = (1+km)(1-km)
|
|
// x[y(2-xy)] = 1-k^2m^2
|
|
// x[y(2-xy)] == 1 (mod m^2)
|
|
// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2
|
|
// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.
|
|
// JS multiply "overflows" differently from C/C++, so care is needed here.
|
|
function bnpInvDigit() {
|
|
var this_array = this.array;
|
|
if(this.t < 1) return 0;
|
|
var x = this_array[0];
|
|
if((x&1) == 0) return 0;
|
|
var y = x&3; // y == 1/x mod 2^2
|
|
y = (y*(2-(x&0xf)*y))&0xf; // y == 1/x mod 2^4
|
|
y = (y*(2-(x&0xff)*y))&0xff; // y == 1/x mod 2^8
|
|
y = (y*(2-(((x&0xffff)*y)&0xffff)))&0xffff; // y == 1/x mod 2^16
|
|
// last step - calculate inverse mod DV directly;
|
|
// assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints
|
|
y = (y*(2-x*y%BI_DV))%BI_DV; // y == 1/x mod 2^dbits
|
|
// we really want the negative inverse, and -DV < y < DV
|
|
return (y>0)?BI_DV-y:-y;
|
|
}
|
|
|
|
// Montgomery reduction
|
|
function Montgomery(m) {
|
|
this.m = m;
|
|
this.mp = m.invDigit();
|
|
this.mpl = this.mp&0x7fff;
|
|
this.mph = this.mp>>15;
|
|
this.um = (1<<(BI_DB-15))-1;
|
|
this.mt2 = 2*m.t;
|
|
}
|
|
|
|
// xR mod m
|
|
function montConvert(x) {
|
|
var r = nbi();
|
|
x.abs().dlShiftTo(this.m.t,r);
|
|
r.divRemTo(this.m,null,r);
|
|
if(x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r,r);
|
|
return r;
|
|
}
|
|
|
|
// x/R mod m
|
|
function montRevert(x) {
|
|
var r = nbi();
|
|
x.copyTo(r);
|
|
this.reduce(r);
|
|
return r;
|
|
}
|
|
|
|
// x = x/R mod m (HAC 14.32)
|
|
function montReduce(x) {
|
|
var x_array = x.array;
|
|
while(x.t <= this.mt2) // pad x so am has enough room later
|
|
x_array[x.t++] = 0;
|
|
for(var i = 0; i < this.m.t; ++i) {
|
|
// faster way of calculating u0 = x[i]*mp mod DV
|
|
var j = x_array[i]&0x7fff;
|
|
var u0 = (j*this.mpl+(((j*this.mph+(x_array[i]>>15)*this.mpl)&this.um)<<15))&BI_DM;
|
|
// use am to combine the multiply-shift-add into one call
|
|
j = i+this.m.t;
|
|
x_array[j] += this.m.am(0,u0,x,i,0,this.m.t);
|
|
// propagate carry
|
|
while(x_array[j] >= BI_DV) { x_array[j] -= BI_DV; x_array[++j]++; }
|
|
}
|
|
x.clamp();
|
|
x.drShiftTo(this.m.t,x);
|
|
if(x.compareTo(this.m) >= 0) x.subTo(this.m,x);
|
|
}
|
|
|
|
// r = "x^2/R mod m"; x != r
|
|
function montSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
|
|
|
|
// r = "xy/R mod m"; x,y != r
|
|
function montMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
|
|
|
|
Montgomery.prototype.convert = montConvert;
|
|
Montgomery.prototype.revert = montRevert;
|
|
Montgomery.prototype.reduce = montReduce;
|
|
Montgomery.prototype.mulTo = montMulTo;
|
|
Montgomery.prototype.sqrTo = montSqrTo;
|
|
|
|
// (protected) true iff this is even
|
|
function bnpIsEven() {
|
|
var this_array = this.array;
|
|
return ((this.t>0)?(this_array[0]&1):this.s) == 0;
|
|
}
|
|
|
|
// (protected) this^e, e < 2^32, doing sqr and mul with "r" (HAC 14.79)
|
|
function bnpExp(e,z) {
|
|
if(e > 0xffffffff || e < 1) return BigInteger.ONE;
|
|
var r = nbi(), r2 = nbi(), g = z.convert(this), i = nbits(e)-1;
|
|
g.copyTo(r);
|
|
while(--i >= 0) {
|
|
z.sqrTo(r,r2);
|
|
if((e&(1<<i)) > 0) z.mulTo(r2,g,r);
|
|
else { var t = r; r = r2; r2 = t; }
|
|
}
|
|
return z.revert(r);
|
|
}
|
|
|
|
// (public) this^e % m, 0 <= e < 2^32
|
|
function bnModPowInt(e,m) {
|
|
var z;
|
|
if(e < 256 || m.isEven()) z = new Classic(m); else z = new Montgomery(m);
|
|
return this.exp(e,z);
|
|
}
|
|
|
|
// protected
|
|
BigInteger.prototype.copyTo = bnpCopyTo;
|
|
BigInteger.prototype.fromInt = bnpFromInt;
|
|
BigInteger.prototype.fromString = bnpFromString;
|
|
BigInteger.prototype.clamp = bnpClamp;
|
|
BigInteger.prototype.dlShiftTo = bnpDLShiftTo;
|
|
BigInteger.prototype.drShiftTo = bnpDRShiftTo;
|
|
BigInteger.prototype.lShiftTo = bnpLShiftTo;
|
|
BigInteger.prototype.rShiftTo = bnpRShiftTo;
|
|
BigInteger.prototype.subTo = bnpSubTo;
|
|
BigInteger.prototype.multiplyTo = bnpMultiplyTo;
|
|
BigInteger.prototype.squareTo = bnpSquareTo;
|
|
BigInteger.prototype.divRemTo = bnpDivRemTo;
|
|
BigInteger.prototype.invDigit = bnpInvDigit;
|
|
BigInteger.prototype.isEven = bnpIsEven;
|
|
BigInteger.prototype.exp = bnpExp;
|
|
|
|
// public
|
|
BigInteger.prototype.toString = bnToString;
|
|
BigInteger.prototype.negate = bnNegate;
|
|
BigInteger.prototype.abs = bnAbs;
|
|
BigInteger.prototype.compareTo = bnCompareTo;
|
|
BigInteger.prototype.bitLength = bnBitLength;
|
|
BigInteger.prototype.mod = bnMod;
|
|
BigInteger.prototype.modPowInt = bnModPowInt;
|
|
|
|
// "constants"
|
|
BigInteger.ZERO = nbv(0);
|
|
BigInteger.ONE = nbv(1);
|
|
// Copyright (c) 2005 Tom Wu
|
|
// All Rights Reserved.
|
|
// See "LICENSE" for details.
|
|
|
|
// Extended JavaScript BN functions, required for RSA private ops.
|
|
|
|
// (public)
|
|
function bnClone() { var r = nbi(); this.copyTo(r); return r; }
|
|
|
|
// (public) return value as integer
|
|
function bnIntValue() {
|
|
var this_array = this.array;
|
|
if(this.s < 0) {
|
|
if(this.t == 1) return this_array[0]-BI_DV;
|
|
else if(this.t == 0) return -1;
|
|
}
|
|
else if(this.t == 1) return this_array[0];
|
|
else if(this.t == 0) return 0;
|
|
// assumes 16 < DB < 32
|
|
return ((this_array[1]&((1<<(32-BI_DB))-1))<<BI_DB)|this_array[0];
|
|
}
|
|
|
|
// (public) return value as byte
|
|
function bnByteValue() {
|
|
var this_array = this.array;
|
|
return (this.t==0)?this.s:(this_array[0]<<24)>>24;
|
|
}
|
|
|
|
// (public) return value as short (assumes DB>=16)
|
|
function bnShortValue() {
|
|
var this_array = this.array;
|
|
return (this.t==0)?this.s:(this_array[0]<<16)>>16;
|
|
}
|
|
|
|
// (protected) return x s.t. r^x < DV
|
|
function bnpChunkSize(r) { return Math.floor(Math.LN2*BI_DB/Math.log(r)); }
|
|
|
|
// (public) 0 if this == 0, 1 if this > 0
|
|
function bnSigNum() {
|
|
var this_array = this.array;
|
|
if(this.s < 0) return -1;
|
|
else if(this.t <= 0 || (this.t == 1 && this_array[0] <= 0)) return 0;
|
|
else return 1;
|
|
}
|
|
|
|
// (protected) convert to radix string
|
|
function bnpToRadix(b) {
|
|
if(b == null) b = 10;
|
|
if(this.signum() == 0 || b < 2 || b > 36) return "0";
|
|
var cs = this.chunkSize(b);
|
|
var a = Math.pow(b,cs);
|
|
var d = nbv(a), y = nbi(), z = nbi(), r = "";
|
|
this.divRemTo(d,y,z);
|
|
while(y.signum() > 0) {
|
|
r = (a+z.intValue()).toString(b).substr(1) + r;
|
|
y.divRemTo(d,y,z);
|
|
}
|
|
return z.intValue().toString(b) + r;
|
|
}
|
|
|
|
// (protected) convert from radix string
|
|
function bnpFromRadix(s,b) {
|
|
this.fromInt(0);
|
|
if(b == null) b = 10;
|
|
var cs = this.chunkSize(b);
|
|
var d = Math.pow(b,cs), mi = false, j = 0, w = 0;
|
|
for(var i = 0; i < s.length; ++i) {
|
|
var x = intAt(s,i);
|
|
if(x < 0) {
|
|
if(s.charAt(i) == "-" && this.signum() == 0) mi = true;
|
|
continue;
|
|
}
|
|
w = b*w+x;
|
|
if(++j >= cs) {
|
|
this.dMultiply(d);
|
|
this.dAddOffset(w,0);
|
|
j = 0;
|
|
w = 0;
|
|
}
|
|
}
|
|
if(j > 0) {
|
|
this.dMultiply(Math.pow(b,j));
|
|
this.dAddOffset(w,0);
|
|
}
|
|
if(mi) BigInteger.ZERO.subTo(this,this);
|
|
}
|
|
|
|
// (protected) alternate constructor
|
|
function bnpFromNumber(a,b,c) {
|
|
if("number" == typeof b) {
|
|
// new BigInteger(int,int,RNG)
|
|
if(a < 2) this.fromInt(1);
|
|
else {
|
|
this.fromNumber(a,c);
|
|
if(!this.testBit(a-1)) // force MSB set
|
|
this.bitwiseTo(BigInteger.ONE.shiftLeft(a-1),op_or,this);
|
|
if(this.isEven()) this.dAddOffset(1,0); // force odd
|
|
while(!this.isProbablePrime(b)) {
|
|
this.dAddOffset(2,0);
|
|
if(this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a-1),this);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// todo: we don't need this.
|
|
// new BigInteger(int,RNG)
|
|
var x = new Array(), t = a&7;
|
|
x.length = (a>>3)+1;
|
|
b.nextBytes(x);
|
|
if(t > 0) x[0] &= ((1<<t)-1); else x[0] = 0;
|
|
this.fromString(x,256);
|
|
}
|
|
}
|
|
|
|
// (public) convert to bigendian byte array
|
|
function bnToByteArray() {
|
|
var this_array = this.array;
|
|
var i = this.t, r = new Array();
|
|
r[0] = this.s;
|
|
var p = BI_DB-(i*BI_DB)%8, d, k = 0;
|
|
if(i-- > 0) {
|
|
if(p < BI_DB && (d = this_array[i]>>p) != (this.s&BI_DM)>>p)
|
|
r[k++] = d|(this.s<<(BI_DB-p));
|
|
while(i >= 0) {
|
|
if(p < 8) {
|
|
d = (this_array[i]&((1<<p)-1))<<(8-p);
|
|
d |= this_array[--i]>>(p+=BI_DB-8);
|
|
}
|
|
else {
|
|
d = (this_array[i]>>(p-=8))&0xff;
|
|
if(p <= 0) { p += BI_DB; --i; }
|
|
}
|
|
if((d&0x80) != 0) d |= -256;
|
|
if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;
|
|
if(k > 0 || d != this.s) r[k++] = d;
|
|
}
|
|
}
|
|
return r;
|
|
}
|
|
|
|
function bnEquals(a) { return(this.compareTo(a)==0); }
|
|
function bnMin(a) { return(this.compareTo(a)<0)?this:a; }
|
|
function bnMax(a) { return(this.compareTo(a)>0)?this:a; }
|
|
|
|
// (protected) r = this op a (bitwise)
|
|
function bnpBitwiseTo(a,op,r) {
|
|
var this_array = this.array;
|
|
var a_array = a.array;
|
|
var r_array = r.array;
|
|
var i, f, m = Math.min(a.t,this.t);
|
|
for(i = 0; i < m; ++i) r_array[i] = op(this_array[i],a_array[i]);
|
|
if(a.t < this.t) {
|
|
f = a.s&BI_DM;
|
|
for(i = m; i < this.t; ++i) r_array[i] = op(this_array[i],f);
|
|
r.t = this.t;
|
|
}
|
|
else {
|
|
f = this.s&BI_DM;
|
|
for(i = m; i < a.t; ++i) r_array[i] = op(f,a_array[i]);
|
|
r.t = a.t;
|
|
}
|
|
r.s = op(this.s,a.s);
|
|
r.clamp();
|
|
}
|
|
|
|
// (public) this & a
|
|
function op_and(x,y) { return x&y; }
|
|
function bnAnd(a) { var r = nbi(); this.bitwiseTo(a,op_and,r); return r; }
|
|
|
|
// (public) this | a
|
|
function op_or(x,y) { return x|y; }
|
|
function bnOr(a) { var r = nbi(); this.bitwiseTo(a,op_or,r); return r; }
|
|
|
|
// (public) this ^ a
|
|
function op_xor(x,y) { return x^y; }
|
|
function bnXor(a) { var r = nbi(); this.bitwiseTo(a,op_xor,r); return r; }
|
|
|
|
// (public) this & ~a
|
|
function op_andnot(x,y) { return x&~y; }
|
|
function bnAndNot(a) { var r = nbi(); this.bitwiseTo(a,op_andnot,r); return r; }
|
|
|
|
// (public) ~this
|
|
function bnNot() {
|
|
var this_array = this.array;
|
|
var r = nbi();
|
|
var r_array = r.array;
|
|
|
|
for(var i = 0; i < this.t; ++i) r_array[i] = BI_DM&~this_array[i];
|
|
r.t = this.t;
|
|
r.s = ~this.s;
|
|
return r;
|
|
}
|
|
|
|
// (public) this << n
|
|
function bnShiftLeft(n) {
|
|
var r = nbi();
|
|
if(n < 0) this.rShiftTo(-n,r); else this.lShiftTo(n,r);
|
|
return r;
|
|
}
|
|
|
|
// (public) this >> n
|
|
function bnShiftRight(n) {
|
|
var r = nbi();
|
|
if(n < 0) this.lShiftTo(-n,r); else this.rShiftTo(n,r);
|
|
return r;
|
|
}
|
|
|
|
// return index of lowest 1-bit in x, x < 2^31
|
|
function lbit(x) {
|
|
if(x == 0) return -1;
|
|
var r = 0;
|
|
if((x&0xffff) == 0) { x >>= 16; r += 16; }
|
|
if((x&0xff) == 0) { x >>= 8; r += 8; }
|
|
if((x&0xf) == 0) { x >>= 4; r += 4; }
|
|
if((x&3) == 0) { x >>= 2; r += 2; }
|
|
if((x&1) == 0) ++r;
|
|
return r;
|
|
}
|
|
|
|
// (public) returns index of lowest 1-bit (or -1 if none)
|
|
function bnGetLowestSetBit() {
|
|
var this_array = this.array;
|
|
for(var i = 0; i < this.t; ++i)
|
|
if(this_array[i] != 0) return i*BI_DB+lbit(this_array[i]);
|
|
if(this.s < 0) return this.t*BI_DB;
|
|
return -1;
|
|
}
|
|
|
|
// return number of 1 bits in x
|
|
function cbit(x) {
|
|
var r = 0;
|
|
while(x != 0) { x &= x-1; ++r; }
|
|
return r;
|
|
}
|
|
|
|
// (public) return number of set bits
|
|
function bnBitCount() {
|
|
var this_array = this.array;
|
|
var r = 0, x = this.s&BI_DM;
|
|
for(var i = 0; i < this.t; ++i) r += cbit(this_array[i]^x);
|
|
return r;
|
|
}
|
|
|
|
// (public) true iff nth bit is set
|
|
function bnTestBit(n) {
|
|
var this_array = this.array;
|
|
var j = Math.floor(n/BI_DB);
|
|
if(j >= this.t) return(this.s!=0);
|
|
return((this_array[j]&(1<<(n%BI_DB)))!=0);
|
|
}
|
|
|
|
// (protected) this op (1<<n)
|
|
function bnpChangeBit(n,op) {
|
|
var r = BigInteger.ONE.shiftLeft(n);
|
|
this.bitwiseTo(r,op,r);
|
|
return r;
|
|
}
|
|
|
|
// (public) this | (1<<n)
|
|
function bnSetBit(n) { return this.changeBit(n,op_or); }
|
|
|
|
// (public) this & ~(1<<n)
|
|
function bnClearBit(n) { return this.changeBit(n,op_andnot); }
|
|
|
|
// (public) this ^ (1<<n)
|
|
function bnFlipBit(n) { return this.changeBit(n,op_xor); }
|
|
|
|
// (protected) r = this + a
|
|
function bnpAddTo(a,r) {
|
|
var this_array = this.array;
|
|
var a_array = a.array;
|
|
var r_array = r.array;
|
|
var i = 0, c = 0, m = Math.min(a.t,this.t);
|
|
while(i < m) {
|
|
c += this_array[i]+a_array[i];
|
|
r_array[i++] = c&BI_DM;
|
|
c >>= BI_DB;
|
|
}
|
|
if(a.t < this.t) {
|
|
c += a.s;
|
|
while(i < this.t) {
|
|
c += this_array[i];
|
|
r_array[i++] = c&BI_DM;
|
|
c >>= BI_DB;
|
|
}
|
|
c += this.s;
|
|
}
|
|
else {
|
|
c += this.s;
|
|
while(i < a.t) {
|
|
c += a_array[i];
|
|
r_array[i++] = c&BI_DM;
|
|
c >>= BI_DB;
|
|
}
|
|
c += a.s;
|
|
}
|
|
r.s = (c<0)?-1:0;
|
|
if(c > 0) r_array[i++] = c;
|
|
else if(c < -1) r_array[i++] = BI_DV+c;
|
|
r.t = i;
|
|
r.clamp();
|
|
}
|
|
|
|
// (public) this + a
|
|
function bnAdd(a) { var r = nbi(); this.addTo(a,r); return r; }
|
|
|
|
// (public) this - a
|
|
function bnSubtract(a) { var r = nbi(); this.subTo(a,r); return r; }
|
|
|
|
// (public) this * a
|
|
function bnMultiply(a) { var r = nbi(); this.multiplyTo(a,r); return r; }
|
|
|
|
// (public) this / a
|
|
function bnDivide(a) { var r = nbi(); this.divRemTo(a,r,null); return r; }
|
|
|
|
// (public) this % a
|
|
function bnRemainder(a) { var r = nbi(); this.divRemTo(a,null,r); return r; }
|
|
|
|
// (public) [this/a,this%a]
|
|
function bnDivideAndRemainder(a) {
|
|
var q = nbi(), r = nbi();
|
|
this.divRemTo(a,q,r);
|
|
return new Array(q,r);
|
|
}
|
|
|
|
// (protected) this *= n, this >= 0, 1 < n < DV
|
|
function bnpDMultiply(n) {
|
|
var this_array = this.array;
|
|
this_array[this.t] = this.am(0,n-1,this,0,0,this.t);
|
|
++this.t;
|
|
this.clamp();
|
|
}
|
|
|
|
// (protected) this += n << w words, this >= 0
|
|
function bnpDAddOffset(n,w) {
|
|
var this_array = this.array;
|
|
while(this.t <= w) this_array[this.t++] = 0;
|
|
this_array[w] += n;
|
|
while(this_array[w] >= BI_DV) {
|
|
this_array[w] -= BI_DV;
|
|
if(++w >= this.t) this_array[this.t++] = 0;
|
|
++this_array[w];
|
|
}
|
|
}
|
|
|
|
// A "null" reducer
|
|
function NullExp() {}
|
|
function nNop(x) { return x; }
|
|
function nMulTo(x,y,r) { x.multiplyTo(y,r); }
|
|
function nSqrTo(x,r) { x.squareTo(r); }
|
|
|
|
NullExp.prototype.convert = nNop;
|
|
NullExp.prototype.revert = nNop;
|
|
NullExp.prototype.mulTo = nMulTo;
|
|
NullExp.prototype.sqrTo = nSqrTo;
|
|
|
|
// (public) this^e
|
|
function bnPow(e) { return this.exp(e,new NullExp()); }
|
|
|
|
// (protected) r = lower n words of "this * a", a.t <= n
|
|
// "this" should be the larger one if appropriate.
|
|
function bnpMultiplyLowerTo(a,n,r) {
|
|
var r_array = r.array;
|
|
var a_array = a.array;
|
|
var i = Math.min(this.t+a.t,n);
|
|
r.s = 0; // assumes a,this >= 0
|
|
r.t = i;
|
|
while(i > 0) r_array[--i] = 0;
|
|
var j;
|
|
for(j = r.t-this.t; i < j; ++i) r_array[i+this.t] = this.am(0,a_array[i],r,i,0,this.t);
|
|
for(j = Math.min(a.t,n); i < j; ++i) this.am(0,a_array[i],r,i,0,n-i);
|
|
r.clamp();
|
|
}
|
|
|
|
// (protected) r = "this * a" without lower n words, n > 0
|
|
// "this" should be the larger one if appropriate.
|
|
function bnpMultiplyUpperTo(a,n,r) {
|
|
var r_array = r.array;
|
|
var a_array = a.array;
|
|
--n;
|
|
var i = r.t = this.t+a.t-n;
|
|
r.s = 0; // assumes a,this >= 0
|
|
while(--i >= 0) r_array[i] = 0;
|
|
for(i = Math.max(n-this.t,0); i < a.t; ++i)
|
|
r_array[this.t+i-n] = this.am(n-i,a_array[i],r,0,0,this.t+i-n);
|
|
r.clamp();
|
|
r.drShiftTo(1,r);
|
|
}
|
|
|
|
// Barrett modular reduction
|
|
function Barrett(m) {
|
|
// setup Barrett
|
|
this.r2 = nbi();
|
|
this.q3 = nbi();
|
|
BigInteger.ONE.dlShiftTo(2*m.t,this.r2);
|
|
this.mu = this.r2.divide(m);
|
|
this.m = m;
|
|
}
|
|
|
|
function barrettConvert(x) {
|
|
if(x.s < 0 || x.t > 2*this.m.t) return x.mod(this.m);
|
|
else if(x.compareTo(this.m) < 0) return x;
|
|
else { var r = nbi(); x.copyTo(r); this.reduce(r); return r; }
|
|
}
|
|
|
|
function barrettRevert(x) { return x; }
|
|
|
|
// x = x mod m (HAC 14.42)
|
|
function barrettReduce(x) {
|
|
x.drShiftTo(this.m.t-1,this.r2);
|
|
if(x.t > this.m.t+1) { x.t = this.m.t+1; x.clamp(); }
|
|
this.mu.multiplyUpperTo(this.r2,this.m.t+1,this.q3);
|
|
this.m.multiplyLowerTo(this.q3,this.m.t+1,this.r2);
|
|
while(x.compareTo(this.r2) < 0) x.dAddOffset(1,this.m.t+1);
|
|
x.subTo(this.r2,x);
|
|
while(x.compareTo(this.m) >= 0) x.subTo(this.m,x);
|
|
}
|
|
|
|
// r = x^2 mod m; x != r
|
|
function barrettSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
|
|
|
|
// r = x*y mod m; x,y != r
|
|
function barrettMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
|
|
|
|
Barrett.prototype.convert = barrettConvert;
|
|
Barrett.prototype.revert = barrettRevert;
|
|
Barrett.prototype.reduce = barrettReduce;
|
|
Barrett.prototype.mulTo = barrettMulTo;
|
|
Barrett.prototype.sqrTo = barrettSqrTo;
|
|
|
|
// (public) this^e % m (HAC 14.85)
|
|
function bnModPow(e,m) {
|
|
var e_array = e.array;
|
|
var i = e.bitLength(), k, r = nbv(1), z;
|
|
if(i <= 0) return r;
|
|
else if(i < 18) k = 1;
|
|
else if(i < 48) k = 3;
|
|
else if(i < 144) k = 4;
|
|
else if(i < 768) k = 5;
|
|
else k = 6;
|
|
if(i < 8)
|
|
z = new Classic(m);
|
|
else if(m.isEven())
|
|
z = new Barrett(m);
|
|
else
|
|
z = new Montgomery(m);
|
|
|
|
// precomputation
|
|
var g = new Array(), n = 3, k1 = k-1, km = (1<<k)-1;
|
|
g[1] = z.convert(this);
|
|
if(k > 1) {
|
|
var g2 = nbi();
|
|
z.sqrTo(g[1],g2);
|
|
while(n <= km) {
|
|
g[n] = nbi();
|
|
z.mulTo(g2,g[n-2],g[n]);
|
|
n += 2;
|
|
}
|
|
}
|
|
|
|
var j = e.t-1, w, is1 = true, r2 = nbi(), t;
|
|
i = nbits(e_array[j])-1;
|
|
while(j >= 0) {
|
|
if(i >= k1) w = (e_array[j]>>(i-k1))&km;
|
|
else {
|
|
w = (e_array[j]&((1<<(i+1))-1))<<(k1-i);
|
|
if(j > 0) w |= e_array[j-1]>>(BI_DB+i-k1);
|
|
}
|
|
|
|
n = k;
|
|
while((w&1) == 0) { w >>= 1; --n; }
|
|
if((i -= n) < 0) { i += BI_DB; --j; }
|
|
if(is1) { // ret == 1, don't bother squaring or multiplying it
|
|
g[w].copyTo(r);
|
|
is1 = false;
|
|
}
|
|
else {
|
|
while(n > 1) { z.sqrTo(r,r2); z.sqrTo(r2,r); n -= 2; }
|
|
if(n > 0) z.sqrTo(r,r2); else { t = r; r = r2; r2 = t; }
|
|
z.mulTo(r2,g[w],r);
|
|
}
|
|
|
|
while(j >= 0 && (e_array[j]&(1<<i)) == 0) {
|
|
z.sqrTo(r,r2); t = r; r = r2; r2 = t;
|
|
if(--i < 0) { i = BI_DB-1; --j; }
|
|
}
|
|
}
|
|
return z.revert(r);
|
|
}
|
|
|
|
// (public) gcd(this,a) (HAC 14.54)
|
|
function bnGCD(a) {
|
|
var x = (this.s<0)?this.negate():this.clone();
|
|
var y = (a.s<0)?a.negate():a.clone();
|
|
if(x.compareTo(y) < 0) { var t = x; x = y; y = t; }
|
|
var i = x.getLowestSetBit(), g = y.getLowestSetBit();
|
|
if(g < 0) return x;
|
|
if(i < g) g = i;
|
|
if(g > 0) {
|
|
x.rShiftTo(g,x);
|
|
y.rShiftTo(g,y);
|
|
}
|
|
while(x.signum() > 0) {
|
|
if((i = x.getLowestSetBit()) > 0) x.rShiftTo(i,x);
|
|
if((i = y.getLowestSetBit()) > 0) y.rShiftTo(i,y);
|
|
if(x.compareTo(y) >= 0) {
|
|
x.subTo(y,x);
|
|
x.rShiftTo(1,x);
|
|
}
|
|
else {
|
|
y.subTo(x,y);
|
|
y.rShiftTo(1,y);
|
|
}
|
|
}
|
|
if(g > 0) y.lShiftTo(g,y);
|
|
return y;
|
|
}
|
|
|
|
// (protected) this % n, n < 2^26
|
|
function bnpModInt(n) {
|
|
var this_array = this.array;
|
|
if(n <= 0) return 0;
|
|
var d = BI_DV%n, r = (this.s<0)?n-1:0;
|
|
if(this.t > 0)
|
|
if(d == 0) r = this_array[0]%n;
|
|
else for(var i = this.t-1; i >= 0; --i) r = (d*r+this_array[i])%n;
|
|
return r;
|
|
}
|
|
|
|
// (public) 1/this % m (HAC 14.61)
|
|
function bnModInverse(m) {
|
|
var ac = m.isEven();
|
|
if((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;
|
|
var u = m.clone(), v = this.clone();
|
|
var a = nbv(1), b = nbv(0), c = nbv(0), d = nbv(1);
|
|
while(u.signum() != 0) {
|
|
while(u.isEven()) {
|
|
u.rShiftTo(1,u);
|
|
if(ac) {
|
|
if(!a.isEven() || !b.isEven()) { a.addTo(this,a); b.subTo(m,b); }
|
|
a.rShiftTo(1,a);
|
|
}
|
|
else if(!b.isEven()) b.subTo(m,b);
|
|
b.rShiftTo(1,b);
|
|
}
|
|
while(v.isEven()) {
|
|
v.rShiftTo(1,v);
|
|
if(ac) {
|
|
if(!c.isEven() || !d.isEven()) { c.addTo(this,c); d.subTo(m,d); }
|
|
c.rShiftTo(1,c);
|
|
}
|
|
else if(!d.isEven()) d.subTo(m,d);
|
|
d.rShiftTo(1,d);
|
|
}
|
|
if(u.compareTo(v) >= 0) {
|
|
u.subTo(v,u);
|
|
if(ac) a.subTo(c,a);
|
|
b.subTo(d,b);
|
|
}
|
|
else {
|
|
v.subTo(u,v);
|
|
if(ac) c.subTo(a,c);
|
|
d.subTo(b,d);
|
|
}
|
|
}
|
|
if(v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;
|
|
if(d.compareTo(m) >= 0) return d.subtract(m);
|
|
if(d.signum() < 0) d.addTo(m,d); else return d;
|
|
if(d.signum() < 0) return d.add(m); else return d;
|
|
}
|
|
|
|
var lowprimes = [2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71,73,79,83,89,97,101,103,107,109,113,127,131,137,139,149,151,157,163,167,173,179,181,191,193,197,199,211,223,227,229,233,239,241,251,257,263,269,271,277,281,283,293,307,311,313,317,331,337,347,349,353,359,367,373,379,383,389,397,401,409,419,421,431,433,439,443,449,457,461,463,467,479,487,491,499,503,509];
|
|
var lplim = (1<<26)/lowprimes[lowprimes.length-1];
|
|
|
|
// (public) test primality with certainty >= 1-.5^t
|
|
function bnIsProbablePrime(t) {
|
|
var i, x = this.abs();
|
|
var x_array = x.array;
|
|
if(x.t == 1 && x_array[0] <= lowprimes[lowprimes.length-1]) {
|
|
for(i = 0; i < lowprimes.length; ++i)
|
|
if(x_array[0] == lowprimes[i]) return true;
|
|
return false;
|
|
}
|
|
if(x.isEven()) return false;
|
|
i = 1;
|
|
while(i < lowprimes.length) {
|
|
var m = lowprimes[i], j = i+1;
|
|
while(j < lowprimes.length && m < lplim) m *= lowprimes[j++];
|
|
m = x.modInt(m);
|
|
while(i < j) if(m%lowprimes[i++] == 0) return false;
|
|
}
|
|
return x.millerRabin(t);
|
|
}
|
|
|
|
// (protected) true if probably prime (HAC 4.24, Miller-Rabin)
|
|
function bnpMillerRabin(t) {
|
|
var n1 = this.subtract(BigInteger.ONE);
|
|
var k = n1.getLowestSetBit();
|
|
if(k <= 0) return false;
|
|
var r = n1.shiftRight(k);
|
|
t = (t+1)>>1;
|
|
if(t > lowprimes.length) t = lowprimes.length;
|
|
var a = nbi();
|
|
for(var i = 0; i < t; ++i) {
|
|
a.fromInt(lowprimes[i]);
|
|
var y = a.modPow(r,this);
|
|
if(y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {
|
|
var j = 1;
|
|
while(j++ < k && y.compareTo(n1) != 0) {
|
|
y = y.modPowInt(2,this);
|
|
if(y.compareTo(BigInteger.ONE) == 0) return false;
|
|
}
|
|
if(y.compareTo(n1) != 0) return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
// protected
|
|
BigInteger.prototype.chunkSize = bnpChunkSize;
|
|
BigInteger.prototype.toRadix = bnpToRadix;
|
|
BigInteger.prototype.fromRadix = bnpFromRadix;
|
|
BigInteger.prototype.fromNumber = bnpFromNumber;
|
|
BigInteger.prototype.bitwiseTo = bnpBitwiseTo;
|
|
BigInteger.prototype.changeBit = bnpChangeBit;
|
|
BigInteger.prototype.addTo = bnpAddTo;
|
|
BigInteger.prototype.dMultiply = bnpDMultiply;
|
|
BigInteger.prototype.dAddOffset = bnpDAddOffset;
|
|
BigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;
|
|
BigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;
|
|
BigInteger.prototype.modInt = bnpModInt;
|
|
BigInteger.prototype.millerRabin = bnpMillerRabin;
|
|
|
|
// public
|
|
BigInteger.prototype.clone = bnClone;
|
|
BigInteger.prototype.intValue = bnIntValue;
|
|
BigInteger.prototype.byteValue = bnByteValue;
|
|
BigInteger.prototype.shortValue = bnShortValue;
|
|
BigInteger.prototype.signum = bnSigNum;
|
|
BigInteger.prototype.toByteArray = bnToByteArray;
|
|
BigInteger.prototype.equals = bnEquals;
|
|
BigInteger.prototype.min = bnMin;
|
|
BigInteger.prototype.max = bnMax;
|
|
BigInteger.prototype.and = bnAnd;
|
|
BigInteger.prototype.or = bnOr;
|
|
BigInteger.prototype.xor = bnXor;
|
|
BigInteger.prototype.andNot = bnAndNot;
|
|
BigInteger.prototype.not = bnNot;
|
|
BigInteger.prototype.shiftLeft = bnShiftLeft;
|
|
BigInteger.prototype.shiftRight = bnShiftRight;
|
|
BigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;
|
|
BigInteger.prototype.bitCount = bnBitCount;
|
|
BigInteger.prototype.testBit = bnTestBit;
|
|
BigInteger.prototype.setBit = bnSetBit;
|
|
BigInteger.prototype.clearBit = bnClearBit;
|
|
BigInteger.prototype.flipBit = bnFlipBit;
|
|
BigInteger.prototype.add = bnAdd;
|
|
BigInteger.prototype.subtract = bnSubtract;
|
|
BigInteger.prototype.multiply = bnMultiply;
|
|
BigInteger.prototype.divide = bnDivide;
|
|
BigInteger.prototype.remainder = bnRemainder;
|
|
BigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;
|
|
BigInteger.prototype.modPow = bnModPow;
|
|
BigInteger.prototype.modInverse = bnModInverse;
|
|
BigInteger.prototype.pow = bnPow;
|
|
BigInteger.prototype.gcd = bnGCD;
|
|
BigInteger.prototype.isProbablePrime = bnIsProbablePrime;
|
|
BigInteger.prototype.am = am4;
|
|
|
|
// end of stuff copied from github.
|
|
},{}],122:[function(require,module,exports){
|
|
/**
|
|
sprintf() for JavaScript 0.7-beta1
|
|
http://www.diveintojavascript.com/projects/javascript-sprintf
|
|
|
|
Copyright (c) Alexandru Marasteanu <alexaholic [at) gmail (dot] com>
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions are met:
|
|
* Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in the
|
|
documentation and/or other materials provided with the distribution.
|
|
* Neither the name of sprintf() for JavaScript nor the
|
|
names of its contributors may be used to endorse or promote products
|
|
derived from this software without specific prior written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
DISCLAIMED. IN NO EVENT SHALL Alexandru Marasteanu BE LIABLE FOR ANY
|
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
Changelog:
|
|
2010.11.07 - 0.7-beta1-node
|
|
- converted it to a node.js compatible module
|
|
|
|
2010.09.06 - 0.7-beta1
|
|
- features: vsprintf, support for named placeholders
|
|
- enhancements: format cache, reduced global namespace pollution
|
|
|
|
2010.05.22 - 0.6:
|
|
- reverted to 0.4 and fixed the bug regarding the sign of the number 0
|
|
Note:
|
|
Thanks to Raphael Pigulla <raph (at] n3rd [dot) org> (http://www.n3rd.org/)
|
|
who warned me about a bug in 0.5, I discovered that the last update was
|
|
a regress. I appologize for that.
|
|
|
|
2010.05.09 - 0.5:
|
|
- bug fix: 0 is now preceeded with a + sign
|
|
- bug fix: the sign was not at the right position on padded results (Kamal Abdali)
|
|
- switched from GPL to BSD license
|
|
|
|
2007.10.21 - 0.4:
|
|
- unit test and patch (David Baird)
|
|
|
|
2007.09.17 - 0.3:
|
|
- bug fix: no longer throws exception on empty paramenters (Hans Pufal)
|
|
|
|
2007.09.11 - 0.2:
|
|
- feature: added argument swapping
|
|
|
|
2007.04.03 - 0.1:
|
|
- initial release
|
|
**/
|
|
|
|
var util = require('util');
|
|
|
|
var sprintf = (function() {
|
|
function get_type(variable) {
|
|
return Object.prototype.toString.call(variable).slice(8, -1).toLowerCase();
|
|
}
|
|
function str_repeat(input, multiplier) {
|
|
for (var output = []; multiplier > 0; output[--multiplier] = input) {/* do nothing */}
|
|
return output.join('');
|
|
}
|
|
|
|
var str_format = function() {
|
|
if (!str_format.cache.hasOwnProperty(arguments[0])) {
|
|
str_format.cache[arguments[0]] = str_format.parse(arguments[0]);
|
|
}
|
|
return str_format.format.call(null, str_format.cache[arguments[0]], arguments);
|
|
};
|
|
|
|
// convert object to simple one line string without indentation or
|
|
// newlines. Note that this implementation does not print array
|
|
// values to their actual place for sparse arrays.
|
|
//
|
|
// For example sparse array like this
|
|
// l = []
|
|
// l[4] = 1
|
|
// Would be printed as "[1]" instead of "[, , , , 1]"
|
|
//
|
|
// If argument 'seen' is not null and array the function will check for
|
|
// circular object references from argument.
|
|
str_format.object_stringify = function(obj, depth, maxdepth, seen) {
|
|
var str = '';
|
|
if (obj != null) {
|
|
switch( typeof(obj) ) {
|
|
case 'function':
|
|
return '[Function' + (obj.name ? ': '+obj.name : '') + ']';
|
|
break;
|
|
case 'object':
|
|
if ( obj instanceof Error) { return '[' + obj.toString() + ']' };
|
|
if (depth >= maxdepth) return '[Object]'
|
|
if (seen) {
|
|
// add object to seen list
|
|
seen = seen.slice(0)
|
|
seen.push(obj);
|
|
}
|
|
if (obj.length != null) { //array
|
|
str += '[';
|
|
var arr = []
|
|
for (var i in obj) {
|
|
if (seen && seen.indexOf(obj[i]) >= 0) arr.push('[Circular]');
|
|
else arr.push(str_format.object_stringify(obj[i], depth+1, maxdepth, seen));
|
|
}
|
|
str += arr.join(', ') + ']';
|
|
} else if ('getMonth' in obj) { // date
|
|
return 'Date(' + obj + ')';
|
|
} else { // object
|
|
str += '{';
|
|
var arr = []
|
|
for (var k in obj) {
|
|
if(obj.hasOwnProperty(k)) {
|
|
if (seen && seen.indexOf(obj[k]) >= 0) arr.push(k + ': [Circular]');
|
|
else arr.push(k +': ' +str_format.object_stringify(obj[k], depth+1, maxdepth, seen));
|
|
}
|
|
}
|
|
str += arr.join(', ') + '}';
|
|
}
|
|
return str;
|
|
break;
|
|
case 'string':
|
|
return '"' + obj + '"';
|
|
break
|
|
}
|
|
}
|
|
return '' + obj;
|
|
}
|
|
|
|
str_format.format = function(parse_tree, argv) {
|
|
var cursor = 1, tree_length = parse_tree.length, node_type = '', arg, output = [], i, k, match, pad, pad_character, pad_length;
|
|
for (i = 0; i < tree_length; i++) {
|
|
node_type = get_type(parse_tree[i]);
|
|
if (node_type === 'string') {
|
|
output.push(parse_tree[i]);
|
|
}
|
|
else if (node_type === 'array') {
|
|
match = parse_tree[i]; // convenience purposes only
|
|
if (match[2]) { // keyword argument
|
|
arg = argv[cursor];
|
|
for (k = 0; k < match[2].length; k++) {
|
|
if (!arg.hasOwnProperty(match[2][k])) {
|
|
throw new Error(sprintf('[sprintf] property "%s" does not exist', match[2][k]));
|
|
}
|
|
arg = arg[match[2][k]];
|
|
}
|
|
}
|
|
else if (match[1]) { // positional argument (explicit)
|
|
arg = argv[match[1]];
|
|
}
|
|
else { // positional argument (implicit)
|
|
arg = argv[cursor++];
|
|
}
|
|
|
|
if (/[^sO]/.test(match[8]) && (get_type(arg) != 'number')) {
|
|
throw new Error(sprintf('[sprintf] expecting number but found %s "' + arg + '"', get_type(arg)));
|
|
}
|
|
switch (match[8]) {
|
|
case 'b': arg = arg.toString(2); break;
|
|
case 'c': arg = String.fromCharCode(arg); break;
|
|
case 'd': arg = parseInt(arg, 10); break;
|
|
case 'e': arg = match[7] ? arg.toExponential(match[7]) : arg.toExponential(); break;
|
|
case 'f': arg = match[7] ? parseFloat(arg).toFixed(match[7]) : parseFloat(arg); break;
|
|
case 'O': arg = str_format.object_stringify(arg, 0, parseInt(match[7]) || 5); break;
|
|
case 'o': arg = arg.toString(8); break;
|
|
case 's': arg = ((arg = String(arg)) && match[7] ? arg.substring(0, match[7]) : arg); break;
|
|
case 'u': arg = Math.abs(arg); break;
|
|
case 'x': arg = arg.toString(16); break;
|
|
case 'X': arg = arg.toString(16).toUpperCase(); break;
|
|
}
|
|
arg = (/[def]/.test(match[8]) && match[3] && arg >= 0 ? '+'+ arg : arg);
|
|
pad_character = match[4] ? match[4] == '0' ? '0' : match[4].charAt(1) : ' ';
|
|
pad_length = match[6] - String(arg).length;
|
|
pad = match[6] ? str_repeat(pad_character, pad_length) : '';
|
|
output.push(match[5] ? arg + pad : pad + arg);
|
|
}
|
|
}
|
|
return output.join('');
|
|
};
|
|
|
|
str_format.cache = {};
|
|
|
|
str_format.parse = function(fmt) {
|
|
var _fmt = fmt, match = [], parse_tree = [], arg_names = 0;
|
|
while (_fmt) {
|
|
if ((match = /^[^\x25]+/.exec(_fmt)) !== null) {
|
|
parse_tree.push(match[0]);
|
|
}
|
|
else if ((match = /^\x25{2}/.exec(_fmt)) !== null) {
|
|
parse_tree.push('%');
|
|
}
|
|
else if ((match = /^\x25(?:([1-9]\d*)\$|\(([^\)]+)\))?(\+)?(0|'[^$])?(-)?(\d+)?(?:\.(\d+))?([b-fosOuxX])/.exec(_fmt)) !== null) {
|
|
if (match[2]) {
|
|
arg_names |= 1;
|
|
var field_list = [], replacement_field = match[2], field_match = [];
|
|
if ((field_match = /^([a-z_][a-z_\d]*)/i.exec(replacement_field)) !== null) {
|
|
field_list.push(field_match[1]);
|
|
while ((replacement_field = replacement_field.substring(field_match[0].length)) !== '') {
|
|
if ((field_match = /^\.([a-z_][a-z_\d]*)/i.exec(replacement_field)) !== null) {
|
|
field_list.push(field_match[1]);
|
|
}
|
|
else if ((field_match = /^\[(\d+)\]/.exec(replacement_field)) !== null) {
|
|
field_list.push(field_match[1]);
|
|
}
|
|
else {
|
|
throw new Error('[sprintf] ' + replacement_field);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
throw new Error('[sprintf] ' + replacement_field);
|
|
}
|
|
match[2] = field_list;
|
|
}
|
|
else {
|
|
arg_names |= 2;
|
|
}
|
|
if (arg_names === 3) {
|
|
throw new Error('[sprintf] mixing positional and named placeholders is not (yet) supported');
|
|
}
|
|
parse_tree.push(match);
|
|
}
|
|
else {
|
|
throw new Error('[sprintf] ' + _fmt);
|
|
}
|
|
_fmt = _fmt.substring(match[0].length);
|
|
}
|
|
return parse_tree;
|
|
};
|
|
|
|
return str_format;
|
|
})();
|
|
|
|
var vsprintf = function(fmt, argv) {
|
|
var argvClone = argv.slice();
|
|
argvClone.unshift(fmt);
|
|
return sprintf.apply(null, argvClone);
|
|
};
|
|
|
|
module.exports = sprintf;
|
|
sprintf.sprintf = sprintf;
|
|
sprintf.vsprintf = vsprintf;
|
|
|
|
},{"util":112}],123:[function(require,module,exports){
|
|
/* Copyright 2010 Membase, Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
/*jshint node:true*/
|
|
|
|
|
|
/**
|
|
* Convert a 16-bit quantity (short integer) from host byte order to network byte order (Little-Endian to Big-Endian).
|
|
*
|
|
* @param {Array|Buffer} b Array of octets or a nodejs Buffer
|
|
* @param {number} i Zero-based index at which to write into b
|
|
* @param {number} v Value to convert
|
|
*/
|
|
exports.htons = function(b, i, v) {
|
|
b[i] = (0xff & (v >> 8));
|
|
b[i + 1] = (0xff & (v));
|
|
};
|
|
|
|
|
|
/**
|
|
* Convert a 16-bit quantity (short integer) from network byte order to host byte order (Big-Endian to Little-Endian).
|
|
*
|
|
* @param {Array|Buffer} b Array of octets or a nodejs Buffer to read value from
|
|
* @param {number} i Zero-based index at which to read from b
|
|
* @returns {number}
|
|
*/
|
|
exports.ntohs = function(b, i) {
|
|
return ((0xff & b[i]) << 8) |
|
|
((0xff & b[i + 1]));
|
|
};
|
|
|
|
|
|
/**
|
|
* Convert a 16-bit quantity (short integer) from network byte order to host byte order (Big-Endian to Little-Endian).
|
|
*
|
|
* @param {string} s String to read value from
|
|
* @param {number} i Zero-based index at which to read from s
|
|
* @returns {number}
|
|
*/
|
|
exports.ntohsStr = function(s, i) {
|
|
return ((0xff & s.charCodeAt(i)) << 8) |
|
|
((0xff & s.charCodeAt(i + 1)));
|
|
};
|
|
|
|
|
|
/**
|
|
* Convert a 32-bit quantity (long integer) from host byte order to network byte order (Little-Endian to Big-Endian).
|
|
*
|
|
* @param {Array|Buffer} b Array of octets or a nodejs Buffer
|
|
* @param {number} i Zero-based index at which to write into b
|
|
* @param {number} v Value to convert
|
|
*/
|
|
exports.htonl = function(b, i, v) {
|
|
b[i] = (0xff & (v >> 24));
|
|
b[i + 1] = (0xff & (v >> 16));
|
|
b[i + 2] = (0xff & (v >> 8));
|
|
b[i + 3] = (0xff & (v));
|
|
};
|
|
|
|
|
|
/**
|
|
* Convert a 32-bit quantity (long integer) from network byte order to host byte order (Big-Endian to Little-Endian).
|
|
*
|
|
* @param {Array|Buffer} b Array of octets or a nodejs Buffer to read value from
|
|
* @param {number} i Zero-based index at which to read from b
|
|
* @returns {number}
|
|
*/
|
|
exports.ntohl = function(b, i) {
|
|
return ((0xff & b[i]) << 24) |
|
|
((0xff & b[i + 1]) << 16) |
|
|
((0xff & b[i + 2]) << 8) |
|
|
((0xff & b[i + 3]));
|
|
};
|
|
|
|
|
|
/**
|
|
* Convert a 32-bit quantity (long integer) from network byte order to host byte order (Big-Endian to Little-Endian).
|
|
*
|
|
* @param {string} s String to read value from
|
|
* @param {number} i Zero-based index at which to read from s
|
|
* @returns {number}
|
|
*/
|
|
exports.ntohlStr = function(s, i) {
|
|
return ((0xff & s.charCodeAt(i)) << 24) |
|
|
((0xff & s.charCodeAt(i + 1)) << 16) |
|
|
((0xff & s.charCodeAt(i + 2)) << 8) |
|
|
((0xff & s.charCodeAt(i + 3)));
|
|
};
|
|
|
|
},{}],124:[function(require,module,exports){
|
|
|
|
module.exports = function(){
|
|
var orig = Error.prepareStackTrace;
|
|
Error.prepareStackTrace = function(_, stack){ return stack; };
|
|
var err = new Error;
|
|
Error.captureStackTrace(err, arguments.callee);
|
|
var stack = err.stack;
|
|
Error.prepareStackTrace = orig;
|
|
return stack;
|
|
};
|
|
|
|
},{}],125:[function(require,module,exports){
|
|
(function (process,global){
|
|
var path = require('path');
|
|
var callsite = require('callsite');
|
|
var realModulePaths = module.paths;
|
|
var realModuleFilename = module.filename;
|
|
|
|
// Decorate the given constructor with some useful
|
|
// object oriented constructs (mainly a convenient inherit()
|
|
// method and the ability to do a super send)
|
|
module.exports = function(constructor) {
|
|
// inherit from the given constructor
|
|
constructor.inherit = function(parent) {
|
|
if (arguments.length > 1) {
|
|
// this allows chaining multiple classes in the call
|
|
parent.inherit(Array.prototype.slice.call(arguments, 1));
|
|
}
|
|
this.super_ = parent;
|
|
this.prototype.__proto__ = parent.prototype;
|
|
this.__proto__ = parent;
|
|
};
|
|
|
|
// invoke the given method of the parent
|
|
constructor.super = function(receiver, method, args) {
|
|
if (!this.super_) return;
|
|
if (typeof method == 'string') {
|
|
// invoke the named method
|
|
return this.super_.prototype[method].apply(receiver, args);
|
|
} else {
|
|
// invoke the constructor of the parent
|
|
return this.super_.apply(receiver, method);
|
|
}
|
|
};
|
|
|
|
// a standarized way to access a cached default instance
|
|
constructor.default = function() {
|
|
if (!this._default) this._default = new this();
|
|
return this._default;
|
|
};
|
|
|
|
// set the parent if one is specified
|
|
if (constructor.parent) {
|
|
constructor.inherit(constructor.parent);
|
|
}
|
|
|
|
return constructor;
|
|
};
|
|
|
|
// load the given module using the given imports
|
|
// @fname the module name (relative paths are relative to the caller's
|
|
// location in the file system
|
|
// @imports namespace for binding values in the loaded module
|
|
var load = function(fname, imports) {
|
|
if((fname.slice(0,2) == './') || (fname.slice(0,3) == '../')) {
|
|
var callerFilename = callsite()[1].getFileName();
|
|
fname = path.resolve(path.dirname(callerFilename), fname);
|
|
}
|
|
|
|
// fake out module path resolution here
|
|
module.paths = module.parent.paths;
|
|
module.filename = module.parent.filename;
|
|
fname = require.resolve(fname);
|
|
module.paths = realModulePaths;
|
|
module.filename = realModuleFilename;
|
|
|
|
var cachedModule = require.cache[fname];
|
|
if (cachedModule) delete require.cache[fname];
|
|
global._imports = imports;
|
|
var answer = require(fname);
|
|
delete require.cache[fname];
|
|
if (cachedModule) require.cache[fname] = cachedModule;
|
|
return answer;
|
|
};
|
|
|
|
var load_browser = function(fname, imports) {
|
|
global._imports = imports;
|
|
var answer;
|
|
try {
|
|
answer = require('!' + fname);
|
|
} catch (e) {
|
|
console.log('SOOP:' + e.message + '\nNote that SOOP requires a custom browserify configuration. please check soop\'s readme');
|
|
throw e;
|
|
}
|
|
return answer;
|
|
};
|
|
|
|
module.exports.load = process.versions ? load : load_browser;
|
|
|
|
// access the imports passed from a call to load()
|
|
module.exports.imports = function() {
|
|
var answer = global._imports || {};
|
|
global._imports = {};
|
|
return answer;
|
|
};
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"),typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
|
|
},{"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"callsite":124,"path":97}],126:[function(require,module,exports){
|
|
(function (process){
|
|
/*
|
|
Copyright (c) 2011 Tim Caswell <tim@creationix.com>
|
|
|
|
MIT License
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
SOFTWARE.
|
|
*/
|
|
|
|
// Inspired by http://github.com/willconant/flow-js, but reimplemented and
|
|
// modified to fit my taste and the node.JS error handling system.
|
|
function Step() {
|
|
var steps = Array.prototype.slice.call(arguments),
|
|
pending, counter, results, lock;
|
|
|
|
// Define the main callback that's given as `this` to the steps.
|
|
function next() {
|
|
|
|
// Check if there are no steps left
|
|
if (steps.length === 0) {
|
|
// Throw uncaught errors
|
|
if (arguments[0]) {
|
|
throw arguments[0];
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Get the next step to execute
|
|
var fn = steps.shift();
|
|
counter = pending = 0;
|
|
results = [];
|
|
|
|
// Run the step in a try..catch block so exceptions don't get out of hand.
|
|
try {
|
|
lock = true;
|
|
var result = fn.apply(next, arguments);
|
|
} catch (e) {
|
|
// Pass any exceptions on through the next callback
|
|
next(e);
|
|
}
|
|
|
|
|
|
// If a syncronous return is used, pass it to the callback
|
|
if (result !== undefined) {
|
|
next(undefined, result);
|
|
}
|
|
lock = false;
|
|
}
|
|
|
|
// Add a special callback generator `this.parallel()` that groups stuff.
|
|
next.parallel = function () {
|
|
var index = 1 + counter++;
|
|
pending++;
|
|
|
|
function check() {
|
|
if (pending === 0) {
|
|
// When they're all done, call the callback
|
|
next.apply(null, results);
|
|
}
|
|
}
|
|
process.nextTick(check); // Ensures that check is called at least once
|
|
|
|
return function () {
|
|
pending--;
|
|
// Compress the error from any result to the first argument
|
|
if (arguments[0]) {
|
|
results[0] = arguments[0];
|
|
}
|
|
// Send the other results as arguments
|
|
results[index] = arguments[1];
|
|
if (!lock) { check(); }
|
|
};
|
|
};
|
|
|
|
// Generates a callback generator for grouped results
|
|
next.group = function () {
|
|
var localCallback = next.parallel();
|
|
var counter = 0;
|
|
var pending = 0;
|
|
var result = [];
|
|
var error = undefined;
|
|
|
|
function check() {
|
|
if (pending === 0) {
|
|
// When group is done, call the callback
|
|
localCallback(error, result);
|
|
}
|
|
}
|
|
process.nextTick(check); // Ensures that check is called at least once
|
|
|
|
// Generates a callback for the group
|
|
return function () {
|
|
var index = counter++;
|
|
pending++;
|
|
return function () {
|
|
pending--;
|
|
// Compress the error from any result to the first argument
|
|
if (arguments[0]) {
|
|
error = arguments[0];
|
|
}
|
|
// Send the other results as arguments
|
|
result[index] = arguments[1];
|
|
if (!lock) { check(); }
|
|
};
|
|
};
|
|
};
|
|
|
|
// Start the engine an pass nothing to the first step.
|
|
next();
|
|
}
|
|
|
|
// Tack on leading and tailing steps for input and output and return
|
|
// the whole thing as a function. Basically turns step calls into function
|
|
// factories.
|
|
Step.fn = function StepFn() {
|
|
var steps = Array.prototype.slice.call(arguments);
|
|
return function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
|
|
// Insert a first step that primes the data stream
|
|
var toRun = [function () {
|
|
this.apply(null, args);
|
|
}].concat(steps);
|
|
|
|
// If the last arg is a function add it as a last step
|
|
if (typeof args[args.length-1] === 'function') {
|
|
toRun.push(args.pop());
|
|
}
|
|
|
|
|
|
Step.apply(null, toRun);
|
|
}
|
|
}
|
|
|
|
|
|
// Hook into commonJS module systems
|
|
if (typeof module !== 'undefined' && "exports" in module) {
|
|
module.exports = Step;
|
|
}
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"))
|
|
},{"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96}],127:[function(require,module,exports){
|
|
// Underscore.js 1.6.0
|
|
// http://underscorejs.org
|
|
// (c) 2009-2014 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
|
|
// Underscore may be freely distributed under the MIT license.
|
|
|
|
(function() {
|
|
|
|
// Baseline setup
|
|
// --------------
|
|
|
|
// Establish the root object, `window` in the browser, or `exports` on the server.
|
|
var root = this;
|
|
|
|
// Save the previous value of the `_` variable.
|
|
var previousUnderscore = root._;
|
|
|
|
// Establish the object that gets returned to break out of a loop iteration.
|
|
var breaker = {};
|
|
|
|
// Save bytes in the minified (but not gzipped) version:
|
|
var ArrayProto = Array.prototype, ObjProto = Object.prototype, FuncProto = Function.prototype;
|
|
|
|
// Create quick reference variables for speed access to core prototypes.
|
|
var
|
|
push = ArrayProto.push,
|
|
slice = ArrayProto.slice,
|
|
concat = ArrayProto.concat,
|
|
toString = ObjProto.toString,
|
|
hasOwnProperty = ObjProto.hasOwnProperty;
|
|
|
|
// All **ECMAScript 5** native function implementations that we hope to use
|
|
// are declared here.
|
|
var
|
|
nativeForEach = ArrayProto.forEach,
|
|
nativeMap = ArrayProto.map,
|
|
nativeReduce = ArrayProto.reduce,
|
|
nativeReduceRight = ArrayProto.reduceRight,
|
|
nativeFilter = ArrayProto.filter,
|
|
nativeEvery = ArrayProto.every,
|
|
nativeSome = ArrayProto.some,
|
|
nativeIndexOf = ArrayProto.indexOf,
|
|
nativeLastIndexOf = ArrayProto.lastIndexOf,
|
|
nativeIsArray = Array.isArray,
|
|
nativeKeys = Object.keys,
|
|
nativeBind = FuncProto.bind;
|
|
|
|
// Create a safe reference to the Underscore object for use below.
|
|
var _ = function(obj) {
|
|
if (obj instanceof _) return obj;
|
|
if (!(this instanceof _)) return new _(obj);
|
|
this._wrapped = obj;
|
|
};
|
|
|
|
// Export the Underscore object for **Node.js**, with
|
|
// backwards-compatibility for the old `require()` API. If we're in
|
|
// the browser, add `_` as a global object via a string identifier,
|
|
// for Closure Compiler "advanced" mode.
|
|
if (typeof exports !== 'undefined') {
|
|
if (typeof module !== 'undefined' && module.exports) {
|
|
exports = module.exports = _;
|
|
}
|
|
exports._ = _;
|
|
} else {
|
|
root._ = _;
|
|
}
|
|
|
|
// Current version.
|
|
_.VERSION = '1.6.0';
|
|
|
|
// Collection Functions
|
|
// --------------------
|
|
|
|
// The cornerstone, an `each` implementation, aka `forEach`.
|
|
// Handles objects with the built-in `forEach`, arrays, and raw objects.
|
|
// Delegates to **ECMAScript 5**'s native `forEach` if available.
|
|
var each = _.each = _.forEach = function(obj, iterator, context) {
|
|
if (obj == null) return obj;
|
|
if (nativeForEach && obj.forEach === nativeForEach) {
|
|
obj.forEach(iterator, context);
|
|
} else if (obj.length === +obj.length) {
|
|
for (var i = 0, length = obj.length; i < length; i++) {
|
|
if (iterator.call(context, obj[i], i, obj) === breaker) return;
|
|
}
|
|
} else {
|
|
var keys = _.keys(obj);
|
|
for (var i = 0, length = keys.length; i < length; i++) {
|
|
if (iterator.call(context, obj[keys[i]], keys[i], obj) === breaker) return;
|
|
}
|
|
}
|
|
return obj;
|
|
};
|
|
|
|
// Return the results of applying the iterator to each element.
|
|
// Delegates to **ECMAScript 5**'s native `map` if available.
|
|
_.map = _.collect = function(obj, iterator, context) {
|
|
var results = [];
|
|
if (obj == null) return results;
|
|
if (nativeMap && obj.map === nativeMap) return obj.map(iterator, context);
|
|
each(obj, function(value, index, list) {
|
|
results.push(iterator.call(context, value, index, list));
|
|
});
|
|
return results;
|
|
};
|
|
|
|
var reduceError = 'Reduce of empty array with no initial value';
|
|
|
|
// **Reduce** builds up a single result from a list of values, aka `inject`,
|
|
// or `foldl`. Delegates to **ECMAScript 5**'s native `reduce` if available.
|
|
_.reduce = _.foldl = _.inject = function(obj, iterator, memo, context) {
|
|
var initial = arguments.length > 2;
|
|
if (obj == null) obj = [];
|
|
if (nativeReduce && obj.reduce === nativeReduce) {
|
|
if (context) iterator = _.bind(iterator, context);
|
|
return initial ? obj.reduce(iterator, memo) : obj.reduce(iterator);
|
|
}
|
|
each(obj, function(value, index, list) {
|
|
if (!initial) {
|
|
memo = value;
|
|
initial = true;
|
|
} else {
|
|
memo = iterator.call(context, memo, value, index, list);
|
|
}
|
|
});
|
|
if (!initial) throw new TypeError(reduceError);
|
|
return memo;
|
|
};
|
|
|
|
// The right-associative version of reduce, also known as `foldr`.
|
|
// Delegates to **ECMAScript 5**'s native `reduceRight` if available.
|
|
_.reduceRight = _.foldr = function(obj, iterator, memo, context) {
|
|
var initial = arguments.length > 2;
|
|
if (obj == null) obj = [];
|
|
if (nativeReduceRight && obj.reduceRight === nativeReduceRight) {
|
|
if (context) iterator = _.bind(iterator, context);
|
|
return initial ? obj.reduceRight(iterator, memo) : obj.reduceRight(iterator);
|
|
}
|
|
var length = obj.length;
|
|
if (length !== +length) {
|
|
var keys = _.keys(obj);
|
|
length = keys.length;
|
|
}
|
|
each(obj, function(value, index, list) {
|
|
index = keys ? keys[--length] : --length;
|
|
if (!initial) {
|
|
memo = obj[index];
|
|
initial = true;
|
|
} else {
|
|
memo = iterator.call(context, memo, obj[index], index, list);
|
|
}
|
|
});
|
|
if (!initial) throw new TypeError(reduceError);
|
|
return memo;
|
|
};
|
|
|
|
// Return the first value which passes a truth test. Aliased as `detect`.
|
|
_.find = _.detect = function(obj, predicate, context) {
|
|
var result;
|
|
any(obj, function(value, index, list) {
|
|
if (predicate.call(context, value, index, list)) {
|
|
result = value;
|
|
return true;
|
|
}
|
|
});
|
|
return result;
|
|
};
|
|
|
|
// Return all the elements that pass a truth test.
|
|
// Delegates to **ECMAScript 5**'s native `filter` if available.
|
|
// Aliased as `select`.
|
|
_.filter = _.select = function(obj, predicate, context) {
|
|
var results = [];
|
|
if (obj == null) return results;
|
|
if (nativeFilter && obj.filter === nativeFilter) return obj.filter(predicate, context);
|
|
each(obj, function(value, index, list) {
|
|
if (predicate.call(context, value, index, list)) results.push(value);
|
|
});
|
|
return results;
|
|
};
|
|
|
|
// Return all the elements for which a truth test fails.
|
|
_.reject = function(obj, predicate, context) {
|
|
return _.filter(obj, function(value, index, list) {
|
|
return !predicate.call(context, value, index, list);
|
|
}, context);
|
|
};
|
|
|
|
// Determine whether all of the elements match a truth test.
|
|
// Delegates to **ECMAScript 5**'s native `every` if available.
|
|
// Aliased as `all`.
|
|
_.every = _.all = function(obj, predicate, context) {
|
|
predicate || (predicate = _.identity);
|
|
var result = true;
|
|
if (obj == null) return result;
|
|
if (nativeEvery && obj.every === nativeEvery) return obj.every(predicate, context);
|
|
each(obj, function(value, index, list) {
|
|
if (!(result = result && predicate.call(context, value, index, list))) return breaker;
|
|
});
|
|
return !!result;
|
|
};
|
|
|
|
// Determine if at least one element in the object matches a truth test.
|
|
// Delegates to **ECMAScript 5**'s native `some` if available.
|
|
// Aliased as `any`.
|
|
var any = _.some = _.any = function(obj, predicate, context) {
|
|
predicate || (predicate = _.identity);
|
|
var result = false;
|
|
if (obj == null) return result;
|
|
if (nativeSome && obj.some === nativeSome) return obj.some(predicate, context);
|
|
each(obj, function(value, index, list) {
|
|
if (result || (result = predicate.call(context, value, index, list))) return breaker;
|
|
});
|
|
return !!result;
|
|
};
|
|
|
|
// Determine if the array or object contains a given value (using `===`).
|
|
// Aliased as `include`.
|
|
_.contains = _.include = function(obj, target) {
|
|
if (obj == null) return false;
|
|
if (nativeIndexOf && obj.indexOf === nativeIndexOf) return obj.indexOf(target) != -1;
|
|
return any(obj, function(value) {
|
|
return value === target;
|
|
});
|
|
};
|
|
|
|
// Invoke a method (with arguments) on every item in a collection.
|
|
_.invoke = function(obj, method) {
|
|
var args = slice.call(arguments, 2);
|
|
var isFunc = _.isFunction(method);
|
|
return _.map(obj, function(value) {
|
|
return (isFunc ? method : value[method]).apply(value, args);
|
|
});
|
|
};
|
|
|
|
// Convenience version of a common use case of `map`: fetching a property.
|
|
_.pluck = function(obj, key) {
|
|
return _.map(obj, _.property(key));
|
|
};
|
|
|
|
// Convenience version of a common use case of `filter`: selecting only objects
|
|
// containing specific `key:value` pairs.
|
|
_.where = function(obj, attrs) {
|
|
return _.filter(obj, _.matches(attrs));
|
|
};
|
|
|
|
// Convenience version of a common use case of `find`: getting the first object
|
|
// containing specific `key:value` pairs.
|
|
_.findWhere = function(obj, attrs) {
|
|
return _.find(obj, _.matches(attrs));
|
|
};
|
|
|
|
// Return the maximum element or (element-based computation).
|
|
// Can't optimize arrays of integers longer than 65,535 elements.
|
|
// See [WebKit Bug 80797](https://bugs.webkit.org/show_bug.cgi?id=80797)
|
|
_.max = function(obj, iterator, context) {
|
|
if (!iterator && _.isArray(obj) && obj[0] === +obj[0] && obj.length < 65535) {
|
|
return Math.max.apply(Math, obj);
|
|
}
|
|
var result = -Infinity, lastComputed = -Infinity;
|
|
each(obj, function(value, index, list) {
|
|
var computed = iterator ? iterator.call(context, value, index, list) : value;
|
|
if (computed > lastComputed) {
|
|
result = value;
|
|
lastComputed = computed;
|
|
}
|
|
});
|
|
return result;
|
|
};
|
|
|
|
// Return the minimum element (or element-based computation).
|
|
_.min = function(obj, iterator, context) {
|
|
if (!iterator && _.isArray(obj) && obj[0] === +obj[0] && obj.length < 65535) {
|
|
return Math.min.apply(Math, obj);
|
|
}
|
|
var result = Infinity, lastComputed = Infinity;
|
|
each(obj, function(value, index, list) {
|
|
var computed = iterator ? iterator.call(context, value, index, list) : value;
|
|
if (computed < lastComputed) {
|
|
result = value;
|
|
lastComputed = computed;
|
|
}
|
|
});
|
|
return result;
|
|
};
|
|
|
|
// Shuffle an array, using the modern version of the
|
|
// [Fisher-Yates shuffle](http://en.wikipedia.org/wiki/Fisher–Yates_shuffle).
|
|
_.shuffle = function(obj) {
|
|
var rand;
|
|
var index = 0;
|
|
var shuffled = [];
|
|
each(obj, function(value) {
|
|
rand = _.random(index++);
|
|
shuffled[index - 1] = shuffled[rand];
|
|
shuffled[rand] = value;
|
|
});
|
|
return shuffled;
|
|
};
|
|
|
|
// Sample **n** random values from a collection.
|
|
// If **n** is not specified, returns a single random element.
|
|
// The internal `guard` argument allows it to work with `map`.
|
|
_.sample = function(obj, n, guard) {
|
|
if (n == null || guard) {
|
|
if (obj.length !== +obj.length) obj = _.values(obj);
|
|
return obj[_.random(obj.length - 1)];
|
|
}
|
|
return _.shuffle(obj).slice(0, Math.max(0, n));
|
|
};
|
|
|
|
// An internal function to generate lookup iterators.
|
|
var lookupIterator = function(value) {
|
|
if (value == null) return _.identity;
|
|
if (_.isFunction(value)) return value;
|
|
return _.property(value);
|
|
};
|
|
|
|
// Sort the object's values by a criterion produced by an iterator.
|
|
_.sortBy = function(obj, iterator, context) {
|
|
iterator = lookupIterator(iterator);
|
|
return _.pluck(_.map(obj, function(value, index, list) {
|
|
return {
|
|
value: value,
|
|
index: index,
|
|
criteria: iterator.call(context, value, index, list)
|
|
};
|
|
}).sort(function(left, right) {
|
|
var a = left.criteria;
|
|
var b = right.criteria;
|
|
if (a !== b) {
|
|
if (a > b || a === void 0) return 1;
|
|
if (a < b || b === void 0) return -1;
|
|
}
|
|
return left.index - right.index;
|
|
}), 'value');
|
|
};
|
|
|
|
// An internal function used for aggregate "group by" operations.
|
|
var group = function(behavior) {
|
|
return function(obj, iterator, context) {
|
|
var result = {};
|
|
iterator = lookupIterator(iterator);
|
|
each(obj, function(value, index) {
|
|
var key = iterator.call(context, value, index, obj);
|
|
behavior(result, key, value);
|
|
});
|
|
return result;
|
|
};
|
|
};
|
|
|
|
// Groups the object's values by a criterion. Pass either a string attribute
|
|
// to group by, or a function that returns the criterion.
|
|
_.groupBy = group(function(result, key, value) {
|
|
_.has(result, key) ? result[key].push(value) : result[key] = [value];
|
|
});
|
|
|
|
// Indexes the object's values by a criterion, similar to `groupBy`, but for
|
|
// when you know that your index values will be unique.
|
|
_.indexBy = group(function(result, key, value) {
|
|
result[key] = value;
|
|
});
|
|
|
|
// Counts instances of an object that group by a certain criterion. Pass
|
|
// either a string attribute to count by, or a function that returns the
|
|
// criterion.
|
|
_.countBy = group(function(result, key) {
|
|
_.has(result, key) ? result[key]++ : result[key] = 1;
|
|
});
|
|
|
|
// Use a comparator function to figure out the smallest index at which
|
|
// an object should be inserted so as to maintain order. Uses binary search.
|
|
_.sortedIndex = function(array, obj, iterator, context) {
|
|
iterator = lookupIterator(iterator);
|
|
var value = iterator.call(context, obj);
|
|
var low = 0, high = array.length;
|
|
while (low < high) {
|
|
var mid = (low + high) >>> 1;
|
|
iterator.call(context, array[mid]) < value ? low = mid + 1 : high = mid;
|
|
}
|
|
return low;
|
|
};
|
|
|
|
// Safely create a real, live array from anything iterable.
|
|
_.toArray = function(obj) {
|
|
if (!obj) return [];
|
|
if (_.isArray(obj)) return slice.call(obj);
|
|
if (obj.length === +obj.length) return _.map(obj, _.identity);
|
|
return _.values(obj);
|
|
};
|
|
|
|
// Return the number of elements in an object.
|
|
_.size = function(obj) {
|
|
if (obj == null) return 0;
|
|
return (obj.length === +obj.length) ? obj.length : _.keys(obj).length;
|
|
};
|
|
|
|
// Array Functions
|
|
// ---------------
|
|
|
|
// Get the first element of an array. Passing **n** will return the first N
|
|
// values in the array. Aliased as `head` and `take`. The **guard** check
|
|
// allows it to work with `_.map`.
|
|
_.first = _.head = _.take = function(array, n, guard) {
|
|
if (array == null) return void 0;
|
|
if ((n == null) || guard) return array[0];
|
|
if (n < 0) return [];
|
|
return slice.call(array, 0, n);
|
|
};
|
|
|
|
// Returns everything but the last entry of the array. Especially useful on
|
|
// the arguments object. Passing **n** will return all the values in
|
|
// the array, excluding the last N. The **guard** check allows it to work with
|
|
// `_.map`.
|
|
_.initial = function(array, n, guard) {
|
|
return slice.call(array, 0, array.length - ((n == null) || guard ? 1 : n));
|
|
};
|
|
|
|
// Get the last element of an array. Passing **n** will return the last N
|
|
// values in the array. The **guard** check allows it to work with `_.map`.
|
|
_.last = function(array, n, guard) {
|
|
if (array == null) return void 0;
|
|
if ((n == null) || guard) return array[array.length - 1];
|
|
return slice.call(array, Math.max(array.length - n, 0));
|
|
};
|
|
|
|
// Returns everything but the first entry of the array. Aliased as `tail` and `drop`.
|
|
// Especially useful on the arguments object. Passing an **n** will return
|
|
// the rest N values in the array. The **guard**
|
|
// check allows it to work with `_.map`.
|
|
_.rest = _.tail = _.drop = function(array, n, guard) {
|
|
return slice.call(array, (n == null) || guard ? 1 : n);
|
|
};
|
|
|
|
// Trim out all falsy values from an array.
|
|
_.compact = function(array) {
|
|
return _.filter(array, _.identity);
|
|
};
|
|
|
|
// Internal implementation of a recursive `flatten` function.
|
|
var flatten = function(input, shallow, output) {
|
|
if (shallow && _.every(input, _.isArray)) {
|
|
return concat.apply(output, input);
|
|
}
|
|
each(input, function(value) {
|
|
if (_.isArray(value) || _.isArguments(value)) {
|
|
shallow ? push.apply(output, value) : flatten(value, shallow, output);
|
|
} else {
|
|
output.push(value);
|
|
}
|
|
});
|
|
return output;
|
|
};
|
|
|
|
// Flatten out an array, either recursively (by default), or just one level.
|
|
_.flatten = function(array, shallow) {
|
|
return flatten(array, shallow, []);
|
|
};
|
|
|
|
// Return a version of the array that does not contain the specified value(s).
|
|
_.without = function(array) {
|
|
return _.difference(array, slice.call(arguments, 1));
|
|
};
|
|
|
|
// Split an array into two arrays: one whose elements all satisfy the given
|
|
// predicate, and one whose elements all do not satisfy the predicate.
|
|
_.partition = function(array, predicate) {
|
|
var pass = [], fail = [];
|
|
each(array, function(elem) {
|
|
(predicate(elem) ? pass : fail).push(elem);
|
|
});
|
|
return [pass, fail];
|
|
};
|
|
|
|
// Produce a duplicate-free version of the array. If the array has already
|
|
// been sorted, you have the option of using a faster algorithm.
|
|
// Aliased as `unique`.
|
|
_.uniq = _.unique = function(array, isSorted, iterator, context) {
|
|
if (_.isFunction(isSorted)) {
|
|
context = iterator;
|
|
iterator = isSorted;
|
|
isSorted = false;
|
|
}
|
|
var initial = iterator ? _.map(array, iterator, context) : array;
|
|
var results = [];
|
|
var seen = [];
|
|
each(initial, function(value, index) {
|
|
if (isSorted ? (!index || seen[seen.length - 1] !== value) : !_.contains(seen, value)) {
|
|
seen.push(value);
|
|
results.push(array[index]);
|
|
}
|
|
});
|
|
return results;
|
|
};
|
|
|
|
// Produce an array that contains the union: each distinct element from all of
|
|
// the passed-in arrays.
|
|
_.union = function() {
|
|
return _.uniq(_.flatten(arguments, true));
|
|
};
|
|
|
|
// Produce an array that contains every item shared between all the
|
|
// passed-in arrays.
|
|
_.intersection = function(array) {
|
|
var rest = slice.call(arguments, 1);
|
|
return _.filter(_.uniq(array), function(item) {
|
|
return _.every(rest, function(other) {
|
|
return _.contains(other, item);
|
|
});
|
|
});
|
|
};
|
|
|
|
// Take the difference between one array and a number of other arrays.
|
|
// Only the elements present in just the first array will remain.
|
|
_.difference = function(array) {
|
|
var rest = concat.apply(ArrayProto, slice.call(arguments, 1));
|
|
return _.filter(array, function(value){ return !_.contains(rest, value); });
|
|
};
|
|
|
|
// Zip together multiple lists into a single array -- elements that share
|
|
// an index go together.
|
|
_.zip = function() {
|
|
var length = _.max(_.pluck(arguments, 'length').concat(0));
|
|
var results = new Array(length);
|
|
for (var i = 0; i < length; i++) {
|
|
results[i] = _.pluck(arguments, '' + i);
|
|
}
|
|
return results;
|
|
};
|
|
|
|
// Converts lists into objects. Pass either a single array of `[key, value]`
|
|
// pairs, or two parallel arrays of the same length -- one of keys, and one of
|
|
// the corresponding values.
|
|
_.object = function(list, values) {
|
|
if (list == null) return {};
|
|
var result = {};
|
|
for (var i = 0, length = list.length; i < length; i++) {
|
|
if (values) {
|
|
result[list[i]] = values[i];
|
|
} else {
|
|
result[list[i][0]] = list[i][1];
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
|
|
// If the browser doesn't supply us with indexOf (I'm looking at you, **MSIE**),
|
|
// we need this function. Return the position of the first occurrence of an
|
|
// item in an array, or -1 if the item is not included in the array.
|
|
// Delegates to **ECMAScript 5**'s native `indexOf` if available.
|
|
// If the array is large and already in sort order, pass `true`
|
|
// for **isSorted** to use binary search.
|
|
_.indexOf = function(array, item, isSorted) {
|
|
if (array == null) return -1;
|
|
var i = 0, length = array.length;
|
|
if (isSorted) {
|
|
if (typeof isSorted == 'number') {
|
|
i = (isSorted < 0 ? Math.max(0, length + isSorted) : isSorted);
|
|
} else {
|
|
i = _.sortedIndex(array, item);
|
|
return array[i] === item ? i : -1;
|
|
}
|
|
}
|
|
if (nativeIndexOf && array.indexOf === nativeIndexOf) return array.indexOf(item, isSorted);
|
|
for (; i < length; i++) if (array[i] === item) return i;
|
|
return -1;
|
|
};
|
|
|
|
// Delegates to **ECMAScript 5**'s native `lastIndexOf` if available.
|
|
_.lastIndexOf = function(array, item, from) {
|
|
if (array == null) return -1;
|
|
var hasIndex = from != null;
|
|
if (nativeLastIndexOf && array.lastIndexOf === nativeLastIndexOf) {
|
|
return hasIndex ? array.lastIndexOf(item, from) : array.lastIndexOf(item);
|
|
}
|
|
var i = (hasIndex ? from : array.length);
|
|
while (i--) if (array[i] === item) return i;
|
|
return -1;
|
|
};
|
|
|
|
// Generate an integer Array containing an arithmetic progression. A port of
|
|
// the native Python `range()` function. See
|
|
// [the Python documentation](http://docs.python.org/library/functions.html#range).
|
|
_.range = function(start, stop, step) {
|
|
if (arguments.length <= 1) {
|
|
stop = start || 0;
|
|
start = 0;
|
|
}
|
|
step = arguments[2] || 1;
|
|
|
|
var length = Math.max(Math.ceil((stop - start) / step), 0);
|
|
var idx = 0;
|
|
var range = new Array(length);
|
|
|
|
while(idx < length) {
|
|
range[idx++] = start;
|
|
start += step;
|
|
}
|
|
|
|
return range;
|
|
};
|
|
|
|
// Function (ahem) Functions
|
|
// ------------------
|
|
|
|
// Reusable constructor function for prototype setting.
|
|
var ctor = function(){};
|
|
|
|
// Create a function bound to a given object (assigning `this`, and arguments,
|
|
// optionally). Delegates to **ECMAScript 5**'s native `Function.bind` if
|
|
// available.
|
|
_.bind = function(func, context) {
|
|
var args, bound;
|
|
if (nativeBind && func.bind === nativeBind) return nativeBind.apply(func, slice.call(arguments, 1));
|
|
if (!_.isFunction(func)) throw new TypeError;
|
|
args = slice.call(arguments, 2);
|
|
return bound = function() {
|
|
if (!(this instanceof bound)) return func.apply(context, args.concat(slice.call(arguments)));
|
|
ctor.prototype = func.prototype;
|
|
var self = new ctor;
|
|
ctor.prototype = null;
|
|
var result = func.apply(self, args.concat(slice.call(arguments)));
|
|
if (Object(result) === result) return result;
|
|
return self;
|
|
};
|
|
};
|
|
|
|
// Partially apply a function by creating a version that has had some of its
|
|
// arguments pre-filled, without changing its dynamic `this` context. _ acts
|
|
// as a placeholder, allowing any combination of arguments to be pre-filled.
|
|
_.partial = function(func) {
|
|
var boundArgs = slice.call(arguments, 1);
|
|
return function() {
|
|
var position = 0;
|
|
var args = boundArgs.slice();
|
|
for (var i = 0, length = args.length; i < length; i++) {
|
|
if (args[i] === _) args[i] = arguments[position++];
|
|
}
|
|
while (position < arguments.length) args.push(arguments[position++]);
|
|
return func.apply(this, args);
|
|
};
|
|
};
|
|
|
|
// Bind a number of an object's methods to that object. Remaining arguments
|
|
// are the method names to be bound. Useful for ensuring that all callbacks
|
|
// defined on an object belong to it.
|
|
_.bindAll = function(obj) {
|
|
var funcs = slice.call(arguments, 1);
|
|
if (funcs.length === 0) throw new Error('bindAll must be passed function names');
|
|
each(funcs, function(f) { obj[f] = _.bind(obj[f], obj); });
|
|
return obj;
|
|
};
|
|
|
|
// Memoize an expensive function by storing its results.
|
|
_.memoize = function(func, hasher) {
|
|
var memo = {};
|
|
hasher || (hasher = _.identity);
|
|
return function() {
|
|
var key = hasher.apply(this, arguments);
|
|
return _.has(memo, key) ? memo[key] : (memo[key] = func.apply(this, arguments));
|
|
};
|
|
};
|
|
|
|
// Delays a function for the given number of milliseconds, and then calls
|
|
// it with the arguments supplied.
|
|
_.delay = function(func, wait) {
|
|
var args = slice.call(arguments, 2);
|
|
return setTimeout(function(){ return func.apply(null, args); }, wait);
|
|
};
|
|
|
|
// Defers a function, scheduling it to run after the current call stack has
|
|
// cleared.
|
|
_.defer = function(func) {
|
|
return _.delay.apply(_, [func, 1].concat(slice.call(arguments, 1)));
|
|
};
|
|
|
|
// Returns a function, that, when invoked, will only be triggered at most once
|
|
// during a given window of time. Normally, the throttled function will run
|
|
// as much as it can, without ever going more than once per `wait` duration;
|
|
// but if you'd like to disable the execution on the leading edge, pass
|
|
// `{leading: false}`. To disable execution on the trailing edge, ditto.
|
|
_.throttle = function(func, wait, options) {
|
|
var context, args, result;
|
|
var timeout = null;
|
|
var previous = 0;
|
|
options || (options = {});
|
|
var later = function() {
|
|
previous = options.leading === false ? 0 : _.now();
|
|
timeout = null;
|
|
result = func.apply(context, args);
|
|
context = args = null;
|
|
};
|
|
return function() {
|
|
var now = _.now();
|
|
if (!previous && options.leading === false) previous = now;
|
|
var remaining = wait - (now - previous);
|
|
context = this;
|
|
args = arguments;
|
|
if (remaining <= 0) {
|
|
clearTimeout(timeout);
|
|
timeout = null;
|
|
previous = now;
|
|
result = func.apply(context, args);
|
|
context = args = null;
|
|
} else if (!timeout && options.trailing !== false) {
|
|
timeout = setTimeout(later, remaining);
|
|
}
|
|
return result;
|
|
};
|
|
};
|
|
|
|
// Returns a function, that, as long as it continues to be invoked, will not
|
|
// be triggered. The function will be called after it stops being called for
|
|
// N milliseconds. If `immediate` is passed, trigger the function on the
|
|
// leading edge, instead of the trailing.
|
|
_.debounce = function(func, wait, immediate) {
|
|
var timeout, args, context, timestamp, result;
|
|
|
|
var later = function() {
|
|
var last = _.now() - timestamp;
|
|
if (last < wait) {
|
|
timeout = setTimeout(later, wait - last);
|
|
} else {
|
|
timeout = null;
|
|
if (!immediate) {
|
|
result = func.apply(context, args);
|
|
context = args = null;
|
|
}
|
|
}
|
|
};
|
|
|
|
return function() {
|
|
context = this;
|
|
args = arguments;
|
|
timestamp = _.now();
|
|
var callNow = immediate && !timeout;
|
|
if (!timeout) {
|
|
timeout = setTimeout(later, wait);
|
|
}
|
|
if (callNow) {
|
|
result = func.apply(context, args);
|
|
context = args = null;
|
|
}
|
|
|
|
return result;
|
|
};
|
|
};
|
|
|
|
// Returns a function that will be executed at most one time, no matter how
|
|
// often you call it. Useful for lazy initialization.
|
|
_.once = function(func) {
|
|
var ran = false, memo;
|
|
return function() {
|
|
if (ran) return memo;
|
|
ran = true;
|
|
memo = func.apply(this, arguments);
|
|
func = null;
|
|
return memo;
|
|
};
|
|
};
|
|
|
|
// Returns the first function passed as an argument to the second,
|
|
// allowing you to adjust arguments, run code before and after, and
|
|
// conditionally execute the original function.
|
|
_.wrap = function(func, wrapper) {
|
|
return _.partial(wrapper, func);
|
|
};
|
|
|
|
// Returns a function that is the composition of a list of functions, each
|
|
// consuming the return value of the function that follows.
|
|
_.compose = function() {
|
|
var funcs = arguments;
|
|
return function() {
|
|
var args = arguments;
|
|
for (var i = funcs.length - 1; i >= 0; i--) {
|
|
args = [funcs[i].apply(this, args)];
|
|
}
|
|
return args[0];
|
|
};
|
|
};
|
|
|
|
// Returns a function that will only be executed after being called N times.
|
|
_.after = function(times, func) {
|
|
return function() {
|
|
if (--times < 1) {
|
|
return func.apply(this, arguments);
|
|
}
|
|
};
|
|
};
|
|
|
|
// Object Functions
|
|
// ----------------
|
|
|
|
// Retrieve the names of an object's properties.
|
|
// Delegates to **ECMAScript 5**'s native `Object.keys`
|
|
_.keys = function(obj) {
|
|
if (!_.isObject(obj)) return [];
|
|
if (nativeKeys) return nativeKeys(obj);
|
|
var keys = [];
|
|
for (var key in obj) if (_.has(obj, key)) keys.push(key);
|
|
return keys;
|
|
};
|
|
|
|
// Retrieve the values of an object's properties.
|
|
_.values = function(obj) {
|
|
var keys = _.keys(obj);
|
|
var length = keys.length;
|
|
var values = new Array(length);
|
|
for (var i = 0; i < length; i++) {
|
|
values[i] = obj[keys[i]];
|
|
}
|
|
return values;
|
|
};
|
|
|
|
// Convert an object into a list of `[key, value]` pairs.
|
|
_.pairs = function(obj) {
|
|
var keys = _.keys(obj);
|
|
var length = keys.length;
|
|
var pairs = new Array(length);
|
|
for (var i = 0; i < length; i++) {
|
|
pairs[i] = [keys[i], obj[keys[i]]];
|
|
}
|
|
return pairs;
|
|
};
|
|
|
|
// Invert the keys and values of an object. The values must be serializable.
|
|
_.invert = function(obj) {
|
|
var result = {};
|
|
var keys = _.keys(obj);
|
|
for (var i = 0, length = keys.length; i < length; i++) {
|
|
result[obj[keys[i]]] = keys[i];
|
|
}
|
|
return result;
|
|
};
|
|
|
|
// Return a sorted list of the function names available on the object.
|
|
// Aliased as `methods`
|
|
_.functions = _.methods = function(obj) {
|
|
var names = [];
|
|
for (var key in obj) {
|
|
if (_.isFunction(obj[key])) names.push(key);
|
|
}
|
|
return names.sort();
|
|
};
|
|
|
|
// Extend a given object with all the properties in passed-in object(s).
|
|
_.extend = function(obj) {
|
|
each(slice.call(arguments, 1), function(source) {
|
|
if (source) {
|
|
for (var prop in source) {
|
|
obj[prop] = source[prop];
|
|
}
|
|
}
|
|
});
|
|
return obj;
|
|
};
|
|
|
|
// Return a copy of the object only containing the whitelisted properties.
|
|
_.pick = function(obj) {
|
|
var copy = {};
|
|
var keys = concat.apply(ArrayProto, slice.call(arguments, 1));
|
|
each(keys, function(key) {
|
|
if (key in obj) copy[key] = obj[key];
|
|
});
|
|
return copy;
|
|
};
|
|
|
|
// Return a copy of the object without the blacklisted properties.
|
|
_.omit = function(obj) {
|
|
var copy = {};
|
|
var keys = concat.apply(ArrayProto, slice.call(arguments, 1));
|
|
for (var key in obj) {
|
|
if (!_.contains(keys, key)) copy[key] = obj[key];
|
|
}
|
|
return copy;
|
|
};
|
|
|
|
// Fill in a given object with default properties.
|
|
_.defaults = function(obj) {
|
|
each(slice.call(arguments, 1), function(source) {
|
|
if (source) {
|
|
for (var prop in source) {
|
|
if (obj[prop] === void 0) obj[prop] = source[prop];
|
|
}
|
|
}
|
|
});
|
|
return obj;
|
|
};
|
|
|
|
// Create a (shallow-cloned) duplicate of an object.
|
|
_.clone = function(obj) {
|
|
if (!_.isObject(obj)) return obj;
|
|
return _.isArray(obj) ? obj.slice() : _.extend({}, obj);
|
|
};
|
|
|
|
// Invokes interceptor with the obj, and then returns obj.
|
|
// The primary purpose of this method is to "tap into" a method chain, in
|
|
// order to perform operations on intermediate results within the chain.
|
|
_.tap = function(obj, interceptor) {
|
|
interceptor(obj);
|
|
return obj;
|
|
};
|
|
|
|
// Internal recursive comparison function for `isEqual`.
|
|
var eq = function(a, b, aStack, bStack) {
|
|
// Identical objects are equal. `0 === -0`, but they aren't identical.
|
|
// See the [Harmony `egal` proposal](http://wiki.ecmascript.org/doku.php?id=harmony:egal).
|
|
if (a === b) return a !== 0 || 1 / a == 1 / b;
|
|
// A strict comparison is necessary because `null == undefined`.
|
|
if (a == null || b == null) return a === b;
|
|
// Unwrap any wrapped objects.
|
|
if (a instanceof _) a = a._wrapped;
|
|
if (b instanceof _) b = b._wrapped;
|
|
// Compare `[[Class]]` names.
|
|
var className = toString.call(a);
|
|
if (className != toString.call(b)) return false;
|
|
switch (className) {
|
|
// Strings, numbers, dates, and booleans are compared by value.
|
|
case '[object String]':
|
|
// Primitives and their corresponding object wrappers are equivalent; thus, `"5"` is
|
|
// equivalent to `new String("5")`.
|
|
return a == String(b);
|
|
case '[object Number]':
|
|
// `NaN`s are equivalent, but non-reflexive. An `egal` comparison is performed for
|
|
// other numeric values.
|
|
return a != +a ? b != +b : (a == 0 ? 1 / a == 1 / b : a == +b);
|
|
case '[object Date]':
|
|
case '[object Boolean]':
|
|
// Coerce dates and booleans to numeric primitive values. Dates are compared by their
|
|
// millisecond representations. Note that invalid dates with millisecond representations
|
|
// of `NaN` are not equivalent.
|
|
return +a == +b;
|
|
// RegExps are compared by their source patterns and flags.
|
|
case '[object RegExp]':
|
|
return a.source == b.source &&
|
|
a.global == b.global &&
|
|
a.multiline == b.multiline &&
|
|
a.ignoreCase == b.ignoreCase;
|
|
}
|
|
if (typeof a != 'object' || typeof b != 'object') return false;
|
|
// Assume equality for cyclic structures. The algorithm for detecting cyclic
|
|
// structures is adapted from ES 5.1 section 15.12.3, abstract operation `JO`.
|
|
var length = aStack.length;
|
|
while (length--) {
|
|
// Linear search. Performance is inversely proportional to the number of
|
|
// unique nested structures.
|
|
if (aStack[length] == a) return bStack[length] == b;
|
|
}
|
|
// Objects with different constructors are not equivalent, but `Object`s
|
|
// from different frames are.
|
|
var aCtor = a.constructor, bCtor = b.constructor;
|
|
if (aCtor !== bCtor && !(_.isFunction(aCtor) && (aCtor instanceof aCtor) &&
|
|
_.isFunction(bCtor) && (bCtor instanceof bCtor))
|
|
&& ('constructor' in a && 'constructor' in b)) {
|
|
return false;
|
|
}
|
|
// Add the first object to the stack of traversed objects.
|
|
aStack.push(a);
|
|
bStack.push(b);
|
|
var size = 0, result = true;
|
|
// Recursively compare objects and arrays.
|
|
if (className == '[object Array]') {
|
|
// Compare array lengths to determine if a deep comparison is necessary.
|
|
size = a.length;
|
|
result = size == b.length;
|
|
if (result) {
|
|
// Deep compare the contents, ignoring non-numeric properties.
|
|
while (size--) {
|
|
if (!(result = eq(a[size], b[size], aStack, bStack))) break;
|
|
}
|
|
}
|
|
} else {
|
|
// Deep compare objects.
|
|
for (var key in a) {
|
|
if (_.has(a, key)) {
|
|
// Count the expected number of properties.
|
|
size++;
|
|
// Deep compare each member.
|
|
if (!(result = _.has(b, key) && eq(a[key], b[key], aStack, bStack))) break;
|
|
}
|
|
}
|
|
// Ensure that both objects contain the same number of properties.
|
|
if (result) {
|
|
for (key in b) {
|
|
if (_.has(b, key) && !(size--)) break;
|
|
}
|
|
result = !size;
|
|
}
|
|
}
|
|
// Remove the first object from the stack of traversed objects.
|
|
aStack.pop();
|
|
bStack.pop();
|
|
return result;
|
|
};
|
|
|
|
// Perform a deep comparison to check if two objects are equal.
|
|
_.isEqual = function(a, b) {
|
|
return eq(a, b, [], []);
|
|
};
|
|
|
|
// Is a given array, string, or object empty?
|
|
// An "empty" object has no enumerable own-properties.
|
|
_.isEmpty = function(obj) {
|
|
if (obj == null) return true;
|
|
if (_.isArray(obj) || _.isString(obj)) return obj.length === 0;
|
|
for (var key in obj) if (_.has(obj, key)) return false;
|
|
return true;
|
|
};
|
|
|
|
// Is a given value a DOM element?
|
|
_.isElement = function(obj) {
|
|
return !!(obj && obj.nodeType === 1);
|
|
};
|
|
|
|
// Is a given value an array?
|
|
// Delegates to ECMA5's native Array.isArray
|
|
_.isArray = nativeIsArray || function(obj) {
|
|
return toString.call(obj) == '[object Array]';
|
|
};
|
|
|
|
// Is a given variable an object?
|
|
_.isObject = function(obj) {
|
|
return obj === Object(obj);
|
|
};
|
|
|
|
// Add some isType methods: isArguments, isFunction, isString, isNumber, isDate, isRegExp.
|
|
each(['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp'], function(name) {
|
|
_['is' + name] = function(obj) {
|
|
return toString.call(obj) == '[object ' + name + ']';
|
|
};
|
|
});
|
|
|
|
// Define a fallback version of the method in browsers (ahem, IE), where
|
|
// there isn't any inspectable "Arguments" type.
|
|
if (!_.isArguments(arguments)) {
|
|
_.isArguments = function(obj) {
|
|
return !!(obj && _.has(obj, 'callee'));
|
|
};
|
|
}
|
|
|
|
// Optimize `isFunction` if appropriate.
|
|
if (typeof (/./) !== 'function') {
|
|
_.isFunction = function(obj) {
|
|
return typeof obj === 'function';
|
|
};
|
|
}
|
|
|
|
// Is a given object a finite number?
|
|
_.isFinite = function(obj) {
|
|
return isFinite(obj) && !isNaN(parseFloat(obj));
|
|
};
|
|
|
|
// Is the given value `NaN`? (NaN is the only number which does not equal itself).
|
|
_.isNaN = function(obj) {
|
|
return _.isNumber(obj) && obj != +obj;
|
|
};
|
|
|
|
// Is a given value a boolean?
|
|
_.isBoolean = function(obj) {
|
|
return obj === true || obj === false || toString.call(obj) == '[object Boolean]';
|
|
};
|
|
|
|
// Is a given value equal to null?
|
|
_.isNull = function(obj) {
|
|
return obj === null;
|
|
};
|
|
|
|
// Is a given variable undefined?
|
|
_.isUndefined = function(obj) {
|
|
return obj === void 0;
|
|
};
|
|
|
|
// Shortcut function for checking if an object has a given property directly
|
|
// on itself (in other words, not on a prototype).
|
|
_.has = function(obj, key) {
|
|
return hasOwnProperty.call(obj, key);
|
|
};
|
|
|
|
// Utility Functions
|
|
// -----------------
|
|
|
|
// Run Underscore.js in *noConflict* mode, returning the `_` variable to its
|
|
// previous owner. Returns a reference to the Underscore object.
|
|
_.noConflict = function() {
|
|
root._ = previousUnderscore;
|
|
return this;
|
|
};
|
|
|
|
// Keep the identity function around for default iterators.
|
|
_.identity = function(value) {
|
|
return value;
|
|
};
|
|
|
|
_.constant = function(value) {
|
|
return function () {
|
|
return value;
|
|
};
|
|
};
|
|
|
|
_.property = function(key) {
|
|
return function(obj) {
|
|
return obj[key];
|
|
};
|
|
};
|
|
|
|
// Returns a predicate for checking whether an object has a given set of `key:value` pairs.
|
|
_.matches = function(attrs) {
|
|
return function(obj) {
|
|
if (obj === attrs) return true; //avoid comparing an object to itself.
|
|
for (var key in attrs) {
|
|
if (attrs[key] !== obj[key])
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
};
|
|
|
|
// Run a function **n** times.
|
|
_.times = function(n, iterator, context) {
|
|
var accum = Array(Math.max(0, n));
|
|
for (var i = 0; i < n; i++) accum[i] = iterator.call(context, i);
|
|
return accum;
|
|
};
|
|
|
|
// Return a random integer between min and max (inclusive).
|
|
_.random = function(min, max) {
|
|
if (max == null) {
|
|
max = min;
|
|
min = 0;
|
|
}
|
|
return min + Math.floor(Math.random() * (max - min + 1));
|
|
};
|
|
|
|
// A (possibly faster) way to get the current timestamp as an integer.
|
|
_.now = Date.now || function() { return new Date().getTime(); };
|
|
|
|
// List of HTML entities for escaping.
|
|
var entityMap = {
|
|
escape: {
|
|
'&': '&',
|
|
'<': '<',
|
|
'>': '>',
|
|
'"': '"',
|
|
"'": '''
|
|
}
|
|
};
|
|
entityMap.unescape = _.invert(entityMap.escape);
|
|
|
|
// Regexes containing the keys and values listed immediately above.
|
|
var entityRegexes = {
|
|
escape: new RegExp('[' + _.keys(entityMap.escape).join('') + ']', 'g'),
|
|
unescape: new RegExp('(' + _.keys(entityMap.unescape).join('|') + ')', 'g')
|
|
};
|
|
|
|
// Functions for escaping and unescaping strings to/from HTML interpolation.
|
|
_.each(['escape', 'unescape'], function(method) {
|
|
_[method] = function(string) {
|
|
if (string == null) return '';
|
|
return ('' + string).replace(entityRegexes[method], function(match) {
|
|
return entityMap[method][match];
|
|
});
|
|
};
|
|
});
|
|
|
|
// If the value of the named `property` is a function then invoke it with the
|
|
// `object` as context; otherwise, return it.
|
|
_.result = function(object, property) {
|
|
if (object == null) return void 0;
|
|
var value = object[property];
|
|
return _.isFunction(value) ? value.call(object) : value;
|
|
};
|
|
|
|
// Add your own custom functions to the Underscore object.
|
|
_.mixin = function(obj) {
|
|
each(_.functions(obj), function(name) {
|
|
var func = _[name] = obj[name];
|
|
_.prototype[name] = function() {
|
|
var args = [this._wrapped];
|
|
push.apply(args, arguments);
|
|
return result.call(this, func.apply(_, args));
|
|
};
|
|
});
|
|
};
|
|
|
|
// Generate a unique integer id (unique within the entire client session).
|
|
// Useful for temporary DOM ids.
|
|
var idCounter = 0;
|
|
_.uniqueId = function(prefix) {
|
|
var id = ++idCounter + '';
|
|
return prefix ? prefix + id : id;
|
|
};
|
|
|
|
// By default, Underscore uses ERB-style template delimiters, change the
|
|
// following template settings to use alternative delimiters.
|
|
_.templateSettings = {
|
|
evaluate : /<%([\s\S]+?)%>/g,
|
|
interpolate : /<%=([\s\S]+?)%>/g,
|
|
escape : /<%-([\s\S]+?)%>/g
|
|
};
|
|
|
|
// When customizing `templateSettings`, if you don't want to define an
|
|
// interpolation, evaluation or escaping regex, we need one that is
|
|
// guaranteed not to match.
|
|
var noMatch = /(.)^/;
|
|
|
|
// Certain characters need to be escaped so that they can be put into a
|
|
// string literal.
|
|
var escapes = {
|
|
"'": "'",
|
|
'\\': '\\',
|
|
'\r': 'r',
|
|
'\n': 'n',
|
|
'\t': 't',
|
|
'\u2028': 'u2028',
|
|
'\u2029': 'u2029'
|
|
};
|
|
|
|
var escaper = /\\|'|\r|\n|\t|\u2028|\u2029/g;
|
|
|
|
// JavaScript micro-templating, similar to John Resig's implementation.
|
|
// Underscore templating handles arbitrary delimiters, preserves whitespace,
|
|
// and correctly escapes quotes within interpolated code.
|
|
_.template = function(text, data, settings) {
|
|
var render;
|
|
settings = _.defaults({}, settings, _.templateSettings);
|
|
|
|
// Combine delimiters into one regular expression via alternation.
|
|
var matcher = new RegExp([
|
|
(settings.escape || noMatch).source,
|
|
(settings.interpolate || noMatch).source,
|
|
(settings.evaluate || noMatch).source
|
|
].join('|') + '|$', 'g');
|
|
|
|
// Compile the template source, escaping string literals appropriately.
|
|
var index = 0;
|
|
var source = "__p+='";
|
|
text.replace(matcher, function(match, escape, interpolate, evaluate, offset) {
|
|
source += text.slice(index, offset)
|
|
.replace(escaper, function(match) { return '\\' + escapes[match]; });
|
|
|
|
if (escape) {
|
|
source += "'+\n((__t=(" + escape + "))==null?'':_.escape(__t))+\n'";
|
|
}
|
|
if (interpolate) {
|
|
source += "'+\n((__t=(" + interpolate + "))==null?'':__t)+\n'";
|
|
}
|
|
if (evaluate) {
|
|
source += "';\n" + evaluate + "\n__p+='";
|
|
}
|
|
index = offset + match.length;
|
|
return match;
|
|
});
|
|
source += "';\n";
|
|
|
|
// If a variable is not specified, place data values in local scope.
|
|
if (!settings.variable) source = 'with(obj||{}){\n' + source + '}\n';
|
|
|
|
source = "var __t,__p='',__j=Array.prototype.join," +
|
|
"print=function(){__p+=__j.call(arguments,'');};\n" +
|
|
source + "return __p;\n";
|
|
|
|
try {
|
|
render = new Function(settings.variable || 'obj', '_', source);
|
|
} catch (e) {
|
|
e.source = source;
|
|
throw e;
|
|
}
|
|
|
|
if (data) return render(data, _);
|
|
var template = function(data) {
|
|
return render.call(this, data, _);
|
|
};
|
|
|
|
// Provide the compiled function source as a convenience for precompilation.
|
|
template.source = 'function(' + (settings.variable || 'obj') + '){\n' + source + '}';
|
|
|
|
return template;
|
|
};
|
|
|
|
// Add a "chain" function, which will delegate to the wrapper.
|
|
_.chain = function(obj) {
|
|
return _(obj).chain();
|
|
};
|
|
|
|
// OOP
|
|
// ---------------
|
|
// If Underscore is called as a function, it returns a wrapped object that
|
|
// can be used OO-style. This wrapper holds altered versions of all the
|
|
// underscore functions. Wrapped objects may be chained.
|
|
|
|
// Helper function to continue chaining intermediate results.
|
|
var result = function(obj) {
|
|
return this._chain ? _(obj).chain() : obj;
|
|
};
|
|
|
|
// Add all of the Underscore functions to the wrapper object.
|
|
_.mixin(_);
|
|
|
|
// Add all mutator Array functions to the wrapper.
|
|
each(['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'], function(name) {
|
|
var method = ArrayProto[name];
|
|
_.prototype[name] = function() {
|
|
var obj = this._wrapped;
|
|
method.apply(obj, arguments);
|
|
if ((name == 'shift' || name == 'splice') && obj.length === 0) delete obj[0];
|
|
return result.call(this, obj);
|
|
};
|
|
});
|
|
|
|
// Add all accessor Array functions to the wrapper.
|
|
each(['concat', 'join', 'slice'], function(name) {
|
|
var method = ArrayProto[name];
|
|
_.prototype[name] = function() {
|
|
return result.call(this, method.apply(this._wrapped, arguments));
|
|
};
|
|
});
|
|
|
|
_.extend(_.prototype, {
|
|
|
|
// Start chaining a wrapped Underscore object.
|
|
chain: function() {
|
|
this._chain = true;
|
|
return this;
|
|
},
|
|
|
|
// Extracts the result from a wrapped and chained object.
|
|
value: function() {
|
|
return this._wrapped;
|
|
}
|
|
|
|
});
|
|
|
|
// AMD registration happens at the end for compatibility with AMD loaders
|
|
// that may not enforce next-turn semantics on modules. Even though general
|
|
// practice for AMD registration is to be anonymous, underscore registers
|
|
// as a named module because, like jQuery, it is a base library that is
|
|
// popular enough to be bundled in a third party lib, but not be part of
|
|
// an AMD load request. Those cases could generate an error when an
|
|
// anonymous define() is called outside of a loader request.
|
|
if (typeof define === 'function' && define.amd) {
|
|
define('underscore', [], function() {
|
|
return _;
|
|
});
|
|
}
|
|
}).call(this);
|
|
|
|
},{}],"kytKTK":[function(require,module,exports){
|
|
(function (Buffer){
|
|
exports.patch = function(Buffers) {
|
|
Buffers.prototype.skip = function (i) {
|
|
if (i == 0) {
|
|
return;
|
|
} else if (i == this.length) {
|
|
this.buffers = [];
|
|
this.length = 0;
|
|
return;
|
|
}
|
|
var pos = this.pos(i);
|
|
this.buffers = this.buffers.slice(pos.buf);
|
|
this.buffers[0] = new Buffer(this.buffers[0].slice(pos.offset));
|
|
this.length -= i;
|
|
};
|
|
};
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80}],"./patches/Buffers.monkey":[function(require,module,exports){
|
|
module.exports=require('kytKTK');
|
|
},{}],"AwmEwz":[function(require,module,exports){
|
|
exports.patch = function(Number) {
|
|
//round to specified number of places
|
|
Number.prototype.round = function(places) {
|
|
if(!places) return Math.round(this);
|
|
var tmp = Math.pow(10,places);
|
|
return Math.round(this * tmp) / tmp;
|
|
};
|
|
};
|
|
|
|
},{}],"./patches/Number.monkey":[function(require,module,exports){
|
|
module.exports=require('AwmEwz');
|
|
},{}],"./util/BinaryParser":[function(require,module,exports){
|
|
module.exports=require('b3ZSD7');
|
|
},{}],"b3ZSD7":[function(require,module,exports){
|
|
(function (Buffer){
|
|
/**
|
|
* Simple synchronous parser based on node-binary.
|
|
*/
|
|
|
|
var imports = require('soop').imports();
|
|
function Parser(buffer)
|
|
{
|
|
this.subject = buffer;
|
|
this.pos = 0;
|
|
};
|
|
|
|
Parser.prototype.buffer = function buffer(len) {
|
|
var buf = this.subject.slice(this.pos, this.pos+len);
|
|
this.pos += len;
|
|
return buf;
|
|
};
|
|
|
|
Parser.prototype.search = function search(needle) {
|
|
var len;
|
|
|
|
if ("string" === typeof needle || Buffer.isBuffer(needle)) {
|
|
// TODO: Slicing is probably too slow
|
|
len = this.subject.slice(this.pos).indexOf(needle);
|
|
if (len !== -1) {
|
|
this.pos += len + needle.length;
|
|
}
|
|
return len;
|
|
}
|
|
if ("number" === typeof needle) {
|
|
needle = needle & 0xff;
|
|
// Search for single byte
|
|
for (var i = this.pos, l = this.subject.length; i < l; i++) {
|
|
if (this.subject[i] == needle) {
|
|
len = i - this.pos;
|
|
this.pos = i+1;
|
|
return len;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Like search(), but returns the skipped bytes
|
|
*/
|
|
Parser.prototype.scan = function scan(needle) {
|
|
var startPos = this.pos;
|
|
var len = this.search(needle);
|
|
if (len !== -1) {
|
|
return this.subject.slice(startPos, startPos+len);
|
|
} else {
|
|
throw new Error('No match');
|
|
}
|
|
};
|
|
|
|
Parser.prototype.eof = function eof() {
|
|
return this.pos >= this.subject.length;
|
|
};
|
|
|
|
// convert byte strings to unsigned little endian numbers
|
|
function decodeLEu (bytes) {
|
|
var acc = 0;
|
|
for (var i = 0; i < bytes.length; i++) {
|
|
acc += Math.pow(256,i) * bytes[i];
|
|
}
|
|
return acc;
|
|
}
|
|
|
|
// convert byte strings to unsigned big endian numbers
|
|
function decodeBEu (bytes) {
|
|
var acc = 0;
|
|
for (var i = 0; i < bytes.length; i++) {
|
|
acc += Math.pow(256, bytes.length - i - 1) * bytes[i];
|
|
}
|
|
return acc;
|
|
}
|
|
|
|
// convert byte strings to signed big endian numbers
|
|
function decodeBEs (bytes) {
|
|
var val = decodeBEu(bytes);
|
|
if ((bytes[0] & 0x80) == 0x80) {
|
|
val -= Math.pow(256, bytes.length);
|
|
}
|
|
return val;
|
|
}
|
|
|
|
// convert byte strings to signed little endian numbers
|
|
function decodeLEs (bytes) {
|
|
var val = decodeLEu(bytes);
|
|
if ((bytes[bytes.length - 1] & 0x80) == 0x80) {
|
|
val -= Math.pow(256, bytes.length);
|
|
}
|
|
return val;
|
|
}
|
|
|
|
function getDecoder(len, fn) {
|
|
return function () {
|
|
var buf = this.buffer(len);
|
|
return fn(buf);
|
|
};
|
|
};
|
|
[ 1, 2, 4, 8 ].forEach(function (bytes) {
|
|
var bits = bytes * 8;
|
|
|
|
Parser.prototype['word' + bits + 'le']
|
|
= Parser.prototype['word' + bits + 'lu']
|
|
= getDecoder(bytes, decodeLEu);
|
|
|
|
Parser.prototype['word' + bits + 'ls']
|
|
= getDecoder(bytes, decodeLEs);
|
|
|
|
Parser.prototype['word' + bits + 'be']
|
|
= Parser.prototype['word' + bits + 'bu']
|
|
= getDecoder(bytes, decodeBEu);
|
|
|
|
Parser.prototype['word' + bits + 'bs']
|
|
= getDecoder(bytes, decodeBEs);
|
|
|
|
Parser.prototype.word8 = Parser.prototype.word8u = Parser.prototype.word8be;
|
|
Parser.prototype.word8s = Parser.prototype.word8bs;
|
|
});
|
|
|
|
Parser.prototype.varInt = function ()
|
|
{
|
|
var firstByte = this.word8();
|
|
switch (firstByte) {
|
|
case 0xFD:
|
|
return this.word16le();
|
|
|
|
case 0xFE:
|
|
return this.word32le();
|
|
|
|
case 0xFF:
|
|
return this.word64le();
|
|
|
|
default:
|
|
return firstByte;
|
|
}
|
|
};
|
|
|
|
Parser.prototype.varStr = function () {
|
|
var len = this.varInt();
|
|
return this.buffer(len);
|
|
};
|
|
|
|
module.exports = require('soop')(Parser);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"buffer":80,"soop":125}],134:[function(require,module,exports){
|
|
|
|
var fs = require('fs');
|
|
var crypto = require('crypto');
|
|
|
|
exports.readFileSync = function(enc_method, enc_passphrase, filename)
|
|
{
|
|
// read entire file into memory
|
|
var fileData = fs.readFileSync(filename, 'binary');
|
|
if (fileData.length < 32)
|
|
throw new Error("Crypted file " + filename + " truncated");
|
|
|
|
// separate into data, hmac parts
|
|
var fileCrypted = fileData.slice(0, -32);
|
|
var fileHmac = fileData.slice(-32);
|
|
|
|
// generate and verify HMAC
|
|
var hmac = crypto.createHmac('sha256', enc_passphrase);
|
|
hmac.update(fileCrypted);
|
|
var digest = hmac.digest('binary');
|
|
|
|
if (digest.toString() != fileHmac.toString())
|
|
throw new Error("Crypted file " + filename + " failed HMAC checksum verification");
|
|
|
|
// decrypt to plaintext
|
|
var decipher = crypto.createDecipher(enc_method, enc_passphrase);
|
|
var dec = decipher.update(fileCrypted, 'binary', 'binary');
|
|
dec += decipher.final('binary');
|
|
return dec;
|
|
};
|
|
|
|
exports.readJFileSync = function(enc_method, enc_passphrase, filename)
|
|
{
|
|
var raw = this.readFileSync(enc_method, enc_passphrase, filename);
|
|
return JSON.parse(raw);
|
|
};
|
|
|
|
exports.writeFileSync = function(enc_method, enc_passphrase, filename, data)
|
|
{
|
|
// encrypt to ciphertext
|
|
var cipher = crypto.createCipher(enc_method, enc_passphrase);
|
|
var crypted = cipher.update(data, 'binary', 'binary');
|
|
crypted += cipher.final('binary');
|
|
|
|
// compute HMAC
|
|
var hmac = crypto.createHmac('sha256', enc_passphrase);
|
|
hmac.update(crypted);
|
|
var digest = hmac.digest('binary');
|
|
|
|
fs.writeFileSync(filename, crypted + digest, 'binary');
|
|
|
|
return true;
|
|
};
|
|
|
|
exports.writeJFileSync = function(enc_method, enc_passphrase, filename, obj)
|
|
{
|
|
var raw = JSON.stringify(obj);
|
|
return this.writeFileSync(enc_method, enc_passphrase, filename, raw);
|
|
};
|
|
|
|
|
|
},{"crypto":84,"fs":76}],"./util/EncodedData":[function(require,module,exports){
|
|
module.exports=require('eLfUFE');
|
|
},{}],"eLfUFE":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
var base58 = imports.base58 || require('../lib/Base58').base58Check;
|
|
|
|
|
|
// Constructor. Takes the following forms:
|
|
// new EncodedData(<base58_address_string>)
|
|
// new EncodedData(<binary_buffer>)
|
|
// new EncodedData(<data>, <encoding>)
|
|
// new EncodedData(<version>, <20-byte-hash>)
|
|
function EncodedData(data, encoding) {
|
|
this.data = data;
|
|
if(!encoding && (typeof data == 'string')) {
|
|
this.__proto__ = this.encodings['base58'];
|
|
} else {
|
|
this.__proto__ = this.encodings[encoding || 'binary'];
|
|
}
|
|
};
|
|
|
|
// get or set the encoding used (transforms data)
|
|
EncodedData.prototype.encoding = function(encoding) {
|
|
if(encoding && (encoding != this._encoding)) {
|
|
this.data = this.as(encoding);
|
|
this.__proto__ = this.encodings[encoding];
|
|
}
|
|
return this._encoding;
|
|
};
|
|
|
|
// answer a new instance having the given encoding
|
|
EncodedData.prototype.withEncoding = function(encoding) {
|
|
return new EncodedData(this.as(encoding), encoding);
|
|
};
|
|
|
|
// answer the data in the given encoding
|
|
EncodedData.prototype.as = function(encoding) {
|
|
if(!encodings[encoding]) throw new Error('invalid encoding');
|
|
return this.converters[encoding].call(this);
|
|
};
|
|
|
|
// validate that we can convert to binary
|
|
EncodedData.prototype._validate = function() {
|
|
this.withEncoding('binary');
|
|
};
|
|
|
|
// Boolean protocol for testing if valid
|
|
EncodedData.prototype.isValid = function() {
|
|
try {
|
|
this.validate();
|
|
return true;
|
|
} catch(e) {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
// subclasses can override to do more stuff
|
|
EncodedData.prototype.validate = function() {
|
|
this._validate();
|
|
};
|
|
|
|
// Boolean protocol for testing if valid
|
|
EncodedData.prototype.isValid = function() {
|
|
try {
|
|
this.validate();
|
|
return true;
|
|
} catch(e) {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
// convert to a string (in base58 form)
|
|
EncodedData.prototype.toString = function() {
|
|
return this.as('base58');
|
|
};
|
|
|
|
// utility
|
|
EncodedData.prototype.doAsBinary = function(callback) {
|
|
var oldEncoding = this.encoding();
|
|
this.encoding('binary');
|
|
callback.apply(this);
|
|
this.encoding(oldEncoding);
|
|
};
|
|
|
|
// Setup support for various address encodings. The object for
|
|
// each encoding inherits from the EncodedData prototype. This
|
|
// allows any encoding to override any method...changing the encoding
|
|
// for an instance will change the encoding it inherits from. Note,
|
|
// this will present some problems for anyone wanting to inherit from
|
|
// EncodedData (we'll deal with that when needed).
|
|
var encodings = {
|
|
'binary': {
|
|
converters: {
|
|
'binary': function() {
|
|
var answer = new Buffer(this.data.length);
|
|
this.data.copy(answer);
|
|
return answer;
|
|
},
|
|
'base58': function() {
|
|
return base58.encode(this.data);
|
|
},
|
|
'hex': function() {
|
|
return this.data.toString('hex');
|
|
},
|
|
},
|
|
|
|
_validate: function() {
|
|
//nothing to do here...we make no assumptions about the data
|
|
},
|
|
},
|
|
|
|
'base58': {
|
|
converters: {
|
|
'binary': function() {
|
|
return base58.decode(this.data);
|
|
},
|
|
'hex': function() {
|
|
return this.withEncoding('binary').as('hex');
|
|
},
|
|
},
|
|
},
|
|
|
|
'hex': {
|
|
converters: {
|
|
'binary': function() {
|
|
return new Buffer(this.data, 'hex');
|
|
},
|
|
'base58': function() {
|
|
return this.withEncoding('binary').as('base58');
|
|
},
|
|
},
|
|
},
|
|
};
|
|
|
|
var no_conversion = function() {return this.data;};
|
|
for(var k in encodings) {
|
|
if(encodings.hasOwnProperty(k)){
|
|
if(!encodings[k].converters[k])
|
|
encodings[k].converters[k] = no_conversion;
|
|
encodings[k]._encoding = k;
|
|
}
|
|
}
|
|
|
|
EncodedData.applyEncodingsTo = function(aClass) {
|
|
var tmp = {};
|
|
for(var k in encodings) {
|
|
var enc = encodings[k];
|
|
var obj = {};
|
|
for(var j in enc) {
|
|
obj[j] = enc[j];
|
|
}
|
|
obj.__proto__ = aClass.prototype;
|
|
tmp[k] = obj;
|
|
}
|
|
aClass.prototype.encodings = tmp;
|
|
};
|
|
|
|
EncodedData.applyEncodingsTo(EncodedData);
|
|
|
|
module.exports = require('soop')(EncodedData);
|
|
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../lib/Base58":"6VqyzY","buffer":80,"soop":125}],"./util/VersionedData":[function(require,module,exports){
|
|
module.exports=require('QLzNQg');
|
|
},{}],"QLzNQg":[function(require,module,exports){
|
|
(function (Buffer){
|
|
var imports = require('soop').imports();
|
|
var base58 = imports.base58 || require('../lib/Base58').base58Check;
|
|
var parent = imports.parent || require('./EncodedData');
|
|
|
|
|
|
function VersionedData(version, payload) {
|
|
if(typeof version != 'number') {
|
|
VersionedData.super(this, arguments);
|
|
return;
|
|
};
|
|
this.data = new Buffer(payload.length + 1);
|
|
this.__proto__ = this.encodings['binary'];
|
|
this.version(version);
|
|
this.payload(payload);
|
|
};
|
|
|
|
VersionedData.parent = parent;
|
|
parent.applyEncodingsTo(VersionedData);
|
|
|
|
// get or set the version data (the first byte of the address)
|
|
VersionedData.prototype.version = function(num) {
|
|
if(num || (num === 0)) {
|
|
this.doAsBinary(function() {this.data.writeUInt8(num, 0);});
|
|
return num;
|
|
}
|
|
return this.as('binary').readUInt8(0);
|
|
};
|
|
|
|
// get or set the payload data (as a Buffer object)
|
|
VersionedData.prototype.payload = function(data) {
|
|
if(data) {
|
|
this.doAsBinary(function() {data.copy(this.data,1);});
|
|
return data;
|
|
}
|
|
return this.as('binary').slice(1);
|
|
};
|
|
|
|
module.exports = require('soop')(VersionedData);
|
|
|
|
}).call(this,require("buffer").Buffer)
|
|
},{"../lib/Base58":"6VqyzY","./EncodedData":"eLfUFE","buffer":80,"soop":125}],139:[function(require,module,exports){
|
|
|
|
/**
|
|
* Used during transcation verification when a source txout is missing.
|
|
*
|
|
* When a transaction is being verified by the memory pool this error causes
|
|
* it to be added to the orphan pool instead of being discarded.
|
|
*/
|
|
function MissingSourceError(msg, missingTxHash) {
|
|
// TODO: Since this happens in normal operation, perhaps we should
|
|
// avoid generating a whole stack trace.
|
|
Error.call(this);
|
|
// This is not compatible with firefox.
|
|
// Error.captureStackTrace(this, arguments.callee);
|
|
this.message = msg;
|
|
this.missingTxHash = missingTxHash;
|
|
this.name = 'MissingSourceError';
|
|
};
|
|
|
|
MissingSourceError.prototype.__proto__ = Error.prototype;
|
|
|
|
exports.MissingSourceError = MissingSourceError;
|
|
|
|
|
|
/**
|
|
* Used in several places to indicate invalid data.
|
|
*
|
|
* We want to distinguish invalid data errors from program errors, so we use
|
|
* this exception to indicate the former.
|
|
*/
|
|
function VerificationError(msg, missingTxHash) {
|
|
// TODO: Since this happens in normal operation, perhaps we should
|
|
// avoid generating a whole stack trace.
|
|
Error.call(this);
|
|
|
|
// This is not compatible with firefox.
|
|
// Error.captureStackTrace(this, arguments.callee);
|
|
this.message = msg;
|
|
this.missingTxHash = missingTxHash;
|
|
this.name = 'VerificationError';
|
|
};
|
|
|
|
VerificationError.prototype.__proto__ = Error.prototype;
|
|
|
|
exports.VerificationError = VerificationError;
|
|
|
|
},{}],140:[function(require,module,exports){
|
|
module.exports = require('./util');
|
|
},{"./util":"ACyo5H"}],"./util/log":[function(require,module,exports){
|
|
module.exports=require('AdF7pF');
|
|
},{}],"AdF7pF":[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var noop = function() {};
|
|
var cl = function() {
|
|
console.log(arguments);
|
|
};
|
|
|
|
var loggers = {
|
|
none: {info: noop, warn: noop, err: noop, debug: noop},
|
|
normal: {info: cl, warn: cl, err: cl, debug: noop},
|
|
debug: {info: cl, warn: cl, err: cl, debug: cl},
|
|
};
|
|
|
|
var config = require('../config');
|
|
if(config.log) {
|
|
module.exports = config.log;
|
|
} else {
|
|
module.exports = loggers[config.logger || 'normal'];
|
|
}
|
|
|
|
},{"../config":"4itQ50"}],143:[function(require,module,exports){
|
|
|
|
// current time, in seconds
|
|
exports.curtime = function curtime()
|
|
{
|
|
return Math.round(Date.now() / 1000);
|
|
}
|
|
|
|
|
|
},{}],"ACyo5H":[function(require,module,exports){
|
|
(function (process,Buffer){
|
|
var crypto = require('crypto');
|
|
var bignum = require('bignum');
|
|
var Binary = require('binary');
|
|
var Put = require('bufferput');
|
|
var buffertools = require('buffertools');
|
|
var jssha = require('jssha');
|
|
var browser;
|
|
var inBrowser = !process.versions;
|
|
if (inBrowser) {
|
|
browser = require('../browser/vendor-bundle.js');
|
|
}
|
|
|
|
var sha256 = exports.sha256 = function(data) {
|
|
return new Buffer(crypto.createHash('sha256').update(data).digest('binary'), 'binary');
|
|
};
|
|
|
|
var sha512hmac = exports.sha512hmac = function (data, key) {
|
|
if (inBrowser) {
|
|
var j = new jssha(data.toString('hex'), 'HEX');
|
|
var hash = j.getHMAC(key.toString('hex'), "HEX", "SHA-512", "HEX");
|
|
hash = new Buffer(hash, 'hex');
|
|
return hash;
|
|
};
|
|
var hmac = crypto.createHmac('sha512', key);
|
|
var hash = hmac.update(data).digest();
|
|
return hash;
|
|
};
|
|
|
|
var ripe160 = exports.ripe160 = function (data) {
|
|
if (!Buffer.isBuffer(data)) {
|
|
throw new Error('arg should be a buffer');
|
|
}
|
|
if (inBrowser) {
|
|
var w = new browser.crypto31.lib.WordArray.init(Crypto.util.bytesToWords(data), data.length);
|
|
var wordArray = browser.crypto31.RIPEMD160(w);
|
|
var words = wordArray.words;
|
|
var answer = [];
|
|
for (var b = 0; b < words.length * 32; b += 8) {
|
|
answer.push((words[b >>> 5] >>> (24 - b % 32)) & 0xFF);
|
|
}
|
|
return new Buffer(answer, 'hex');
|
|
}
|
|
return new Buffer(crypto.createHash('rmd160').update(data).digest('binary'), 'binary');
|
|
};
|
|
|
|
var sha1 = exports.sha1 = function(data) {
|
|
return new Buffer(crypto.createHash('sha1').update(data).digest('binary'), 'binary');
|
|
};
|
|
|
|
var twoSha256 = exports.twoSha256 = function(data) {
|
|
return sha256(sha256(data));
|
|
};
|
|
|
|
var sha256ripe160 = exports.sha256ripe160 = function(data) {
|
|
return ripe160(sha256(data));
|
|
};
|
|
|
|
/**
|
|
* Format a block hash like the official client does.
|
|
*/
|
|
var formatHash = exports.formatHash = function(hash) {
|
|
var hashEnd = new Buffer(10);
|
|
hash.copy(hashEnd, 0, 22, 32);
|
|
return buffertools.reverse(hashEnd).toString('hex');
|
|
};
|
|
|
|
/**
|
|
* Display the whole hash, as hex, in correct endian order.
|
|
*/
|
|
var formatHashFull = exports.formatHashFull = function(hash) {
|
|
var copy = new Buffer(hash.length);
|
|
hash.copy(copy);
|
|
var hex = buffertools.toHex(buffertools.reverse(copy));
|
|
return hex;
|
|
};
|
|
|
|
/**
|
|
* Format a block hash like Block Explorer does.
|
|
*
|
|
* Formats a block hash by removing leading zeros and truncating to 10 characters.
|
|
*/
|
|
var formatHashAlt = exports.formatHashAlt = function(hash) {
|
|
var hex = formatHashFull(hash);
|
|
hex = hex.replace(/^0*/, '');
|
|
return hex.substr(0, 10);
|
|
};
|
|
|
|
var formatBuffer = exports.formatBuffer = function(buffer, maxLen) {
|
|
// Calculate amount of bytes to display
|
|
if (maxLen === null) {
|
|
maxLen = 10;
|
|
}
|
|
if (maxLen > buffer.length || maxLen === 0) {
|
|
maxLen = buffer.length;
|
|
}
|
|
|
|
// Copy those bytes into a temporary buffer
|
|
var temp = new Buffer(maxLen);
|
|
buffer.copy(temp, 0, 0, maxLen);
|
|
|
|
// Format as string
|
|
var output = buffertools.toHex(temp);
|
|
if (temp.length < buffer.length) {
|
|
output += "...";
|
|
}
|
|
return output;
|
|
};
|
|
|
|
var valueToBigInt = exports.valueToBigInt = function(valueBuffer) {
|
|
if (Buffer.isBuffer(valueBuffer)) {
|
|
return bignum.fromBuffer(valueBuffer, {
|
|
endian: 'little',
|
|
size: 8
|
|
});
|
|
} else {
|
|
return valueBuffer;
|
|
}
|
|
};
|
|
|
|
var bigIntToValue = exports.bigIntToValue = function(valueBigInt) {
|
|
if (Buffer.isBuffer(valueBigInt)) {
|
|
return valueBigInt;
|
|
} else {
|
|
return valueBigInt.toBuffer({
|
|
endian: 'little',
|
|
size: 8
|
|
});
|
|
}
|
|
};
|
|
|
|
var fitsInNBits = function(integer, n) {
|
|
// TODO: make this efficient!!!
|
|
return integer.toString(2).replace('-', '').length < n;
|
|
};
|
|
exports.bytesNeededToStore = bytesNeededToStore = function(integer) {
|
|
if (integer === 0) return 0;
|
|
return Math.ceil(((integer).toString(2).replace('-', '').length + 1) / 8);
|
|
};
|
|
|
|
exports.negativeBuffer = negativeBuffer = function(b) {
|
|
// implement two-complement negative
|
|
var c = new Buffer(b.length);
|
|
// negate each byte
|
|
for (var i = 0; i < b.length; i++) {
|
|
c[i] = ~b[i];
|
|
if (c[i] < 0) c[i] += 256;
|
|
}
|
|
// add one
|
|
for (var i = b.length - 1; i >= 0; i--) {
|
|
c[i] += 1;
|
|
if (c[i] >= 256) c[i] -= 256;
|
|
if (c[i] !== 0) break;
|
|
}
|
|
return c;
|
|
};
|
|
|
|
/*
|
|
* Transforms an integer into a buffer using two-complement encoding
|
|
* For example, 1 is encoded as 01 and -1 is encoded as ff
|
|
* For more info see:
|
|
* http://en.wikipedia.org/wiki/Signed_number_representations#Two.27s_complement
|
|
*/
|
|
exports.intToBuffer2C = function(integer) {
|
|
var size = bytesNeededToStore(integer);
|
|
var buf = new Put();
|
|
var s = integer.toString(16);
|
|
var neg = s[0] === '-';
|
|
s = s.replace('-', '');
|
|
for (var i = 0; i < size; i++) {
|
|
var si = s.substring(s.length - 2 * (i + 1), s.length - 2 * (i));
|
|
if (si.lenght === 1) {
|
|
si = '0' + si;
|
|
}
|
|
var pi = parseInt(si, 16);
|
|
buf.word8(pi);
|
|
}
|
|
var ret = buf.buffer();
|
|
if (neg) {
|
|
ret = buffertools.reverse(ret);
|
|
ret = negativeBuffer(ret);
|
|
ret = buffertools.reverse(ret);
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
|
|
var padSign = function(b) {
|
|
var c;
|
|
if (b[0] & 0x80) {
|
|
c = new Buffer(b.length + 1);
|
|
b.copy(c, 1);
|
|
c[0] = 0;
|
|
} else {
|
|
c = b;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
|
|
/*
|
|
* Transforms an integer into a buffer using sign+magnitude encoding
|
|
* For example, 1 is encoded as 01 and -1 is encoded as 81
|
|
* For more info see:
|
|
* http://en.wikipedia.org/wiki/Signed_number_representations#Signed_magnitude_representation
|
|
*/
|
|
exports.intToBufferSM = function(v) {
|
|
if ("number" === typeof v) {
|
|
v = bignum(v);
|
|
}
|
|
var b, c;
|
|
var cmp = v.cmp(0);
|
|
if (cmp > 0) {
|
|
b = v.toBuffer();
|
|
c = padSign(b);
|
|
c = buffertools.reverse(c);
|
|
} else if (cmp == 0) {
|
|
c = new Buffer([]);
|
|
} else {
|
|
b = v.neg().toBuffer();
|
|
c = padSign(b);
|
|
c[0] |= 0x80;
|
|
c = buffertools.reverse(c);
|
|
}
|
|
return c;
|
|
};
|
|
|
|
/*
|
|
* Reverse of intToBufferSM
|
|
*/
|
|
exports.bufferSMToInt = function(v) {
|
|
if (!v.length) {
|
|
return bignum(0);
|
|
}
|
|
// Arithmetic operands must be in range [-2^31...2^31]
|
|
if (v.length > 4) {
|
|
throw new Error('Bigint cast overflow (> 4 bytes)');
|
|
}
|
|
|
|
var w = new Buffer(v.length);
|
|
v.copy(w);
|
|
w = buffertools.reverse(w);
|
|
var isNeg = w[0] & 0x80;
|
|
if (isNeg) {
|
|
w[0] &= 0x7f;
|
|
return bignum.fromBuffer(w).neg();
|
|
} else {
|
|
return bignum.fromBuffer(w);
|
|
}
|
|
};
|
|
|
|
|
|
|
|
var formatValue = exports.formatValue = function(valueBuffer) {
|
|
var value = valueToBigInt(valueBuffer).toString();
|
|
var integerPart = value.length > 8 ? value.substr(0, value.length - 8) : '0';
|
|
var decimalPart = value.length > 8 ? value.substr(value.length - 8) : value;
|
|
while (decimalPart.length < 8) {
|
|
decimalPart = "0" + decimalPart;
|
|
}
|
|
decimalPart = decimalPart.replace(/0*$/, '');
|
|
while (decimalPart.length < 2) {
|
|
decimalPart += "0";
|
|
}
|
|
return integerPart + "." + decimalPart;
|
|
};
|
|
|
|
var reFullVal = /^\s*(\d+)\.(\d+)/;
|
|
var reFracVal = /^\s*\.(\d+)/;
|
|
var reWholeVal = /^\s*(\d+)/;
|
|
|
|
function padFrac(frac) {
|
|
frac = frac.substr(0, 8); //truncate to 8 decimal places
|
|
while (frac.length < 8)
|
|
frac = frac + '0';
|
|
return frac;
|
|
}
|
|
|
|
function parseFullValue(res) {
|
|
return bignum(res[1]).mul('100000000').add(padFrac(res[2]));
|
|
}
|
|
|
|
function parseFracValue(res) {
|
|
return bignum(padFrac(res[1]));
|
|
}
|
|
|
|
function parseWholeValue(res) {
|
|
return bignum(res[1]).mul('100000000');
|
|
}
|
|
|
|
exports.parseValue = function parseValue(valueStr) {
|
|
if (typeof valueStr !== 'string')
|
|
valueStr = valueStr.toString();
|
|
|
|
var res = valueStr.match(reFullVal);
|
|
if (res)
|
|
return parseFullValue(res);
|
|
|
|
res = valueStr.match(reFracVal);
|
|
if (res)
|
|
return parseFracValue(res);
|
|
|
|
res = valueStr.match(reWholeVal);
|
|
if (res)
|
|
return parseWholeValue(res);
|
|
|
|
return undefined;
|
|
};
|
|
|
|
// Utility that synchronizes function calls based on a key
|
|
var createSynchrotron = exports.createSynchrotron = function(fn) {
|
|
var table = {};
|
|
return function(key) {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
var run = function() {
|
|
// Function fn() will call when it finishes
|
|
args[0] = function next() {
|
|
if (table[key]) {
|
|
if (table[key].length) {
|
|
table[key].shift()();
|
|
} else {
|
|
delete table[key];
|
|
}
|
|
}
|
|
};
|
|
|
|
fn.apply(null, args);
|
|
};
|
|
|
|
if (!table[key]) {
|
|
table[key] = [];
|
|
run();
|
|
} else {
|
|
table[key].push(run);
|
|
}
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Decode difficulty bits.
|
|
*
|
|
* This function calculates the difficulty target given the difficulty bits.
|
|
*/
|
|
var decodeDiffBits = exports.decodeDiffBits = function(diffBits, asBigInt) {
|
|
diffBits = +diffBits;
|
|
|
|
var target = bignum(diffBits & 0xffffff);
|
|
/*
|
|
* shiftLeft is not implemented on the bignum browser
|
|
*
|
|
* target = target.shiftLeft(8*((diffBits >>> 24) - 3));
|
|
*/
|
|
|
|
var mov = 8*((diffBits >>> 24) - 3);
|
|
while (mov-- > 0)
|
|
target = target.mul(2);
|
|
|
|
if (asBigInt) {
|
|
return target;
|
|
}
|
|
|
|
// Convert to buffer
|
|
var diffBuf = target.toBuffer();
|
|
var targetBuf = new Buffer(32);
|
|
buffertools.fill(targetBuf, 0);
|
|
diffBuf.copy(targetBuf, 32 - diffBuf.length);
|
|
return targetBuf;
|
|
};
|
|
|
|
/**
|
|
* Encode difficulty bits.
|
|
*
|
|
* This function calculates the compact difficulty, given a difficulty target.
|
|
*/
|
|
var encodeDiffBits = exports.encodeDiffBits = function encodeDiffBits(target) {
|
|
if (Buffer.isBuffer(target)) {
|
|
target = bignum.fromBuffer(target);
|
|
} else if ("function" === typeof target.toBuffer) { // duck-typing bignum
|
|
// Nothing to do
|
|
} else {
|
|
throw new Error("Incorrect variable type for difficulty");
|
|
}
|
|
|
|
var mpiBuf = target.toBuffer("mpint");
|
|
var size = mpiBuf.length - 4;
|
|
|
|
var compact = size << 24;
|
|
if (size >= 1) compact |= mpiBuf[4] << 16;
|
|
if (size >= 2) compact |= mpiBuf[5] << 8;
|
|
if (size >= 3) compact |= mpiBuf[6];
|
|
|
|
return compact;
|
|
};
|
|
|
|
/**
|
|
* Calculate "difficulty".
|
|
*
|
|
* This function calculates the maximum difficulty target divided by the given
|
|
* difficulty target.
|
|
*/
|
|
var calcDifficulty = exports.calcDifficulty = function(target) {
|
|
if (!Buffer.isBuffer(target)) {
|
|
target = decodeDiffBits(target);
|
|
}
|
|
var targetBigint = bignum.fromBuffer(target, {
|
|
order: 'forward'
|
|
});
|
|
var maxBigint = bignum.fromBuffer(MAX_TARGET, {
|
|
order: 'forward'
|
|
});
|
|
return maxBigint.div(targetBigint).toNumber();
|
|
};
|
|
|
|
var reverseBytes32 = exports.reverseBytes32 = function(data) {
|
|
if (data.length % 4) {
|
|
throw new Error("Util.reverseBytes32(): Data length must be multiple of 4");
|
|
}
|
|
var put = new Put();
|
|
var parser = Binary.parse(data);
|
|
while (!parser.eof()) {
|
|
var word = parser.word32le('word').vars.word;
|
|
put.word32be(word);
|
|
}
|
|
return put.buffer();
|
|
};
|
|
|
|
|
|
var getVarIntSize = exports.getVarIntSize = function getVarIntSize(i) {
|
|
|
|
if (i < 253) {
|
|
// unsigned char
|
|
return 1;
|
|
} else if (i < 0x10000) {
|
|
// unsigned short (LE)
|
|
return 3;
|
|
} else if (i < 0x100000000) {
|
|
// unsigned int (LE)
|
|
return 5;
|
|
} else {
|
|
// unsigned long long (LE)
|
|
return 9;
|
|
}
|
|
};
|
|
|
|
var varIntBuf = exports.varIntBuf = function varIntBuf(n) {
|
|
var buf = undefined;
|
|
if (n < 253) {
|
|
buf = new Buffer(1);
|
|
buf.writeUInt8(n, 0);
|
|
} else if (n < 0x10000) {
|
|
buf = new Buffer(1 + 2);
|
|
buf.writeUInt8(253, 0);
|
|
buf.writeUInt16LE(n, 1);
|
|
} else if (n < 0x100000000) {
|
|
buf = new Buffer(1 + 4);
|
|
buf.writeUInt8(254, 0);
|
|
buf.writeUInt32LE(n, 1);
|
|
} else {
|
|
buf = new Buffer(1 + 8);
|
|
buf.writeUInt8(255, 0);
|
|
buf.writeInt32LE(n & -1, 1);
|
|
buf.writeUInt32LE(Math.floor(n / 0x100000000), 5);
|
|
}
|
|
|
|
return buf;
|
|
};
|
|
|
|
var varStrBuf = exports.varStrBuf = function varStrBuf(s) {
|
|
return Buffer.concat([varIntBuf(s.length), s]);
|
|
};
|
|
|
|
// Initializations
|
|
exports.NULL_HASH = buffertools.fill(new Buffer(32), 0);
|
|
exports.EMPTY_BUFFER = new Buffer(0);
|
|
exports.ZERO_VALUE = buffertools.fill(new Buffer(8), 0);
|
|
var INT64_MAX = new Buffer('ffffffffffffffff', 'hex');
|
|
exports.INT64_MAX = INT64_MAX;
|
|
|
|
// How much of Bitcoin's internal integer coin representation
|
|
// makes 1 BTC
|
|
exports.COIN = 100000000;
|
|
|
|
var MAX_TARGET = exports.MAX_TARGET = new Buffer('00000000FFFF0000000000000000000000000000000000000000000000000000', 'hex');
|
|
|
|
}).call(this,require("/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js"),require("buffer").Buffer)
|
|
},{"../browser/vendor-bundle.js":3,"/Users/ematiu/devel/node/bitcore/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js":96,"bignum":58,"binary":69,"buffer":80,"bufferput":"aXRuS6","buffertools":"fugeBw","crypto":84,"jssha":117}],"./util/util":[function(require,module,exports){
|
|
module.exports=require('ACyo5H');
|
|
},{}]},{},[])
|