Feature: Add endpoint for PSBT nonWitnessUtxo inclusion

This commit is contained in:
junderw 2022-09-05 23:13:45 +09:00
parent 9047cb5998
commit f062132636
No known key found for this signature in database
GPG Key ID: B256185D3A971908

View File

@ -1,5 +1,6 @@
import { Application, Request, Response } from 'express';
import axios from 'axios';
import * as bitcoinjs from 'bitcoinjs-lib';
import config from '../../config';
import websocketHandler from '../websocket-handler';
import mempool from '../mempool';
@ -95,6 +96,7 @@ class BitcoinRoutes {
.get(config.MEMPOOL.API_URL_PREFIX + 'mempool', this.getMempool)
.get(config.MEMPOOL.API_URL_PREFIX + 'mempool/txids', this.getMempoolTxIds)
.get(config.MEMPOOL.API_URL_PREFIX + 'mempool/recent', this.getRecentMempoolTransactions)
.post(config.MEMPOOL.API_URL_PREFIX + 'psbt/addparents', this.postPsbtCompletion)
.get(config.MEMPOOL.API_URL_PREFIX + 'tx/:txId', this.getTransaction)
.post(config.MEMPOOL.API_URL_PREFIX + 'tx', this.$postTransaction)
.get(config.MEMPOOL.API_URL_PREFIX + 'tx/:txId/hex', this.getRawTransaction)
@ -241,6 +243,48 @@ class BitcoinRoutes {
}
}
/**
* Takes the PSBT as text/plain body, parses it, and adds the full
* parent transaction to each input that doesn't already have it.
* This is used for BTCPayServer / Trezor users which need access to
* the full parent transaction even with segwit inputs.
* It will respond with a text/plain PSBT in the same format (hex|base64).
*/
private async postPsbtCompletion(req: Request, res: Response) {
res.setHeader('content-type', 'text/plain');
try {
let psbt: bitcoinjs.Psbt;
let format: 'hex' | 'base64';
try {
psbt = bitcoinjs.Psbt.fromBase64(req.body);
format = 'base64';
} catch(e1) {
try {
psbt = bitcoinjs.Psbt.fromHex(req.body);
format = 'hex';
} catch(e2) {
throw new Error(`Unable to parse PSBT`);
}
}
for (const [index, input] of psbt.data.inputs.entries()) {
if (!input.nonWitnessUtxo) {
// Buffer.from ensures it won't be modified in place by reverse()
const txid = Buffer.from(psbt.txInputs[index].hash).reverse().toString('hex');
const transaction: IEsploraApi.Transaction = await bitcoinApi.$getRawTransaction(txid, true);
if (!transaction.hex) {
throw new Error(`Couldn't get transaction hex for ${txid}`);
}
psbt.updateInput(index, {
nonWitnessUtxo: Buffer.from(transaction.hex, 'hex'),
});
}
}
res.send(format === 'hex' ? psbt.toHex() : psbt.toBase64());
} catch (e: any) {
res.status(500).send(e instanceof Error ? e.message : e);
}
}
private async getTransactionStatus(req: Request, res: Response) {
try {
const transaction = await transactionUtils.$getTransactionExtended(req.params.txId, true);