Browse Source

TransactionBuilder: re-order in order of usage

hk-custom-address
Daniel Cousens 8 years ago
committed by Daniel Cousens
parent
commit
7059b177eb
  1. 132
      src/transaction_builder.js

132
src/transaction_builder.js

@ -84,6 +84,37 @@ function expandInput (scriptSig, redeemScript) {
} }
} }
function fixMultisigOrder (input, transaction, vin) {
if (input.redeemScriptType !== 'multisig' || !input.redeemScript) return
if (input.pubKeys.length === input.signatures.length) return
var unmatched = input.signatures.concat()
var hashType = input.hashType || Transaction.SIGHASH_ALL
var hash = transaction.hashForSignature(vin, input.redeemScript, hashType)
input.signatures = input.pubKeys.map(function (pubKey, y) {
var keyPair = ECPair.fromPublicKeyBuffer(pubKey)
var match
// check for a signature
unmatched.some(function (signature, i) {
// skip if undefined || OP_0
if (!signature) return false
// skip if signature does not match pubKey
if (!keyPair.verify(hash, signature)) return false
// remove matched signature from unmatched
unmatched[i] = undefined
match = signature
return true
})
return match
})
}
function expandOutput (script, scriptType, ourPubKey) { function expandOutput (script, scriptType, ourPubKey) {
typeforce(types.Buffer, script) typeforce(types.Buffer, script)
@ -122,50 +153,6 @@ function expandOutput (script, scriptType, ourPubKey) {
} }
} }
function buildInput (input, allowIncomplete) {
var signatures = input.signatures
var scriptType = input.redeemScriptType || input.prevOutType
var scriptSig
switch (scriptType) {
case 'pubkeyhash':
case 'pubkey':
if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided')
var pkSignature = signatures[0].toScriptSignature(input.hashType)
if (scriptType === 'pubkeyhash') {
scriptSig = bscript.pubKeyHashInput(pkSignature, input.pubKeys[0])
} else {
scriptSig = bscript.pubKeyInput(pkSignature)
}
break
// ref https://github.com/bitcoin/bitcoin/blob/d612837814020ae832499d18e6ee5eb919a87907/src/script/sign.cpp#L232
case 'multisig':
signatures = signatures.map(function (signature) {
return (signature && signature.toScriptSignature(input.hashType)) || ops.OP_0
})
if (!allowIncomplete) {
// remove blank signatures
signatures = signatures.filter(function (x) { return x !== ops.OP_0 })
}
scriptSig = bscript.multisigInput(signatures, allowIncomplete ? undefined : input.redeemScript)
break
default: return
}
// wrap as scriptHash if necessary
if (input.prevOutType === 'scripthash') {
scriptSig = bscript.scriptHashInput(scriptSig, input.redeemScript)
}
return scriptSig
}
function prepareInput (input, kpPubKey, redeemScript, hashType) { function prepareInput (input, kpPubKey, redeemScript, hashType) {
if (redeemScript) { if (redeemScript) {
var redeemScriptHash = bcrypto.hash160(redeemScript) var redeemScriptHash = bcrypto.hash160(redeemScript)
@ -212,35 +199,48 @@ function prepareInput (input, kpPubKey, redeemScript, hashType) {
input.hashType = hashType input.hashType = hashType
} }
function fixMultisigOrder (input, transaction, vin) { function buildInput (input, allowIncomplete) {
if (input.redeemScriptType !== 'multisig' || !input.redeemScript) return var signatures = input.signatures
if (input.pubKeys.length === input.signatures.length) return var scriptType = input.redeemScriptType || input.prevOutType
var scriptSig
var unmatched = input.signatures.concat() switch (scriptType) {
var hashType = input.hashType || Transaction.SIGHASH_ALL case 'pubkeyhash':
var hash = transaction.hashForSignature(vin, input.redeemScript, hashType) case 'pubkey':
if (signatures.length < 1 || !signatures[0]) throw new Error('Not enough signatures provided')
input.signatures = input.pubKeys.map(function (pubKey, y) { var pkSignature = signatures[0].toScriptSignature(input.hashType)
var keyPair = ECPair.fromPublicKeyBuffer(pubKey) if (scriptType === 'pubkeyhash') {
var match scriptSig = bscript.pubKeyHashInput(pkSignature, input.pubKeys[0])
} else {
scriptSig = bscript.pubKeyInput(pkSignature)
}
// check for a signature break
unmatched.some(function (signature, i) {
// skip if undefined || OP_0
if (!signature) return false
// skip if signature does not match pubKey // ref https://github.com/bitcoin/bitcoin/blob/d612837814020ae832499d18e6ee5eb919a87907/src/script/sign.cpp#L232
if (!keyPair.verify(hash, signature)) return false case 'multisig':
signatures = signatures.map(function (signature) {
return (signature && signature.toScriptSignature(input.hashType)) || ops.OP_0
})
// remove matched signature from unmatched if (!allowIncomplete) {
unmatched[i] = undefined // remove blank signatures
match = signature signatures = signatures.filter(function (x) { return x !== ops.OP_0 })
}
return true scriptSig = bscript.multisigInput(signatures, allowIncomplete ? undefined : input.redeemScript)
}) break
return match default: return
}) }
// wrap as scriptHash if necessary
if (input.prevOutType === 'scripthash') {
scriptSig = bscript.scriptHashInput(scriptSig, input.redeemScript)
}
return scriptSig
} }
function TransactionBuilder (network) { function TransactionBuilder (network) {

Loading…
Cancel
Save