Browse Source

Merge pull request #206 from dcousens/base58rework

Base58 Check rework
hk-custom-address
Kyle Drake 11 years ago
parent
commit
9b5dfbd865
  1. 12
      src/address.js
  2. 21
      src/base58check.js
  3. 22
      src/eckey.js
  4. 6
      src/ecpubkey.js
  5. 22
      src/hdnode.js
  6. 7
      src/message.js
  7. 12
      test/base58check.js
  8. 6
      test/eckey.js
  9. 5
      test/ecpubkey.js
  10. 322
      test/fixtures/base58check.json
  11. 9
      test/integration/p2sh.js

12
src/address.js

@ -28,9 +28,11 @@ function Address(hash, version) {
// Import functions // Import functions
Address.fromBase58Check = function(string) { Address.fromBase58Check = function(string) {
var decode = base58check.decode(string) var payload = base58check.decode(string)
var version = payload.readUInt8(0)
var hash = payload.slice(1)
return new Address(decode.payload, decode.version) return new Address(hash, version)
} }
Address.fromScriptPubKey = function(script, network) { Address.fromScriptPubKey = function(script, network) {
@ -51,7 +53,11 @@ Address.fromScriptPubKey = function(script, network) {
// Export functions // Export functions
Address.prototype.toBase58Check = function () { Address.prototype.toBase58Check = function () {
return base58check.encode(this.hash, this.version) var payload = new Buffer(21)
payload.writeUInt8(this.version, 0)
this.hash.copy(payload, 1)
return base58check.encode(payload)
} }
Address.prototype.toScriptPubKey = function() { Address.prototype.toScriptPubKey = function() {

21
src/base58check.js

@ -4,13 +4,11 @@ var base58 = require('./base58')
var crypto = require('./crypto') var crypto = require('./crypto')
// Encode a buffer as a base58-check-encoded string // Encode a buffer as a base58-check-encoded string
function encode(payload, version) { function encode(payload) {
var version = new Buffer([version]) var checksum = crypto.hash256(payload).slice(0, 4)
var message = Buffer.concat([version, payload])
var checksum = crypto.hash256(message).slice(0, 4)
return base58.encode(Buffer.concat([ return base58.encode(Buffer.concat([
message, payload,
checksum checksum
])) ]))
} }
@ -19,20 +17,13 @@ function encode(payload, version) {
function decode(string) { function decode(string) {
var buffer = base58.decode(string) var buffer = base58.decode(string)
var message = buffer.slice(0, -4) var payload = buffer.slice(0, -4)
var checksum = buffer.slice(-4) var checksum = buffer.slice(-4)
var newChecksum = crypto.hash256(message).slice(0, 4) var newChecksum = crypto.hash256(payload).slice(0, 4)
assert.deepEqual(newChecksum, checksum, 'Invalid checksum') assert.deepEqual(newChecksum, checksum, 'Invalid checksum')
var version = message.readUInt8(0) return payload
var payload = message.slice(1)
return {
version: version,
payload: payload,
checksum: checksum
}
} }
module.exports = { module.exports = {

22
src/eckey.js

@ -22,13 +22,16 @@ function ECKey(D, compressed) {
// Static constructors // Static constructors
ECKey.fromWIF = function(string) { ECKey.fromWIF = function(string) {
var decode = base58check.decode(string) var payload = base58check.decode(string)
var payload = decode.payload
var compressed = false var compressed = false
// Ignore the version byte
payload = payload.slice(1)
if (payload.length === 33) { if (payload.length === 33) {
assert.strictEqual(payload[32], 0x01, 'Invalid compression flag') assert.strictEqual(payload[32], 0x01, 'Invalid compression flag')
// Truncate the compression flag
payload = payload.slice(0, -1) payload = payload.slice(0, -1)
compressed = true compressed = true
} }
@ -50,15 +53,20 @@ ECKey.makeRandom = function(compressed, rng) {
} }
// Export functions // Export functions
ECKey.prototype.toWIF = function(version) { ECKey.prototype.toWIF = function(network) {
version = version || networks.bitcoin.wif network = network || networks.bitcoin
var bufferLen = this.pub.compressed ? 34 : 33
var buffer = new Buffer(bufferLen)
buffer.writeUInt8(network.wif, 0)
this.D.toBuffer(32).copy(buffer, 1)
var buffer = this.D.toBuffer(32)
if (this.pub.compressed) { if (this.pub.compressed) {
buffer = Buffer.concat([buffer, new Buffer([0x01])]) buffer.writeUInt8(0x01, 33)
} }
return base58check.encode(buffer, version) return base58check.encode(buffer)
} }
// Operations // Operations

6
src/ecpubkey.js

@ -30,10 +30,10 @@ ECPubKey.fromHex = function(hex) {
} }
// Operations // Operations
ECPubKey.prototype.getAddress = function(version) { ECPubKey.prototype.getAddress = function(network) {
version = version || networks.bitcoin.pubKeyHash network = network || networks.bitcoin
return new Address(crypto.hash160(this.toBuffer()), version) return new Address(crypto.hash160(this.toBuffer()), network.pubKeyHash)
} }
ECPubKey.prototype.verify = function(hash, signature) { ECPubKey.prototype.verify = function(hash, signature) {

22
src/hdnode.js

@ -1,5 +1,5 @@
var assert = require('assert') var assert = require('assert')
var base58 = require('./base58') var base58check = require('./base58check')
var BigInteger = require('bigi') var BigInteger = require('bigi')
var crypto = require('./crypto') var crypto = require('./crypto')
@ -68,15 +68,7 @@ HDNode.fromSeedHex = function(hex, network) {
} }
HDNode.fromBase58 = function(string) { HDNode.fromBase58 = function(string) {
var buffer = base58.decode(string) return HDNode.fromBuffer(base58check.decode(string))
var payload = buffer.slice(0, -4)
var checksum = buffer.slice(-4)
var newChecksum = crypto.hash256(payload).slice(0, 4)
assert.deepEqual(newChecksum, checksum, 'Invalid checksum')
return HDNode.fromBuffer(payload)
} }
HDNode.fromBuffer = function(buffer) { HDNode.fromBuffer = function(buffer) {
@ -144,17 +136,11 @@ HDNode.prototype.getFingerprint = function() {
} }
HDNode.prototype.getAddress = function() { HDNode.prototype.getAddress = function() {
return this.pubKey.getAddress(this.network.pubKeyHash) return this.pubKey.getAddress(this.network)
} }
HDNode.prototype.toBase58 = function(isPrivate) { HDNode.prototype.toBase58 = function(isPrivate) {
var buffer = this.toBuffer(isPrivate) return base58check.encode(this.toBuffer(isPrivate))
var checksum = crypto.hash256(buffer).slice(0, 4)
return base58.encode(Buffer.concat([
buffer,
checksum
]))
} }
HDNode.prototype.toBuffer = function(isPrivate) { HDNode.prototype.toBuffer = function(isPrivate) {

7
src/message.js

@ -37,10 +37,9 @@ function sign(key, message, network) {
// TODO: network could be implied from address // TODO: network could be implied from address
function verify(address, compactSig, message, network) { function verify(address, compactSig, message, network) {
if (typeof address === 'string') { if (address instanceof Address) {
address = Address.fromBase58Check(address) address = address.toString()
} }
network = network || networks.bitcoin network = network || networks.bitcoin
var hash = magicHash(message, network) var hash = magicHash(message, network)
@ -49,7 +48,7 @@ function verify(address, compactSig, message, network) {
var Q = ecdsa.recoverPubKey(ecparams, e, parsed.signature, parsed.i) var Q = ecdsa.recoverPubKey(ecparams, e, parsed.signature, parsed.i)
var pubKey = new ECPubKey(Q, parsed.compressed) var pubKey = new ECPubKey(Q, parsed.compressed)
return pubKey.getAddress(address.version).toString() === address.toString() return pubKey.getAddress(network).toString() === address
} }
module.exports = { module.exports = {

12
test/base58check.js

@ -10,20 +10,16 @@ describe('base58check', function() {
fixtures.valid.forEach(function(f) { fixtures.valid.forEach(function(f) {
it('can decode ' + f.string, function() { it('can decode ' + f.string, function() {
var actual = base58check.decode(f.string) var actual = base58check.decode(f.string)
var expected = { var expected = h2b(f.payload)
version: f.decode.version || 0,
payload: h2b(f.decode.payload),
checksum: h2b(f.decode.checksum)
}
assert.deepEqual(actual, expected) assert.deepEqual(actual, expected)
}) })
}) })
fixtures.invalid.forEach(function(f) { fixtures.invalid.forEach(function(f) {
it('throws on ' + f.description, function() { it('throws on ' + f, function() {
assert.throws(function() { assert.throws(function() {
base58check.decode(f.string) base58check.decode(f)
}, /Invalid checksum/) }, /Invalid checksum/)
}) })
}) })
@ -32,7 +28,7 @@ describe('base58check', function() {
describe('encode', function() { describe('encode', function() {
fixtures.valid.forEach(function(f) { fixtures.valid.forEach(function(f) {
it('can encode ' + f.string, function() { it('can encode ' + f.string, function() {
var actual = base58check.encode(h2b(f.decode.payload), f.decode.version) var actual = base58check.encode(h2b(f.payload))
var expected = f.string var expected = f.string
assert.strictEqual(actual, expected) assert.strictEqual(actual, expected)

6
test/eckey.js

@ -1,11 +1,11 @@
var assert = require('assert') var assert = require('assert')
var crypto = require('../src/crypto') var crypto = require('../src/crypto')
var networks = require('../src/networks')
var BigInteger = require('bigi') var BigInteger = require('bigi')
var ECKey = require('../src/eckey') var ECKey = require('../src/eckey')
var fixtures = require('./fixtures/eckey.json') var fixtures = require('./fixtures/eckey.json')
var networks = require('../src/networks')
describe('ECKey', function() { describe('ECKey', function() {
describe('constructor', function() { describe('constructor', function() {
@ -66,8 +66,8 @@ describe('ECKey', function() {
f.WIFs.forEach(function(wif) { f.WIFs.forEach(function(wif) {
it('exports ' + wif.string + ' correctly', function() { it('exports ' + wif.string + ' correctly', function() {
var privKey = ECKey.fromWIF(wif.string) var privKey = ECKey.fromWIF(wif.string)
var version = networks[wif.network].wif var network = networks[wif.network]
var result = privKey.toWIF(version) var result = privKey.toWIF(network)
assert.equal(result, wif.string) assert.equal(result, wif.string)
}) })

5
test/ecpubkey.js

@ -1,5 +1,6 @@
var assert = require('assert') var assert = require('assert')
var crypto = require('../src/crypto') var crypto = require('../src/crypto')
var networks = require('../src/networks')
var sec = require('../src/sec') var sec = require('../src/sec')
var ecparams = sec('secp256k1') var ecparams = sec('secp256k1')
@ -71,9 +72,9 @@ describe('ECPubKey', function() {
it('supports alternative networks', function() { it('supports alternative networks', function() {
var pubKey = new ECPubKey(Q) var pubKey = new ECPubKey(Q)
var address = pubKey.getAddress(0x09) var address = pubKey.getAddress(networks.testnet)
assert.equal(address.version, 0x09) assert.equal(address.version, networks.testnet.pubKeyHash)
assert.equal(address.hash.toString('hex'), fixtures.compressed.hash160) assert.equal(address.hash.toString('hex'), fixtures.compressed.hash160)
}) })
}) })

322
test/fixtures/base58check.json

@ -2,424 +2,208 @@
"valid": [ "valid": [
{ {
"string": "1AGNa15ZQXAZUgFiqJ2i7Z2DPU2J6hW62i", "string": "1AGNa15ZQXAZUgFiqJ2i7Z2DPU2J6hW62i",
"decode": { "payload": "0065a16059864a2fdbc7c99a4723a8395bc6f188eb"
"payload": "65a16059864a2fdbc7c99a4723a8395bc6f188eb",
"checksum": "c046b2ff"
}
},
{
"string": "1AGNa15ZQXAZUgFiqJ2i7Z2DPU2J6hW62i",
"decode": {
"version": 0,
"payload": "65a16059864a2fdbc7c99a4723a8395bc6f188eb",
"checksum": "c046b2ff"
}
}, },
{ {
"string": "3CMNFxN1oHBc4R1EpboAL5yzHGgE611Xou", "string": "3CMNFxN1oHBc4R1EpboAL5yzHGgE611Xou",
"decode": { "payload": "0574f209f6ea907e2ea48f74fae05782ae8a665257"
"version": 5,
"payload": "74f209f6ea907e2ea48f74fae05782ae8a665257",
"checksum": "a2353eb8"
}
}, },
{ {
"string": "mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs", "string": "mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs",
"decode": { "payload": "6f53c0307d6851aa0ce7825ba883c6bd9ad242b486"
"version": 111,
"payload": "53c0307d6851aa0ce7825ba883c6bd9ad242b486",
"checksum": "3aa11ed6"
}
}, },
{ {
"string": "2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br", "string": "2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br",
"decode": { "payload": "c46349a418fc4578d10a372b54b45c280cc8c4382f"
"version": 196,
"payload": "6349a418fc4578d10a372b54b45c280cc8c4382f",
"checksum": "c4a2b951"
}
}, },
{ {
"string": "5Kd3NBUAdUnhyzenEwVLy9pBKxSwXvE9FMPyR4UKZvpe6E3AgLr", "string": "5Kd3NBUAdUnhyzenEwVLy9pBKxSwXvE9FMPyR4UKZvpe6E3AgLr",
"decode": { "payload": "80eddbdc1168f1daeadbd3e44c1e3f8f5a284c2029f78ad26af98583a499de5b19"
"version": 128,
"payload": "eddbdc1168f1daeadbd3e44c1e3f8f5a284c2029f78ad26af98583a499de5b19",
"checksum": "13a4f863"
}
}, },
{ {
"string": "Kz6UJmQACJmLtaQj5A3JAge4kVTNQ8gbvXuwbmCj7bsaabudb3RD", "string": "Kz6UJmQACJmLtaQj5A3JAge4kVTNQ8gbvXuwbmCj7bsaabudb3RD",
"decode": { "payload": "8055c9bccb9ed68446d1b75273bbce89d7fe013a8acd1625514420fb2aca1a21c401"
"version": 128,
"payload": "55c9bccb9ed68446d1b75273bbce89d7fe013a8acd1625514420fb2aca1a21c401",
"checksum": "99a16fd4"
}
}, },
{ {
"string": "9213qJab2HNEpMpYNBa7wHGFKKbkDn24jpANDs2huN3yi4J11ko", "string": "9213qJab2HNEpMpYNBa7wHGFKKbkDn24jpANDs2huN3yi4J11ko",
"decode": { "payload": "ef36cb93b9ab1bdabf7fb9f2c04f1b9cc879933530ae7842398eef5a63a56800c2"
"version": 239,
"payload": "36cb93b9ab1bdabf7fb9f2c04f1b9cc879933530ae7842398eef5a63a56800c2",
"checksum": "6a84391c"
}
}, },
{ {
"string": "cTpB4YiyKiBcPxnefsDpbnDxFDffjqJob8wGCEDXxgQ7zQoMXJdH", "string": "cTpB4YiyKiBcPxnefsDpbnDxFDffjqJob8wGCEDXxgQ7zQoMXJdH",
"decode": { "payload": "efb9f4892c9e8282028fea1d2667c4dc5213564d41fc5783896a0d843fc15089f301"
"version": 239,
"payload": "b9f4892c9e8282028fea1d2667c4dc5213564d41fc5783896a0d843fc15089f301",
"checksum": "b008368c"
}
}, },
{ {
"string": "1Ax4gZtb7gAit2TivwejZHYtNNLT18PUXJ", "string": "1Ax4gZtb7gAit2TivwejZHYtNNLT18PUXJ",
"decode": { "payload": "006d23156cbbdcc82a5a47eee4c2c7c583c18b6bf4"
"version": 0,
"payload": "6d23156cbbdcc82a5a47eee4c2c7c583c18b6bf4",
"checksum": "0680e989"
}
}, },
{ {
"string": "3QjYXhTkvuj8qPaXHTTWb5wjXhdsLAAWVy", "string": "3QjYXhTkvuj8qPaXHTTWb5wjXhdsLAAWVy",
"decode": { "payload": "05fcc5460dd6e2487c7d75b1963625da0e8f4c5975"
"version": 5,
"payload": "fcc5460dd6e2487c7d75b1963625da0e8f4c5975",
"checksum": "8b495dfc"
}
}, },
{ {
"string": "n3ZddxzLvAY9o7184TB4c6FJasAybsw4HZ", "string": "n3ZddxzLvAY9o7184TB4c6FJasAybsw4HZ",
"decode": { "payload": "6ff1d470f9b02370fdec2e6b708b08ac431bf7a5f7"
"version": 111,
"payload": "f1d470f9b02370fdec2e6b708b08ac431bf7a5f7",
"checksum": "b663687c"
}
}, },
{ {
"string": "2NBFNJTktNa7GZusGbDbGKRZTxdK9VVez3n", "string": "2NBFNJTktNa7GZusGbDbGKRZTxdK9VVez3n",
"decode": { "payload": "c4c579342c2c4c9220205e2cdc285617040c924a0a"
"version": 196,
"payload": "c579342c2c4c9220205e2cdc285617040c924a0a",
"checksum": "ba7d11ad"
}
}, },
{ {
"string": "5K494XZwps2bGyeL71pWid4noiSNA2cfCibrvRWqcHSptoFn7rc", "string": "5K494XZwps2bGyeL71pWid4noiSNA2cfCibrvRWqcHSptoFn7rc",
"decode": { "payload": "80a326b95ebae30164217d7a7f57d72ab2b54e3be64928a19da0210b9568d4015e"
"version": 128,
"payload": "a326b95ebae30164217d7a7f57d72ab2b54e3be64928a19da0210b9568d4015e",
"checksum": "171a2dfd"
}
}, },
{ {
"string": "L1RrrnXkcKut5DEMwtDthjwRcTTwED36thyL1DebVrKuwvohjMNi", "string": "L1RrrnXkcKut5DEMwtDthjwRcTTwED36thyL1DebVrKuwvohjMNi",
"decode": { "payload": "807d998b45c219a1e38e99e7cbd312ef67f77a455a9b50c730c27f02c6f730dfb401"
"version": 128,
"payload": "7d998b45c219a1e38e99e7cbd312ef67f77a455a9b50c730c27f02c6f730dfb401",
"checksum": "cd40f1cb"
}
}, },
{ {
"string": "93DVKyFYwSN6wEo3E2fCrFPUp17FtrtNi2Lf7n4G3garFb16CRj", "string": "93DVKyFYwSN6wEo3E2fCrFPUp17FtrtNi2Lf7n4G3garFb16CRj",
"decode": { "payload": "efd6bca256b5abc5602ec2e1c121a08b0da2556587430bcf7e1898af2224885203"
"version": 239,
"payload": "d6bca256b5abc5602ec2e1c121a08b0da2556587430bcf7e1898af2224885203",
"checksum": "f0bf192e"
}
}, },
{ {
"string": "cTDVKtMGVYWTHCb1AFjmVbEbWjvKpKqKgMaR3QJxToMSQAhmCeTN", "string": "cTDVKtMGVYWTHCb1AFjmVbEbWjvKpKqKgMaR3QJxToMSQAhmCeTN",
"decode": { "payload": "efa81ca4e8f90181ec4b61b6a7eb998af17b2cb04de8a03b504b9e34c4c61db7d901"
"version": 239,
"payload": "a81ca4e8f90181ec4b61b6a7eb998af17b2cb04de8a03b504b9e34c4c61db7d901",
"checksum": "7f570e65"
}
}, },
{ {
"string": "1C5bSj1iEGUgSTbziymG7Cn18ENQuT36vv", "string": "1C5bSj1iEGUgSTbziymG7Cn18ENQuT36vv",
"decode": { "payload": "007987ccaa53d02c8873487ef919677cd3db7a6912"
"version": 0,
"payload": "7987ccaa53d02c8873487ef919677cd3db7a6912",
"checksum": "11a2e19b"
}
}, },
{ {
"string": "3AnNxabYGoTxYiTEZwFEnerUoeFXK2Zoks", "string": "3AnNxabYGoTxYiTEZwFEnerUoeFXK2Zoks",
"decode": { "payload": "0563bcc565f9e68ee0189dd5cc67f1b0e5f02f45cb"
"version": 5,
"payload": "63bcc565f9e68ee0189dd5cc67f1b0e5f02f45cb",
"checksum": "35e6c138"
}
}, },
{ {
"string": "n3LnJXCqbPjghuVs8ph9CYsAe4Sh4j97wk", "string": "n3LnJXCqbPjghuVs8ph9CYsAe4Sh4j97wk",
"decode": { "payload": "6fef66444b5b17f14e8fae6e7e19b045a78c54fd79"
"version": 111,
"payload": "ef66444b5b17f14e8fae6e7e19b045a78c54fd79",
"checksum": "7d48bb7f"
}
}, },
{ {
"string": "2NB72XtkjpnATMggui83aEtPawyyKvnbX2o", "string": "2NB72XtkjpnATMggui83aEtPawyyKvnbX2o",
"decode": { "payload": "c4c3e55fceceaa4391ed2a9677f4a4d34eacd021a0"
"version": 196,
"payload": "c3e55fceceaa4391ed2a9677f4a4d34eacd021a0",
"checksum": "03bf5e60"
}
}, },
{ {
"string": "5KaBW9vNtWNhc3ZEDyNCiXLPdVPHCikRxSBWwV9NrpLLa4LsXi9", "string": "5KaBW9vNtWNhc3ZEDyNCiXLPdVPHCikRxSBWwV9NrpLLa4LsXi9",
"decode": { "payload": "80e75d936d56377f432f404aabb406601f892fd49da90eb6ac558a733c93b47252"
"version": 128,
"payload": "e75d936d56377f432f404aabb406601f892fd49da90eb6ac558a733c93b47252",
"checksum": "135d832a"
}
}, },
{ {
"string": "L1axzbSyynNYA8mCAhzxkipKkfHtAXYF4YQnhSKcLV8YXA874fgT", "string": "L1axzbSyynNYA8mCAhzxkipKkfHtAXYF4YQnhSKcLV8YXA874fgT",
"decode": { "payload": "808248bd0375f2f75d7e274ae544fb920f51784480866b102384190b1addfbaa5c01"
"version": 128,
"payload": "8248bd0375f2f75d7e274ae544fb920f51784480866b102384190b1addfbaa5c01",
"checksum": "2daf30c0"
}
}, },
{ {
"string": "927CnUkUbasYtDwYwVn2j8GdTuACNnKkjZ1rpZd2yBB1CLcnXpo", "string": "927CnUkUbasYtDwYwVn2j8GdTuACNnKkjZ1rpZd2yBB1CLcnXpo",
"decode": { "payload": "ef44c4f6a096eac5238291a94cc24c01e3b19b8d8cef72874a079e00a242237a52"
"version": 239,
"payload": "44c4f6a096eac5238291a94cc24c01e3b19b8d8cef72874a079e00a242237a52",
"checksum": "bc446ce4"
}
}, },
{ {
"string": "cUcfCMRjiQf85YMzzQEk9d1s5A4K7xL5SmBCLrezqXFuTVefyhY7", "string": "cUcfCMRjiQf85YMzzQEk9d1s5A4K7xL5SmBCLrezqXFuTVefyhY7",
"decode": { "payload": "efd1de707020a9059d6d3abaf85e17967c6555151143db13dbb06db78df0f15c6901"
"version": 239,
"payload": "d1de707020a9059d6d3abaf85e17967c6555151143db13dbb06db78df0f15c6901",
"checksum": "db1645ec"
}
}, },
{ {
"string": "1Gqk4Tv79P91Cc1STQtU3s1W6277M2CVWu", "string": "1Gqk4Tv79P91Cc1STQtU3s1W6277M2CVWu",
"decode": { "payload": "00adc1cc2081a27206fae25792f28bbc55b831549d"
"version": 0,
"payload": "adc1cc2081a27206fae25792f28bbc55b831549d",
"checksum": "4c762f7e"
}
}, },
{ {
"string": "33vt8ViH5jsr115AGkW6cEmEz9MpvJSwDk", "string": "33vt8ViH5jsr115AGkW6cEmEz9MpvJSwDk",
"decode": { "payload": "05188f91a931947eddd7432d6e614387e32b244709"
"version": 5,
"payload": "188f91a931947eddd7432d6e614387e32b244709",
"checksum": "36a8fc53"
}
}, },
{ {
"string": "mhaMcBxNh5cqXm4aTQ6EcVbKtfL6LGyK2H", "string": "mhaMcBxNh5cqXm4aTQ6EcVbKtfL6LGyK2H",
"decode": { "payload": "6f1694f5bc1a7295b600f40018a618a6ea48eeb498"
"version": 111,
"payload": "1694f5bc1a7295b600f40018a618a6ea48eeb498",
"checksum": "047e9722"
}
}, },
{ {
"string": "2MxgPqX1iThW3oZVk9KoFcE5M4JpiETssVN", "string": "2MxgPqX1iThW3oZVk9KoFcE5M4JpiETssVN",
"decode": { "payload": "c43b9b3fd7a50d4f08d1a5b0f62f644fa7115ae2f3"
"version": 196,
"payload": "3b9b3fd7a50d4f08d1a5b0f62f644fa7115ae2f3",
"checksum": "0fd6ecc5"
}
}, },
{ {
"string": "5HtH6GdcwCJA4ggWEL1B3jzBBUB8HPiBi9SBc5h9i4Wk4PSeApR", "string": "5HtH6GdcwCJA4ggWEL1B3jzBBUB8HPiBi9SBc5h9i4Wk4PSeApR",
"decode": { "payload": "80091035445ef105fa1bb125eccfb1882f3fe69592265956ade751fd095033d8d0"
"version": 128,
"payload": "091035445ef105fa1bb125eccfb1882f3fe69592265956ade751fd095033d8d0",
"checksum": "9c5b305a"
}
}, },
{ {
"string": "L2xSYmMeVo3Zek3ZTsv9xUrXVAmrWxJ8Ua4cw8pkfbQhcEFhkXT8", "string": "L2xSYmMeVo3Zek3ZTsv9xUrXVAmrWxJ8Ua4cw8pkfbQhcEFhkXT8",
"decode": { "payload": "80ab2b4bcdfc91d34dee0ae2a8c6b6668dadaeb3a88b9859743156f462325187af01"
"version": 128,
"payload": "ab2b4bcdfc91d34dee0ae2a8c6b6668dadaeb3a88b9859743156f462325187af01",
"checksum": "271c74db"
}
}, },
{ {
"string": "92xFEve1Z9N8Z641KQQS7ByCSb8kGjsDzw6fAmjHN1LZGKQXyMq", "string": "92xFEve1Z9N8Z641KQQS7ByCSb8kGjsDzw6fAmjHN1LZGKQXyMq",
"decode": { "payload": "efb4204389cef18bbe2b353623cbf93e8678fbc92a475b664ae98ed594e6cf0856"
"version": 239,
"payload": "b4204389cef18bbe2b353623cbf93e8678fbc92a475b664ae98ed594e6cf0856",
"checksum": "13985bb8"
}
}, },
{ {
"string": "cVM65tdYu1YK37tNoAyGoJTR13VBYFva1vg9FLuPAsJijGvG6NEA", "string": "cVM65tdYu1YK37tNoAyGoJTR13VBYFva1vg9FLuPAsJijGvG6NEA",
"decode": { "payload": "efe7b230133f1b5489843260236b06edca25f66adb1be455fbd38d4010d48faeef01"
"version": 239,
"payload": "e7b230133f1b5489843260236b06edca25f66adb1be455fbd38d4010d48faeef01",
"checksum": "45de6187"
}
}, },
{ {
"string": "1JwMWBVLtiqtscbaRHai4pqHokhFCbtoB4", "string": "1JwMWBVLtiqtscbaRHai4pqHokhFCbtoB4",
"decode": { "payload": "00c4c1b72491ede1eedaca00618407ee0b772cad0d"
"version": 0,
"payload": "c4c1b72491ede1eedaca00618407ee0b772cad0d",
"checksum": "27a58137"
}
}, },
{ {
"string": "3QCzvfL4ZRvmJFiWWBVwxfdaNBT8EtxB5y", "string": "3QCzvfL4ZRvmJFiWWBVwxfdaNBT8EtxB5y",
"decode": { "payload": "05f6fe69bcb548a829cce4c57bf6fff8af3a5981f9"
"version": 5,
"payload": "f6fe69bcb548a829cce4c57bf6fff8af3a5981f9",
"checksum": "80242330"
}
}, },
{ {
"string": "mizXiucXRCsEriQCHUkCqef9ph9qtPbZZ6", "string": "mizXiucXRCsEriQCHUkCqef9ph9qtPbZZ6",
"decode": { "payload": "6f261f83568a098a8638844bd7aeca039d5f2352c0"
"version": 111,
"payload": "261f83568a098a8638844bd7aeca039d5f2352c0",
"checksum": "56892155"
}
}, },
{ {
"string": "2NEWDzHWwY5ZZp8CQWbB7ouNMLqCia6YRda", "string": "2NEWDzHWwY5ZZp8CQWbB7ouNMLqCia6YRda",
"decode": { "payload": "c4e930e1834a4d234702773951d627cce82fbb5d2e"
"version": 196,
"payload": "e930e1834a4d234702773951d627cce82fbb5d2e",
"checksum": "3ac92331"
}
}, },
{ {
"string": "5KQmDryMNDcisTzRp3zEq9e4awRmJrEVU1j5vFRTKpRNYPqYrMg", "string": "5KQmDryMNDcisTzRp3zEq9e4awRmJrEVU1j5vFRTKpRNYPqYrMg",
"decode": { "payload": "80d1fab7ab7385ad26872237f1eb9789aa25cc986bacc695e07ac571d6cdac8bc0"
"version": 128,
"payload": "d1fab7ab7385ad26872237f1eb9789aa25cc986bacc695e07ac571d6cdac8bc0",
"checksum": "d6f7166b"
}
}, },
{ {
"string": "L39Fy7AC2Hhj95gh3Yb2AU5YHh1mQSAHgpNixvm27poizcJyLtUi", "string": "L39Fy7AC2Hhj95gh3Yb2AU5YHh1mQSAHgpNixvm27poizcJyLtUi",
"decode": { "payload": "80b0bbede33ef254e8376aceb1510253fc3550efd0fcf84dcd0c9998b288f166b301"
"version": 128,
"payload": "b0bbede33ef254e8376aceb1510253fc3550efd0fcf84dcd0c9998b288f166b301",
"checksum": "e396e2cb"
}
}, },
{ {
"string": "91cTVUcgydqyZLgaANpf1fvL55FH53QMm4BsnCADVNYuWuqdVys", "string": "91cTVUcgydqyZLgaANpf1fvL55FH53QMm4BsnCADVNYuWuqdVys",
"decode": { "payload": "ef037f4192c630f399d9271e26c575269b1d15be553ea1a7217f0cb8513cef41cb"
"version": 239,
"payload": "037f4192c630f399d9271e26c575269b1d15be553ea1a7217f0cb8513cef41cb",
"checksum": "504ffa32"
}
}, },
{ {
"string": "cQspfSzsgLeiJGB2u8vrAiWpCU4MxUT6JseWo2SjXy4Qbzn2fwDw", "string": "cQspfSzsgLeiJGB2u8vrAiWpCU4MxUT6JseWo2SjXy4Qbzn2fwDw",
"decode": { "payload": "ef6251e205e8ad508bab5596bee086ef16cd4b239e0cc0c5d7c4e6035441e7d5de01"
"version": 239,
"payload": "6251e205e8ad508bab5596bee086ef16cd4b239e0cc0c5d7c4e6035441e7d5de01",
"checksum": "070c22e6"
}
}, },
{ {
"string": "19dcawoKcZdQz365WpXWMhX6QCUpR9SY4r", "string": "19dcawoKcZdQz365WpXWMhX6QCUpR9SY4r",
"decode": { "payload": "005eadaf9bb7121f0f192561a5a62f5e5f54210292"
"version": 0,
"payload": "5eadaf9bb7121f0f192561a5a62f5e5f54210292",
"checksum": "ef7a7be3"
}
}, },
{ {
"string": "37Sp6Rv3y4kVd1nQ1JV5pfqXccHNyZm1x3", "string": "37Sp6Rv3y4kVd1nQ1JV5pfqXccHNyZm1x3",
"decode": { "payload": "053f210e7277c899c3a155cc1c90f4106cbddeec6e"
"version": 5,
"payload": "3f210e7277c899c3a155cc1c90f4106cbddeec6e",
"checksum": "0f6e5c98"
}
}, },
{ {
"string": "myoqcgYiehufrsnnkqdqbp69dddVDMopJu", "string": "myoqcgYiehufrsnnkqdqbp69dddVDMopJu",
"decode": { "payload": "6fc8a3c2a09a298592c3e180f02487cd91ba3400b5"
"version": 111,
"payload": "c8a3c2a09a298592c3e180f02487cd91ba3400b5",
"checksum": "30c3769a"
}
}, },
{ {
"string": "2N7FuwuUuoTBrDFdrAZ9KxBmtqMLxce9i1C", "string": "2N7FuwuUuoTBrDFdrAZ9KxBmtqMLxce9i1C",
"decode": { "payload": "c499b31df7c9068d1481b596578ddbb4d3bd90baeb"
"version": 196,
"payload": "99b31df7c9068d1481b596578ddbb4d3bd90baeb",
"checksum": "02eca37f"
}
}, },
{ {
"string": "5KL6zEaMtPRXZKo1bbMq7JDjjo1bJuQcsgL33je3oY8uSJCR5b4", "string": "5KL6zEaMtPRXZKo1bbMq7JDjjo1bJuQcsgL33je3oY8uSJCR5b4",
"decode": { "payload": "80c7666842503db6dc6ea061f092cfb9c388448629a6fe868d068c42a488b478ae"
"version": 128,
"payload": "c7666842503db6dc6ea061f092cfb9c388448629a6fe868d068c42a488b478ae",
"checksum": "88451297"
}
}, },
{ {
"string": "KwV9KAfwbwt51veZWNscRTeZs9CKpojyu1MsPnaKTF5kz69H1UN2", "string": "KwV9KAfwbwt51veZWNscRTeZs9CKpojyu1MsPnaKTF5kz69H1UN2",
"decode": { "payload": "8007f0803fc5399e773555ab1e8939907e9badacc17ca129e67a2f5f2ff84351dd01"
"version": 128,
"payload": "07f0803fc5399e773555ab1e8939907e9badacc17ca129e67a2f5f2ff84351dd01",
"checksum": "004d964f"
}
}, },
{ {
"string": "93N87D6uxSBzwXvpokpzg8FFmfQPmvX4xHoWQe3pLdYpbiwT5YV", "string": "93N87D6uxSBzwXvpokpzg8FFmfQPmvX4xHoWQe3pLdYpbiwT5YV",
"decode": { "payload": "efea577acfb5d1d14d3b7b195c321566f12f87d2b77ea3a53f68df7ebf8604a801"
"version": 239,
"payload": "ea577acfb5d1d14d3b7b195c321566f12f87d2b77ea3a53f68df7ebf8604a801",
"checksum": "babac8c2"
}
}, },
{ {
"string": "cMxXusSihaX58wpJ3tNuuUcZEQGt6DKJ1wEpxys88FFaQCYjku9h", "string": "cMxXusSihaX58wpJ3tNuuUcZEQGt6DKJ1wEpxys88FFaQCYjku9h",
"decode": { "payload": "ef0b3b34f0958d8a268193a9814da92c3e8b58b4a4378a542863e34ac289cd830c01"
"version": 239,
"payload": "0b3b34f0958d8a268193a9814da92c3e8b58b4a4378a542863e34ac289cd830c01",
"checksum": "61073840"
}
}, },
{ {
"string": "13p1ijLwsnrcuyqcTvJXkq2ASdXqcnEBLE", "string": "13p1ijLwsnrcuyqcTvJXkq2ASdXqcnEBLE",
"decode": { "payload": "001ed467017f043e91ed4c44b4e8dd674db211c4e6"
"version": 0,
"payload": "1ed467017f043e91ed4c44b4e8dd674db211c4e6",
"checksum": "6cc37d7b"
}
}, },
{ {
"string": "3ALJH9Y951VCGcVZYAdpA3KchoP9McEj1G", "string": "3ALJH9Y951VCGcVZYAdpA3KchoP9McEj1G",
"decode": { "payload": "055ece0cadddc415b1980f001785947120acdb36fc"
"version": 5,
"payload": "5ece0cadddc415b1980f001785947120acdb36fc",
"checksum": "b43c48af"
}
} }
], ],
"invalid": [ "invalid": [
{ "Z9inZq4e2HGQRZQezDjFMmqgUE8NwMRok",
"description": "bad version byte", "3HK7MezAm6qEZQUMPRf8jX7wDv6zig6Ky8",
"string": "Z9inZq4e2HGQRZQezDjFMmqgUE8NwMRok" "3AW8j12DUk8mgA7kkfZ1BrrzCVFuH1LsXS"
},
{
"description": "bad payload",
"string": "3HK7MezAm6qEZQUMPRf8jX7wDv6zig6Ky8"
},
{
"description": "bad SHA256 checksum",
"string": "3AW8j12DUk8mgA7kkfZ1BrrzCVFuH1LsXS"
}
] ]
} }

9
test/integration/p2sh.js

@ -24,21 +24,22 @@ describe('Bitcoin-js', function() {
return ECKey.fromWIF(wif) return ECKey.fromWIF(wif)
}) })
// how much to withdraw if we run dry
var coldAmount = 2e4 var coldAmount = 2e4
var outputAmount = 1e4 var outputAmount = 1e4
var pubKeys = privKeys.map(function(eck) { return eck.pub }) var pubKeys = privKeys.map(function(eck) { return eck.pub })
var redeemScript = Script.createMultisigScriptPubKey(2, pubKeys) var redeemScript = Script.createMultisigScriptPubKey(2, pubKeys)
var multisigAddress = new Address(redeemScript.getHash(), networks.testnet.scriptHash).toString() var scriptPubKey = Script.createP2SHScriptPubKey(redeemScript.getHash())
// Send some testnet coins to the multisig address to ensure it has some unspents for later var multisigAddress = Address.fromScriptPubKey(scriptPubKey, networks.testnet).toString()
// Attempt to send funds to the source address, providing some unspents for later
helloblock.faucet.withdraw(multisigAddress, coldAmount, function(err) { helloblock.faucet.withdraw(multisigAddress, coldAmount, function(err) {
if (err) return done(err) if (err) return done(err)
}) })
// make a random private key // make a random private key
var targetAddress = ECKey.makeRandom().pub.getAddress(networks.testnet.pubKeyHash).toString() var targetAddress = ECKey.makeRandom().pub.getAddress(networks.testnet).toString()
// get latest unspents from the multisigAddress // get latest unspents from the multisigAddress
helloblock.addresses.getUnspents(multisigAddress, function(err, resp, resource) { helloblock.addresses.getUnspents(multisigAddress, function(err, resp, resource) {

Loading…
Cancel
Save