Browse Source

buildStack, and make tests pass

hk-custom-address
Thomas Kerin 8 years ago
committed by Daniel Cousens
parent
commit
b6ae10b2f1
  1. 108
      src/transaction_builder.js
  2. 5
      test/templates.js

108
src/transaction_builder.js

@ -6,6 +6,9 @@ var ops = require('bitcoin-ops')
var typeforce = require('typeforce') var typeforce = require('typeforce')
var types = require('./types') var types = require('./types')
var scriptTypes = bscript.types var scriptTypes = bscript.types
var SIGNABLE = [bscript.types.P2PKH, bscript.types.P2PK, bscript.types.MULTISIG]
var P2SH = SIGNABLE.concat([bscript.types.P2WPKH, bscript.types.P2WSH])
var EMPTY_SCRIPT = new Buffer(0)
var ECPair = require('./ecpair') var ECPair = require('./ecpair')
var ECSignature = require('./ecsignature') var ECSignature = require('./ecsignature')
@ -223,31 +226,14 @@ function prepareInput (input, kpPubKey, redeemScript) {
} }
} }
var EMPTY_SCRIPT = new Buffer(0) function buildStack (type, signatures, pubKeys, allowIncomplete) {
if (type === scriptTypes.P2PKH) {
function buildInput (input, allowIncomplete) {
var signatures = input.signatures
var scriptType = input.redeemScriptType || input.prevOutType
var stack
switch (scriptType) {
case scriptTypes.P2WPKH:
if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided') if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided')
stack = bscript.witnessPubKeyHash.input.encodeStack(signatures[0], input.pubKeys[0]) return bscript.pubKeyHash.input.encodeStack(signatures[0], pubKeys[0])
break } else if (type === scriptTypes.P2PK) {
case scriptTypes.P2PKH:
if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided')
stack = bscript.pubKeyHash.input.encodeStack(signatures[0], input.pubKeys[0])
break
case scriptTypes.P2PK:
if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided') if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided')
stack = bscript.pubKey.input.encodeStack(signatures[0]) return bscript.pubKey.input.encodeStack(signatures[0])
break } else {
// ref https://github.com/bitcoin/bitcoin/blob/d612837814020ae832499d18e6ee5eb919a87907/src/script/sign.cpp#L232
case scriptTypes.MULTISIG:
signatures = signatures.map(function (signature) { signatures = signatures.map(function (signature) {
return signature || ops.OP_0 return signature || ops.OP_0
}) })
@ -257,38 +243,58 @@ function buildInput (input, allowIncomplete) {
signatures = signatures.filter(function (x) { return x !== ops.OP_0 }) signatures = signatures.filter(function (x) { return x !== ops.OP_0 })
} }
stack = bscript.multisig.input.encodeStack(signatures, allowIncomplete ? undefined : input.redeemScript) return bscript.multisig.input.encodeStack(signatures /* see if it's necessary first */)
break
default: return
} }
var script, witness
// encode witness for P2WPKH if necessary
if (input.prevOutType === scriptTypes.P2WPKH ||
input.redeemScriptType === scriptTypes.P2WPKH) {
witness = stack
script = EMPTY_SCRIPT
} }
// no witness? plain old script! function buildInput (input, allowIncomplete) {
if (witness === undefined) { var scriptType = input.prevOutType
script = bscript.compilePushOnly(stack) var sig = []
var witness = []
if (SIGNABLE.indexOf(scriptType) !== -1) {
sig = buildStack(scriptType, input.signatures, input.pubKeys, input.script, allowIncomplete)
}
var p2sh = false
if (scriptType === bscript.types.P2SH) {
// We can remove this error later when we have a guarantee prepareInput
// rejects unsignabale scripts - it MUST be signable at this point.
if (P2SH.indexOf(input.redeemScriptType) === -1) {
throw new Error('Impossible to sign this type')
}
p2sh = true
if (SIGNABLE.indexOf(input.redeemScriptType) !== -1) {
sig = buildStack(input.redeemScriptType, input.signatures, input.pubKeys, allowIncomplete)
}
// If it wasn't SIGNABLE, it's witness, defer to that
scriptType = input.redeemScriptType
}
if (scriptType === bscript.types.P2WPKH) {
// P2WPKH is a special case of P2PKH
witness = buildStack(bscript.types.P2PKH, input.signatures, input.pubKeys, allowIncomplete)
} else if (scriptType === bscript.types.P2WSH) {
// We can remove this check later
if (SIGNABLE.indexOf(input.witnessScriptType) !== -1) {
witness = buildStack(input.witnessScriptType, input.signatures, input.pubKeys, allowIncomplete)
witness.push(input.witnessScript)
} else {
// We can remove this error later when we have a guarantee prepareInput
// rejects unsignble scripts - it MUST be signable at this point.
throw new Error()
} }
// wrap as scriptHash if necessary scriptType = input.witnessScriptType
if (input.prevOutType === scriptTypes.P2SH) {
script = bscript.scriptHash.input.encode(script, input.redeemScript)
} }
// falsy is easier // append redeemScript if necessary
if (script === EMPTY_SCRIPT) { if (p2sh) {
script = undefined sig.push(input.redeemScript)
} }
return { return {
script: script, type: scriptType,
script: bscript.compile(sig),
witness: witness witness: witness
} }
} }
@ -461,15 +467,17 @@ TransactionBuilder.prototype.__build = function (allowIncomplete) {
this.inputs.forEach(function (input, i) { this.inputs.forEach(function (input, i) {
var scriptType = input.redeemScriptType || input.prevOutType var scriptType = input.redeemScriptType || input.prevOutType
if (!scriptType && !allowIncomplete) throw new Error('Transaction is not complete') if (!scriptType && !allowIncomplete) throw new Error('Transaction is not complete')
var result = buildInput(input, allowIncomplete) var result = buildInput(input, allowIncomplete)
// skip if no result // skip if no result
if (!result && allowIncomplete) return if (!allowIncomplete) {
if (result && result.script) return tx.setInputScript(i, result.script) if (SIGNABLE.indexOf(result.type) === -1 && result.type !== bscript.types.P2WPKH) {
if (result && result.witness) return tx.setWitness(i, result.witness) throw new Error(result.type + ' not supported')
}
}
throw new Error(scriptType + ' not supported') tx.setInputScript(i, result.script)
tx.setWitness(i, result.witness)
}) })
if (!allowIncomplete) { if (!allowIncomplete) {

5
test/templates.js

@ -61,7 +61,10 @@ describe('script-templates', function () {
describe(name + '.input.check', function () { describe(name + '.input.check', function () {
fixtures.valid.forEach(function (f) { fixtures.valid.forEach(function (f) {
var expected = name.toLowerCase() === f.type // Temporary - while we don't have witnessKeyHash.input, etc.
if (name.toLowerCase() === bscript.types.P2WPKH) return
if (name.toLowerCase() === bscript.types.P2WSH) return
var expected = name.toLowerCase() === f.type.toLowerCase()
if (inputType && f.input) { if (inputType && f.input) {
var input = bscript.fromASM(f.input) var input = bscript.fromASM(f.input)

Loading…
Cancel
Save