var $ = require('preconditions').singleton(); var _ = require('lodash'); var bitcore = require('bitcore-lib'); var crypto = bitcore.crypto; var encoding = bitcore.encoding; var secp256k1 = require('secp256k1'); var Utils = {}; Utils.getMissingFields = function(obj, args) { args = [].concat(args); if (!_.isObject(obj)) return args; var missing = _.filter(args, function(arg) { return !obj.hasOwnProperty(arg); }); return missing; }; /** * * @desc rounds a JAvascript number * @param number * @return {number} */ Utils.strip = function(number) { return parseFloat(number.toPrecision(12)); } /* TODO: It would be nice to be compatible with bitcoind signmessage. How * the hash is calculated there? */ Utils.hashMessage = function(text, noReverse) { $.checkArgument(text); var buf = new Buffer(text); var ret = crypto.Hash.sha256sha256(buf); if (!noReverse) { ret = new bitcore.encoding.BufferReader(ret).readReverse(); } return ret; }; Utils.verifyMessage = function(text, signature, publicKey) { $.checkArgument(text); var hash = Utils.hashMessage(text, true); var sig = this._tryImportSignature(signature); if (!sig) { return false; } var publicKeyBuffer = this._tryImportPublicKey(publicKey); if (!publicKeyBuffer) { return false; } return this._tryVerifyMessage(hash, sig, publicKeyBuffer); }; Utils._tryImportPublicKey = function(publicKey) { var publicKeyBuffer = publicKey; try { if (!Buffer.isBuffer(publicKey)) { publicKeyBuffer = new Buffer(publicKey, 'hex'); } return publicKeyBuffer; } catch (e) { return false; } }; Utils._tryImportSignature = function(signature) { try { var signatureBuffer = signature; if (!Buffer.isBuffer(signature)) { signatureBuffer = new Buffer(signature, 'hex'); } return secp256k1.signatureImport(signatureBuffer); } catch (e) { return false; } }; Utils._tryVerifyMessage = function(hash, sig, publicKeyBuffer) { try { return secp256k1.verify(hash, sig, publicKeyBuffer); } catch (e) { return false; } }; Utils.formatAmount = function(satoshis, unit, opts) { var UNITS = { btc: { toSatoshis: 100000000, maxDecimals: 6, minDecimals: 2, }, bit: { toSatoshis: 100, maxDecimals: 0, minDecimals: 0, }, sat: { toSatoshis: 1, maxDecimals: 0, minDecimals: 0, } }; $.shouldBeNumber(satoshis); $.checkArgument(_.contains(_.keys(UNITS), unit)); function addSeparators(nStr, thousands, decimal, minDecimals) { nStr = nStr.replace('.', decimal); var x = nStr.split(decimal); var x0 = x[0]; var x1 = x[1]; x1 = _.dropRightWhile(x1, function(n, i) { return n == '0' && i >= minDecimals; }).join(''); var x2 = x.length > 1 ? decimal + x1 : ''; x0 = x0.replace(/\B(?=(\d{3})+(?!\d))/g, thousands); return x0 + x2; } opts = opts || {}; var u = _.assign(UNITS[unit], opts); var amount = (satoshis / u.toSatoshis).toFixed(u.maxDecimals); return addSeparators(amount, opts.thousandsSeparator || ',', opts.decimalSeparator || '.', u.minDecimals); }; Utils.formatAmountInBtc = function(amount) { return Utils.formatAmount(amount, 'btc', { minDecimals: 8, maxDecimals: 8, }) + 'btc'; }; Utils.formatUtxos = function(utxos) { if (_.isEmpty(utxos)) return 'none'; return _.map([].concat(utxos), function(i) { var amount = Utils.formatAmountInBtc(i.satoshis); var confirmations = i.confirmations ? i.confirmations + 'c' : 'u'; return amount + '/' + confirmations; }).join(', '); }; Utils.formatRatio = function(ratio) { return (ratio * 100.).toFixed(4) + '%'; }; Utils.formatSize = function(size) { return (size / 1000.).toFixed(4) + 'kB'; }; Utils.parseVersion = function(version) { var v = {}; if (!version) return null; var x = version.split('-'); if (x.length != 2) { v.agent = version; return v; } v.agent = _.contains(['bwc', 'bws'], x[0]) ? 'bwc' : x[0]; x = x[1].split('.'); v.major = parseInt(x[0]); v.minor = parseInt(x[1]); v.patch = parseInt(x[2]); return v; }; Utils.checkValueInCollection = function(value, collection) { if (!value || !_.isString(value)) return false; return _.contains(_.values(collection), value); }; module.exports = Utils;