bitcoinjs-lib/ts_src/transaction_builder.ts

968 lines
27 KiB
TypeScript
Raw Normal View History

2019-03-07 14:03:04 +09:00
import * as baddress from './address';
2019-03-03 23:07:49 +09:00
import { reverseBuffer } from './bufferutils';
2019-03-07 14:03:04 +09:00
import * as classify from './classify';
import * as bcrypto from './crypto';
2019-03-03 23:07:49 +09:00
import { ECPairInterface } from './ecpair';
import * as ECPair from './ecpair';
2019-03-07 14:03:04 +09:00
import { Network } from './networks';
import * as networks from './networks';
2019-03-03 23:07:49 +09:00
import { Payment } from './payments';
import * as payments from './payments';
2019-03-07 14:03:04 +09:00
import * as bscript from './script';
2019-03-03 23:07:49 +09:00
import { OPS as ops } from './script';
2019-03-07 14:03:04 +09:00
import { Output, Transaction } from './transaction';
import * as types from './types';
2019-03-03 23:07:49 +09:00
const typeforce = require('typeforce');
const SCRIPT_TYPES = classify.types;
2019-03-07 14:03:04 +09:00
type MaybeBuffer = Buffer | undefined;
type TxbSignatures = Buffer[] | MaybeBuffer[];
type TxbPubkeys = MaybeBuffer[];
type TxbWitness = Buffer[];
2019-03-03 23:07:49 +09:00
type TxbScriptType = string;
type TxbScript = Buffer;
2014-06-16 16:05:31 +10:00
2018-12-28 11:56:03 +09:00
interface TxbInput {
2019-03-03 23:07:49 +09:00
value?: number;
hasWitness?: boolean;
signScript?: TxbScript;
signType?: TxbScriptType;
prevOutScript?: TxbScript;
redeemScript?: TxbScript;
redeemScriptType?: TxbScriptType;
prevOutType?: TxbScriptType;
pubkeys?: TxbPubkeys;
signatures?: TxbSignatures;
witness?: TxbWitness;
witnessScript?: TxbScript;
witnessScriptType?: TxbScriptType;
script?: TxbScript;
sequence?: number;
scriptSig?: TxbScript;
maxSignatures?: number;
2018-12-28 11:56:03 +09:00
}
interface TxbOutput {
2019-03-03 23:07:49 +09:00
type: string;
pubkeys?: TxbPubkeys;
signatures?: TxbSignatures;
maxSignatures?: number;
2018-12-28 11:56:03 +09:00
}
function txIsString(tx: Buffer | string | Transaction): tx is string {
2019-03-03 23:07:49 +09:00
return typeof tx === 'string' || tx instanceof String;
2018-12-28 11:56:03 +09:00
}
function txIsTransaction(tx: Buffer | string | Transaction): tx is Transaction {
2019-03-03 23:07:49 +09:00
return tx instanceof Transaction;
2018-12-28 11:56:03 +09:00
}
export class TransactionBuilder {
2019-03-03 23:07:49 +09:00
static fromTransaction(
transaction: Transaction,
network?: Network,
): TransactionBuilder {
const txb = new TransactionBuilder(network);
2018-12-28 11:56:03 +09:00
// Copy transaction fields
2019-03-03 23:07:49 +09:00
txb.setVersion(transaction.version);
txb.setLockTime(transaction.locktime);
2018-12-28 11:56:03 +09:00
// Copy outputs (done first to avoid signature invalidation)
transaction.outs.forEach(txOut => {
2019-03-07 14:03:04 +09:00
txb.addOutput(txOut.script, (txOut as Output).value);
2019-03-03 23:07:49 +09:00
});
2018-12-28 11:56:03 +09:00
// Copy inputs
transaction.ins.forEach(txIn => {
txb.__addInputUnsafe(txIn.hash, txIn.index, {
sequence: txIn.sequence,
script: txIn.script,
2019-03-03 23:07:49 +09:00
witness: txIn.witness,
});
});
2018-12-28 11:56:03 +09:00
// fix some things not possible through the public API
2019-03-07 14:03:04 +09:00
txb.__INPUTS.forEach((input, i) => {
2019-03-03 23:07:49 +09:00
fixMultisigOrder(input, transaction, i);
});
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
return txb;
2018-12-28 11:56:03 +09:00
}
2019-03-07 14:03:04 +09:00
private __PREV_TX_SET: { [index: string]: boolean };
private __INPUTS: TxbInput[];
private __TX: Transaction;
2019-03-20 15:25:48 +09:00
// WARNING: maximumFeeRate is __NOT__ to be relied on,
// it's just another potential safety mechanism (safety in-depth)
constructor(
public network: Network = networks.bitcoin,
public maximumFeeRate: number = 2500,
) {
2019-03-07 14:03:04 +09:00
this.__PREV_TX_SET = {};
this.__INPUTS = [];
this.__TX = new Transaction();
this.__TX.version = 2;
}
2019-03-03 23:07:49 +09:00
setLockTime(locktime: number): void {
typeforce(types.UInt32, locktime);
2018-12-28 11:56:03 +09:00
// if any signatures exist, throw
2019-03-03 23:07:49 +09:00
if (
2019-03-07 14:03:04 +09:00
this.__INPUTS.some(input => {
2019-03-03 23:07:49 +09:00
if (!input.signatures) return false;
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
return input.signatures.some(s => s !== undefined);
})
) {
throw new Error('No, this would invalidate signatures');
2018-12-28 11:56:03 +09:00
}
2019-03-07 14:03:04 +09:00
this.__TX.locktime = locktime;
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
setVersion(version: number): void {
typeforce(types.UInt32, version);
2018-12-28 11:56:03 +09:00
// XXX: this might eventually become more complex depending on what the versions represent
2019-03-07 14:03:04 +09:00
this.__TX.version = version;
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
addInput(
txHash: Buffer | string | Transaction,
vout: number,
sequence: number,
prevOutScript: Buffer,
): number {
2018-12-28 11:56:03 +09:00
if (!this.__canModifyInputs()) {
2019-03-03 23:07:49 +09:00
throw new Error('No, this would invalidate signatures');
2018-12-28 11:56:03 +09:00
}
2019-03-07 14:03:04 +09:00
let value: number | undefined;
2018-12-28 11:56:03 +09:00
// is it a hex string?
if (txIsString(txHash)) {
// transaction hashs's are displayed in reverse order, un-reverse it
2019-03-03 23:07:49 +09:00
txHash = reverseBuffer(Buffer.from(txHash, 'hex'));
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
// is it a Transaction object?
2018-12-28 11:56:03 +09:00
} else if (txIsTransaction(txHash)) {
2019-03-03 23:07:49 +09:00
const txOut = txHash.outs[vout];
prevOutScript = txOut.script;
2019-03-07 14:03:04 +09:00
value = (txOut as Output).value;
2018-12-28 11:56:03 +09:00
2019-03-07 14:03:04 +09:00
txHash = txHash.getHash(false) as Buffer;
2018-12-28 11:56:03 +09:00
}
return this.__addInputUnsafe(txHash, vout, {
2019-03-07 14:03:04 +09:00
sequence,
prevOutScript,
value,
2019-03-03 23:07:49 +09:00
});
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
addOutput(scriptPubKey: string | Buffer, value: number): number {
2018-12-28 11:56:03 +09:00
if (!this.__canModifyOutputs()) {
2019-03-03 23:07:49 +09:00
throw new Error('No, this would invalidate signatures');
2018-12-28 11:56:03 +09:00
}
// Attempt to get a script if it's a base58 or bech32 address string
if (typeof scriptPubKey === 'string') {
2019-03-03 23:07:49 +09:00
scriptPubKey = baddress.toOutputScript(scriptPubKey, this.network);
2018-12-28 11:56:03 +09:00
}
2019-03-07 14:03:04 +09:00
return this.__TX.addOutput(scriptPubKey, value);
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
build(): Transaction {
return this.__build(false);
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
buildIncomplete(): Transaction {
return this.__build(true);
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
sign(
vin: number,
keyPair: ECPairInterface,
redeemScript: Buffer,
hashType: number,
witnessValue: number,
witnessScript: Buffer,
): void {
2018-12-28 11:56:03 +09:00
// TODO: remove keyPair.network matching in 4.0.0
2019-03-03 23:07:49 +09:00
if (keyPair.network && keyPair.network !== this.network)
throw new TypeError('Inconsistent network');
2019-03-07 14:03:04 +09:00
if (!this.__INPUTS[vin]) throw new Error('No input at index: ' + vin);
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
hashType = hashType || Transaction.SIGHASH_ALL;
if (this.__needsOutputs(hashType))
throw new Error('Transaction needs outputs');
2018-12-28 11:56:03 +09:00
2019-03-07 14:03:04 +09:00
const input = this.__INPUTS[vin];
2018-12-28 11:56:03 +09:00
// if redeemScript was previously provided, enforce consistency
2019-03-03 23:07:49 +09:00
if (
input.redeemScript !== undefined &&
redeemScript &&
!input.redeemScript.equals(redeemScript)
) {
throw new Error('Inconsistent redeemScript');
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
const ourPubKey = keyPair.publicKey || keyPair.getPublicKey!();
2018-12-28 11:56:03 +09:00
if (!canSign(input)) {
if (witnessValue !== undefined) {
2019-03-03 23:07:49 +09:00
if (input.value !== undefined && input.value !== witnessValue)
2019-03-07 14:03:04 +09:00
throw new Error('Input did not match witnessValue');
2019-03-03 23:07:49 +09:00
typeforce(types.Satoshi, witnessValue);
input.value = witnessValue;
2018-12-28 11:56:03 +09:00
}
if (!canSign(input)) {
2019-03-03 23:07:49 +09:00
const prepared = prepareInput(
input,
ourPubKey,
redeemScript,
witnessScript,
);
2018-12-28 11:56:03 +09:00
// updates inline
2019-03-03 23:07:49 +09:00
Object.assign(input, prepared);
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
if (!canSign(input)) throw Error(input.prevOutType + ' not supported');
2018-12-28 11:56:03 +09:00
}
// ready to sign
2019-03-03 23:07:49 +09:00
let signatureHash: Buffer;
2018-12-28 11:56:03 +09:00
if (input.hasWitness) {
2019-03-07 14:03:04 +09:00
signatureHash = this.__TX.hashForWitnessV0(
2019-03-03 23:07:49 +09:00
vin,
2019-03-07 14:03:04 +09:00
input.signScript as Buffer,
input.value as number,
2019-03-03 23:07:49 +09:00
hashType,
);
2018-12-28 11:56:03 +09:00
} else {
2019-03-07 14:03:04 +09:00
signatureHash = this.__TX.hashForSignature(
2019-03-03 23:07:49 +09:00
vin,
2019-03-07 14:03:04 +09:00
input.signScript as Buffer,
2019-03-03 23:07:49 +09:00
hashType,
);
2018-12-28 11:56:03 +09:00
}
// enforce in order signing of public keys
const signed = input.pubkeys!.some((pubKey, i) => {
2019-03-03 23:07:49 +09:00
if (!ourPubKey.equals(pubKey!)) return false;
if (input.signatures![i]) throw new Error('Signature already exists');
2018-12-28 11:56:03 +09:00
// TODO: add tests
if (ourPubKey.length !== 33 && input.hasWitness) {
2019-03-03 23:07:49 +09:00
throw new Error(
'BIP143 rejects uncompressed public keys in P2WPKH or P2WSH',
);
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
const signature = keyPair.sign(signatureHash);
input.signatures![i] = bscript.signature.encode(signature, hashType);
return true;
});
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
if (!signed) throw new Error('Key pair cannot sign for this input');
2018-12-28 11:56:03 +09:00
}
2019-03-07 14:03:04 +09:00
private __addInputUnsafe(
txHash: Buffer,
vout: number,
options: TxbInput,
): number {
if (Transaction.isCoinbaseHash(txHash)) {
throw new Error('coinbase inputs not supported');
}
const prevTxOut = txHash.toString('hex') + ':' + vout;
if (this.__PREV_TX_SET[prevTxOut] !== undefined)
throw new Error('Duplicate TxOut: ' + prevTxOut);
let input: TxbInput = {};
// derive what we can from the scriptSig
if (options.script !== undefined) {
input = expandInput(options.script, options.witness || []);
}
// if an input value was given, retain it
if (options.value !== undefined) {
input.value = options.value;
}
// derive what we can from the previous transactions output script
if (!input.prevOutScript && options.prevOutScript) {
let prevOutType;
if (!input.pubkeys && !input.signatures) {
const expanded = expandOutput(options.prevOutScript);
if (expanded.pubkeys) {
input.pubkeys = expanded.pubkeys;
input.signatures = expanded.signatures;
}
prevOutType = expanded.type;
}
input.prevOutScript = options.prevOutScript;
input.prevOutType = prevOutType || classify.output(options.prevOutScript);
}
const vin = this.__TX.addInput(
txHash,
vout,
options.sequence,
options.scriptSig,
);
this.__INPUTS[vin] = input;
this.__PREV_TX_SET[prevTxOut] = true;
return vin;
}
private __build(allowIncomplete?: boolean): Transaction {
if (!allowIncomplete) {
if (!this.__TX.ins.length) throw new Error('Transaction has no inputs');
if (!this.__TX.outs.length) throw new Error('Transaction has no outputs');
}
const tx = this.__TX.clone();
// create script signatures from inputs
this.__INPUTS.forEach((input, i) => {
if (!input.prevOutType && !allowIncomplete)
throw new Error('Transaction is not complete');
const result = build(input.prevOutType!, input, allowIncomplete);
if (!result) {
if (!allowIncomplete && input.prevOutType === SCRIPT_TYPES.NONSTANDARD)
throw new Error('Unknown input type');
if (!allowIncomplete) throw new Error('Not enough information');
return;
}
tx.setInputScript(i, result.input!);
tx.setWitness(i, result.witness!);
});
if (!allowIncomplete) {
// do not rely on this, its merely a last resort
if (this.__overMaximumFees(tx.virtualSize())) {
throw new Error('Transaction has absurd fees');
}
}
return tx;
}
2019-03-03 23:07:49 +09:00
private __canModifyInputs(): boolean {
2019-03-07 14:03:04 +09:00
return this.__INPUTS.every(input => {
2019-03-03 23:07:49 +09:00
if (!input.signatures) return true;
2018-12-28 11:56:03 +09:00
return input.signatures.every(signature => {
2019-03-03 23:07:49 +09:00
if (!signature) return true;
const hashType = signatureHashType(signature);
2018-12-28 11:56:03 +09:00
// if SIGHASH_ANYONECANPAY is set, signatures would not
// be invalidated by more inputs
2019-03-03 23:07:49 +09:00
return (hashType & Transaction.SIGHASH_ANYONECANPAY) !== 0;
});
});
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
private __needsOutputs(signingHashType: number): boolean {
2018-12-28 11:56:03 +09:00
if (signingHashType === Transaction.SIGHASH_ALL) {
2019-03-07 14:03:04 +09:00
return this.__TX.outs.length === 0;
2018-12-28 11:56:03 +09:00
}
// if inputs are being signed with SIGHASH_NONE, we don't strictly need outputs
// .build() will fail, but .buildIncomplete() is OK
2019-03-03 23:07:49 +09:00
return (
2019-03-07 14:03:04 +09:00
this.__TX.outs.length === 0 &&
this.__INPUTS.some(input => {
2019-03-03 23:07:49 +09:00
if (!input.signatures) return false;
return input.signatures.some(signature => {
if (!signature) return false; // no signature, no issue
const hashType = signatureHashType(signature);
if (hashType & Transaction.SIGHASH_NONE) return false; // SIGHASH_NONE doesn't care about outputs
return true; // SIGHASH_* does care
});
2018-12-28 11:56:03 +09:00
})
2019-03-03 23:07:49 +09:00
);
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
private __canModifyOutputs(): boolean {
2019-03-07 14:03:04 +09:00
const nInputs = this.__TX.ins.length;
const nOutputs = this.__TX.outs.length;
2018-12-28 11:56:03 +09:00
2019-03-07 14:03:04 +09:00
return this.__INPUTS.every(input => {
2019-03-03 23:07:49 +09:00
if (input.signatures === undefined) return true;
2018-12-28 11:56:03 +09:00
return input.signatures.every(signature => {
2019-03-03 23:07:49 +09:00
if (!signature) return true;
const hashType = signatureHashType(signature);
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
const hashTypeMod = hashType & 0x1f;
if (hashTypeMod === Transaction.SIGHASH_NONE) return true;
2018-12-28 11:56:03 +09:00
if (hashTypeMod === Transaction.SIGHASH_SINGLE) {
// if SIGHASH_SINGLE is set, and nInputs > nOutputs
// some signatures would be invalidated by the addition
// of more outputs
2019-03-03 23:07:49 +09:00
return nInputs <= nOutputs;
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
return false;
});
});
2018-12-28 11:56:03 +09:00
}
2019-03-03 23:07:49 +09:00
private __overMaximumFees(bytes: number): boolean {
2018-12-28 11:56:03 +09:00
// not all inputs will have .value defined
2019-03-07 14:03:04 +09:00
const incoming = this.__INPUTS.reduce((a, x) => a + (x.value! >>> 0), 0);
2018-12-28 11:56:03 +09:00
// but all outputs do, and if we have any input value
// we can immediately determine if the outputs are too small
2019-03-07 14:03:04 +09:00
const outgoing = this.__TX.outs.reduce(
(a, x) => a + (x as Output).value,
0,
);
2019-03-03 23:07:49 +09:00
const fee = incoming - outgoing;
const feeRate = fee / bytes;
2018-12-28 11:56:03 +09:00
2019-03-03 23:07:49 +09:00
return feeRate > this.maximumFeeRate;
2018-12-28 11:56:03 +09:00
}
}
2019-03-03 23:07:49 +09:00
function expandInput(
scriptSig: Buffer,
2019-03-07 14:03:04 +09:00
witnessStack: Buffer[],
2019-03-03 23:07:49 +09:00
type?: string,
scriptPubKey?: Buffer,
): TxbInput {
if (scriptSig.length === 0 && witnessStack.length === 0) return {};
2018-06-27 17:29:18 +10:00
if (!type) {
2019-03-03 23:07:49 +09:00
let ssType: string | undefined = classify.input(scriptSig, true);
let wsType: string | undefined = classify.witness(witnessStack, true);
if (ssType === SCRIPT_TYPES.NONSTANDARD) ssType = undefined;
if (wsType === SCRIPT_TYPES.NONSTANDARD) wsType = undefined;
type = ssType || wsType;
2018-06-27 17:29:18 +10:00
}
2017-01-02 17:45:23 +01:00
switch (type) {
2018-06-27 17:29:18 +10:00
case SCRIPT_TYPES.P2WPKH: {
2019-03-03 23:07:49 +09:00
const { output, pubkey, signature } = payments.p2wpkh({
witness: witnessStack,
});
2018-06-27 17:29:18 +10:00
return {
prevOutScript: output,
prevOutType: SCRIPT_TYPES.P2WPKH,
pubkeys: [pubkey],
2019-03-03 23:07:49 +09:00
signatures: [signature],
};
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
case SCRIPT_TYPES.P2PKH: {
2019-03-03 23:07:49 +09:00
const { output, pubkey, signature } = payments.p2pkh({
input: scriptSig,
});
2018-06-27 17:29:18 +10:00
return {
prevOutScript: output,
prevOutType: SCRIPT_TYPES.P2PKH,
pubkeys: [pubkey],
2019-03-03 23:07:49 +09:00
signatures: [signature],
};
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
case SCRIPT_TYPES.P2PK: {
2019-03-03 23:07:49 +09:00
const { signature } = payments.p2pk({ input: scriptSig });
2018-06-27 17:29:18 +10:00
return {
prevOutType: SCRIPT_TYPES.P2PK,
pubkeys: [undefined],
2019-03-03 23:07:49 +09:00
signatures: [signature],
};
2017-01-02 17:45:23 +01:00
}
2018-09-03 14:06:30 +10:00
case SCRIPT_TYPES.P2MS: {
2019-03-03 23:07:49 +09:00
const { m, pubkeys, signatures } = payments.p2ms(
{
input: scriptSig,
output: scriptPubKey,
},
{ allowIncomplete: true },
);
2018-06-27 17:29:18 +10:00
return {
2018-09-03 14:06:30 +10:00
prevOutType: SCRIPT_TYPES.P2MS,
2019-03-07 14:03:04 +09:00
pubkeys,
signatures,
2019-03-03 23:07:49 +09:00
maxSignatures: m,
};
2017-01-02 17:45:23 +01:00
}
2018-06-27 17:29:18 +10:00
}
2017-01-02 17:45:23 +01:00
2018-06-27 17:29:18 +10:00
if (type === SCRIPT_TYPES.P2SH) {
const { output, redeem } = payments.p2sh({
input: scriptSig,
2019-03-03 23:07:49 +09:00
witness: witnessStack,
});
const outputType = classify.output(redeem!.output!);
const expanded = expandInput(
redeem!.input!,
redeem!.witness!,
outputType,
redeem!.output,
);
if (!expanded.prevOutType) return {};
2017-01-02 17:45:23 +01:00
2018-06-27 17:29:18 +10:00
return {
prevOutScript: output,
prevOutType: SCRIPT_TYPES.P2SH,
redeemScript: redeem!.output,
2018-06-27 17:29:18 +10:00
redeemScriptType: expanded.prevOutType,
witnessScript: expanded.witnessScript,
witnessScriptType: expanded.witnessScriptType,
2017-01-02 17:45:23 +01:00
2018-06-27 17:29:18 +10:00
pubkeys: expanded.pubkeys,
2019-03-03 23:07:49 +09:00
signatures: expanded.signatures,
};
2017-01-02 17:45:23 +01:00
}
2018-06-27 17:29:18 +10:00
if (type === SCRIPT_TYPES.P2WSH) {
const { output, redeem } = payments.p2wsh({
input: scriptSig,
2019-03-03 23:07:49 +09:00
witness: witnessStack,
});
const outputType = classify.output(redeem!.output!);
let expanded;
2018-06-27 17:29:18 +10:00
if (outputType === SCRIPT_TYPES.P2WPKH) {
2019-03-03 23:07:49 +09:00
expanded = expandInput(redeem!.input!, redeem!.witness!, outputType);
2018-06-27 17:29:18 +10:00
} else {
2019-03-03 23:07:49 +09:00
expanded = expandInput(
bscript.compile(redeem!.witness!),
[],
outputType,
redeem!.output,
);
2018-06-27 17:29:18 +10:00
}
2019-03-03 23:07:49 +09:00
if (!expanded.prevOutType) return {};
2018-06-27 17:29:18 +10:00
return {
prevOutScript: output,
prevOutType: SCRIPT_TYPES.P2WSH,
witnessScript: redeem!.output,
2018-06-27 17:29:18 +10:00
witnessScriptType: expanded.prevOutType,
2017-01-02 17:45:23 +01:00
2018-06-27 17:29:18 +10:00
pubkeys: expanded.pubkeys,
2019-03-03 23:07:49 +09:00
signatures: expanded.signatures,
};
2017-01-02 17:45:23 +01:00
}
2018-06-27 17:29:18 +10:00
return {
prevOutType: SCRIPT_TYPES.NONSTANDARD,
2019-03-03 23:07:49 +09:00
prevOutScript: scriptSig,
};
2017-01-02 17:45:23 +01:00
}
// could be done in expandInput, but requires the original Transaction for hashForSignature
2019-03-03 23:07:49 +09:00
function fixMultisigOrder(
input: TxbInput,
transaction: Transaction,
vin: number,
): void {
if (input.redeemScriptType !== SCRIPT_TYPES.P2MS || !input.redeemScript)
return;
if (input.pubkeys!.length === input.signatures!.length) return;
2019-03-03 23:07:49 +09:00
const unmatched = input.signatures!.concat();
input.signatures = input.pubkeys!.map(pubKey => {
2019-03-03 23:07:49 +09:00
const keyPair = ECPair.fromPublicKey(pubKey!);
let match: Buffer | undefined;
// check for a signature
2018-12-28 11:56:03 +09:00
unmatched.some((signature, i) => {
// skip if undefined || OP_0
2019-03-03 23:07:49 +09:00
if (!signature) return false;
// TODO: avoid O(n) hashForSignature
2019-03-03 23:07:49 +09:00
const parsed = bscript.signature.decode(signature);
const hash = transaction.hashForSignature(
vin,
input.redeemScript!,
parsed.hashType,
);
// skip if signature does not match pubKey
2019-03-03 23:07:49 +09:00
if (!keyPair.verify(hash, parsed.signature)) return false;
// remove matched signature from unmatched
2019-03-03 23:07:49 +09:00
unmatched[i] = undefined;
match = signature;
2019-03-03 23:07:49 +09:00
return true;
});
2019-03-03 23:07:49 +09:00
return match;
});
}
2019-03-03 23:07:49 +09:00
function expandOutput(script: Buffer, ourPubKey?: Buffer): TxbOutput {
typeforce(types.Buffer, script);
const type = classify.output(script);
2018-06-27 17:29:18 +10:00
switch (type) {
case SCRIPT_TYPES.P2PKH: {
2019-03-03 23:07:49 +09:00
if (!ourPubKey) return { type };
2018-06-27 17:29:18 +10:00
// does our hash160(pubKey) match the output scripts?
2019-03-03 23:07:49 +09:00
const pkh1 = payments.p2pkh({ output: script }).hash;
const pkh2 = bcrypto.hash160(ourPubKey);
if (!pkh1!.equals(pkh2)) return { type };
2018-06-27 17:29:18 +10:00
return {
type,
pubkeys: [ourPubKey],
2019-03-03 23:07:49 +09:00
signatures: [undefined],
};
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
case SCRIPT_TYPES.P2WPKH: {
2019-03-03 23:07:49 +09:00
if (!ourPubKey) return { type };
2016-12-14 15:41:24 +11:00
2018-06-27 17:29:18 +10:00
// does our hash160(pubKey) match the output scripts?
2019-03-03 23:07:49 +09:00
const wpkh1 = payments.p2wpkh({ output: script }).hash;
const wpkh2 = bcrypto.hash160(ourPubKey);
if (!wpkh1!.equals(wpkh2)) return { type };
2016-12-14 15:41:24 +11:00
2018-06-27 17:29:18 +10:00
return {
type,
pubkeys: [ourPubKey],
2019-03-03 23:07:49 +09:00
signatures: [undefined],
};
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
case SCRIPT_TYPES.P2PK: {
2019-03-03 23:07:49 +09:00
const p2pk = payments.p2pk({ output: script });
2018-06-27 17:29:18 +10:00
return {
type,
pubkeys: [p2pk.pubkey],
2019-03-03 23:07:49 +09:00
signatures: [undefined],
};
2018-06-27 17:29:18 +10:00
}
2016-09-27 21:08:48 +10:00
2018-09-03 14:06:30 +10:00
case SCRIPT_TYPES.P2MS: {
2019-03-03 23:07:49 +09:00
const p2ms = payments.p2ms({ output: script });
2018-06-27 17:29:18 +10:00
return {
type,
pubkeys: p2ms.pubkeys,
signatures: p2ms.pubkeys!.map((): undefined => undefined),
2019-03-03 23:07:49 +09:00
maxSignatures: p2ms.m,
};
2018-06-27 17:29:18 +10:00
}
2016-09-27 21:08:48 +10:00
}
2019-03-03 23:07:49 +09:00
return { type };
}
2019-03-03 23:07:49 +09:00
function prepareInput(
input: TxbInput,
ourPubKey: Buffer,
redeemScript: Buffer,
witnessScript: Buffer,
): TxbInput {
2018-06-27 17:29:18 +10:00
if (redeemScript && witnessScript) {
2019-03-07 14:03:04 +09:00
const p2wsh = payments.p2wsh({
redeem: { output: witnessScript },
}) as Payment;
const p2wshAlt = payments.p2wsh({ output: redeemScript }) as Payment;
const p2sh = payments.p2sh({ redeem: { output: redeemScript } }) as Payment;
const p2shAlt = payments.p2sh({ redeem: p2wsh }) as Payment;
2018-06-27 17:29:18 +10:00
// enforces P2SH(P2WSH(...))
2019-03-03 23:07:49 +09:00
if (!p2wsh.hash!.equals(p2wshAlt.hash!))
throw new Error('Witness script inconsistent with prevOutScript');
if (!p2sh.hash!.equals(p2shAlt.hash!))
throw new Error('Redeem script inconsistent with prevOutScript');
const expanded = expandOutput(p2wsh.redeem!.output!, ourPubKey);
if (!expanded.pubkeys)
throw new Error(
expanded.type +
' not supported as witnessScript (' +
bscript.toASM(witnessScript) +
')',
);
2018-12-28 11:56:03 +09:00
if (input.signatures && input.signatures.some(x => x !== undefined)) {
2019-03-03 23:07:49 +09:00
expanded.signatures = input.signatures;
2018-06-27 17:29:18 +10:00
}
2019-03-07 14:03:04 +09:00
const signScript = witnessScript;
2019-03-03 23:07:49 +09:00
if (expanded.type === SCRIPT_TYPES.P2WPKH)
throw new Error('P2SH(P2WSH(P2WPKH)) is a consensus failure');
2018-06-27 17:29:18 +10:00
return {
redeemScript,
2018-06-27 17:29:18 +10:00
redeemScriptType: SCRIPT_TYPES.P2WSH,
witnessScript,
2018-06-27 17:29:18 +10:00
witnessScriptType: expanded.type,
2018-06-27 17:29:18 +10:00
prevOutType: SCRIPT_TYPES.P2SH,
prevOutScript: p2sh.output,
2018-06-27 17:29:18 +10:00
hasWitness: true,
signScript,
2018-06-27 17:29:18 +10:00
signType: expanded.type,
2018-06-27 17:29:18 +10:00
pubkeys: expanded.pubkeys,
signatures: expanded.signatures,
2019-03-03 23:07:49 +09:00
maxSignatures: expanded.maxSignatures,
};
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
if (redeemScript) {
2019-03-07 14:03:04 +09:00
const p2sh = payments.p2sh({ redeem: { output: redeemScript } }) as Payment;
2018-06-27 17:29:18 +10:00
if (input.prevOutScript) {
2019-03-03 23:07:49 +09:00
let p2shAlt;
2018-06-27 17:29:18 +10:00
try {
2019-03-07 14:03:04 +09:00
p2shAlt = payments.p2sh({ output: input.prevOutScript }) as Payment;
2019-03-03 23:07:49 +09:00
} catch (e) {
throw new Error('PrevOutScript must be P2SH');
}
if (!p2sh.hash!.equals(p2shAlt.hash!))
throw new Error('Redeem script inconsistent with prevOutScript');
}
const expanded = expandOutput(p2sh.redeem!.output!, ourPubKey);
if (!expanded.pubkeys)
throw new Error(
expanded.type +
' not supported as redeemScript (' +
bscript.toASM(redeemScript) +
')',
);
2018-12-28 11:56:03 +09:00
if (input.signatures && input.signatures.some(x => x !== undefined)) {
2019-03-03 23:07:49 +09:00
expanded.signatures = input.signatures;
2018-06-27 17:29:18 +10:00
}
2019-03-03 23:07:49 +09:00
let signScript = redeemScript;
2018-06-27 17:29:18 +10:00
if (expanded.type === SCRIPT_TYPES.P2WPKH) {
2019-03-03 23:07:49 +09:00
signScript = payments.p2pkh({ pubkey: expanded.pubkeys[0] }).output!;
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
return {
redeemScript,
2018-06-27 17:29:18 +10:00
redeemScriptType: expanded.type,
2018-06-27 17:29:18 +10:00
prevOutType: SCRIPT_TYPES.P2SH,
prevOutScript: p2sh.output,
hasWitness: expanded.type === SCRIPT_TYPES.P2WPKH,
signScript,
2018-06-27 17:29:18 +10:00
signType: expanded.type,
2018-06-27 17:29:18 +10:00
pubkeys: expanded.pubkeys,
signatures: expanded.signatures,
2019-03-03 23:07:49 +09:00
maxSignatures: expanded.maxSignatures,
};
}
2018-06-27 17:29:18 +10:00
if (witnessScript) {
2019-03-03 23:07:49 +09:00
const p2wsh = payments.p2wsh({ redeem: { output: witnessScript } });
2018-06-27 17:29:18 +10:00
if (input.prevOutScript) {
2019-03-03 23:07:49 +09:00
const p2wshAlt = payments.p2wsh({ output: input.prevOutScript });
if (!p2wsh.hash!.equals(p2wshAlt.hash!))
throw new Error('Witness script inconsistent with prevOutScript');
}
const expanded = expandOutput(p2wsh.redeem!.output!, ourPubKey);
if (!expanded.pubkeys)
throw new Error(
expanded.type +
' not supported as witnessScript (' +
bscript.toASM(witnessScript) +
')',
);
2018-12-28 11:56:03 +09:00
if (input.signatures && input.signatures.some(x => x !== undefined)) {
2019-03-03 23:07:49 +09:00
expanded.signatures = input.signatures;
2016-12-22 13:18:45 +01:00
}
2019-03-07 14:03:04 +09:00
const signScript = witnessScript;
2019-03-03 23:07:49 +09:00
if (expanded.type === SCRIPT_TYPES.P2WPKH)
throw new Error('P2WSH(P2WPKH) is a consensus failure');
2018-06-27 17:29:18 +10:00
return {
witnessScript,
2018-06-27 17:29:18 +10:00
witnessScriptType: expanded.type,
2018-06-27 17:29:18 +10:00
prevOutType: SCRIPT_TYPES.P2WSH,
prevOutScript: p2wsh.output,
2018-06-27 17:29:18 +10:00
hasWitness: true,
signScript,
2018-06-27 17:29:18 +10:00
signType: expanded.type,
2016-12-14 15:41:24 +11:00
2018-06-27 17:29:18 +10:00
pubkeys: expanded.pubkeys,
signatures: expanded.signatures,
2019-03-03 23:07:49 +09:00
maxSignatures: expanded.maxSignatures,
};
2018-06-27 17:29:18 +10:00
}
2018-06-27 17:29:18 +10:00
if (input.prevOutType && input.prevOutScript) {
// embedded scripts are not possible without extra information
2019-03-03 23:07:49 +09:00
if (input.prevOutType === SCRIPT_TYPES.P2SH)
throw new Error(
'PrevOutScript is ' + input.prevOutType + ', requires redeemScript',
);
if (input.prevOutType === SCRIPT_TYPES.P2WSH)
throw new Error(
'PrevOutScript is ' + input.prevOutType + ', requires witnessScript',
);
if (!input.prevOutScript) throw new Error('PrevOutScript is missing');
const expanded = expandOutput(input.prevOutScript, ourPubKey);
if (!expanded.pubkeys)
throw new Error(
expanded.type +
' not supported (' +
bscript.toASM(input.prevOutScript) +
')',
);
2018-12-28 11:56:03 +09:00
if (input.signatures && input.signatures.some(x => x !== undefined)) {
2019-03-03 23:07:49 +09:00
expanded.signatures = input.signatures;
2016-12-22 13:18:45 +01:00
}
2019-03-03 23:07:49 +09:00
let signScript = input.prevOutScript;
if (expanded.type === SCRIPT_TYPES.P2WPKH) {
2019-03-07 14:03:04 +09:00
signScript = payments.p2pkh({ pubkey: expanded.pubkeys[0] })
.output as Buffer;
}
2018-06-27 17:29:18 +10:00
return {
prevOutType: expanded.type,
prevOutScript: input.prevOutScript,
hasWitness: expanded.type === SCRIPT_TYPES.P2WPKH,
signScript,
2018-06-27 17:29:18 +10:00
signType: expanded.type,
pubkeys: expanded.pubkeys,
signatures: expanded.signatures,
2019-03-03 23:07:49 +09:00
maxSignatures: expanded.maxSignatures,
};
2016-12-14 15:41:24 +11:00
}
2019-03-03 23:07:49 +09:00
const prevOutScript = payments.p2pkh({ pubkey: ourPubKey }).output;
2018-06-27 17:29:18 +10:00
return {
prevOutType: SCRIPT_TYPES.P2PKH,
2019-03-07 14:03:04 +09:00
prevOutScript,
2018-06-27 17:29:18 +10:00
hasWitness: false,
signScript: prevOutScript,
signType: SCRIPT_TYPES.P2PKH,
2018-06-27 17:29:18 +10:00
pubkeys: [ourPubKey],
2019-03-03 23:07:49 +09:00
signatures: [undefined],
};
2018-06-27 17:29:18 +10:00
}
2016-12-14 15:41:24 +11:00
2019-03-03 23:07:49 +09:00
function build(
type: string,
input: TxbInput,
allowIncomplete?: boolean,
): Payment | undefined {
2019-03-07 14:03:04 +09:00
const pubkeys = (input.pubkeys || []) as Buffer[];
let signatures = (input.signatures || []) as Buffer[];
2018-06-27 17:29:18 +10:00
switch (type) {
case SCRIPT_TYPES.P2PKH: {
2019-03-03 23:07:49 +09:00
if (pubkeys.length === 0) break;
if (signatures.length === 0) break;
2018-06-27 17:29:18 +10:00
2019-03-03 23:07:49 +09:00
return payments.p2pkh({ pubkey: pubkeys[0], signature: signatures[0] });
2018-06-27 17:29:18 +10:00
}
case SCRIPT_TYPES.P2WPKH: {
2019-03-03 23:07:49 +09:00
if (pubkeys.length === 0) break;
if (signatures.length === 0) break;
2018-06-27 17:29:18 +10:00
2019-03-03 23:07:49 +09:00
return payments.p2wpkh({ pubkey: pubkeys[0], signature: signatures[0] });
2018-06-27 17:29:18 +10:00
}
case SCRIPT_TYPES.P2PK: {
2019-03-03 23:07:49 +09:00
if (pubkeys.length === 0) break;
if (signatures.length === 0) break;
2018-06-27 17:29:18 +10:00
2019-03-03 23:07:49 +09:00
return payments.p2pk({ signature: signatures[0] });
2018-06-27 17:29:18 +10:00
}
2018-09-03 14:06:30 +10:00
case SCRIPT_TYPES.P2MS: {
2019-03-03 23:07:49 +09:00
const m = input.maxSignatures;
2018-06-27 17:29:18 +10:00
if (allowIncomplete) {
2019-03-03 23:07:49 +09:00
signatures = signatures.map(x => x || ops.OP_0);
2018-06-27 17:29:18 +10:00
} else {
2019-03-03 23:07:49 +09:00
signatures = signatures.filter(x => x);
2018-06-27 17:29:18 +10:00
}
2018-09-26 14:57:03 +10:00
// if the transaction is not not complete (complete), or if signatures.length === m, validate
// otherwise, the number of OP_0's may be >= m, so don't validate (boo)
2019-03-03 23:07:49 +09:00
const validate = !allowIncomplete || m === signatures.length;
return payments.p2ms(
{ m, pubkeys, signatures },
{ allowIncomplete, validate },
);
2018-06-27 17:29:18 +10:00
}
case SCRIPT_TYPES.P2SH: {
2019-03-03 23:07:49 +09:00
const redeem = build(input.redeemScriptType!, input, allowIncomplete);
if (!redeem) return;
2018-06-27 17:29:18 +10:00
return payments.p2sh({
redeem: {
output: redeem.output || input.redeemScript,
input: redeem.input,
2019-03-03 23:07:49 +09:00
witness: redeem.witness,
},
});
2018-06-27 17:29:18 +10:00
}
case SCRIPT_TYPES.P2WSH: {
2019-03-03 23:07:49 +09:00
const redeem = build(input.witnessScriptType!, input, allowIncomplete);
if (!redeem) return;
2018-06-27 17:29:18 +10:00
return payments.p2wsh({
redeem: {
output: input.witnessScript,
input: redeem.input,
2019-03-03 23:07:49 +09:00
witness: redeem.witness,
},
});
2018-06-27 17:29:18 +10:00
}
2016-12-14 15:41:24 +11:00
}
}
2019-03-03 23:07:49 +09:00
function canSign(input: TxbInput): boolean {
return (
input.signScript !== undefined &&
2018-06-27 17:29:18 +10:00
input.signType !== undefined &&
input.pubkeys !== undefined &&
input.signatures !== undefined &&
2018-06-27 17:29:18 +10:00
input.signatures.length === input.pubkeys.length &&
input.pubkeys.length > 0 &&
2019-03-03 23:07:49 +09:00
(input.hasWitness === false || input.value !== undefined)
);
}
2019-03-03 23:07:49 +09:00
function signatureHashType(buffer: Buffer): number {
return buffer.readUInt8(buffer.length - 1);
}