Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(btc): support including multi-origin UTXOs #228

Merged
merged 3 commits into from
Jun 18, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 9 additions & 0 deletions .changeset/orange-mice-trade.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
---
"@rgbpp-sdk/btc": minor
---

Support including multi-origin UTXOs in the same transaction

- Add `pubkeyMap` option in the sendUtxos(), sendRgbppUtxos() and sendRbf() API
- Rename `inputsPubkey` option to `pubkeyMap` in the sendRbf() API
- Delete `onlyProvableUtxos` option from the sendRgbppUtxos() API
15 changes: 15 additions & 0 deletions packages/btc/src/address.ts
Original file line number Diff line number Diff line change
Expand Up @@ -210,3 +210,18 @@ function getAddressTypeDust(addressType: AddressType) {
return 546;
}
}

/**
* Add address/pubkey pair to a Record<address, pubkey> map
*/
export function addAddressToPubkeyMap(
pubkeyMap: Record<string, string>,
address: string,
pubkey?: string,
): Record<string, string> {
const newMap = { ...pubkeyMap };
if (pubkey) {
newMap[address] = pubkey;
}
return newMap;
}
16 changes: 2 additions & 14 deletions packages/btc/src/api/sendRbf.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ import { isOpReturnScriptPubkey } from '../transaction/embed';
import { networkTypeToNetwork } from '../preset/network';
import { networkTypeToConfig } from '../preset/config';
import { createSendUtxosBuilder } from './sendUtxos';
import { isP2trScript } from '../script';
import { bitcoin } from '../bitcoin';

