313 lines
6.3 KiB
JavaScript
313 lines
6.3 KiB
JavaScript
/*!
|
|
* output.js - output object for bcoin
|
|
* Copyright (c) 2014-2015, Fedor Indutny (MIT License)
|
|
* Copyright (c) 2014-2016, Christopher Jeffrey (MIT License).
|
|
* https://github.com/bcoin-org/bcoin
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
var assert = require('assert');
|
|
var util = require('../utils/util');
|
|
var Amount = require('../btc/amount');
|
|
var Network = require('../protocol/network');
|
|
var Script = require('../script/script');
|
|
var StaticWriter = require('../utils/staticwriter');
|
|
var BufferReader = require('../utils/reader');
|
|
var consensus = require('../protocol/consensus');
|
|
var policy = require('../protocol/policy');
|
|
|
|
/**
|
|
* Represents a transaction output.
|
|
* @exports Output
|
|
* @constructor
|
|
* @param {NakedOutput} options
|
|
* @property {Amount} value - Value in satoshis.
|
|
* @property {Script} script
|
|
*/
|
|
|
|
function Output(options) {
|
|
if (!(this instanceof Output))
|
|
return new Output(options);
|
|
|
|
this.value = 0;
|
|
this.script = new Script();
|
|
this.mutable = false;
|
|
this._address = null;
|
|
|
|
if (options)
|
|
this.fromOptions(options);
|
|
}
|
|
|
|
/**
|
|
* Inject properties from options object.
|
|
* @private
|
|
* @param {NakedOutput} options
|
|
*/
|
|
|
|
Output.prototype.fromOptions = function fromOptions(options) {
|
|
assert(options, 'Output data is required.');
|
|
|
|
if (options.value) {
|
|
assert(util.isNumber(options.value));
|
|
this.value = options.value;
|
|
}
|
|
|
|
if (options.script)
|
|
this.script.fromOptions(options.script);
|
|
|
|
if (options.address)
|
|
this.script.fromAddress(options.address);
|
|
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Instantiate output from options object.
|
|
* @param {NakedOutput} options
|
|
* @returns {Output}
|
|
*/
|
|
|
|
Output.fromOptions = function fromOptions(options) {
|
|
return new Output().fromOptions(options);
|
|
};
|
|
|
|
/**
|
|
* Get the script type as a string.
|
|
* @returns {ScriptType} type
|
|
*/
|
|
|
|
Output.prototype.getType = function getType() {
|
|
return Script.typesByVal[this.script.getType()].toLowerCase();
|
|
};
|
|
|
|
/**
|
|
* Get the address.
|
|
* @returns {Address} address
|
|
*/
|
|
|
|
Output.prototype.getAddress = function getAddress() {
|
|
var address = this._address;
|
|
|
|
if (!address) {
|
|
address = this.script.getAddress();
|
|
if (!this.mutable)
|
|
this._address = address;
|
|
}
|
|
|
|
return address;
|
|
};
|
|
|
|
/**
|
|
* Get the address hash.
|
|
* @param {String?} enc
|
|
* @returns {Hash} hash
|
|
*/
|
|
|
|
Output.prototype.getHash = function getHash(enc) {
|
|
var address = this.getAddress();
|
|
if (!address)
|
|
return;
|
|
return address.getHash(enc);
|
|
};
|
|
|
|
/**
|
|
* Convert the input to a more user-friendly object.
|
|
* @returns {Object}
|
|
*/
|
|
|
|
Output.prototype.inspect = function inspect() {
|
|
return {
|
|
type: this.getType(),
|
|
value: Amount.btc(this.value),
|
|
script: this.script,
|
|
address: this.getAddress()
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Convert the output to an object suitable
|
|
* for JSON serialization.
|
|
* @returns {Object}
|
|
*/
|
|
|
|
Output.prototype.toJSON = function toJSON() {
|
|
return this.getJSON();
|
|
};
|
|
|
|
/**
|
|
* Convert the output to an object suitable
|
|
* for JSON serialization.
|
|
* @param {Network} network
|
|
* @returns {Object}
|
|
*/
|
|
|
|
Output.prototype.getJSON = function getJSON(network) {
|
|
var address = this.getAddress();
|
|
|
|
network = Network.get(network);
|
|
|
|
if (address)
|
|
address = address.toBase58(network);
|
|
|
|
return {
|
|
value: Amount.btc(this.value),
|
|
script: this.script.toJSON(),
|
|
address: address
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Calculate the dust threshold for this
|
|
* output, based on serialize size and rate.
|
|
* @param {Rate?} rate
|
|
* @returns {Amount}
|
|
*/
|
|
|
|
Output.prototype.getDustThreshold = function getDustThreshold(rate) {
|
|
var scale = consensus.WITNESS_SCALE_FACTOR;
|
|
var size;
|
|
|
|
if (this.script.isUnspendable())
|
|
return 0;
|
|
|
|
size = this.getSize();
|
|
|
|
if (this.script.isProgram()) {
|
|
// 75% segwit discount applied to script size.
|
|
size += 32 + 4 + 1 + (107 / scale | 0) + 4;
|
|
} else {
|
|
size += 32 + 4 + 1 + 107 + 4;
|
|
}
|
|
|
|
return 3 * policy.getMinFee(size, rate);
|
|
};
|
|
|
|
/**
|
|
* Calculate size of serialized output.
|
|
* @returns {Number}
|
|
*/
|
|
|
|
Output.prototype.getSize = function getSize() {
|
|
return 8 + this.script.getVarSize();
|
|
};
|
|
|
|
/**
|
|
* Test whether the output should be considered dust.
|
|
* @param {Rate?} rate
|
|
* @returns {Boolean}
|
|
*/
|
|
|
|
Output.prototype.isDust = function isDust(rate) {
|
|
return this.value < this.getDustThreshold(rate);
|
|
};
|
|
|
|
/**
|
|
* Inject properties from a JSON object.
|
|
* @private
|
|
* @param {Object} json
|
|
*/
|
|
|
|
Output.prototype.fromJSON = function fromJSON(json) {
|
|
assert(typeof json.value === 'string');
|
|
this.value = Amount.value(json.value);
|
|
this.script.fromJSON(json.script);
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Instantiate an Output from a jsonified output object.
|
|
* @param {Object} json - The jsonified output object.
|
|
* @returns {Output}
|
|
*/
|
|
|
|
Output.fromJSON = function fromJSON(json) {
|
|
return new Output().fromJSON(json);
|
|
};
|
|
|
|
/**
|
|
* Write the output to a buffer writer.
|
|
* @param {BufferWriter} bw
|
|
*/
|
|
|
|
Output.prototype.toWriter = function toWriter(bw) {
|
|
bw.write64(this.value);
|
|
bw.writeVarBytes(this.script.toRaw());
|
|
return bw;
|
|
};
|
|
|
|
/**
|
|
* Serialize the output.
|
|
* @param {String?} enc - Encoding, can be `'hex'` or null.
|
|
* @returns {Buffer|String}
|
|
*/
|
|
|
|
Output.prototype.toRaw = function toRaw() {
|
|
var size = this.getSize();
|
|
return this.toWriter(new StaticWriter(size)).render();
|
|
};
|
|
|
|
/**
|
|
* Inject properties from buffer reader.
|
|
* @private
|
|
* @param {BufferReader} br
|
|
*/
|
|
|
|
Output.prototype.fromReader = function fromReader(br) {
|
|
this.value = br.read64();
|
|
this.script.fromRaw(br.readVarBytes());
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Inject properties from serialized data.
|
|
* @private
|
|
* @param {Buffer} data
|
|
*/
|
|
|
|
Output.prototype.fromRaw = function fromRaw(data) {
|
|
return this.fromReader(new BufferReader(data));
|
|
};
|
|
|
|
/**
|
|
* Instantiate an output from a buffer reader.
|
|
* @param {BufferReader} br
|
|
* @returns {Output}
|
|
*/
|
|
|
|
Output.fromReader = function fromReader(br) {
|
|
return new Output().fromReader(br);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an output from a serialized Buffer.
|
|
* @param {Buffer} data
|
|
* @param {String?} enc - Encoding, can be `'hex'` or null.
|
|
* @returns {Output}
|
|
*/
|
|
|
|
Output.fromRaw = function fromRaw(data, enc) {
|
|
if (typeof data === 'string')
|
|
data = new Buffer(data, enc);
|
|
return new Output().fromRaw(data);
|
|
};
|
|
|
|
/**
|
|
* Test an object to see if it is an Output.
|
|
* @param {Object} obj
|
|
* @returns {Boolean}
|
|
*/
|
|
|
|
Output.isOutput = function isOutput(obj) {
|
|
return obj
|
|
&& obj.value !== undefined
|
|
&& obj.script !== undefined
|
|
&& typeof obj.getAddress === 'function';
|
|
};
|
|
|
|
/*
|
|
* Expose
|
|
*/
|
|
|
|
module.exports = Output;
|