require('buffertools'); var crypto = require('crypto'); var bignum = require('bignum'); var Binary = require('binary'); var Put = require('put'); //var logger = require('./logger'); var ccmodule = require('bindings')('nativetools'); exports.ccmodule = ccmodule; exports.BitcoinKey = ccmodule.BitcoinKey; var sha256 = exports.sha256 = function (data) { return new Buffer(crypto.createHash('sha256').update(data).digest('binary'), 'binary'); }; var ripe160 = exports.ripe160 = function (data) { 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)); }; var sha256midstate = exports.sha256midstate = ccmodule.sha256_midstate; /** * Format a block hash like the official client does. */ var formatHash = exports.formatHash = function (hash) { // Make a copy, because reverse() and toHex() are destructive. var hashEnd = new Buffer(10); hash.copy(hashEnd, 0, 22, 32); return hashEnd.reverse().toString('hex'); }; /** * Display the whole hash, as hex, in correct endian order. */ var formatHashFull = exports.formatHashFull = function (hash) { // Make a copy, because reverse() and toHex() are destructive. var copy = new Buffer(hash.length); hash.copy(copy); var hex = copy.reverse().toHex(); 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 = temp.toHex(); 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 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 pubKeyHashToAddress = exports.pubKeyHashToAddress = function (pubKeyHash, addressVersion) { if (!pubKeyHash) return ""; var put = Put(); // Version if(addressVersion) { put.word8le(addressVersion); } else { put.word8le(0); } // Hash put.put(pubKeyHash); // Checksum (four bytes) put.put(twoSha256(put.buffer()).slice(0,4)); return encodeBase58(put.buffer()); }; var addressToPubKeyHash = exports.addressToPubKeyHash = function (address) { // Trim address = String(address).replace(/\s/g, ''); // Check sanity if (!address.match(/^[1-9A-HJ-NP-Za-km-z]{27,35}$/)) { //logger.warn("Not a valid Bitcoin address"); return null; } // Decode var buffer = decodeBase58(address); // Parse var parser = Binary.parse(buffer); parser.word8('version'); parser.buffer('hash', 20); parser.buffer('checksum', 4); // Check checksum var checksum = twoSha256(buffer.slice(0, 21)).slice(0, 4); if (checksum.compare(parser.vars.checksum) !== 0) { //logger.warn("Checksum comparison failed"); return null; } return parser.vars.hash; }; // 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); } }; }; /** * Generate a random 64-bit number. * * With ideas from node-uuid: * Copyright (c) 2010 Robert Kieffer * https://github.com/broofa/node-uuid/ * * @returns Buffer random nonce */ var generateNonce = exports.generateNonce = function () { var b32 = 0x100000000, ff = 0xff; var b = new Buffer(8), i = 0; // Generate eight random bytes r = Math.random()*b32; b[i++] = r & ff; b[i++] = (r=r>>>8) & ff; b[i++] = (r=r>>>8) & ff; b[i++] = (r=r>>>8) & ff; r = Math.random()*b32; b[i++] = r & ff; b[i++] = (r=r>>>8) & ff; b[i++] = (r=r>>>8) & ff; b[i++] = (r=r>>>8) & ff; return b; }; /** * 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); target = target.shiftLeft(8*((diffBits >>> 24) - 3)); if (asBigInt) { return target; } // Convert to buffer var diffBuf = target.toBuffer(); var targetBuf = new Buffer(32).fill(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 = 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 < 0xFD) { // unsigned char return 1; } else if (i <= 1<<16) { // unsigned short (LE) return 3; } else if (i <= 1<<32) { // unsigned int (LE) return 5; } else { // unsigned long long (LE) return 9; } }; // Initializations try { var NULL_HASH = exports.NULL_HASH = new Buffer(32).fill(0); var EMPTY_BUFFER = exports.EMPTY_BUFFER = new Buffer(0); var ZERO_VALUE = exports.ZERO_VALUE = new Buffer(8).fill(0); var INT64_MAX = exports.INT64_MAX = decodeHex("ffffffffffffffff"); // How much of Bitcoin's internal integer coin representation // makes 1 BTC var COIN = exports.COIN = 100000000; var MAX_TARGET = exports.MAX_TARGET = decodeHex('00000000FFFF0000000000000000000000000000000000000000000000000000'); } catch (e) { //logger.error("Error while generating utility constants:\n" + // (e.stack ? e.stack : e.toString())); process.exit(1); }