Merge pull request #375 from bitcoinjs/pushdatafix

Scripts, fix invalid push data int parsing
This commit is contained in:
Daniel Cousens 2015-03-05 18:45:23 +11:00
commit 6a86f22e45
11 changed files with 237 additions and 114 deletions

View file

@ -27,16 +27,19 @@ function readPushDataInt (buffer, offset) {
// 8 bit // 8 bit
} else if (opcode === opcodes.OP_PUSHDATA1) { } else if (opcode === opcodes.OP_PUSHDATA1) {
if (offset + 2 > buffer.length) return null
number = buffer.readUInt8(offset + 1) number = buffer.readUInt8(offset + 1)
size = 2 size = 2
// 16 bit // 16 bit
} else if (opcode === opcodes.OP_PUSHDATA2) { } else if (opcode === opcodes.OP_PUSHDATA2) {
if (offset + 3 > buffer.length) return null
number = buffer.readUInt16LE(offset + 1) number = buffer.readUInt16LE(offset + 1)
size = 3 size = 3
// 32 bit // 32 bit
} else { } else {
if (offset + 5 > buffer.length) return null
assert.equal(opcode, opcodes.OP_PUSHDATA4, 'Unexpected opcode') assert.equal(opcode, opcodes.OP_PUSHDATA4, 'Unexpected opcode')
number = buffer.readUInt32LE(offset + 1) number = buffer.readUInt32LE(offset + 1)

View file

@ -38,8 +38,14 @@ Script.fromBuffer = function (buffer) {
// data chunk // data chunk
if ((opcode > opcodes.OP_0) && (opcode <= opcodes.OP_PUSHDATA4)) { if ((opcode > opcodes.OP_0) && (opcode <= opcodes.OP_PUSHDATA4)) {
var d = bufferutils.readPushDataInt(buffer, i) var d = bufferutils.readPushDataInt(buffer, i)
// did reading a pushDataInt fail? return non-chunked script
if (d === null) return new Script(buffer, [])
i += d.size i += d.size
// attempt to read too much data?
if (i + d.number > buffer.length) return new Script(buffer, [])
var data = buffer.slice(i, i + d.number) var data = buffer.slice(i, i + d.number)
i += d.number i += d.number

View file

@ -73,15 +73,12 @@ function isScriptHashInput (script, allowIncomplete) {
if (!Buffer.isBuffer(lastChunk)) return false if (!Buffer.isBuffer(lastChunk)) return false
var scriptSig = Script.fromChunks(script.chunks.slice(0, -1)) var scriptSig = Script.fromChunks(script.chunks.slice(0, -1))
var scriptPubKey var redeemScript = Script.fromBuffer(lastChunk)
try { // is redeemScript a valid script?
scriptPubKey = Script.fromBuffer(lastChunk) if (redeemScript.chunks.length === 0) return false
} catch (e) {
return false
}
return classifyInput(scriptSig, allowIncomplete) === classifyOutput(scriptPubKey) return classifyInput(scriptSig, allowIncomplete) === classifyOutput(redeemScript)
} }
function isScriptHashOutput (script) { function isScriptHashOutput (script) {

View file

@ -32,6 +32,17 @@ describe('bufferutils', function () {
assert.equal(d.size, buffer.length) assert.equal(d.size, buffer.length)
}) })
}) })
fixtures.invalid.readPushDataInt.forEach(function (f) {
if (!f.hexPD) return
it('decodes ' + f.hexPD + ' as null', function () {
var buffer = new Buffer(f.hexPD, 'hex')
var n = bufferutils.readPushDataInt(buffer, 0)
assert.equal(n, null)
})
})
}) })
describe('readUInt64LE', function () { describe('readUInt64LE', function () {
@ -44,7 +55,7 @@ describe('bufferutils', function () {
}) })
}) })
fixtures.invalid.forEach(function (f) { fixtures.invalid.readUInt64LE.forEach(function (f) {
it('throws on ' + f.description, function () { it('throws on ' + f.description, function () {
var buffer = new Buffer(f.hex64, 'hex') var buffer = new Buffer(f.hex64, 'hex')
@ -66,7 +77,7 @@ describe('bufferutils', function () {
}) })
}) })
fixtures.invalid.forEach(function (f) { fixtures.invalid.readUInt64LE.forEach(function (f) {
it('throws on ' + f.description, function () { it('throws on ' + f.description, function () {
var buffer = new Buffer(f.hexVI, 'hex') var buffer = new Buffer(f.hexVI, 'hex')
@ -151,7 +162,7 @@ describe('bufferutils', function () {
}) })
}) })
fixtures.invalid.forEach(function (f) { fixtures.invalid.readUInt64LE.forEach(function (f) {
it('throws on ' + f.description, function () { it('throws on ' + f.description, function () {
var buffer = new Buffer(8) var buffer = new Buffer(8)
buffer.fill(0) buffer.fill(0)
@ -174,7 +185,7 @@ describe('bufferutils', function () {
}) })
}) })
fixtures.invalid.forEach(function (f) { fixtures.invalid.readUInt64LE.forEach(function (f) {
it('throws on ' + f.description, function () { it('throws on ' + f.description, function () {
var buffer = new Buffer(9) var buffer = new Buffer(9)
buffer.fill(0) buffer.fill(0)

View file

@ -82,7 +82,8 @@
"hexVI": "ffffffffffffff1f00" "hexVI": "ffffffffffffff1f00"
} }
], ],
"invalid": [ "invalid": {
"readUInt64LE": [
{ {
"description": "n === 2^53", "description": "n === 2^53",
"exception": "value is larger than maximum value for type", "exception": "value is larger than maximum value for type",
@ -97,5 +98,20 @@
"hexVI": "ff0100000000000020", "hexVI": "ff0100000000000020",
"dec": 9007199254740993 "dec": 9007199254740993
} }
],
"readPushDataInt": [
{
"description": "OP_PUSHDATA1, no size",
"hexPD": "4c"
},
{
"description": "OP_PUSHDATA2, no size",
"hexPD": "4d"
},
{
"description": "OP_PUSHDATA4, no size",
"hexPD": "4e"
}
] ]
} }
}

View file

@ -1,86 +1,96 @@
{ {
"valid": [ "valid": [
{ {
"description": "pay-to-PubKey",
"hex": "21031f1e68f82112b373f0fe980b3a89d212d2b5c01fb51eb25acb8b4c4b4299ce95ac",
"type": "pubkey",
"hash": "26e645ab170255f2a0a82d29e48f35b14ae7c826",
"pubKey": "031f1e68f82112b373f0fe980b3a89d212d2b5c01fb51eb25acb8b4c4b4299ce95",
"asm": "031f1e68f82112b373f0fe980b3a89d212d2b5c01fb51eb25acb8b4c4b4299ce95 OP_CHECKSIG", "asm": "031f1e68f82112b373f0fe980b3a89d212d2b5c01fb51eb25acb8b4c4b4299ce95 OP_CHECKSIG",
"scriptPubKey": true "description": "pay-to-PubKey",
"hash": "26e645ab170255f2a0a82d29e48f35b14ae7c826",
"hex": "21031f1e68f82112b373f0fe980b3a89d212d2b5c01fb51eb25acb8b4c4b4299ce95ac",
"pubKey": "031f1e68f82112b373f0fe980b3a89d212d2b5c01fb51eb25acb8b4c4b4299ce95"
}, },
{ {
"description": "P2SH ScriptPubKey",
"hex": "a914e8c300c87986efa84c37c0519929019ef86eb5b487",
"type": "scripthash",
"hash": "0ba47b56a573bab4b430ad6ed3ec79270e04b066",
"asm": "OP_HASH160 e8c300c87986efa84c37c0519929019ef86eb5b4 OP_EQUAL", "asm": "OP_HASH160 e8c300c87986efa84c37c0519929019ef86eb5b4 OP_EQUAL",
"scriptPubKey": true "description": "P2SH ScriptPubKey",
"hash": "0ba47b56a573bab4b430ad6ed3ec79270e04b066",
"hex": "a914e8c300c87986efa84c37c0519929019ef86eb5b487"
}, },
{ {
"description": "PubKeyHash ScriptPubKey",
"hex": "76a9145a3acbc7bbcc97c5ff16f5909c9d7d3fadb293a888ac",
"type": "pubkeyhash",
"hash": "a5313f33d5c7b81674b35f7f3febc3522ef234db",
"asm": "OP_DUP OP_HASH160 5a3acbc7bbcc97c5ff16f5909c9d7d3fadb293a8 OP_EQUALVERIFY OP_CHECKSIG", "asm": "OP_DUP OP_HASH160 5a3acbc7bbcc97c5ff16f5909c9d7d3fadb293a8 OP_EQUALVERIFY OP_CHECKSIG",
"scriptPubKey": true "description": "PubKeyHash ScriptPubKey",
"hash": "a5313f33d5c7b81674b35f7f3febc3522ef234db",
"hex": "76a9145a3acbc7bbcc97c5ff16f5909c9d7d3fadb293a888ac"
}, },
{ {
"description": "pubKeyHash scriptSig",
"hex": "48304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f30141040cd2d2ce17a1e9b2b3b2cb294d40eecf305a25b7e7bfdafae6bb2639f4ee399b3637706c3d377ec4ab781355add443ae864b134c5e523001c442186ea60f0eb8",
"type": "pubkeyhash",
"hash": "b9bac2a5c5c29bb27c382d41fa3d179c646c78fd",
"asm": "304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301 040cd2d2ce17a1e9b2b3b2cb294d40eecf305a25b7e7bfdafae6bb2639f4ee399b3637706c3d377ec4ab781355add443ae864b134c5e523001c442186ea60f0eb8", "asm": "304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301 040cd2d2ce17a1e9b2b3b2cb294d40eecf305a25b7e7bfdafae6bb2639f4ee399b3637706c3d377ec4ab781355add443ae864b134c5e523001c442186ea60f0eb8",
"scriptPubKey": false "description": "pubKeyHash scriptSig",
"hash": "b9bac2a5c5c29bb27c382d41fa3d179c646c78fd",
"hex": "48304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f30141040cd2d2ce17a1e9b2b3b2cb294d40eecf305a25b7e7bfdafae6bb2639f4ee399b3637706c3d377ec4ab781355add443ae864b134c5e523001c442186ea60f0eb8"
}, },
{ {
"description": "pubKey scriptSig",
"hex": "48304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301",
"type": "pubkey",
"hash": "44d9982c3e79452e02ef5816976a0e20a0ec1cba",
"signature": "304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301",
"asm": "304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301", "asm": "304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301",
"scriptPubKey": false "description": "pubKey scriptSig",
"hash": "44d9982c3e79452e02ef5816976a0e20a0ec1cba",
"hex": "48304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301",
"signature": "304502206becda98cecf7a545d1a640221438ff8912d9b505ede67e0138485111099f696022100ccd616072501310acba10feb97cecc918e21c8e92760cd35144efec7622938f301"
}, },
{ {
"description": "Valid multisig script",
"hex": "5121032487c2a32f7c8d57d2a93906a6457afd00697925b0e6e145d89af6d3bca330162102308673d16987eaa010e540901cc6fe3695e758c19f46ce604e174dac315e685a52ae",
"type": "multisig",
"hash": "f1c98f0b74ecabcf78ae20dfa224bb6666051fbe",
"asm": "OP_TRUE 032487c2a32f7c8d57d2a93906a6457afd00697925b0e6e145d89af6d3bca33016 02308673d16987eaa010e540901cc6fe3695e758c19f46ce604e174dac315e685a OP_2 OP_CHECKMULTISIG", "asm": "OP_TRUE 032487c2a32f7c8d57d2a93906a6457afd00697925b0e6e145d89af6d3bca33016 02308673d16987eaa010e540901cc6fe3695e758c19f46ce604e174dac315e685a OP_2 OP_CHECKMULTISIG",
"scriptPubKey": true "description": "Valid multisig script",
"hash": "f1c98f0b74ecabcf78ae20dfa224bb6666051fbe",
"hex": "5121032487c2a32f7c8d57d2a93906a6457afd00697925b0e6e145d89af6d3bca330162102308673d16987eaa010e540901cc6fe3695e758c19f46ce604e174dac315e685a52ae"
}, },
{ {
"description": "mutisig scriptSig",
"hex": "0047304402202b29881db1b4cc128442d955e906d41c77365ed9a8392b584be12d980b236459022009da4bc60d09280aa26f4f981bfbed94eb7263d92920961e48a7f3f0991895b101483045022100871708a7597c1dbebff2a5527a56a1f2b49d73e35cd825a07285f5f29f5766d8022003bd7ac25334e9a6d6020cc8ba1be67a8c70dca8e7063ea0547d79c45b9bc12601",
"type": "multisig",
"hash": "b1ef3ae2c77b356eff81049aad7dfd2eeb34c6f5",
"asm": "OP_0 304402202b29881db1b4cc128442d955e906d41c77365ed9a8392b584be12d980b236459022009da4bc60d09280aa26f4f981bfbed94eb7263d92920961e48a7f3f0991895b101 3045022100871708a7597c1dbebff2a5527a56a1f2b49d73e35cd825a07285f5f29f5766d8022003bd7ac25334e9a6d6020cc8ba1be67a8c70dca8e7063ea0547d79c45b9bc12601", "asm": "OP_0 304402202b29881db1b4cc128442d955e906d41c77365ed9a8392b584be12d980b236459022009da4bc60d09280aa26f4f981bfbed94eb7263d92920961e48a7f3f0991895b101 3045022100871708a7597c1dbebff2a5527a56a1f2b49d73e35cd825a07285f5f29f5766d8022003bd7ac25334e9a6d6020cc8ba1be67a8c70dca8e7063ea0547d79c45b9bc12601",
"scriptPubKey": false "description": "mutisig scriptSig",
"hash": "b1ef3ae2c77b356eff81049aad7dfd2eeb34c6f5",
"hex": "0047304402202b29881db1b4cc128442d955e906d41c77365ed9a8392b584be12d980b236459022009da4bc60d09280aa26f4f981bfbed94eb7263d92920961e48a7f3f0991895b101483045022100871708a7597c1dbebff2a5527a56a1f2b49d73e35cd825a07285f5f29f5766d8022003bd7ac25334e9a6d6020cc8ba1be67a8c70dca8e7063ea0547d79c45b9bc12601"
}, },
{ {
"description": "OP_RETURN script",
"hex": "6a2606deadbeef03f895a2ad89fb6d696497af486cb7c644a27aa568c7a18dd06113401115185474",
"type": "nulldata",
"hash": "ec88f016655477663455fe6a8e83508c348ea145",
"asm": "OP_RETURN 06deadbeef03f895a2ad89fb6d696497af486cb7c644a27aa568c7a18dd06113401115185474", "asm": "OP_RETURN 06deadbeef03f895a2ad89fb6d696497af486cb7c644a27aa568c7a18dd06113401115185474",
"scriptPubKey": true "description": "OP_RETURN script",
"hash": "ec88f016655477663455fe6a8e83508c348ea145",
"hex": "6a2606deadbeef03f895a2ad89fb6d696497af486cb7c644a27aa568c7a18dd06113401115185474"
}, },
{ {
"description": "Non standard script",
"hex": "aa206fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d619000000000087",
"type": "nonstandard",
"hash": "3823382e70d1930989813d3459988e0d7c2861d8",
"asm": "OP_HASH256 6fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000 OP_EQUAL", "asm": "OP_HASH256 6fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000 OP_EQUAL",
"scriptPubKey": true "description": "Non standard script",
"hash": "3823382e70d1930989813d3459988e0d7c2861d8",
"hex": "aa206fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d619000000000087"
}, },
{ {
"description": "Invalid multisig script",
"hex": "000000ae",
"type": "nonstandard",
"hash": "62ede8963f9387544935f168745262f703dab1fb",
"asm": "OP_0 OP_0 OP_0 OP_CHECKMULTISIG", "asm": "OP_0 OP_0 OP_0 OP_CHECKMULTISIG",
"scriptPubKey": true "description": "Invalid multisig script",
"hash": "62ede8963f9387544935f168745262f703dab1fb",
"hex": "000000ae"
},
{
"asm": "",
"description": "Not enough data: OP_1",
"hash": "c51b66bced5e4491001bd702669770dccf440982",
"hex": "01"
},
{
"asm": "",
"description": "Not enough data: OP_2",
"hash": "d48ce86c698f246829921ba9fb2a844ae2adba67",
"hex": "0201"
},
{
"asm": "",
"description": "Not enough data: OP_PUSHDATA1 0x02",
"hash": "b663ef01a96ff65bec84a3fb14688d6ff7fc617c",
"hex": "4c0201"
},
{
"asm": "",
"description": "Not enough data: OP_PUSHDATA2 0xffff",
"hash": "b4d2fac2836232e59d7b1628f64f24bce3cb4478",
"hex": "4dffff01"
},
{
"asm": "",
"description": "Not enough data: OP_PUSHDATA4 0xffffffff",
"hash": "941db1ca32faf29e1338fb966bb56d98fbce4823",
"hex": "4effffffff01"
} }
] ]
} }

View file

@ -98,6 +98,12 @@
"redeemScript": "OP_2 0479be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b8 04c6047f9441ed7d6d3045406e95c07cd85c778e4b8cef3ca7abac09b95c709ee51ae168fea63dc339a3c58419466ceaeef7f632653266d0e1236431a950cfe52a OP_2 OP_CHECKMULTISIG", "redeemScript": "OP_2 0479be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b8 04c6047f9441ed7d6d3045406e95c07cd85c778e4b8cef3ca7abac09b95c709ee51ae168fea63dc339a3c58419466ceaeef7f632653266d0e1236431a950cfe52a OP_2 OP_CHECKMULTISIG",
"redeemScriptSig": "OP_0 OP_0 30450221009c92c1ae1767ac04e424da7f6db045d979b08cde86b1ddba48621d59a109d818022004f5bb21ad72255177270abaeb2d7940ac18f1e5ca1f53db4f3fd1045647a8a801", "redeemScriptSig": "OP_0 OP_0 30450221009c92c1ae1767ac04e424da7f6db045d979b08cde86b1ddba48621d59a109d818022004f5bb21ad72255177270abaeb2d7940ac18f1e5ca1f53db4f3fd1045647a8a801",
"scriptSig": "OP_0 OP_0 30450221009c92c1ae1767ac04e424da7f6db045d979b08cde86b1ddba48621d59a109d818022004f5bb21ad72255177270abaeb2d7940ac18f1e5ca1f53db4f3fd1045647a8a801 52410479be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b84104c6047f9441ed7d6d3045406e95c07cd85c778e4b8cef3ca7abac09b95c709ee51ae168fea63dc339a3c58419466ceaeef7f632653266d0e1236431a950cfe52a52ae" "scriptSig": "OP_0 OP_0 30450221009c92c1ae1767ac04e424da7f6db045d979b08cde86b1ddba48621d59a109d818022004f5bb21ad72255177270abaeb2d7940ac18f1e5ca1f53db4f3fd1045647a8a801 52410479be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b84104c6047f9441ed7d6d3045406e95c07cd85c778e4b8cef3ca7abac09b95c709ee51ae168fea63dc339a3c58419466ceaeef7f632653266d0e1236431a950cfe52a52ae"
},
{
"type": "scripthash",
"redeemScript": "OP_0",
"redeemScriptSig": "OP_0",
"scriptSigHex": "000100"
} }
], ],
"invalid": { "invalid": {
@ -113,7 +119,7 @@
"scriptSig": "OP_0 304402207515cf147d201f411092e6be5a64a6006f9308fad7b2a8fdaab22cd86ce764c202200974b8aca7bf51dbf54150d3884e1ae04f675637b926ec33bf75939446f6ca2801 3045022100ef253c1faa39e65115872519e5f0a33bbecf430c0f35cf562beabbad4da24d8d02201742be8ee49812a73adea3007c9641ce6725c32cd44ddb8e3a3af460015d140501 OP_RESERVED" "scriptSig": "OP_0 304402207515cf147d201f411092e6be5a64a6006f9308fad7b2a8fdaab22cd86ce764c202200974b8aca7bf51dbf54150d3884e1ae04f675637b926ec33bf75939446f6ca2801 3045022100ef253c1faa39e65115872519e5f0a33bbecf430c0f35cf562beabbad4da24d8d02201742be8ee49812a73adea3007c9641ce6725c32cd44ddb8e3a3af460015d140501 OP_RESERVED"
}, },
{ {
"description": "signature forms invalid script", "description": "redeemScript is a signature, therefore not a valid script",
"scriptSig": "OP_0 3045022100e12b17b3a4c80c401a1687487bd2bafee9e5f1f8f1ffc6180ce186672ad7b43a02205e316d1e5e71822f5ef301b694e578fa9c94af4f5f098c952c833f4691307f4e01" "scriptSig": "OP_0 3045022100e12b17b3a4c80c401a1687487bd2bafee9e5f1f8f1ffc6180ce186672ad7b43a02205e316d1e5e71822f5ef301b694e578fa9c94af4f5f098c952c833f4691307f4e01"
} }
], ],

View file

@ -173,6 +173,57 @@
], ],
"locktime": 0 "locktime": 0
} }
},
{
"description": "Transaction that ignores script chunking rules - Bug #367",
"id": "ebc9fa1196a59e192352d76c0f6e73167046b9d37b8302b6bb6968dfd279b767",
"hash": "67b779d2df6869bbb602837bd3b9467016736e0f6cd75223199ea59611fac9eb",
"hex": "01000000019ac03d5ae6a875d970128ef9086cef276a1919684a6988023cc7254691d97e6d010000006b4830450221009d41dc793ba24e65f571473d40b299b6459087cea1509f0d381740b1ac863cb6022039c425906fcaf51b2b84d8092569fb3213de43abaff2180e2a799d4fcb4dd0aa012102d5ede09a8ae667d0f855ef90325e27f6ce35bbe60a1e6e87af7f5b3c652140fdffffffff080100000000000000010101000000000000000202010100000000000000014c0100000000000000034c02010100000000000000014d0100000000000000044dffff010100000000000000014e0100000000000000064effffffff0100000000",
"raw": {
"version": 1,
"locktime": 0,
"ins": [
{
"hash": "9ac03d5ae6a875d970128ef9086cef276a1919684a6988023cc7254691d97e6d",
"index": 1,
"script": "30450221009d41dc793ba24e65f571473d40b299b6459087cea1509f0d381740b1ac863cb6022039c425906fcaf51b2b84d8092569fb3213de43abaff2180e2a799d4fcb4dd0aa01 02d5ede09a8ae667d0f855ef90325e27f6ce35bbe60a1e6e87af7f5b3c652140fd"
}
],
"outs": [
{
"data": "01",
"value": 1
},
{
"data": "0201",
"value": 1
},
{
"data": "4c",
"value": 1
},
{
"data": "4c0201",
"value": 1
},
{
"data": "4d",
"value": 1
},
{
"data": "4dffff01",
"value": 1
},
{
"data": "4e",
"value": 1
},
{
"data": "4effffffff01",
"value": 1
}
]
}
} }
], ],
"invalid": { "invalid": {

View file

@ -28,8 +28,13 @@ describe('Script', function () {
describe('fromASM/toASM', function () { describe('fromASM/toASM', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
if (!f.asm) return
it('decodes/encodes ' + f.description, function () { it('decodes/encodes ' + f.description, function () {
assert.equal(Script.fromASM(f.asm).toASM(), f.asm) var script = Script.fromASM(f.asm)
assert.equal(script.toASM(), f.asm)
assert.equal(script.toHex(), f.hex)
}) })
}) })
}) })
@ -37,14 +42,17 @@ describe('Script', function () {
describe('fromHex/toHex', function () { describe('fromHex/toHex', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
it('decodes/encodes ' + f.description, function () { it('decodes/encodes ' + f.description, function () {
assert.equal(Script.fromHex(f.hex).toHex(), f.hex) var script = Script.fromHex(f.hex)
assert.equal(script.toASM(), f.asm)
assert.equal(script.toHex(), f.hex)
}) })
}) })
}) })
describe('getHash', function () { describe('getHash', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
it('produces a HASH160 of "' + f.asm + '"', function () { it('produces a HASH160 of ' + f.description, function () {
var script = Script.fromHex(f.hex) var script = Script.fromHex(f.hex)
assert.equal(script.getHash().toString('hex'), f.hash) assert.equal(script.getHash().toString('hex'), f.hash)

View file

@ -64,9 +64,15 @@ describe('Scripts', function () {
var expected = type.toLowerCase() === f.type var expected = type.toLowerCase() === f.type
if (inputFn && f.scriptSig) { if (inputFn && f.scriptSig) {
it('returns ' + expected + ' for ' + f.scriptSig, function () { var script
var script = Script.fromASM(f.scriptSig)
if (f.scriptSig) {
script = Script.fromASM(f.scriptSig)
} else {
script = Script.fromHex(f.scriptSigHex)
}
it('returns ' + expected + ' for ' + f.scriptSig, function () {
assert.equal(inputFn(script), expected) assert.equal(inputFn(script), expected)
}) })
@ -74,8 +80,6 @@ describe('Scripts', function () {
var expectedIncomplete = type.toLowerCase() === f.typeIncomplete var expectedIncomplete = type.toLowerCase() === f.typeIncomplete
it('returns ' + expected + ' for ' + f.scriptSig, function () { it('returns ' + expected + ' for ' + f.scriptSig, function () {
var script = Script.fromASM(f.scriptSig)
assert.equal(inputFn(script, true), expectedIncomplete) assert.equal(inputFn(script, true), expectedIncomplete)
}) })
} }
@ -85,9 +89,15 @@ describe('Scripts', function () {
if (!(inputFnName in fixtures.invalid)) return if (!(inputFnName in fixtures.invalid)) return
fixtures.invalid[inputFnName].forEach(function (f) { fixtures.invalid[inputFnName].forEach(function (f) {
if (inputFn && f.scriptSig) { if (inputFn && (f.scriptSig || f.scriptSigHex)) {
it('returns false for ' + f.scriptSig, function () { it('returns false for ' + f.description + ' (' + (f.scriptSig || f.scriptSigHex) + ')', function () {
var script = Script.fromASM(f.scriptSig) var script
if (f.scriptSig) {
script = Script.fromASM(f.scriptSig)
} else {
script = Script.fromHex(f.scriptSigHex)
}
assert.equal(inputFn(script), false) assert.equal(inputFn(script), false)
}) })
@ -112,7 +122,7 @@ describe('Scripts', function () {
fixtures.invalid[outputFnName].forEach(function (f) { fixtures.invalid[outputFnName].forEach(function (f) {
if (outputFn && f.scriptPubKey) { if (outputFn && f.scriptPubKey) {
it('returns false for ' + f.scriptPubKey, function () { it('returns false for ' + f.description + ' (' + f.scriptPubKey + ')', function () {
var script = Script.fromASM(f.scriptPubKey) var script = Script.fromASM(f.scriptPubKey)
assert.equal(outputFn(script), false) assert.equal(outputFn(script), false)
@ -240,7 +250,11 @@ describe('Scripts', function () {
it('returns ' + f.scriptSig, function () { it('returns ' + f.scriptSig, function () {
var scriptSig = scripts.scriptHashInput(redeemScriptSig, redeemScript) var scriptSig = scripts.scriptHashInput(redeemScriptSig, redeemScript)
if (f.scriptSig) {
assert.equal(scriptSig.toASM(), f.scriptSig) assert.equal(scriptSig.toASM(), f.scriptSig)
} else {
assert.equal(scriptSig.toHex(), f.scriptSigHex)
}
}) })
}) })
}) })
@ -248,10 +262,10 @@ describe('Scripts', function () {
describe('scriptHashOutput', function () { describe('scriptHashOutput', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
if (f.type !== 'scripthash') return if (f.type !== 'scripthash') return
if (!f.scriptPubKey) return
var redeemScript = Script.fromASM(f.redeemScript)
it('returns ' + f.scriptPubKey, function () { it('returns ' + f.scriptPubKey, function () {
var redeemScript = Script.fromASM(f.redeemScript)
var scriptPubKey = scripts.scriptHashOutput(redeemScript.getHash()) var scriptPubKey = scripts.scriptHashOutput(redeemScript.getHash())
assert.equal(scriptPubKey.toASM(), f.scriptPubKey) assert.equal(scriptPubKey.toASM(), f.scriptPubKey)

View file

@ -18,7 +18,8 @@ describe('Transaction', function () {
var script var script
if (txIn.data) { if (txIn.data) {
script = new Script(new Buffer(txIn.data, 'hex'), []) var data = new Buffer(txIn.data, 'hex')
script = new Script(data, [])
} else if (txIn.script) { } else if (txIn.script) {
script = Script.fromASM(txIn.script) script = Script.fromASM(txIn.script)
} }
@ -27,7 +28,16 @@ describe('Transaction', function () {
}) })
raw.outs.forEach(function (txOut) { raw.outs.forEach(function (txOut) {
tx.addOutput(Script.fromASM(txOut.script), txOut.value) var script
if (txOut.data) {
var data = new Buffer(txOut.data, 'hex')
script = new Script(data, [])
} else if (txOut.script) {
script = Script.fromASM(txOut.script)
}
tx.addOutput(script, txOut.value)
}) })
return tx return tx
@ -35,10 +45,10 @@ describe('Transaction', function () {
describe('fromBuffer/fromHex', function () { describe('fromBuffer/fromHex', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
it('imports ' + f.id + ' correctly', function () { it('imports ' + f.description + ' (' + f.id + ')', function () {
var actual = Transaction.fromHex(f.hex) var actual = Transaction.fromHex(f.hex)
assert.deepEqual(actual.toHex(), f.hex) assert.equal(actual.toHex(), f.hex, actual.toHex())
}) })
}) })
@ -53,10 +63,10 @@ describe('Transaction', function () {
describe('toBuffer/toHex', function () { describe('toBuffer/toHex', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
it('exports ' + f.id + ' correctly', function () { it('exports ' + f.description + ' (' + f.id + ')', function () {
var actual = fromRaw(f.raw) var actual = fromRaw(f.raw)
assert.deepEqual(actual.toHex(), f.hex) assert.equal(actual.toHex(), f.hex, actual.toHex())
}) })
}) })
}) })
@ -108,19 +118,10 @@ describe('Transaction', function () {
}) })
describe('addOutput', function () { describe('addOutput', function () {
fixtures.valid.forEach(function (f) { it('returns an index', function () {
it('should add the outputs for ' + f.id + ' correctly', function () {
var tx = new Transaction() var tx = new Transaction()
assert.equal(tx.addOutput(Script.EMPTY, 0), 0)
f.raw.outs.forEach(function (txOut, i) { assert.equal(tx.addOutput(Script.EMPTY, 0), 1)
var scriptPubKey = Script.fromASM(txOut.script)
var j = tx.addOutput(scriptPubKey, txOut.value)
assert.equal(i, j)
assert.equal(tx.outs[i].script, scriptPubKey)
assert.equal(tx.outs[i].value, txOut.value)
})
})
}) })
}) })
@ -158,7 +159,7 @@ describe('Transaction', function () {
it('should return the hash for ' + f.id, function () { it('should return the hash for ' + f.id, function () {
var tx = Transaction.fromHex(f.hex) var tx = Transaction.fromHex(f.hex)
assert.deepEqual(tx.getHash().toString('hex'), f.hash) assert.equal(tx.getHash().toString('hex'), f.hash)
}) })
}) })
}) })