Browse Source

Merge pull request #688 from eordano/feature/addressMethods

Add check functions to Address
patch-2
Manuel Aráoz 10 years ago
parent
commit
22b1193e32
  1. 32
      lib/address.js
  2. 134
      test/address.js

32
lib/address.js

@ -44,7 +44,7 @@ function Address(data, network, type) {
throw new TypeError('Second argument must be "livenet" or "testnet".'); throw new TypeError('Second argument must be "livenet" or "testnet".');
} }
if (type && (type !== 'pubkeyhash' && type !== 'scripthash')) { if (type && (type !== Address.PayToPublicKeyHash && type !== Address.PayToScriptHash)) {
throw new TypeError('Third argument must be "pubkeyhash" or "scripthash".'); throw new TypeError('Third argument must be "pubkeyhash" or "scripthash".');
} }
@ -69,7 +69,7 @@ function Address(data, network, type) {
// set defaults if not set // set defaults if not set
info.network = info.network || network || networks.defaultNetwork.name; info.network = info.network || network || networks.defaultNetwork.name;
info.type = info.type || type || 'pubkeyhash'; info.type = info.type || type || Address.PayToPublicKeyHash;
Object.defineProperty(this, 'hashBuffer', { Object.defineProperty(this, 'hashBuffer', {
configurable: false, configurable: false,
@ -87,9 +87,11 @@ function Address(data, network, type) {
}); });
return this; return this;
} }
Address.PayToPublicKeyHash = 'pubkeyhash';
Address.PayToScriptHash = 'scripthash';
/** /**
* *
* Internal function to transform a hash buffer * Internal function to transform a hash buffer
@ -182,7 +184,7 @@ Address._transformPublicKey = function(pubkey){
throw new TypeError('Address must be an instance of PublicKey.'); throw new TypeError('Address must be an instance of PublicKey.');
} }
info.hashBuffer = Hash.sha256ripemd160(pubkey.toBuffer()); info.hashBuffer = Hash.sha256ripemd160(pubkey.toBuffer());
info.type = 'pubkeyhash'; info.type = Address.PayToPublicKeyHash;
return info; return info;
}; };
@ -200,7 +202,7 @@ Address._transformScript = function(script){
throw new TypeError('Address must be an instance of Script.'); throw new TypeError('Address must be an instance of Script.');
} }
info.hashBuffer = Hash.sha256ripemd160(script.toBuffer()); info.hashBuffer = Hash.sha256ripemd160(script.toBuffer());
info.type = 'scripthash'; info.type = Address.PayToScriptHash;
return info; return info;
}; };
@ -247,7 +249,7 @@ Address.fromPublicKey = function(data, network){
*/ */
Address.fromPublicKeyHash = function(hash, network) { Address.fromPublicKeyHash = function(hash, network) {
var info = Address._transformHash(hash); var info = Address._transformHash(hash);
return new Address(info.hashBuffer, network, 'pubkeyhash'); return new Address(info.hashBuffer, network, Address.PayToPublicKeyHash);
}; };
/** /**
@ -260,7 +262,7 @@ Address.fromPublicKeyHash = function(hash, network) {
*/ */
Address.fromScriptHash = function(hash, network) { Address.fromScriptHash = function(hash, network) {
var info = Address._transformHash(hash); var info = Address._transformHash(hash);
return new Address(info.hashBuffer, network, 'scripthash'); return new Address(info.hashBuffer, network, Address.PayToScriptHash);
}; };
/** /**
@ -346,6 +348,22 @@ Address.isValid = function(data, network, type) {
return !Address.getValidationError(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 * Will return a buffer representation of the address

134
test/address.js

@ -1,5 +1,7 @@
'use strict'; 'use strict';
/* jshint maxstatements: 30 */
var should = require('chai').should(); var should = require('chai').should();
var bitcore = require('..'); var bitcore = require('..');
@ -13,29 +15,28 @@ describe('Address', function() {
var pubkeyhash = new Buffer('3c3fa3d4adcaf8f52d5b1843975e122548269937', 'hex'); var pubkeyhash = new Buffer('3c3fa3d4adcaf8f52d5b1843975e122548269937', 'hex');
var buf = Buffer.concat([new Buffer([0]), pubkeyhash]); var buf = Buffer.concat([new Buffer([0]), pubkeyhash]);
var str = '16VZnHwRhwrExfeHFHGjwrgEMq8VcYPs9r'; var str = '16VZnHwRhwrExfeHFHGjwrgEMq8VcYPs9r';
var strTest = 'n28S35tqEMbt6vNad7A5K3mZ7vdn8dZ86X';
it('should throw an error because of missing data', function() { it('should throw an error because of missing data', function() {
(function() { (function() {
var a = new Address(); return new Address();
}).should.throw('First argument is required, please include address data.'); }).should.throw('First argument is required, please include address data.');
}); });
it('should throw an error because of bad network param', function() { it('should throw an error because of bad network param', function() {
(function(){ (function(){
var a = new Address(validAddresses[0], 'main', 'pubkeyhash'); return new Address(PKHLivenet[0], 'main', 'pubkeyhash');
}).should.throw('Second argument must be "livenet" or "testnet".'); }).should.throw('Second argument must be "livenet" or "testnet".');
}); });
it('should throw an error because of bad type param', function() { it('should throw an error because of bad type param', function() {
(function() { (function() {
var a = new Address(validAddresses[0], 'livenet', 'pubkey'); return new Address(PKHLivenet[0], 'livenet', 'pubkey');
}).should.throw('Third argument must be "pubkeyhash" or "scripthash"'); }).should.throw('Third argument must be "pubkeyhash" or "scripthash"');
}); });
// livenet valid // livenet valid
var validAddresses = [ var PKHLivenet = [
'15vkcKf7gB23wLAnZLmbVuMiiVDc1Nm4a2', '15vkcKf7gB23wLAnZLmbVuMiiVDc1Nm4a2',
'1A6ut1tWnUq1SEQLMr4ttDh24wcbJ5o9TT', '1A6ut1tWnUq1SEQLMr4ttDh24wcbJ5o9TT',
'1BpbpfLdY7oBS9gK7aDXgvMgr1DPvNhEB2', '1BpbpfLdY7oBS9gK7aDXgvMgr1DPvNhEB2',
@ -43,7 +44,7 @@ describe('Address', function() {
]; ];
// livenet p2sh // livenet p2sh
var validp2shAddresses = [ var P2SHLivenet = [
'342ftSRCvFHfCeFFBuz4xwbeqnDw6BGUey', '342ftSRCvFHfCeFFBuz4xwbeqnDw6BGUey',
'33vt8ViH5jsr115AGkW6cEmEz9MpvJSwDk', '33vt8ViH5jsr115AGkW6cEmEz9MpvJSwDk',
'37Sp6Rv3y4kVd1nQ1JV5pfqXccHNyZm1x3', '37Sp6Rv3y4kVd1nQ1JV5pfqXccHNyZm1x3',
@ -51,7 +52,7 @@ describe('Address', function() {
]; ];
// testnet p2sh // testnet p2sh
var testValidp2shAddresses = [ var P2SHTestnet = [
'2N7FuwuUuoTBrDFdrAZ9KxBmtqMLxce9i1C', '2N7FuwuUuoTBrDFdrAZ9KxBmtqMLxce9i1C',
'2NEWDzHWwY5ZZp8CQWbB7ouNMLqCia6YRda', '2NEWDzHWwY5ZZp8CQWbB7ouNMLqCia6YRda',
'2MxgPqX1iThW3oZVk9KoFcE5M4JpiETssVN', '2MxgPqX1iThW3oZVk9KoFcE5M4JpiETssVN',
@ -75,7 +76,7 @@ describe('Address', function() {
]; ];
//testnet valid //testnet valid
var testValidAddresses = [ var PKHTestnet = [
'n28S35tqEMbt6vNad7A5K3mZ7vdn8dZ86X', 'n28S35tqEMbt6vNad7A5K3mZ7vdn8dZ86X',
'n45x3R2w2jaSC62BMa9MeJCd3TXxgvDEmm', 'n45x3R2w2jaSC62BMa9MeJCd3TXxgvDEmm',
'mursDVxqNQmmwWHACpM9VHwVVSfTddGsEM', 'mursDVxqNQmmwWHACpM9VHwVVSfTddGsEM',
@ -100,42 +101,42 @@ describe('Address', function() {
}); });
it('should validate addresses', function() { it('should validate addresses', function() {
for(var i=0;i<validAddresses.length;i++){ for (var i = 0; i < PKHLivenet.length; i++) {
var error = Address.getValidationError(validAddresses[i]); var error = Address.getValidationError(PKHLivenet[i]);
should.not.exist(error); should.not.exist(error);
} }
}); });
it('should validate p2sh addresses', function() { it('should validate p2sh addresses', function() {
for(var i=0;i<validp2shAddresses.length;i++){ for (var i = 0; i < P2SHLivenet.length; i++) {
var error = Address.getValidationError(validp2shAddresses[i]); var error = Address.getValidationError(P2SHLivenet[i]);
should.not.exist(error); should.not.exist(error);
} }
}); });
it('should validate testnet p2sh addresses', function() { it('should validate testnet p2sh addresses', function() {
for(var i=0;i<testValidp2shAddresses.length;i++){ for (var i = 0; i < P2SHTestnet.length; i++) {
var error = Address.getValidationError(testValidp2shAddresses[i], 'testnet'); var error = Address.getValidationError(P2SHTestnet[i], 'testnet');
should.not.exist(error); should.not.exist(error);
} }
}); });
it('should not validate addresses with params', function() { it('should not validate addresses with params', function() {
for(var i=0;i<validAddresses.length;i++){ for (var i = 0; i < PKHLivenet.length; i++) {
var error = Address.getValidationError(validAddresses[i], 'testnet'); var error = Address.getValidationError(PKHLivenet[i], 'testnet');
should.exist(error); should.exist(error);
} }
}); });
it('should validate addresses with params', function() { it('should validate addresses with params', function() {
for(var i=0;i<validAddresses.length;i++){ for (var i = 0; i < PKHLivenet.length; i++) {
var error = Address.getValidationError(validAddresses[i], 'livenet'); var error = Address.getValidationError(PKHLivenet[i], 'livenet');
should.not.exist(error); should.not.exist(error);
} }
}); });
it('should not validate because of an invalid checksum', function() { it('should not validate because of an invalid checksum', function() {
for(var i=0;i<badChecksums.length;i++){ for (var i = 0; i < badChecksums.length; i++) {
var error = Address.getValidationError(badChecksums[i], 'livenet', 'pubkeyhash'); var error = Address.getValidationError(badChecksums[i], 'livenet', 'pubkeyhash');
should.exist(error); should.exist(error);
error.message.should.equal('Checksum mismatch'); error.message.should.equal('Checksum mismatch');
@ -143,8 +144,8 @@ describe('Address', function() {
}); });
it('should not validate because of mismatched network', function() { it('should not validate because of mismatched network', function() {
for(var i=0;i<validAddresses.length;i++){ for (var i = 0; i < PKHLivenet.length; i++) {
var error = Address.getValidationError(validAddresses[i], 'testnet', 'pubkeyhash'); var error = Address.getValidationError(PKHLivenet[i], 'testnet', 'pubkeyhash');
should.exist(error); should.exist(error);
error.message.should.equal('Address has mismatched network type.'); error.message.should.equal('Address has mismatched network type.');
} }
@ -152,15 +153,15 @@ describe('Address', function() {
}); });
it('should not validate because of a mismatched type', function() { it('should not validate because of a mismatched type', function() {
for(var i=0;i<validAddresses.length;i++){ for (var i = 0; i < PKHLivenet.length; i++) {
var error = Address.getValidationError(validAddresses[i], 'livenet', 'scripthash'); var error = Address.getValidationError(PKHLivenet[i], 'livenet', 'scripthash');
should.exist(error); should.exist(error);
error.message.should.equal('Address has mismatched type.'); error.message.should.equal('Address has mismatched type.');
} }
}); });
it('should not validate because of non-base58 characters', function() { it('should not validate because of non-base58 characters', function() {
for(var i=0;i<nonBase58.length;i++){ for (var i = 0; i < nonBase58.length; i++) {
var error = Address.getValidationError(nonBase58[i], 'livenet', 'pubkeyhash'); var error = Address.getValidationError(nonBase58[i], 'livenet', 'pubkeyhash');
should.exist(error); should.exist(error);
error.message.should.equal('Non-base58 character'); error.message.should.equal('Non-base58 character');
@ -168,22 +169,22 @@ describe('Address', function() {
}); });
it('should not validate addresses', function() { it('should not validate addresses', function() {
for(var i=0;i<badChecksums.length;i++){ for (var i = 0; i < badChecksums.length; i++) {
var error = Address.getValidationError(badChecksums[i]); var error = Address.getValidationError(badChecksums[i]);
should.exist(error); should.exist(error);
} }
}); });
it('should validate testnet addresses', function() { it('should validate testnet addresses', function() {
for(var i=0;i<testValidAddresses.length;i++){ for (var i = 0; i < PKHTestnet.length; i++) {
var error = Address.getValidationError(testValidAddresses[i], 'testnet'); var error = Address.getValidationError(PKHTestnet[i], 'testnet');
should.not.exist(error); should.not.exist(error);
} }
}); });
it('should not validate testnet addresses because of mismatched network', function() { it('should not validate testnet addresses because of mismatched network', function() {
for(var i=0;i<testValidAddresses.length;i++){ for (var i = 0; i < PKHTestnet.length; i++) {
var error = Address.getValidationError(testValidAddresses[i], 'livenet', 'pubkeyhash'); var error = Address.getValidationError(PKHTestnet[i], 'livenet', 'pubkeyhash');
should.exist(error); should.exist(error);
error.message.should.equal('Address has mismatched network type.'); error.message.should.equal('Address has mismatched network type.');
} }
@ -194,71 +195,71 @@ describe('Address', function() {
describe('encodings', function() { describe('encodings', function() {
it('should make an address from a buffer', function() { it('should make an address from a buffer', function() {
var a = Address.fromBuffer(buf).toString().should.equal(str); Address.fromBuffer(buf).toString().should.equal(str);
var b = new Address(buf).toString().should.equal(str); new Address(buf).toString().should.equal(str);
var c = Address(buf).toString().should.equal(str); Address(buf).toString().should.equal(str);
}); });
it('should make an address from a string', function() { it('should make an address from a string', function() {
var a = Address.fromString(str).toString().should.equal(str); Address.fromString(str).toString().should.equal(str);
var b = new Address(str).toString().should.equal(str); new Address(str).toString().should.equal(str);
}); });
it('should error because of unrecognized data format', function() { it('should error because of unrecognized data format', function() {
(function() { (function() {
var a = new Address(new Error()); return new Address(new Error());
}).should.throw('First argument is an unrecognized data format.'); }).should.throw('First argument is an unrecognized data format.');
}); });
it('should error because of incorrect format for pubkey hash', function() { it('should error because of incorrect format for pubkey hash', function() {
(function() { (function() {
var a = new Address.fromPublicKeyHash('notahash'); return new Address.fromPublicKeyHash('notahash');
}).should.throw('Address supplied is not a buffer.'); }).should.throw('Address supplied is not a buffer.');
}); });
it('should error because of incorrect format for script hash', function() { it('should error because of incorrect format for script hash', function() {
(function() { (function() {
var a = new Address.fromScriptHash('notascript'); return new Address.fromScriptHash('notascript');
}).should.throw('Address supplied is not a buffer.'); }).should.throw('Address supplied is not a buffer.');
}); });
it('should error because of incorrect type for transform buffer', function() { it('should error because of incorrect type for transform buffer', function() {
(function() { (function() {
var info = Address._transformBuffer('notabuffer'); return Address._transformBuffer('notabuffer');
}).should.throw('Address supplied is not a buffer.'); }).should.throw('Address supplied is not a buffer.');
}); });
it('should error because of incorrect length buffer for transform buffer', function() { it('should error because of incorrect length buffer for transform buffer', function() {
(function() { (function() {
var info = Address._transformBuffer(new Buffer(20)); return Address._transformBuffer(new Buffer(20));
}).should.throw('Address buffers must be exactly 21 bytes.'); }).should.throw('Address buffers must be exactly 21 bytes.');
}); });
it('should error because of incorrect type for pubkey transform', function() { it('should error because of incorrect type for pubkey transform', function() {
(function() { (function() {
var info = Address._transformPublicKey(new Buffer(20)); return Address._transformPublicKey(new Buffer(20));
}).should.throw('Address must be an instance of PublicKey.'); }).should.throw('Address must be an instance of PublicKey.');
}); });
it('should error because of incorrect type for script transform', function() { it('should error because of incorrect type for script transform', function() {
(function() { (function() {
var info = Address._transformScript(new Buffer(20)); return Address._transformScript(new Buffer(20));
}).should.throw('Address must be an instance of Script.'); }).should.throw('Address must be an instance of Script.');
}); });
it('should error because of incorrect type for string transform', function() { it('should error because of incorrect type for string transform', function() {
(function() { (function() {
var info = Address._transformString(new Buffer(20)); return Address._transformString(new Buffer(20));
}).should.throw('Address supplied is not a string.'); }).should.throw('Address supplied is not a string.');
}); });
it('should make an address from a pubkey hash buffer', function() { it('should make an address from a pubkey hash buffer', function() {
var hash = pubkeyhash; //use the same hash var hash = pubkeyhash; //use the same hash
var a = Address.fromPublicKeyHash(hash).toString().should.equal(str); Address.fromPublicKeyHash(hash).toString().should.equal(str);
var b = Address.fromPublicKeyHash(hash, 'testnet'); var b = Address.fromPublicKeyHash(hash, 'testnet');
b.network.should.equal('testnet'); b.network.should.equal('testnet');
b.type.should.equal('pubkeyhash'); b.type.should.equal('pubkeyhash');
var c = new Address(hash).toString().should.equal(str); new Address(hash).toString().should.equal(str);
}); });
it('should make an address using the default network', function() { it('should make an address using the default network', function() {
@ -275,7 +276,7 @@ describe('Address', function() {
it('should throw an error for invalid length hashBuffer', function() { it('should throw an error for invalid length hashBuffer', function() {
(function() { (function() {
var a = Address.fromPublicKeyHash(buf); return Address.fromPublicKeyHash(buf);
}).should.throw('Address hashbuffers must be exactly 20 bytes.'); }).should.throw('Address hashbuffers must be exactly 20 bytes.');
}); });
@ -294,7 +295,7 @@ describe('Address', function() {
}); });
it('should make this address from a script', function() { it('should make this address from a script', function() {
var s = Script.fromString("OP_CHECKMULTISIG"); var s = Script.fromString('OP_CHECKMULTISIG');
var buf = s.toBuffer(); var buf = s.toBuffer();
var a = Address.fromScript(s); var a = Address.fromScript(s);
a.toString().should.equal('3BYmEwgV2vANrmfRymr1mFnHXgLjD6gAWm'); a.toString().should.equal('3BYmEwgV2vANrmfRymr1mFnHXgLjD6gAWm');
@ -305,7 +306,7 @@ describe('Address', function() {
}); });
it('should make this address from other script', function() { it('should make this address from other script', function() {
var s = Script.fromString("OP_CHECKSIG OP_HASH160"); var s = Script.fromString('OP_CHECKSIG OP_HASH160');
var a = Address.fromScript(s); var a = Address.fromScript(s);
a.toString().should.equal('347iRqVwks5r493N1rsLN4k9J7Ljg488W7'); a.toString().should.equal('347iRqVwks5r493N1rsLN4k9J7Ljg488W7');
var b = new Address(s); var b = new Address(s);
@ -321,22 +322,22 @@ describe('Address', function() {
}); });
it('should derive from this known address string testnet', function() { it('should derive from this known address string testnet', function() {
var a = new Address(testValidAddresses[0], 'testnet'); var a = new Address(PKHTestnet[0], 'testnet');
var b = new Address(a.toString()); var b = new Address(a.toString());
b.toString().should.equal(testValidAddresses[0]); b.toString().should.equal(PKHTestnet[0]);
b.network.should.equal('testnet'); b.network.should.equal('testnet');
}); });
it('should derive from this known address string livenet scripthash', function() { it('should derive from this known address string livenet scripthash', function() {
var a = new Address(validp2shAddresses[0], 'livenet', 'scripthash'); var a = new Address(P2SHLivenet[0], 'livenet', 'scripthash');
var b = new Address(a.toString()); var b = new Address(a.toString());
b.toString().should.equal(validp2shAddresses[0]); b.toString().should.equal(P2SHLivenet[0]);
}); });
it('should derive from this known address string testnet scripthash', function() { it('should derive from this known address string testnet scripthash', function() {
var address = new Address(testValidp2shAddresses[0], 'testnet', 'scripthash'); var address = new Address(P2SHTestnet[0], 'testnet', 'scripthash');
address = new Address(address.toString()); address = new Address(address.toString());
address.toString().should.equal(testValidp2shAddresses[0]); address.toString().should.equal(P2SHTestnet[0]);
}); });
}); });
@ -358,30 +359,43 @@ describe('Address', function() {
}); });
it('should output a scripthash address', function() { it('should output a scripthash address', function() {
var address = new Address(validp2shAddresses[0]); var address = new Address(P2SHLivenet[0]);
address.toString().should.equal(validp2shAddresses[0]); address.toString().should.equal(P2SHLivenet[0]);
}); });
it('should output a testnet scripthash address', function() { it('should output a testnet scripthash address', function() {
var address = new Address(testValidp2shAddresses[0]); var address = new Address(P2SHTestnet[0]);
address.toString().should.equal(testValidp2shAddresses[0]); address.toString().should.equal(P2SHTestnet[0]);
}); });
it('should output a testnet pubkeyhash address', function() { it('should output a testnet pubkeyhash address', function() {
var address = new Address(testValidAddresses[0]); var address = new Address(PKHTestnet[0]);
address.toString().should.equal(testValidAddresses[0]); address.toString().should.equal(PKHTestnet[0]);
}); });
}); });
describe('#inspect', function() { describe('#inspect', function() {
it('should output formatted output correctly', function() { it('should output formatted output correctly', function() {
var address = new Address(str); var address = new Address(str);
var output = '<Address: 16VZnHwRhwrExfeHFHGjwrgEMq8VcYPs9r, type: pubkeyhash, network: livenet>'; var output = '<Address: 16VZnHwRhwrExfeHFHGjwrgEMq8VcYPs9r, type: pubkeyhash, network: livenet>';
address.inspect().should.equal(output); address.inspect().should.equal(output);
}); });
});
describe('questions about the address', function() {
it('should detect a P2SH address', function() {
new Address(P2SHLivenet[0]).isPayToScriptHash().should.equal(true);
new Address(P2SHLivenet[0]).isPayToPublicKeyHash().should.equal(false);
new Address(P2SHTestnet[0]).isPayToScriptHash().should.equal(true);
new Address(P2SHTestnet[0]).isPayToPublicKeyHash().should.equal(false);
});
it('should detect a Pay To PubkeyHash address', function() {
new Address(PKHLivenet[0]).isPayToPublicKeyHash().should.equal(true);
new Address(PKHLivenet[0]).isPayToScriptHash().should.equal(false);
new Address(PKHTestnet[0]).isPayToPublicKeyHash().should.equal(true);
new Address(PKHTestnet[0]).isPayToScriptHash().should.equal(false);
});
}); });
}); });

Loading…
Cancel
Save