2013-02-17 06:39:15 +01:00
|
|
|
var BigInteger = require('./jsbn/jsbn');
|
|
|
|
var Crypto = require('./crypto-js/crypto');
|
2012-08-17 09:14:07 +02:00
|
|
|
|
2013-02-17 06:39:15 +01:00
|
|
|
module.exports = {
|
2012-01-11 10:41:52 +01:00
|
|
|
/**
|
|
|
|
* Cross-browser compatibility version of Array.isArray.
|
|
|
|
*/
|
|
|
|
isArray: Array.isArray || function(o)
|
|
|
|
{
|
2012-01-11 02:40:45 +01:00
|
|
|
return Object.prototype.toString.call(o) === '[object Array]';
|
|
|
|
},
|
2012-01-11 10:41:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an array of a certain length filled with a specific value.
|
|
|
|
*/
|
|
|
|
makeFilledArray: function (len, val)
|
|
|
|
{
|
2012-01-11 02:40:45 +01:00
|
|
|
var array = [];
|
|
|
|
var i = 0;
|
|
|
|
while (i < len) {
|
|
|
|
array[i++] = val;
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
},
|
2013-10-07 14:21:00 +02:00
|
|
|
/**
|
|
|
|
* Create a byte array representing a number with the given length
|
|
|
|
*/
|
|
|
|
numToBytes: function(num,bytes) {
|
2013-10-08 12:34:15 +02:00
|
|
|
if (bytes === undefined) bytes = 8;
|
|
|
|
if (bytes == 0) return [];
|
2013-10-08 11:35:19 +02:00
|
|
|
else return [num % 256].concat(module.exports.numToBytes(Math.floor(num / 256),bytes-1));
|
2013-10-07 14:21:00 +02:00
|
|
|
},
|
2013-10-08 12:34:15 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a byte array representing a number with the given length
|
|
|
|
*/
|
|
|
|
bytesToNum: function(bytes) {
|
|
|
|
if (bytes.length == 0) return 0;
|
2013-10-08 13:44:03 +02:00
|
|
|
else return bytes[0] + 256 * module.exports.bytesToNum(bytes.slice(1));
|
2013-10-08 12:34:15 +02:00
|
|
|
},
|
2012-01-11 10:41:52 +01:00
|
|
|
/**
|
|
|
|
* Turn an integer into a "var_int".
|
|
|
|
*
|
|
|
|
* "var_int" is a variable length integer used by Bitcoin's binary format.
|
|
|
|
*
|
|
|
|
* Returns a byte array.
|
|
|
|
*/
|
2013-10-07 14:21:00 +02:00
|
|
|
numToVarInt: function(num) {
|
|
|
|
var m = module.exports;
|
|
|
|
if (num < 253) return [num];
|
|
|
|
else if (num < 65536) return [253].concat(m.numToBytes(num,2));
|
|
|
|
else if (num < 4294967296) return [254].concat(m.numToBytes(num,4));
|
|
|
|
else return [253].concat(m.numToBytes(num,8));
|
2012-01-11 02:40:45 +01:00
|
|
|
},
|
2012-01-11 10:41:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse a Bitcoin value byte array, returning a BigInteger.
|
|
|
|
*/
|
|
|
|
valueToBigInt: function (valueBuffer)
|
|
|
|
{
|
2012-01-11 02:40:45 +01:00
|
|
|
if (valueBuffer instanceof BigInteger) return valueBuffer;
|
2011-05-08 15:36:11 +02:00
|
|
|
|
2012-01-11 02:40:45 +01:00
|
|
|
// Prepend zero byte to prevent interpretation as negative integer
|
|
|
|
return BigInteger.fromByteArrayUnsigned(valueBuffer);
|
|
|
|
},
|
2012-01-11 10:41:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2012-01-11 02:40:45 +01: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;
|
|
|
|
},
|
2012-01-11 10:41:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2011-08-18 06:01:00 +02:00
|
|
|
parseValue: function (valueString) {
|
2012-01-11 10:41:52 +01:00
|
|
|
// TODO: Detect other number formats (e.g. comma as decimal separator)
|
2012-01-11 02:40:45 +01:00
|
|
|
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)));
|
2011-08-18 06:01:00 +02:00
|
|
|
return value;
|
|
|
|
},
|
2012-01-11 10:41:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate RIPEMD160(SHA256(data)).
|
|
|
|
*
|
|
|
|
* Takes an arbitrary byte array as inputs and returns the hash as a byte
|
|
|
|
* array.
|
|
|
|
*/
|
2012-01-11 02:40:45 +01:00
|
|
|
sha256ripe160: function (data) {
|
|
|
|
return Crypto.RIPEMD160(Crypto.SHA256(data, {asBytes: true}), {asBytes: true});
|
2013-10-21 20:00:31 +02:00
|
|
|
},
|
|
|
|
error: function(msg) {
|
|
|
|
throw new Error(msg);
|
2012-01-11 02:40:45 +01:00
|
|
|
}
|
2011-05-04 18:02:56 +02:00
|
|
|
};
|