You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
476 lines
15 KiB
476 lines
15 KiB
'use strict';
|
|
|
|
var _ = require('lodash');
|
|
var $ = require('./util/preconditions');
|
|
var base58check = require('./encoding/base58check');
|
|
var Networks = require('./networks');
|
|
var Hash = require('./crypto/hash');
|
|
var JSUtil = require('./util/js');
|
|
|
|
/**
|
|
* Instantiate an address from an address String or Buffer, a public key or script hash Buffer,
|
|
* or an instance of {@link PublicKey} or {@link Script}.
|
|
*
|
|
* This is an immutable class, and if the first parameter provided to this constructor is an
|
|
* `Address` instance, the same argument will be returned.
|
|
*
|
|
* An address has two key properties: `network` and `type`. The type is either
|
|
* `Address.PayToPublicKeyHash` (value is the `'pubkeyhash'` string)
|
|
* or `Address.PayToScriptHash` (the string `'scripthash'`). The network is an instance of {@link Network}.
|
|
* You can quickly check whether an address is of a given kind by using the methods
|
|
* `isPayToPublicKeyHash` and `isPayToScriptHash`
|
|
*
|
|
* @example
|
|
* ```javascript
|
|
* // validate that an input field is valid
|
|
* var error = Address.getValidationError(input, 'testnet');
|
|
* if (!error) {
|
|
* var address = Address(input, 'testnet');
|
|
* } else {
|
|
* // invalid network or checksum (typo?)
|
|
* var message = error.messsage;
|
|
* }
|
|
*
|
|
* // get an address from a public key
|
|
* var address = Address(publicKey, 'testnet').toString();
|
|
* ```
|
|
*
|
|
* @param {*} data - The encoded data in various formats
|
|
* @param {Network|String|number} [network] - The network: 'livenet' or 'testnet'
|
|
* @param {String} [type] - The type of address: 'script' or 'pubkey'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
* @constructor
|
|
*/
|
|
function Address(data, network, type) {
|
|
/* jshint maxcomplexity: 12 */
|
|
/* jshint maxstatements: 20 */
|
|
|
|
if (!(this instanceof Address)) {
|
|
return new Address(data, network, type);
|
|
}
|
|
|
|
if (_.isArray(data) && _.isNumber(network)) {
|
|
return Address.createMultisig(data, network, type);
|
|
}
|
|
|
|
if (data instanceof Address) {
|
|
// Immutable instance
|
|
return data;
|
|
}
|
|
|
|
$.checkArgument(data, new TypeError('First argument is required, please include address data.'));
|
|
|
|
if (network && !Networks.get(network)) {
|
|
throw new TypeError('Second argument must be "livenet" or "testnet".');
|
|
}
|
|
|
|
if (type && (type !== Address.PayToPublicKeyHash && type !== Address.PayToScriptHash)) {
|
|
throw new TypeError('Third argument must be "pubkeyhash" or "scripthash".');
|
|
}
|
|
|
|
var info = this._classifyArguments(data, network, type);
|
|
|
|
// set defaults if not set
|
|
info.network = info.network || Networks.get(network) || Networks.defaultNetwork;
|
|
info.type = info.type || type || Address.PayToPublicKeyHash;
|
|
|
|
Object.defineProperty(this, 'hashBuffer', {
|
|
configurable: false,
|
|
value: info.hashBuffer
|
|
});
|
|
|
|
Object.defineProperty(this, 'network', {
|
|
configurable: false,
|
|
value: info.network
|
|
});
|
|
|
|
Object.defineProperty(this, 'type', {
|
|
configurable: false,
|
|
value: info.type
|
|
});
|
|
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Internal function used to split different kinds of arguments of the constructor
|
|
* @param {*} data - The encoded data in various formats
|
|
* @param {Network|String|number} [network] - The network: 'livenet' or 'testnet'
|
|
* @param {String} [type] - The type of address: 'script' or 'pubkey'
|
|
* @returns {Object} An "info" object with "type", "network", and "hashBuffer"
|
|
*/
|
|
Address.prototype._classifyArguments = function(data, network, type) {
|
|
/* jshint maxcomplexity: 10 */
|
|
// transform and validate input data
|
|
if ((data instanceof Buffer || data instanceof Uint8Array) && data.length === 20) {
|
|
return Address._transformHash(data);
|
|
} else if ((data instanceof Buffer || data instanceof Uint8Array) && data.length === 21) {
|
|
return Address._transformBuffer(data, network, type);
|
|
} else if (data.constructor && (data.constructor.name && data.constructor.name === 'PublicKey')) {
|
|
return Address._transformPublicKey(data);
|
|
} else if (data.constructor && (data.constructor.name && data.constructor.name === 'Script')) {
|
|
return Address._transformScript(data, network);
|
|
} else if (typeof(data) === 'string') {
|
|
return Address._transformString(data, network, type);
|
|
} else {
|
|
throw new TypeError('First argument is an unrecognized data format.');
|
|
}
|
|
};
|
|
|
|
/** @static */
|
|
Address.PayToPublicKeyHash = 'pubkeyhash';
|
|
/** @static */
|
|
Address.PayToScriptHash = 'scripthash';
|
|
|
|
/**
|
|
* @param {Buffer} hash - An instance of a hash Buffer
|
|
* @returns {Object} An object with keys: hashBuffer
|
|
* @private
|
|
*/
|
|
Address._transformHash = function(hash){
|
|
var info = {};
|
|
if (!(hash instanceof Buffer) && !(hash instanceof Uint8Array)) {
|
|
throw new TypeError('Address supplied is not a buffer.');
|
|
}
|
|
if (hash.length !== 20) {
|
|
throw new TypeError('Address hashbuffers must be exactly 20 bytes.');
|
|
}
|
|
info.hashBuffer = hash;
|
|
return info;
|
|
};
|
|
|
|
/**
|
|
* Internal function to discover the network and type based on the first data byte
|
|
*
|
|
* @param {Buffer} buffer - An instance of a hex encoded address Buffer
|
|
* @returns {Object} An object with keys: network and type
|
|
* @private
|
|
*/
|
|
Address._classifyFromVersion = function(buffer){
|
|
var version = {};
|
|
version.network = Networks.get(buffer[0]);
|
|
switch (buffer[0]) { // the version byte
|
|
case Networks.livenet.pubkeyhash:
|
|
version.type = Address.PayToPublicKeyHash;
|
|
break;
|
|
|
|
case Networks.livenet.scripthash:
|
|
version.type = Address.PayToScriptHash;
|
|
break;
|
|
|
|
case Networks.testnet.pubkeyhash:
|
|
version.type = Address.PayToPublicKeyHash;
|
|
break;
|
|
|
|
case Networks.testnet.scripthash:
|
|
version.type = Address.PayToScriptHash;
|
|
break;
|
|
}
|
|
return version;
|
|
};
|
|
|
|
/**
|
|
* Internal function to transform a bitcoin address buffer
|
|
*
|
|
* @param {Buffer} buffer - An instance of a hex encoded address Buffer
|
|
* @param {String} [network] - The network: 'livenet' or 'testnet'
|
|
* @param {String} [type] - The type: 'pubkeyhash' or 'scripthash'
|
|
* @returns {Object} An object with keys: hashBuffer, network and type
|
|
* @private
|
|
*/
|
|
Address._transformBuffer = function(buffer, network, type){
|
|
/* jshint maxcomplexity: 9 */
|
|
var info = {};
|
|
if (!(buffer instanceof Buffer) && !(buffer instanceof Uint8Array)) {
|
|
throw new TypeError('Address supplied is not a buffer.');
|
|
}
|
|
if (buffer.length !== 1 + 20) {
|
|
throw new TypeError('Address buffers must be exactly 21 bytes.');
|
|
}
|
|
|
|
network = Networks.get(network);
|
|
var bufferVersion = Address._classifyFromVersion(buffer);
|
|
|
|
if (!bufferVersion.network || (network && network !== bufferVersion.network)) {
|
|
throw new TypeError('Address has mismatched network type.');
|
|
}
|
|
|
|
if (!bufferVersion.type || ( type && type !== bufferVersion.type )) {
|
|
throw new TypeError('Address has mismatched type.');
|
|
}
|
|
|
|
info.hashBuffer = buffer.slice(1);
|
|
info.network = bufferVersion.network;
|
|
info.type = bufferVersion.type;
|
|
return info;
|
|
};
|
|
|
|
/**
|
|
* Internal function to transform a {@link PublicKey}
|
|
*
|
|
* @param {PublicKey} pubkey - An instance of PublicKey
|
|
* @returns {Object} An object with keys: hashBuffer, type
|
|
* @private
|
|
*/
|
|
Address._transformPublicKey = function(pubkey){
|
|
var info = {};
|
|
if (!pubkey.constructor || (pubkey.constructor.name && pubkey.constructor.name !== 'PublicKey')) {
|
|
throw new TypeError('Address must be an instance of PublicKey.');
|
|
}
|
|
info.hashBuffer = Hash.sha256ripemd160(pubkey.toBuffer());
|
|
info.type = Address.PayToPublicKeyHash;
|
|
return info;
|
|
};
|
|
|
|
/**
|
|
* Internal function to transform a {@link Script} into a `info` object.
|
|
*
|
|
* @param {Script} script - An instance of Script
|
|
* @returns {Object} An object with keys: hashBuffer, type
|
|
* @private
|
|
*/
|
|
Address._transformScript = function(script, network){
|
|
var info = {};
|
|
if (!script.constructor || (script.constructor.name && script.constructor.name !== 'Script')) {
|
|
throw new TypeError('Address must be an instance of Script.');
|
|
}
|
|
info.network = network || Networks.defaultNetwork;
|
|
info.hashBuffer = Hash.sha256ripemd160(script.toBuffer());
|
|
info.type = Address.PayToScriptHash;
|
|
return info;
|
|
};
|
|
|
|
/**
|
|
* Creates a P2SH address from a set of public keys and a threshold.
|
|
*
|
|
* The addresses will be sorted lexicographically, as that is the trend in bitcoin.
|
|
* To create an address from unsorted public keys, use the {@link Script#buildMultisigOut}
|
|
* interface.
|
|
*
|
|
* @param {Array} publicKeys - a set of public keys to create an address
|
|
* @param {number} threshold - the number of signatures needed to release the funds
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @return {Address}
|
|
*/
|
|
Address.createMultisig = function(publicKeys, threshold, network) {
|
|
var Script = require('./script');
|
|
return new Address(Script.buildMultisigOut(publicKeys, threshold), network || Networks.defaultNetwork);
|
|
};
|
|
|
|
/**
|
|
* Internal function to transform a bitcoin address string
|
|
*
|
|
* @param {String} data
|
|
* @param {String|Network} [network] - either a Network instance, 'livenet', or 'testnet'
|
|
* @param {String} [type] - The type: 'pubkeyhash' or 'scripthash'
|
|
* @returns {Object} An object with keys: hashBuffer, network and type
|
|
* @private
|
|
*/
|
|
Address._transformString = function(data, network, type){
|
|
if( typeof(data) !== 'string' ) {
|
|
throw new TypeError('Address supplied is not a string.');
|
|
}
|
|
var addressBuffer = base58check.decode(data);
|
|
var info = Address._transformBuffer(addressBuffer, network, type);
|
|
return info;
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from a PublicKey instance
|
|
*
|
|
* @param {PublicKey} data
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
*/
|
|
Address.fromPublicKey = function(data, network){
|
|
var info = Address._transformPublicKey(data);
|
|
network = network || Networks.defaultNetwork;
|
|
return new Address(info.hashBuffer, network, info.type);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from a ripemd160 public key hash
|
|
*
|
|
* @param {Buffer} hash - An instance of buffer of the hash
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
*/
|
|
Address.fromPublicKeyHash = function(hash, network) {
|
|
var info = Address._transformHash(hash);
|
|
return new Address(info.hashBuffer, network, Address.PayToPublicKeyHash);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from a ripemd160 script hash
|
|
*
|
|
* @param {Buffer} hash - An instance of buffer of the hash
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
*/
|
|
Address.fromScriptHash = function(hash, network) {
|
|
var info = Address._transformHash(hash);
|
|
return new Address(info.hashBuffer, network, Address.PayToScriptHash);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from a Script
|
|
*
|
|
* @param {Script} script - An instance of Script
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
*/
|
|
Address.fromScript = function(script, network) {
|
|
var info = Address._transformScript(script, network);
|
|
return new Address(info.hashBuffer, network, info.type);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from a buffer of the address
|
|
*
|
|
* @param {Buffer} buffer - An instance of buffer of the address
|
|
* @param {String|Network} [network] - either a Network instance, 'livenet', or 'testnet'
|
|
* @param {String} [type] - The type of address: 'script' or 'pubkey'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
*/
|
|
Address.fromBuffer = function(buffer, network, type) {
|
|
var info = Address._transformBuffer(buffer, network, type);
|
|
return new Address(info.hashBuffer, info.network, info.type);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from an address string
|
|
*
|
|
* @param {String} str - An string of the bitcoin address
|
|
* @param {String|Network} [network] - either a Network instance, 'livenet', or 'testnet'
|
|
* @param {String} [type] - The type of address: 'script' or 'pubkey'
|
|
* @returns {Address} A new valid and frozen instance of an Address
|
|
*/
|
|
Address.fromString = function(str, network, type) {
|
|
var info = Address._transformString(str, network, type);
|
|
return new Address(info.hashBuffer, info.network, info.type);
|
|
};
|
|
|
|
/**
|
|
* Instantiate an address from JSON
|
|
*
|
|
* @param {String} json - An JSON string or Object with keys: hash, network and type
|
|
* @returns {Address} A new valid instance of an Address
|
|
*/
|
|
Address.fromJSON = function fromJSON(json) {
|
|
if (JSUtil.isValidJSON(json)) {
|
|
json = JSON.parse(json);
|
|
}
|
|
$.checkState(
|
|
JSUtil.isHexa(json.hash),
|
|
'Unexpected hash property, "' + json.hash + '", expected to be hex.'
|
|
);
|
|
var hashBuffer = new Buffer(json.hash, 'hex');
|
|
return new Address(hashBuffer, json.network, json.type);
|
|
};
|
|
|
|
/**
|
|
* Will return a validation error if exists
|
|
*
|
|
* @example
|
|
* ```javascript
|
|
* // a network mismatch error
|
|
* var error = Address.getValidationError('15vkcKf7gB23wLAnZLmbVuMiiVDc1Nm4a2', 'testnet');
|
|
* ```
|
|
*
|
|
* @param {String} data - The encoded data
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @param {String} type - The type of address: 'script' or 'pubkey'
|
|
* @returns {null|Error} The corresponding error message
|
|
*/
|
|
Address.getValidationError = function(data, network, type) {
|
|
var error;
|
|
try {
|
|
/* jshint nonew: false */
|
|
new Address(data, network, type);
|
|
} catch (e) {
|
|
error = e;
|
|
}
|
|
return error;
|
|
};
|
|
|
|
/**
|
|
* Will return a boolean if an address is valid
|
|
*
|
|
* @example
|
|
* ```javascript
|
|
* assert(Address.isValid('15vkcKf7gB23wLAnZLmbVuMiiVDc1Nm4a2', 'livenet'));
|
|
* ```
|
|
*
|
|
* @param {String} data - The encoded data
|
|
* @param {String|Network} network - either a Network instance, 'livenet', or 'testnet'
|
|
* @param {String} type - The type of address: 'script' or 'pubkey'
|
|
* @returns {boolean} The corresponding error message
|
|
*/
|
|
Address.isValid = function(data, network, type) {
|
|
return !Address.getValidationError(data, network, type);
|
|
};
|
|
|
|
/**
|
|
* Returns true if an address is of pay to public key hash type
|
|
* @return boolean
|
|
*/
|
|
Address.prototype.isPayToPublicKeyHash = function() {
|
|
return this.type === Address.PayToPublicKeyHash;
|
|
};
|
|
|
|
/**
|
|
* Returns true if an address is of pay to script hash type
|
|
* @return boolean
|
|
*/
|
|
Address.prototype.isPayToScriptHash = function() {
|
|
return this.type === Address.PayToScriptHash;
|
|
};
|
|
|
|
/**
|
|
* Will return a buffer representation of the address
|
|
*
|
|
* @returns {Buffer} Bitcoin address buffer
|
|
*/
|
|
Address.prototype.toBuffer = function() {
|
|
var version = new Buffer([this.network[this.type]]);
|
|
var buf = Buffer.concat([version, this.hashBuffer]);
|
|
return buf;
|
|
};
|
|
|
|
/**
|
|
* @returns {Object} A plain object with the address information
|
|
*/
|
|
Address.prototype.toObject = function toObject() {
|
|
return {
|
|
hash: this.hashBuffer.toString('hex'),
|
|
type: this.type,
|
|
network: this.network.toString()
|
|
};
|
|
};
|
|
|
|
/**
|
|
* @returns {String} A JSON representation of a plain object with the address information
|
|
*/
|
|
Address.prototype.toJSON = function toJSON() {
|
|
return JSON.stringify(this.toObject());
|
|
};
|
|
|
|
/**
|
|
* Will return a the string representation of the address
|
|
*
|
|
* @returns {String} Bitcoin address
|
|
*/
|
|
Address.prototype.toString = function() {
|
|
return base58check.encode(this.toBuffer());
|
|
};
|
|
|
|
/**
|
|
* Will return a string formatted for the console
|
|
*
|
|
* @returns {String} Bitcoin address
|
|
*/
|
|
Address.prototype.inspect = function() {
|
|
return '<Address: ' + this.toString() + ', type: '+this.type+', network: '+this.network+'>';
|
|
};
|
|
|
|
module.exports = Address;
|
|
|