|
@ -352,30 +352,119 @@ class Psbt extends bip174_1.Psbt { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
exports.Psbt = Psbt; |
|
|
exports.Psbt = Psbt; |
|
|
function addNonWitnessTxCache(cache, input, inputIndex) { |
|
|
function canFinalize(input, script, scriptType) { |
|
|
cache.__NON_WITNESS_UTXO_BUF_CACHE[inputIndex] = input.nonWitnessUtxo; |
|
|
switch (scriptType) { |
|
|
const tx = transaction_1.Transaction.fromBuffer(input.nonWitnessUtxo); |
|
|
case 'pubkey': |
|
|
cache.__NON_WITNESS_UTXO_TX_CACHE[inputIndex] = tx; |
|
|
case 'pubkeyhash': |
|
|
const self = cache; |
|
|
case 'witnesspubkeyhash': |
|
|
const selfIndex = inputIndex; |
|
|
return hasSigs(1, input.partialSig); |
|
|
delete input.nonWitnessUtxo; |
|
|
case 'multisig': |
|
|
Object.defineProperty(input, 'nonWitnessUtxo', { |
|
|
const p2ms = payments.p2ms({ output: script }); |
|
|
enumerable: true, |
|
|
return hasSigs(p2ms.m, input.partialSig); |
|
|
get() { |
|
|
default: |
|
|
const buf = self.__NON_WITNESS_UTXO_BUF_CACHE[selfIndex]; |
|
|
return false; |
|
|
const txCache = self.__NON_WITNESS_UTXO_TX_CACHE[selfIndex]; |
|
|
|
|
|
if (buf !== undefined) { |
|
|
|
|
|
return buf; |
|
|
|
|
|
} else { |
|
|
|
|
|
const newBuf = txCache.toBuffer(); |
|
|
|
|
|
self.__NON_WITNESS_UTXO_BUF_CACHE[selfIndex] = newBuf; |
|
|
|
|
|
return newBuf; |
|
|
|
|
|
} |
|
|
} |
|
|
}, |
|
|
} |
|
|
set(data) { |
|
|
function hasSigs(neededSigs, partialSig) { |
|
|
self.__NON_WITNESS_UTXO_BUF_CACHE[selfIndex] = data; |
|
|
if (!partialSig) return false; |
|
|
}, |
|
|
if (partialSig.length > neededSigs) throw new Error('Too many signatures'); |
|
|
|
|
|
return partialSig.length === neededSigs; |
|
|
|
|
|
} |
|
|
|
|
|
function isFinalized(input) { |
|
|
|
|
|
return !!input.finalScriptSig || !!input.finalScriptWitness; |
|
|
|
|
|
} |
|
|
|
|
|
function isPaymentFactory(payment) { |
|
|
|
|
|
return script => { |
|
|
|
|
|
try { |
|
|
|
|
|
payment({ output: script }); |
|
|
|
|
|
return true; |
|
|
|
|
|
} catch (err) { |
|
|
|
|
|
return false; |
|
|
|
|
|
} |
|
|
|
|
|
}; |
|
|
|
|
|
} |
|
|
|
|
|
const isP2MS = isPaymentFactory(payments.p2ms); |
|
|
|
|
|
const isP2PK = isPaymentFactory(payments.p2pk); |
|
|
|
|
|
const isP2PKH = isPaymentFactory(payments.p2pkh); |
|
|
|
|
|
const isP2WPKH = isPaymentFactory(payments.p2wpkh); |
|
|
|
|
|
function check32Bit(num) { |
|
|
|
|
|
if ( |
|
|
|
|
|
typeof num !== 'number' || |
|
|
|
|
|
num !== Math.floor(num) || |
|
|
|
|
|
num > 0xffffffff || |
|
|
|
|
|
num < 0 |
|
|
|
|
|
) { |
|
|
|
|
|
throw new Error('Invalid 32 bit integer'); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
function checkFees(psbt, cache, opts) { |
|
|
|
|
|
const feeRate = cache.__FEE_RATE || psbt.getFeeRate(); |
|
|
|
|
|
const vsize = cache.__EXTRACTED_TX.virtualSize(); |
|
|
|
|
|
const satoshis = feeRate * vsize; |
|
|
|
|
|
if (feeRate >= opts.maximumFeeRate) { |
|
|
|
|
|
throw new Error( |
|
|
|
|
|
`Warning: You are paying around ${(satoshis / 1e8).toFixed(8)} in ` + |
|
|
|
|
|
`fees, which is ${feeRate} satoshi per byte for a transaction ` + |
|
|
|
|
|
`with a VSize of ${vsize} bytes (segwit counted as 0.25 byte per ` + |
|
|
|
|
|
`byte). Use setMaximumFeeRate method to raise your threshold, or ` + |
|
|
|
|
|
`pass true to the first arg of extractTransaction.`, |
|
|
|
|
|
); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
function checkInputsForPartialSig(inputs, action) { |
|
|
|
|
|
inputs.forEach(input => { |
|
|
|
|
|
let throws = false; |
|
|
|
|
|
if ((input.partialSig || []).length === 0) return; |
|
|
|
|
|
input.partialSig.forEach(pSig => { |
|
|
|
|
|
const { hashType } = bscript.signature.decode(pSig.signature); |
|
|
|
|
|
const whitelist = []; |
|
|
|
|
|
const isAnyoneCanPay = |
|
|
|
|
|
hashType & transaction_1.Transaction.SIGHASH_ANYONECANPAY; |
|
|
|
|
|
if (isAnyoneCanPay) whitelist.push('addInput'); |
|
|
|
|
|
const hashMod = hashType & 0x1f; |
|
|
|
|
|
switch (hashMod) { |
|
|
|
|
|
case transaction_1.Transaction.SIGHASH_ALL: |
|
|
|
|
|
break; |
|
|
|
|
|
case transaction_1.Transaction.SIGHASH_SINGLE: |
|
|
|
|
|
case transaction_1.Transaction.SIGHASH_NONE: |
|
|
|
|
|
whitelist.push('addOutput'); |
|
|
|
|
|
whitelist.push('setSequence'); |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
|
|
|
if (whitelist.indexOf(action) === -1) { |
|
|
|
|
|
throws = true; |
|
|
|
|
|
} |
|
|
|
|
|
}); |
|
|
|
|
|
if (throws) { |
|
|
|
|
|
throw new Error('Can not modify transaction, signatures exist.'); |
|
|
|
|
|
} |
|
|
|
|
|
}); |
|
|
|
|
|
} |
|
|
|
|
|
function checkScriptForPubkey(pubkey, script, action) { |
|
|
|
|
|
const pubkeyHash = crypto_1.hash160(pubkey); |
|
|
|
|
|
const decompiled = bscript.decompile(script); |
|
|
|
|
|
if (decompiled === null) throw new Error('Unknown script error'); |
|
|
|
|
|
const hasKey = decompiled.some(element => { |
|
|
|
|
|
if (typeof element === 'number') return false; |
|
|
|
|
|
return element.equals(pubkey) || element.equals(pubkeyHash); |
|
|
}); |
|
|
}); |
|
|
|
|
|
if (!hasKey) { |
|
|
|
|
|
throw new Error( |
|
|
|
|
|
`Can not ${action} for this input with the key ${pubkey.toString('hex')}`, |
|
|
|
|
|
); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
function checkTxEmpty(tx) { |
|
|
|
|
|
const isEmpty = tx.ins.every( |
|
|
|
|
|
input => |
|
|
|
|
|
input.script && |
|
|
|
|
|
input.script.length === 0 && |
|
|
|
|
|
input.witness && |
|
|
|
|
|
input.witness.length === 0, |
|
|
|
|
|
); |
|
|
|
|
|
if (!isEmpty) { |
|
|
|
|
|
throw new Error('Format Error: Transaction ScriptSigs are not empty'); |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
function checkTxForDupeIns(tx, cache) { |
|
|
function checkTxForDupeIns(tx, cache) { |
|
|
tx.ins.forEach(input => { |
|
|
tx.ins.forEach(input => { |
|
@ -390,18 +479,23 @@ function checkTxInputCache(cache, input) { |
|
|
if (cache.__TX_IN_CACHE[key]) throw new Error('Duplicate input detected.'); |
|
|
if (cache.__TX_IN_CACHE[key]) throw new Error('Duplicate input detected.'); |
|
|
cache.__TX_IN_CACHE[key] = 1; |
|
|
cache.__TX_IN_CACHE[key] = 1; |
|
|
} |
|
|
} |
|
|
function isFinalized(input) { |
|
|
function scriptCheckerFactory(payment, paymentScriptName) { |
|
|
return !!input.finalScriptSig || !!input.finalScriptWitness; |
|
|
return (inputIndex, scriptPubKey, redeemScript) => { |
|
|
} |
|
|
const redeemScriptOutput = payment({ |
|
|
function getHashAndSighashType(inputs, inputIndex, pubkey, cache) { |
|
|
redeem: { output: redeemScript }, |
|
|
const input = utils_1.checkForInput(inputs, inputIndex); |
|
|
}).output; |
|
|
const { hash, sighashType, script } = getHashForSig(inputIndex, input, cache); |
|
|
if (!scriptPubKey.equals(redeemScriptOutput)) { |
|
|
checkScriptForPubkey(pubkey, script, 'sign'); |
|
|
throw new Error( |
|
|
return { |
|
|
`${paymentScriptName} for input #${inputIndex} doesn't match the scriptPubKey in the prevout`, |
|
|
hash, |
|
|
); |
|
|
sighashType, |
|
|
} |
|
|
}; |
|
|
}; |
|
|
} |
|
|
} |
|
|
|
|
|
const checkRedeemScript = scriptCheckerFactory(payments.p2sh, 'Redeem script'); |
|
|
|
|
|
const checkWitnessScript = scriptCheckerFactory( |
|
|
|
|
|
payments.p2wsh, |
|
|
|
|
|
'Witness script', |
|
|
|
|
|
); |
|
|
function getFinalScripts( |
|
|
function getFinalScripts( |
|
|
script, |
|
|
script, |
|
|
scriptType, |
|
|
scriptType, |
|
@ -437,81 +531,14 @@ function getFinalScripts( |
|
|
finalScriptWitness, |
|
|
finalScriptWitness, |
|
|
}; |
|
|
}; |
|
|
} |
|
|
} |
|
|
function getSortedSigs(script, partialSig) { |
|
|
function getHashAndSighashType(inputs, inputIndex, pubkey, cache) { |
|
|
const p2ms = payments.p2ms({ output: script }); |
|
|
const input = utils_1.checkForInput(inputs, inputIndex); |
|
|
// for each pubkey in order of p2ms script
|
|
|
const { hash, sighashType, script } = getHashForSig(inputIndex, input, cache); |
|
|
return p2ms.pubkeys |
|
|
checkScriptForPubkey(pubkey, script, 'sign'); |
|
|
.map(pk => { |
|
|
return { |
|
|
// filter partialSig array by pubkey being equal
|
|
|
hash, |
|
|
return ( |
|
|
sighashType, |
|
|
partialSig.filter(ps => { |
|
|
}; |
|
|
return ps.pubkey.equals(pk); |
|
|
|
|
|
})[0] || {} |
|
|
|
|
|
).signature; |
|
|
|
|
|
// Any pubkey without a match will return undefined
|
|
|
|
|
|
// this last filter removes all the undefined items in the array.
|
|
|
|
|
|
}) |
|
|
|
|
|
.filter(v => !!v); |
|
|
|
|
|
} |
|
|
|
|
|
function getPayment(script, scriptType, partialSig) { |
|
|
|
|
|
let payment; |
|
|
|
|
|
switch (scriptType) { |
|
|
|
|
|
case 'multisig': |
|
|
|
|
|
const sigs = getSortedSigs(script, partialSig); |
|
|
|
|
|
payment = payments.p2ms({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
signatures: sigs, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
case 'pubkey': |
|
|
|
|
|
payment = payments.p2pk({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
signature: partialSig[0].signature, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
case 'pubkeyhash': |
|
|
|
|
|
payment = payments.p2pkh({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
pubkey: partialSig[0].pubkey, |
|
|
|
|
|
signature: partialSig[0].signature, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
case 'witnesspubkeyhash': |
|
|
|
|
|
payment = payments.p2wpkh({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
pubkey: partialSig[0].pubkey, |
|
|
|
|
|
signature: partialSig[0].signature, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
|
|
|
return payment; |
|
|
|
|
|
} |
|
|
|
|
|
function canFinalize(input, script, scriptType) { |
|
|
|
|
|
switch (scriptType) { |
|
|
|
|
|
case 'pubkey': |
|
|
|
|
|
case 'pubkeyhash': |
|
|
|
|
|
case 'witnesspubkeyhash': |
|
|
|
|
|
return hasSigs(1, input.partialSig); |
|
|
|
|
|
case 'multisig': |
|
|
|
|
|
const p2ms = payments.p2ms({ output: script }); |
|
|
|
|
|
return hasSigs(p2ms.m, input.partialSig); |
|
|
|
|
|
default: |
|
|
|
|
|
return false; |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
function checkScriptForPubkey(pubkey, script, action) { |
|
|
|
|
|
const pubkeyHash = crypto_1.hash160(pubkey); |
|
|
|
|
|
const decompiled = bscript.decompile(script); |
|
|
|
|
|
if (decompiled === null) throw new Error('Unknown script error'); |
|
|
|
|
|
const hasKey = decompiled.some(element => { |
|
|
|
|
|
if (typeof element === 'number') return false; |
|
|
|
|
|
return element.equals(pubkey) || element.equals(pubkeyHash); |
|
|
|
|
|
}); |
|
|
|
|
|
if (!hasKey) { |
|
|
|
|
|
throw new Error( |
|
|
|
|
|
`Can not ${action} for this input with the key ${pubkey.toString('hex')}`, |
|
|
|
|
|
); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
} |
|
|
function getHashForSig(inputIndex, input, cache) { |
|
|
function getHashForSig(inputIndex, input, cache) { |
|
|
const unsignedTx = cache.__TX; |
|
|
const unsignedTx = cache.__TX; |
|
@ -597,43 +624,87 @@ function getHashForSig(inputIndex, input, cache) { |
|
|
hash, |
|
|
hash, |
|
|
}; |
|
|
}; |
|
|
} |
|
|
} |
|
|
function scriptCheckerFactory(payment, paymentScriptName) { |
|
|
function getInputAdder(cache) { |
|
|
return (inputIndex, scriptPubKey, redeemScript) => { |
|
|
const selfCache = cache; |
|
|
const redeemScriptOutput = payment({ |
|
|
return (_inputData, txBuf) => { |
|
|
redeem: { output: redeemScript }, |
|
|
if ( |
|
|
}).output; |
|
|
!txBuf || |
|
|
if (!scriptPubKey.equals(redeemScriptOutput)) { |
|
|
_inputData.hash === undefined || |
|
|
throw new Error( |
|
|
_inputData.index === undefined || |
|
|
`${paymentScriptName} for input #${inputIndex} doesn't match the scriptPubKey in the prevout`, |
|
|
(!Buffer.isBuffer(_inputData.hash) && |
|
|
); |
|
|
typeof _inputData.hash !== 'string') || |
|
|
} |
|
|
typeof _inputData.index !== 'number' |
|
|
}; |
|
|
) { |
|
|
} |
|
|
throw new Error('Error adding input.'); |
|
|
const checkRedeemScript = scriptCheckerFactory(payments.p2sh, 'Redeem script'); |
|
|
|
|
|
const checkWitnessScript = scriptCheckerFactory( |
|
|
|
|
|
payments.p2wsh, |
|
|
|
|
|
'Witness script', |
|
|
|
|
|
); |
|
|
|
|
|
function isPaymentFactory(payment) { |
|
|
|
|
|
return script => { |
|
|
|
|
|
try { |
|
|
|
|
|
payment({ output: script }); |
|
|
|
|
|
return true; |
|
|
|
|
|
} catch (err) { |
|
|
|
|
|
return false; |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
const prevHash = Buffer.isBuffer(_inputData.hash) |
|
|
|
|
|
? _inputData.hash |
|
|
|
|
|
: bufferutils_1.reverseBuffer(Buffer.from(_inputData.hash, 'hex')); |
|
|
|
|
|
// Check if input already exists in cache.
|
|
|
|
|
|
const input = { hash: prevHash, index: _inputData.index }; |
|
|
|
|
|
checkTxInputCache(selfCache, input); |
|
|
|
|
|
selfCache.__TX.ins.push( |
|
|
|
|
|
Object.assign({}, input, { |
|
|
|
|
|
script: Buffer.alloc(0), |
|
|
|
|
|
sequence: |
|
|
|
|
|
_inputData.sequence || transaction_1.Transaction.DEFAULT_SEQUENCE, |
|
|
|
|
|
witness: [], |
|
|
|
|
|
}), |
|
|
|
|
|
); |
|
|
|
|
|
return selfCache.__TX.toBuffer(); |
|
|
}; |
|
|
}; |
|
|
} |
|
|
} |
|
|
const isP2WPKH = isPaymentFactory(payments.p2wpkh); |
|
|
function getOutputAdder(cache) { |
|
|
const isP2PKH = isPaymentFactory(payments.p2pkh); |
|
|
const selfCache = cache; |
|
|
const isP2MS = isPaymentFactory(payments.p2ms); |
|
|
return (_outputData, txBuf) => { |
|
|
const isP2PK = isPaymentFactory(payments.p2pk); |
|
|
if ( |
|
|
function classifyScript(script) { |
|
|
!txBuf || |
|
|
if (isP2WPKH(script)) return 'witnesspubkeyhash'; |
|
|
_outputData.script === undefined || |
|
|
if (isP2PKH(script)) return 'pubkeyhash'; |
|
|
_outputData.value === undefined || |
|
|
if (isP2MS(script)) return 'multisig'; |
|
|
!Buffer.isBuffer(_outputData.script) || |
|
|
if (isP2PK(script)) return 'pubkey'; |
|
|
typeof _outputData.value !== 'number' |
|
|
return 'nonstandard'; |
|
|
) { |
|
|
|
|
|
throw new Error('Error adding output.'); |
|
|
|
|
|
} |
|
|
|
|
|
selfCache.__TX.outs.push({ |
|
|
|
|
|
script: _outputData.script, |
|
|
|
|
|
value: _outputData.value, |
|
|
|
|
|
}); |
|
|
|
|
|
return selfCache.__TX.toBuffer(); |
|
|
|
|
|
}; |
|
|
|
|
|
} |
|
|
|
|
|
function getPayment(script, scriptType, partialSig) { |
|
|
|
|
|
let payment; |
|
|
|
|
|
switch (scriptType) { |
|
|
|
|
|
case 'multisig': |
|
|
|
|
|
const sigs = getSortedSigs(script, partialSig); |
|
|
|
|
|
payment = payments.p2ms({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
signatures: sigs, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
case 'pubkey': |
|
|
|
|
|
payment = payments.p2pk({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
signature: partialSig[0].signature, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
case 'pubkeyhash': |
|
|
|
|
|
payment = payments.p2pkh({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
pubkey: partialSig[0].pubkey, |
|
|
|
|
|
signature: partialSig[0].signature, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
case 'witnesspubkeyhash': |
|
|
|
|
|
payment = payments.p2wpkh({ |
|
|
|
|
|
output: script, |
|
|
|
|
|
pubkey: partialSig[0].pubkey, |
|
|
|
|
|
signature: partialSig[0].signature, |
|
|
|
|
|
}); |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
|
|
|
return payment; |
|
|
} |
|
|
} |
|
|
function getScriptFromInput(inputIndex, input, cache) { |
|
|
function getScriptFromInput(inputIndex, input, cache) { |
|
|
const unsignedTx = cache.__TX; |
|
|
const unsignedTx = cache.__TX; |
|
@ -674,32 +745,21 @@ function getScriptFromInput(inputIndex, input, cache) { |
|
|
} |
|
|
} |
|
|
return res; |
|
|
return res; |
|
|
} |
|
|
} |
|
|
function hasSigs(neededSigs, partialSig) { |
|
|
function getSortedSigs(script, partialSig) { |
|
|
if (!partialSig) return false; |
|
|
const p2ms = payments.p2ms({ output: script }); |
|
|
if (partialSig.length > neededSigs) throw new Error('Too many signatures'); |
|
|
// for each pubkey in order of p2ms script
|
|
|
return partialSig.length === neededSigs; |
|
|
return p2ms.pubkeys |
|
|
} |
|
|
.map(pk => { |
|
|
function witnessStackToScriptWitness(witness) { |
|
|
// filter partialSig array by pubkey being equal
|
|
|
let buffer = Buffer.allocUnsafe(0); |
|
|
return ( |
|
|
function writeSlice(slice) { |
|
|
partialSig.filter(ps => { |
|
|
buffer = Buffer.concat([buffer, Buffer.from(slice)]); |
|
|
return ps.pubkey.equals(pk); |
|
|
} |
|
|
})[0] || {} |
|
|
function writeVarInt(i) { |
|
|
).signature; |
|
|
const currentLen = buffer.length; |
|
|
// Any pubkey without a match will return undefined
|
|
|
const varintLen = varuint.encodingLength(i); |
|
|
// this last filter removes all the undefined items in the array.
|
|
|
buffer = Buffer.concat([buffer, Buffer.allocUnsafe(varintLen)]); |
|
|
}) |
|
|
varuint.encode(i, buffer, currentLen); |
|
|
.filter(v => !!v); |
|
|
} |
|
|
|
|
|
function writeVarSlice(slice) { |
|
|
|
|
|
writeVarInt(slice.length); |
|
|
|
|
|
writeSlice(slice); |
|
|
|
|
|
} |
|
|
|
|
|
function writeVector(vector) { |
|
|
|
|
|
writeVarInt(vector.length); |
|
|
|
|
|
vector.forEach(writeVarSlice); |
|
|
|
|
|
} |
|
|
|
|
|
writeVector(witness); |
|
|
|
|
|
return buffer; |
|
|
|
|
|
} |
|
|
} |
|
|
function scriptWitnessToWitnessStack(buffer) { |
|
|
function scriptWitnessToWitnessStack(buffer) { |
|
|
let offset = 0; |
|
|
let offset = 0; |
|
@ -723,118 +783,52 @@ function scriptWitnessToWitnessStack(buffer) { |
|
|
} |
|
|
} |
|
|
return readVector(); |
|
|
return readVector(); |
|
|
} |
|
|
} |
|
|
function range(n) { |
|
|
function witnessStackToScriptWitness(witness) { |
|
|
return [...Array(n).keys()]; |
|
|
let buffer = Buffer.allocUnsafe(0); |
|
|
} |
|
|
function writeSlice(slice) { |
|
|
function checkTxEmpty(tx) { |
|
|
buffer = Buffer.concat([buffer, Buffer.from(slice)]); |
|
|
const isEmpty = tx.ins.every( |
|
|
|
|
|
input => |
|
|
|
|
|
input.script && |
|
|
|
|
|
input.script.length === 0 && |
|
|
|
|
|
input.witness && |
|
|
|
|
|
input.witness.length === 0, |
|
|
|
|
|
); |
|
|
|
|
|
if (!isEmpty) { |
|
|
|
|
|
throw new Error('Format Error: Transaction ScriptSigs are not empty'); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
function checkInputsForPartialSig(inputs, action) { |
|
|
|
|
|
inputs.forEach(input => { |
|
|
|
|
|
let throws = false; |
|
|
|
|
|
if ((input.partialSig || []).length === 0) return; |
|
|
|
|
|
input.partialSig.forEach(pSig => { |
|
|
|
|
|
const { hashType } = bscript.signature.decode(pSig.signature); |
|
|
|
|
|
const whitelist = []; |
|
|
|
|
|
const isAnyoneCanPay = |
|
|
|
|
|
hashType & transaction_1.Transaction.SIGHASH_ANYONECANPAY; |
|
|
|
|
|
if (isAnyoneCanPay) whitelist.push('addInput'); |
|
|
|
|
|
const hashMod = hashType & 0x1f; |
|
|
|
|
|
switch (hashMod) { |
|
|
|
|
|
case transaction_1.Transaction.SIGHASH_ALL: |
|
|
|
|
|
break; |
|
|
|
|
|
case transaction_1.Transaction.SIGHASH_SINGLE: |
|
|
|
|
|
case transaction_1.Transaction.SIGHASH_NONE: |
|
|
|
|
|
whitelist.push('addOutput'); |
|
|
|
|
|
whitelist.push('setSequence'); |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
|
|
|
if (whitelist.indexOf(action) === -1) { |
|
|
|
|
|
throws = true; |
|
|
|
|
|
} |
|
|
} |
|
|
}); |
|
|
function writeVarInt(i) { |
|
|
if (throws) { |
|
|
const currentLen = buffer.length; |
|
|
throw new Error('Can not modify transaction, signatures exist.'); |
|
|
const varintLen = varuint.encodingLength(i); |
|
|
|
|
|
buffer = Buffer.concat([buffer, Buffer.allocUnsafe(varintLen)]); |
|
|
|
|
|
varuint.encode(i, buffer, currentLen); |
|
|
} |
|
|
} |
|
|
}); |
|
|
function writeVarSlice(slice) { |
|
|
} |
|
|
writeVarInt(slice.length); |
|
|
function nonWitnessUtxoTxFromCache(cache, input, inputIndex) { |
|
|
writeSlice(slice); |
|
|
if (!cache.__NON_WITNESS_UTXO_TX_CACHE[inputIndex]) { |
|
|
|
|
|
addNonWitnessTxCache(cache, input, inputIndex); |
|
|
|
|
|
} |
|
|
} |
|
|
return cache.__NON_WITNESS_UTXO_TX_CACHE[inputIndex]; |
|
|
function writeVector(vector) { |
|
|
} |
|
|
writeVarInt(vector.length); |
|
|
function getInputAdder(cache) { |
|
|
vector.forEach(writeVarSlice); |
|
|
const selfCache = cache; |
|
|
|
|
|
return (_inputData, txBuf) => { |
|
|
|
|
|
if ( |
|
|
|
|
|
!txBuf || |
|
|
|
|
|
_inputData.hash === undefined || |
|
|
|
|
|
_inputData.index === undefined || |
|
|
|
|
|
(!Buffer.isBuffer(_inputData.hash) && |
|
|
|
|
|
typeof _inputData.hash !== 'string') || |
|
|
|
|
|
typeof _inputData.index !== 'number' |
|
|
|
|
|
) { |
|
|
|
|
|
throw new Error('Error adding input.'); |
|
|
|
|
|
} |
|
|
} |
|
|
const prevHash = Buffer.isBuffer(_inputData.hash) |
|
|
writeVector(witness); |
|
|
? _inputData.hash |
|
|
return buffer; |
|
|
: bufferutils_1.reverseBuffer(Buffer.from(_inputData.hash, 'hex')); |
|
|
|
|
|
// Check if input already exists in cache.
|
|
|
|
|
|
const input = { hash: prevHash, index: _inputData.index }; |
|
|
|
|
|
checkTxInputCache(selfCache, input); |
|
|
|
|
|
selfCache.__TX.ins.push( |
|
|
|
|
|
Object.assign({}, input, { |
|
|
|
|
|
script: Buffer.alloc(0), |
|
|
|
|
|
sequence: |
|
|
|
|
|
_inputData.sequence || transaction_1.Transaction.DEFAULT_SEQUENCE, |
|
|
|
|
|
witness: [], |
|
|
|
|
|
}), |
|
|
|
|
|
); |
|
|
|
|
|
return selfCache.__TX.toBuffer(); |
|
|
|
|
|
}; |
|
|
|
|
|
} |
|
|
} |
|
|
function getOutputAdder(cache) { |
|
|
function addNonWitnessTxCache(cache, input, inputIndex) { |
|
|
const selfCache = cache; |
|
|
cache.__NON_WITNESS_UTXO_BUF_CACHE[inputIndex] = input.nonWitnessUtxo; |
|
|
return (_outputData, txBuf) => { |
|
|
const tx = transaction_1.Transaction.fromBuffer(input.nonWitnessUtxo); |
|
|
if ( |
|
|
cache.__NON_WITNESS_UTXO_TX_CACHE[inputIndex] = tx; |
|
|
!txBuf || |
|
|
const self = cache; |
|
|
_outputData.script === undefined || |
|
|
const selfIndex = inputIndex; |
|
|
_outputData.value === undefined || |
|
|
delete input.nonWitnessUtxo; |
|
|
!Buffer.isBuffer(_outputData.script) || |
|
|
Object.defineProperty(input, 'nonWitnessUtxo', { |
|
|
typeof _outputData.value !== 'number' |
|
|
enumerable: true, |
|
|
) { |
|
|
get() { |
|
|
throw new Error('Error adding output.'); |
|
|
const buf = self.__NON_WITNESS_UTXO_BUF_CACHE[selfIndex]; |
|
|
|
|
|
const txCache = self.__NON_WITNESS_UTXO_TX_CACHE[selfIndex]; |
|
|
|
|
|
if (buf !== undefined) { |
|
|
|
|
|
return buf; |
|
|
|
|
|
} else { |
|
|
|
|
|
const newBuf = txCache.toBuffer(); |
|
|
|
|
|
self.__NON_WITNESS_UTXO_BUF_CACHE[selfIndex] = newBuf; |
|
|
|
|
|
return newBuf; |
|
|
} |
|
|
} |
|
|
selfCache.__TX.outs.push({ |
|
|
}, |
|
|
script: _outputData.script, |
|
|
set(data) { |
|
|
value: _outputData.value, |
|
|
self.__NON_WITNESS_UTXO_BUF_CACHE[selfIndex] = data; |
|
|
|
|
|
}, |
|
|
}); |
|
|
}); |
|
|
return selfCache.__TX.toBuffer(); |
|
|
|
|
|
}; |
|
|
|
|
|
} |
|
|
|
|
|
function checkFees(psbt, cache, opts) { |
|
|
|
|
|
const feeRate = cache.__FEE_RATE || psbt.getFeeRate(); |
|
|
|
|
|
const vsize = cache.__EXTRACTED_TX.virtualSize(); |
|
|
|
|
|
const satoshis = feeRate * vsize; |
|
|
|
|
|
if (feeRate >= opts.maximumFeeRate) { |
|
|
|
|
|
throw new Error( |
|
|
|
|
|
`Warning: You are paying around ${(satoshis / 1e8).toFixed(8)} in ` + |
|
|
|
|
|
`fees, which is ${feeRate} satoshi per byte for a transaction ` + |
|
|
|
|
|
`with a VSize of ${vsize} bytes (segwit counted as 0.25 byte per ` + |
|
|
|
|
|
`byte). Use setMaximumFeeRate method to raise your threshold, or ` + |
|
|
|
|
|
`pass true to the first arg of extractTransaction.`, |
|
|
|
|
|
); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
} |
|
|
function inputFinalizeGetAmts(inputs, tx, cache, mustFinalize, getAmounts) { |
|
|
function inputFinalizeGetAmts(inputs, tx, cache, mustFinalize, getAmounts) { |
|
|
let inputAmount = 0; |
|
|
let inputAmount = 0; |
|
@ -857,13 +851,19 @@ function inputFinalizeGetAmts(inputs, tx, cache, mustFinalize, getAmounts) { |
|
|
}); |
|
|
}); |
|
|
return inputAmount; |
|
|
return inputAmount; |
|
|
} |
|
|
} |
|
|
function check32Bit(num) { |
|
|
function nonWitnessUtxoTxFromCache(cache, input, inputIndex) { |
|
|
if ( |
|
|
if (!cache.__NON_WITNESS_UTXO_TX_CACHE[inputIndex]) { |
|
|
typeof num !== 'number' || |
|
|
addNonWitnessTxCache(cache, input, inputIndex); |
|
|
num !== Math.floor(num) || |
|
|
|
|
|
num > 0xffffffff || |
|
|
|
|
|
num < 0 |
|
|
|
|
|
) { |
|
|
|
|
|
throw new Error('Invalid 32 bit integer'); |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
return cache.__NON_WITNESS_UTXO_TX_CACHE[inputIndex]; |
|
|
|
|
|
} |
|
|
|
|
|
function classifyScript(script) { |
|
|
|
|
|
if (isP2WPKH(script)) return 'witnesspubkeyhash'; |
|
|
|
|
|
if (isP2PKH(script)) return 'pubkeyhash'; |
|
|
|
|
|
if (isP2MS(script)) return 'multisig'; |
|
|
|
|
|
if (isP2PK(script)) return 'pubkey'; |
|
|
|
|
|
return 'nonstandard'; |
|
|
|
|
|
} |
|
|
|
|
|
function range(n) { |
|
|
|
|
|
return [...Array(n).keys()]; |
|
|
} |
|
|
} |
|
|