export interface SendRbfProps {
Expand All @@ -23,7 +22,7 @@ export interface SendRbfProps {
requireGreaterFeeAndRate?: boolean;

// EXPERIMENTAL: the below props are unstable and can be altered at any time
inputsPubkey?: Record<string, string>; // Record<address, pubkey>
pubkeyMap?: Record<string, string>; // Record<address, pubkey>
}

export async function createSendRbfBuilder(props: SendRbfProps): Promise<{
Expand All @@ -43,18 +42,6 @@ export async function createSendRbfBuilder(props: SendRbfProps): Promise<{
if (!utxo) {
throw TxBuildError.withComment(ErrorCodes.CANNOT_FIND_UTXO, `hash: ${hash}, index: ${input.index}`);
}

// Ensure each P2TR input has a corresponding pubkey
const fromPubkey = utxo.address === props.from ? props.fromPubkey : undefined;
const inputPubkey = props.inputsPubkey?.[utxo.address];
const pubkey = inputPubkey ?? fromPubkey;
if (pubkey) {
utxo.pubkey = pubkey;
}
if (isP2trScript(utxo.scriptPk) && !utxo.pubkey) {
throw TxBuildError.withComment(ErrorCodes.MISSING_PUBKEY, utxo.address);
}

inputs.push(utxo);
}

Expand Down Expand Up @@ -151,6 +138,7 @@ export async function createSendRbfBuilder(props: SendRbfProps): Promise<{
from: props.from,
source: props.source,
feeRate: props.feeRate,
pubkeyMap: props.pubkeyMap,
fromPubkey: props.fromPubkey,
minUtxoSatoshi: props.minUtxoSatoshi,
onlyConfirmedUtxos: props.onlyConfirmedUtxos ?? true,
Expand Down
21 changes: 5 additions & 16 deletions packages/btc/src/api/sendRgbppUtxos.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ export interface SendRgbppUtxosProps {
excludeUtxos?: BaseOutput[];

// EXPERIMENTAL: the below props are unstable and can be altered at any time
onlyProvableUtxos?: boolean;
pubkeyMap?: Record<string, string>; // Record<address, pubkey>
}

/**
Expand All @@ -45,8 +45,6 @@ export async function createSendRgbppUtxosBuilder(props: SendRgbppUtxosProps): P
feeRate: number;
changeIndex: number;
}> {
const onlyProvableUtxos = props.onlyProvableUtxos ?? true;

const btcInputs: Utxo[] = [];
const btcOutputs: InitOutput[] = [];
let lastCkbTypeOutputIndex = -1;
Expand Down Expand Up @@ -86,33 +84,23 @@ export async function createSendRgbppUtxosBuilder(props: SendRgbppUtxosProps): P
for (let i = 0; i < ckbVirtualTx.inputs.length; i++) {
const { lockArgs, isRgbppLock } = ckbLiveCells[i];

// If input.lock == RgbppLock, add to inputs if:
// Add to inputs if all the following conditions are met:
// 1. input.lock.args can be unpacked to RgbppLockArgs
// 2. utxo can be found via the DataSource.getUtxo() API
// 3. utxo.scriptPk == addressToScriptPk(props.from)
// 4. utxo is not duplicated in the inputs
// 3. utxo is not duplicated in the inputs
if (isRgbppLock) {
const args = lockArgs!;
const utxo = btcUtxos[i];
if (!utxo) {
throw TxBuildError.withComment(ErrorCodes.CANNOT_FIND_UTXO, `hash: ${args.btcTxid}, index: ${args.outIndex}`);
}
if (onlyProvableUtxos && utxo.address !== props.from) {
throw TxBuildError.withComment(
ErrorCodes.REFERENCED_UNPROVABLE_UTXO,
`hash: ${args.btcTxid}, index: ${args.outIndex}`,
);
}

const foundInInputs = btcInputs.some((v) => v.txid === utxo.txid && v.vout === utxo.vout);
if (foundInInputs) {
continue;
}

btcInputs.push({
...utxo,
pubkey: props.fromPubkey, // For P2TR addresses, a pubkey is required
});
btcInputs.push(utxo);
}
}

Expand Down Expand Up @@ -179,6 +167,7 @@ export async function createSendRgbppUtxosBuilder(props: SendRgbppUtxosProps): P
minUtxoSatoshi: props.minUtxoSatoshi,
onlyConfirmedUtxos: props.onlyConfirmedUtxos,
excludeUtxos: props.excludeUtxos,
pubkeyMap: props.pubkeyMap,
});
}

Expand Down
24 changes: 17 additions & 7 deletions packages/btc/src/api/sendUtxos.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
import { bitcoin } from '../bitcoin';
import { DataSource } from '../query/source';
import { BaseOutput, Utxo } from '../transaction/utxo';
import { TxBuilder, InitOutput } from '../transaction/build';
import { BaseOutput, Utxo, prepareUtxoInputs } from '../transaction/utxo';
import { addAddressToPubkeyMap } from '../address';

export interface SendUtxosProps {
inputs: Utxo[];
Expand All @@ -17,6 +18,7 @@ export interface SendUtxosProps {

// EXPERIMENTAL: the below props are unstable and can be altered at any time
skipInputsValidation?: boolean;
pubkeyMap?: Record<string, string>; // Record<address, pubkey>
}

export async function createSendUtxosBuilder(props: SendUtxosProps): Promise<{
Expand All @@ -32,16 +34,24 @@ export async function createSendUtxosBuilder(props: SendUtxosProps): Promise<{
onlyConfirmedUtxos: props.onlyConfirmedUtxos,
});

tx.addInputs(props.inputs);
tx.addOutputs(props.outputs);
// Prepare the UTXO inputs:
// 1. Fill pubkey for each P2TR UTXO, and throw if the corresponding pubkey is not found
// 2. Throw if unconfirmed UTXOs are found (if onlyConfirmedUtxos == true && skipInputsValidation == false)
const pubkeyMap = addAddressToPubkeyMap(props.pubkeyMap ?? {}, props.from, props.fromPubkey);
const inputs = await prepareUtxoInputs({
utxos: props.inputs,
source: props.source,
requireConfirmed: props.onlyConfirmedUtxos && !props.skipInputsValidation,
requirePubkey: true,
pubkeyMap,
});

if (props.onlyConfirmedUtxos && !props.skipInputsValidation) {
await tx.validateInputs();
}
tx.addInputs(inputs);
tx.addOutputs(props.outputs);

const paid = await tx.payFee({
address: props.from,
publicKey: props.fromPubkey,
publicKey: pubkeyMap[props.from],
changeAddress: props.changeAddress,
excludeUtxos: props.excludeUtxos,
});
Expand Down
24 changes: 10 additions & 14 deletions packages/btc/src/transaction/build.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { bitcoin } from '../bitcoin';
import { DataSource } from '../query/source';
import { ErrorCodes, TxBuildError } from '../error';
import { NetworkType, RgbppBtcConfig } from '../preset/types';
import { AddressType, addressToScriptPublicKeyHex, getAddressType, isSupportedFromAddress } from '../address';
import { isSupportedFromAddress } from '../address';
import { dataToOpReturnScriptPubkey, isOpReturnScriptPubkey } from './embed';
import { networkTypeToConfig } from '../preset/config';
import { BaseOutput, Utxo, utxoToInput } from './utxo';
Expand Down Expand Up @@ -213,8 +213,7 @@ export class TxBuilder {
}

// Calculate network fee
const addressType = getAddressType(address);
currentFee = await this.calculateFee(addressType, currentFeeRate);
currentFee = await this.calculateFee(currentFeeRate);

// If (fee = previousFee ±1), the fee is considered acceptable/expected.
isFeeExpected = [-1, 0, 1].includes(currentFee - previousFee);
Expand Down Expand Up @@ -471,14 +470,14 @@ export class TxBuilder {
});
}

async calculateFee(addressType: AddressType, feeRate?: number): Promise<number> {
async calculateFee(feeRate?: number): Promise<number> {
if (!feeRate && !this.feeRate) {
throw TxBuildError.withComment(ErrorCodes.INVALID_FEE_RATE, `${feeRate ?? this.feeRate}`);
}

const currentFeeRate = feeRate ?? this.feeRate!;

const psbt = await this.createEstimatedPsbt(addressType);
const psbt = await this.createEstimatedPsbt();
const tx = psbt.extractTransaction(true);

const inputs = tx.ins.length;
Expand All @@ -490,20 +489,17 @@ export class TxBuilder {
return Math.ceil(virtualSize * currentFeeRate);
}

async createEstimatedPsbt(addressType: AddressType): Promise<bitcoin.Psbt> {
const estimate = FeeEstimator.fromRandom(addressType, this.networkType);
const estimateScriptPk = addressToScriptPublicKeyHex(estimate.address, this.networkType);
async createEstimatedPsbt(): Promise<bitcoin.Psbt> {
const estimator = FeeEstimator.fromRandom(this.networkType);

const tx = this.clone();
const utxos = tx.inputs.map((input) => input.utxo);
tx.inputs = utxos.map((utxo) => {
utxo.scriptPk = estimateScriptPk;
utxo.pubkey = estimate.publicKey;
return utxoToInput(utxo);
tx.inputs = tx.inputs.map((input) => {
const replacedUtxo = estimator.replaceUtxo(input.utxo);
return utxoToInput(replacedUtxo);
});

const psbt = tx.toPsbt();
await estimate.signPsbt(psbt);
await estimator.signPsbt(psbt);
return psbt;
}

Expand Down
108 changes: 76 additions & 32 deletions packages/btc/src/transaction/fee.ts
Original file line number Diff line number Diff line change
@@ -1,61 +1,105 @@
import { ECPairInterface } from 'ecpair';
import { AddressType } from '../address';
import { NetworkType } from '../preset/types';
import { toXOnly, tweakSigner } from '../utils';
import { networkTypeToNetwork } from '../preset/network';
import { AddressType, publicKeyToAddress } from '../address';
import { isP2trScript, isP2wpkhScript } from '../script';
import { bitcoin, ECPair, isTaprootInput } from '../bitcoin';
import { toXOnly, tweakSigner } from '../utils';
import { Utxo } from './utxo';

interface FeeEstimateAccount {
payment: bitcoin.Payment;
addressType: AddressType;
address: string;
scriptPubkey: string;
tapInternalKey?: Buffer;
}

export class FeeEstimator {
public networkType: NetworkType;
public addressType: AddressType;
public network: bitcoin.Network;

private readonly keyPair: ECPairInterface;
public publicKey: string;
public address: string;
public readonly pubkey: string;
public accounts: {
p2wpkh: FeeEstimateAccount;
p2tr: FeeEstimateAccount;
};

constructor(wif: string, networkType: NetworkType, addressType: AddressType) {
constructor(wif: string, networkType: NetworkType) {
const network = networkTypeToNetwork(networkType);
const keyPair = ECPair.fromWIF(wif, network);
this.networkType = networkType;
this.network = network;

const keyPair = ECPair.fromWIF(wif, network);
this.pubkey = keyPair.publicKey.toString('hex');
this.keyPair = keyPair;
this.publicKey = keyPair.publicKey.toString('hex');
this.address = publicKeyToAddress(this.publicKey, addressType, networkType);

this.addressType = addressType;
this.networkType = networkType;
this.network = network;
const p2wpkh = bitcoin.payments.p2wpkh({
pubkey: keyPair.publicKey,
network,
});
const p2tr = bitcoin.payments.p2tr({
internalPubkey: toXOnly(keyPair.publicKey),
network,
});
this.accounts = {
p2wpkh: {
payment: p2wpkh,
address: p2wpkh.address!,
addressType: AddressType.P2WPKH,
scriptPubkey: p2wpkh.output!.toString('hex'),
},
p2tr: {
payment: p2tr,
address: p2tr.address!,
addressType: AddressType.P2TR,
tapInternalKey: toXOnly(keyPair.publicKey),
scriptPubkey: p2tr.output!.toString('hex'),
},
};
}

static fromRandom(addressType: AddressType, networkType: NetworkType) {
static fromRandom(networkType: NetworkType) {
const network = networkTypeToNetwork(networkType);
const keyPair = ECPair.makeRandom({ network });
return new FeeEstimator(keyPair.toWIF(), networkType, addressType);
return new FeeEstimator(keyPair.toWIF(), networkType);
}

replaceUtxo(utxo: Utxo): Utxo {
if (utxo.addressType === AddressType.P2WPKH || isP2wpkhScript(utxo.scriptPk)) {
utxo.scriptPk = this.accounts.p2wpkh.scriptPubkey;
utxo.pubkey = this.pubkey;
}
if (utxo.addressType === AddressType.P2TR || isP2trScript(utxo.scriptPk)) {
utxo.scriptPk = this.accounts.p2tr.scriptPubkey;
utxo.pubkey = this.pubkey;
}

return utxo;
}

async signPsbt(psbt: bitcoin.Psbt): Promise<bitcoin.Psbt> {
psbt.data.inputs.forEach((v) => {
const isNotSigned = !(v.finalScriptSig || v.finalScriptWitness);
const isP2TR = this.addressType === AddressType.P2TR;
const lostInternalPubkey = !v.tapInternalKey;
// Special measures taken for compatibility with certain applications.
if (isNotSigned && isP2TR && lostInternalPubkey) {
const tapInternalKey = toXOnly(Buffer.from(this.publicKey, 'hex'));
const { output } = bitcoin.payments.p2tr({
internalPubkey: tapInternalKey,
network: networkTypeToNetwork(this.networkType),
});
if (v.witnessUtxo?.script.toString('hex') == output?.toString('hex')) {
v.tapInternalKey = tapInternalKey;
}
}
// Tweak signer for P2TR inputs
const tweakedSigner = tweakSigner(this.keyPair, {
network: this.network,
});

psbt.data.inputs.forEach((input, index) => {
// Fill tapInternalKey for P2TR inputs if missing
if (input.witnessUtxo) {
const isNotSigned = !(input.finalScriptSig || input.finalScriptWitness);
const isP2trInput = isP2trScript(input.witnessUtxo.script);
const lostInternalPubkey = !input.tapInternalKey;
if (isNotSigned && isP2trInput && lostInternalPubkey) {
if (input.witnessUtxo.script.toString('hex') === this.accounts.p2tr.scriptPubkey) {
input.tapInternalKey = this.accounts.p2tr.tapInternalKey!;
}
}
}

// Sign P2WPKH/P2TR inputs
if (isTaprootInput(input)) {
const tweakedSigner = tweakSigner(this.keyPair, {
network: this.network,
});
psbt.signInput(index, tweakedSigner);
} else {
psbt.signInput(index, this.keyPair);
Expand Down
Loading
Loading