/** * output.js - output object for bcoin * Copyright (c) 2014-2015, Fedor Indutny (MIT License) * https://github.com/indutny/bcoin */ var bn = require('bn.js'); var bcoin = require('../bcoin'); var utils = bcoin.utils; var assert = utils.assert; var constants = bcoin.protocol.constants; /** * Output */ function Output(options) { var value; if (!(this instanceof Output)) return new Output(options); assert(typeof options.script !== 'string'); value = options.value; if (typeof value === 'number') { assert(value % 1 === 0); value = new bn(value); } this.value = utils.satoshi(value || new bn(0)); // this.script = options.script ? options.script.slice() : []; this.script = options.script || []; this._size = options._size || 0; this._offset = options._offset || 0; // For safety: do not allow usage of // Numbers, do not allow negative values. assert(typeof value !== 'number'); assert(!this.value.isNeg()) assert(this.value.bitLength() <= 63); assert(!(this.value.toArray('be', 8)[0] & 0x80)); // if (options.script && options.script._raw) // utils.hidden(this.script, '_raw', options.script._raw); } Output.prototype.__defineGetter__('data', function() { var data; if (this._data) return this._data; data = Output.getData(this); if (this.script.length) utils.hidden(this, '_data', data); return data; }); Output.prototype.__defineGetter__('type', function() { return this.data.type; }); Output.prototype.__defineGetter__('signature', function() { return this.signatures[0]; }); Output.prototype.__defineGetter__('key', function() { return this.keys[0]; }); Output.prototype.__defineGetter__('hash160', function() { return this.data.scriptHash || this.hashes[0]; }); Output.prototype.__defineGetter__('id', function() { return this.address || this.getID(); }); Output.prototype.__defineGetter__('address', function() { return this.data.address; }); Output.prototype.__defineGetter__('signatures', function() { return this.data.signatures || []; }); Output.prototype.__defineGetter__('keys', function() { return this.data.keys || []; }); Output.prototype.__defineGetter__('hashes', function() { return this.data.hashes || []; }); Output.prototype.__defineGetter__('addresses', function() { return this.data.addresses || []; }); Output.prototype.__defineGetter__('scriptHash', function() { return this.data.scriptHash; }); Output.prototype.__defineGetter__('scriptAddress', function() { return this.data.scriptAddress; }); Output.prototype.__defineGetter__('m', function() { return this.data.m || 1; }); Output.prototype.__defineGetter__('n', function() { return this.data.n || this.m; }); Output.prototype.__defineGetter__('locktime', function() { return bcoin.script.getLocktime(this.script); }); Output.prototype.__defineGetter__('flags', function() { return this.data.flags; }); Output.prototype.__defineGetter__('text', function() { return this.data.text; }); // Schema and defaults for data object: // { // type: String, // subtype: String, // side: 'output', // signatures: Array, // keys: Array, // hashes: Array, // addresses: Array, // redeem: Array, // scriptHash: Array, // scriptAddress: String, // m: Number, // n: Number, // height: Number, // flags: Array, // text: String, // locktime: Number, // value: bn, // script: Array, // seq: Number, // prev: String, // index: Number, // none: Boolean // } Output.getData = function getData(output) { var def; assert(output instanceof Output); def = { side: 'output', value: output.value, script: output.script }; return utils.merge(def, bcoin.script.getOutputData(output.script)); }; Output.prototype.getData = function getData() { return Output.getData(this); }; Output.prototype.getType = function getType() { return bcoin.script.getOutputType(this.script); }; Output.prototype.getAddress = function getAddress() { return bcoin.script.getOutputAddress(this.script); }; Output.prototype.getID = function getID() { var data = bcoin.script.encode(this.script); var hash = utils.toHex(utils.ripesha(data)); return '[' + this.type + ':' + hash.slice(0, 7) + ']'; }; Output.prototype.testScript = function testScript(key, hash, keys, scriptHash, type) { if (!type || type === 'pubkey') { if (key) { if (bcoin.script.isPubkey(this.script, key)) return true; } } if (!type || type === 'pubkeyhash') { if (hash) { if (bcoin.script.isPubkeyhash(this.script, hash)) return true; } } if (!type || type === 'multisig') { if (keys) { if (bcoin.script.isMultisig(this.script, keys)) return true; } } if (!type || type === 'scripthash') { if (scriptHash) { if (bcoin.script.isScripthash(this.script, scriptHash)) return true; } } return false; }; Output.prototype.test = function test(addressTable) { var address = this.getAddress(); if (!address) return false; if (typeof addressTable === 'string') addressTable = [addressTable]; if (Array.isArray(addressTable)) { addressTable = addressTable.reduce(function(out, address) { out[address] = true; return out; }, {}); } if (addressTable[address] != null) return true; return false; }; Output.prototype.getSigops = function getSigops(accurate) { return bcoin.script.getSigops(this.script, accurate); }; Output.prototype.inspect = function inspect() { return { type: this.type, address: this.address, keys: this.keys.map(utils.toHex), hashes: this.hashes.map(utils.toHex), addresses: this.addresses, scriptAddress: this.scriptAddress, m: this.m, n: this.n, text: this.text, locktime: this.locktime, hash: this.hash, index: this.index, height: this.height, value: utils.btc(this.value), script: bcoin.script.format(this.script) }; }; Output.prototype.toJSON = function toJSON() { return { value: utils.btc(this.value), script: utils.toHex(bcoin.script.encode(this.script)) }; }; Output._fromJSON = function _fromJSON(json) { return { value: utils.satoshi(json.value), script: bcoin.script.decode(new Buffer(json.script, 'hex')) }; }; Output.fromJSON = function fromJSON(json) { return new Output(Output._fromJSON(json)); }; Output.prototype.toCompact = function toCompact() { return { type: 'output', output: this.toRaw('hex') }; }; Output._fromCompact = function _fromCompact(json) { return Output._fromRaw(json.output, 'hex'); }; Output.fromCompact = function fromCompact(json) { return new Output(Output._fromCompact(json)); }; Output.prototype.toRaw = function toRaw(enc) { var data = bcoin.protocol.framer.output(this); if (enc === 'hex') data = utils.toHex(data); return data; }; Output._fromRaw = function _fromRaw(data, enc) { var parser = new bcoin.protocol.parser(); if (enc === 'hex') data = new Buffer(data, 'hex'); data = parser.parseOutput(data); return data; }; Output.fromRaw = function fromRaw(data, enc) { return new Output(Output._fromRaw(data, enc)); }; /** * Expose */ module.exports = Output;