@bitgo-beta/abstract-utxo 1.6.1-alpha.451 → 1.6.1-alpha.453
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/src/abstractUtxoCoin.d.ts +10 -8
- package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/cjs/src/abstractUtxoCoin.js +8 -12
- package/dist/cjs/src/impl/bch/bch.d.ts +4 -2
- package/dist/cjs/src/impl/bch/bch.d.ts.map +1 -1
- package/dist/cjs/src/impl/bch/bch.js +4 -3
- package/dist/cjs/src/impl/bch/tbch.d.ts +2 -0
- package/dist/cjs/src/impl/bch/tbch.d.ts.map +1 -1
- package/dist/cjs/src/impl/bch/tbch.js +3 -36
- package/dist/cjs/src/impl/bcha/bcha.d.ts +3 -2
- package/dist/cjs/src/impl/bcha/bcha.d.ts.map +1 -1
- package/dist/cjs/src/impl/bcha/bcha.js +4 -37
- package/dist/cjs/src/impl/bcha/tbcha.d.ts +2 -0
- package/dist/cjs/src/impl/bcha/tbcha.d.ts.map +1 -1
- package/dist/cjs/src/impl/bcha/tbcha.js +3 -36
- package/dist/cjs/src/impl/bsv/bsv.d.ts +3 -2
- package/dist/cjs/src/impl/bsv/bsv.d.ts.map +1 -1
- package/dist/cjs/src/impl/bsv/bsv.js +4 -37
- package/dist/cjs/src/impl/bsv/tbsv.d.ts +2 -0
- package/dist/cjs/src/impl/bsv/tbsv.d.ts.map +1 -1
- package/dist/cjs/src/impl/bsv/tbsv.js +3 -36
- package/dist/cjs/src/impl/btc/btc.d.ts +4 -2
- package/dist/cjs/src/impl/btc/btc.d.ts.map +1 -1
- package/dist/cjs/src/impl/btc/btc.js +4 -37
- package/dist/cjs/src/impl/btc/tbtc.d.ts +2 -0
- package/dist/cjs/src/impl/btc/tbtc.d.ts.map +1 -1
- package/dist/cjs/src/impl/btc/tbtc.js +3 -36
- package/dist/cjs/src/impl/btc/tbtc4.d.ts +2 -0
- package/dist/cjs/src/impl/btc/tbtc4.d.ts.map +1 -1
- package/dist/cjs/src/impl/btc/tbtc4.js +3 -36
- package/dist/cjs/src/impl/btc/tbtcbgsig.d.ts +2 -0
- package/dist/cjs/src/impl/btc/tbtcbgsig.d.ts.map +1 -1
- package/dist/cjs/src/impl/btc/tbtcbgsig.js +3 -36
- package/dist/cjs/src/impl/btc/tbtcsig.d.ts +2 -0
- package/dist/cjs/src/impl/btc/tbtcsig.d.ts.map +1 -1
- package/dist/cjs/src/impl/btc/tbtcsig.js +3 -36
- package/dist/cjs/src/impl/btg/btg.d.ts +4 -2
- package/dist/cjs/src/impl/btg/btg.d.ts.map +1 -1
- package/dist/cjs/src/impl/btg/btg.js +4 -37
- package/dist/cjs/src/impl/dash/dash.d.ts +4 -2
- package/dist/cjs/src/impl/dash/dash.d.ts.map +1 -1
- package/dist/cjs/src/impl/dash/dash.js +4 -37
- package/dist/cjs/src/impl/dash/tdash.d.ts +2 -0
- package/dist/cjs/src/impl/dash/tdash.d.ts.map +1 -1
- package/dist/cjs/src/impl/dash/tdash.js +3 -36
- package/dist/cjs/src/impl/doge/doge.d.ts +4 -2
- package/dist/cjs/src/impl/doge/doge.d.ts.map +1 -1
- package/dist/cjs/src/impl/doge/doge.js +4 -4
- package/dist/cjs/src/impl/doge/tdoge.d.ts +2 -0
- package/dist/cjs/src/impl/doge/tdoge.d.ts.map +1 -1
- package/dist/cjs/src/impl/doge/tdoge.js +3 -36
- package/dist/cjs/src/impl/ltc/ltc.d.ts +4 -2
- package/dist/cjs/src/impl/ltc/ltc.d.ts.map +1 -1
- package/dist/cjs/src/impl/ltc/ltc.js +4 -3
- package/dist/cjs/src/impl/ltc/tltc.d.ts +2 -0
- package/dist/cjs/src/impl/ltc/tltc.d.ts.map +1 -1
- package/dist/cjs/src/impl/ltc/tltc.js +3 -2
- package/dist/cjs/src/impl/zec/tzec.d.ts +2 -0
- package/dist/cjs/src/impl/zec/tzec.d.ts.map +1 -1
- package/dist/cjs/src/impl/zec/tzec.js +3 -36
- package/dist/cjs/src/impl/zec/zec.d.ts +4 -2
- package/dist/cjs/src/impl/zec/zec.d.ts.map +1 -1
- package/dist/cjs/src/impl/zec/zec.js +4 -37
- package/dist/cjs/src/names.d.ts +12 -4
- package/dist/cjs/src/names.d.ts.map +1 -1
- package/dist/cjs/src/names.js +66 -57
- package/dist/cjs/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/backupKeyRecovery.js +32 -17
- package/dist/cjs/src/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/crossChainRecovery.js +25 -18
- package/dist/cjs/src/recovery/psbt.d.ts +6 -6
- package/dist/cjs/src/recovery/psbt.d.ts.map +1 -1
- package/dist/cjs/src/recovery/psbt.js +50 -36
- package/dist/cjs/src/tnumber.d.ts +4 -0
- package/dist/cjs/src/tnumber.d.ts.map +1 -0
- package/dist/cjs/src/tnumber.js +12 -0
- package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts +1 -1
- package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/SigningError.js +1 -1
- package/dist/cjs/src/transaction/fixedScript/{signPsbt.d.ts → signPsbtUtxolib.d.ts} +4 -13
- package/dist/cjs/src/transaction/fixedScript/signPsbtUtxolib.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbtUtxolib.js +157 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts +2 -7
- package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.js +3 -17
- package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts +7 -0
- package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/signTransaction.js +25 -6
- package/dist/cjs/src/wallet.js +2 -2
- package/dist/cjs/test/unit/coins.js +18 -1
- package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +2 -2
- package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.js +4 -6
- package/dist/cjs/tsconfig.tsbuildinfo +1 -1
- package/dist/esm/abstractUtxoCoin.d.ts +10 -8
- package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/esm/abstractUtxoCoin.js +9 -13
- package/dist/esm/impl/bch/bch.d.ts +4 -2
- package/dist/esm/impl/bch/bch.d.ts.map +1 -1
- package/dist/esm/impl/bch/bch.js +4 -3
- package/dist/esm/impl/bch/tbch.d.ts +2 -0
- package/dist/esm/impl/bch/tbch.d.ts.map +1 -1
- package/dist/esm/impl/bch/tbch.js +3 -3
- package/dist/esm/impl/bcha/bcha.d.ts +3 -2
- package/dist/esm/impl/bcha/bcha.d.ts.map +1 -1
- package/dist/esm/impl/bcha/bcha.js +4 -4
- package/dist/esm/impl/bcha/tbcha.d.ts +2 -0
- package/dist/esm/impl/bcha/tbcha.d.ts.map +1 -1
- package/dist/esm/impl/bcha/tbcha.js +3 -3
- package/dist/esm/impl/bsv/bsv.d.ts +3 -2
- package/dist/esm/impl/bsv/bsv.d.ts.map +1 -1
- package/dist/esm/impl/bsv/bsv.js +4 -4
- package/dist/esm/impl/bsv/tbsv.d.ts +2 -0
- package/dist/esm/impl/bsv/tbsv.d.ts.map +1 -1
- package/dist/esm/impl/bsv/tbsv.js +3 -3
- package/dist/esm/impl/btc/btc.d.ts +4 -2
- package/dist/esm/impl/btc/btc.d.ts.map +1 -1
- package/dist/esm/impl/btc/btc.js +4 -4
- package/dist/esm/impl/btc/tbtc.d.ts +2 -0
- package/dist/esm/impl/btc/tbtc.d.ts.map +1 -1
- package/dist/esm/impl/btc/tbtc.js +3 -3
- package/dist/esm/impl/btc/tbtc4.d.ts +2 -0
- package/dist/esm/impl/btc/tbtc4.d.ts.map +1 -1
- package/dist/esm/impl/btc/tbtc4.js +3 -3
- package/dist/esm/impl/btc/tbtcbgsig.d.ts +2 -0
- package/dist/esm/impl/btc/tbtcbgsig.d.ts.map +1 -1
- package/dist/esm/impl/btc/tbtcbgsig.js +3 -3
- package/dist/esm/impl/btc/tbtcsig.d.ts +2 -0
- package/dist/esm/impl/btc/tbtcsig.d.ts.map +1 -1
- package/dist/esm/impl/btc/tbtcsig.js +3 -3
- package/dist/esm/impl/btg/btg.d.ts +4 -2
- package/dist/esm/impl/btg/btg.d.ts.map +1 -1
- package/dist/esm/impl/btg/btg.js +4 -4
- package/dist/esm/impl/dash/dash.d.ts +4 -2
- package/dist/esm/impl/dash/dash.d.ts.map +1 -1
- package/dist/esm/impl/dash/dash.js +4 -4
- package/dist/esm/impl/dash/tdash.d.ts +2 -0
- package/dist/esm/impl/dash/tdash.d.ts.map +1 -1
- package/dist/esm/impl/dash/tdash.js +3 -3
- package/dist/esm/impl/doge/doge.d.ts +4 -2
- package/dist/esm/impl/doge/doge.d.ts.map +1 -1
- package/dist/esm/impl/doge/doge.js +4 -4
- package/dist/esm/impl/doge/tdoge.d.ts +2 -0
- package/dist/esm/impl/doge/tdoge.d.ts.map +1 -1
- package/dist/esm/impl/doge/tdoge.js +3 -3
- package/dist/esm/impl/ltc/ltc.d.ts +4 -2
- package/dist/esm/impl/ltc/ltc.d.ts.map +1 -1
- package/dist/esm/impl/ltc/ltc.js +4 -3
- package/dist/esm/impl/ltc/tltc.d.ts +2 -0
- package/dist/esm/impl/ltc/tltc.d.ts.map +1 -1
- package/dist/esm/impl/ltc/tltc.js +3 -2
- package/dist/esm/impl/zec/tzec.d.ts +2 -0
- package/dist/esm/impl/zec/tzec.d.ts.map +1 -1
- package/dist/esm/impl/zec/tzec.js +3 -3
- package/dist/esm/impl/zec/zec.d.ts +4 -2
- package/dist/esm/impl/zec/zec.d.ts.map +1 -1
- package/dist/esm/impl/zec/zec.js +4 -4
- package/dist/esm/names.d.ts +12 -4
- package/dist/esm/names.d.ts.map +1 -1
- package/dist/esm/names.js +61 -54
- package/dist/esm/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/backupKeyRecovery.js +33 -18
- package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/crossChainRecovery.js +26 -19
- package/dist/esm/recovery/psbt.d.ts +6 -6
- package/dist/esm/recovery/psbt.d.ts.map +1 -1
- package/dist/esm/recovery/psbt.js +50 -36
- package/dist/esm/tnumber.d.ts +4 -0
- package/dist/esm/tnumber.d.ts.map +1 -0
- package/dist/esm/tnumber.js +9 -0
- package/dist/esm/transaction/fixedScript/SigningError.d.ts +1 -1
- package/dist/esm/transaction/fixedScript/SigningError.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/SigningError.js +1 -1
- package/dist/esm/transaction/fixedScript/{signPsbt.d.ts → signPsbtUtxolib.d.ts} +4 -13
- package/dist/esm/transaction/fixedScript/signPsbtUtxolib.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/signPsbtUtxolib.js +117 -0
- package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts +2 -7
- package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/signPsbtWasm.js +3 -17
- package/dist/esm/transaction/fixedScript/signTransaction.d.ts +7 -0
- package/dist/esm/transaction/fixedScript/signTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/signTransaction.js +25 -7
- package/dist/esm/wallet.js +2 -2
- package/package.json +11 -11
- package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts.map +0 -1
- package/dist/cjs/src/transaction/fixedScript/signPsbt.js +0 -174
- package/dist/esm/transaction/fixedScript/signPsbt.d.ts.map +0 -1
- package/dist/esm/transaction/fixedScript/signPsbt.js +0 -134
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
-
import { fixedScriptWallet, utxolibCompat } from '@bitgo/wasm-utxo';
|
|
2
|
+
import { CoinName, fixedScriptWallet, utxolibCompat } from '@bitgo/wasm-utxo';
|
|
3
3
|
type RootWalletKeys = utxolib.bitgo.RootWalletKeys;
|
|
4
4
|
type WalletUnspent<TNumber extends number | bigint> = utxolib.bitgo.WalletUnspent<TNumber>;
|
|
5
5
|
type ChainCode = utxolib.bitgo.ChainCode;
|
|
@@ -41,7 +41,7 @@ export interface CreateEmptyWasmPsbtOptions {
|
|
|
41
41
|
* @param options - Optional settings (e.g., blockHeight for Zcash)
|
|
42
42
|
* @returns A wasm-utxo BitGoPsbt instance
|
|
43
43
|
*/
|
|
44
|
-
export declare function createEmptyWasmPsbt(
|
|
44
|
+
export declare function createEmptyWasmPsbt(coinName: CoinName, rootWalletKeys: RootWalletKeys, options?: CreateEmptyWasmPsbtOptions): fixedScriptWallet.BitGoPsbt;
|
|
45
45
|
/**
|
|
46
46
|
* Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.
|
|
47
47
|
* Handles taproot inputs by setting the appropriate signPath.
|
|
@@ -60,7 +60,7 @@ export declare function addWalletInputsToWasmPsbt(wasmPsbt: fixedScriptWallet.Bi
|
|
|
60
60
|
* @param network - The network (used to convert address to script)
|
|
61
61
|
* @returns The output index
|
|
62
62
|
*/
|
|
63
|
-
export declare function addOutputToWasmPsbt(wasmPsbt: fixedScriptWallet.BitGoPsbt, address: string, value: bigint,
|
|
63
|
+
export declare function addOutputToWasmPsbt(wasmPsbt: fixedScriptWallet.BitGoPsbt, address: string, value: bigint, coinName: CoinName): number;
|
|
64
64
|
/**
|
|
65
65
|
* Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.
|
|
66
66
|
*
|
|
@@ -68,7 +68,7 @@ export declare function addOutputToWasmPsbt(wasmPsbt: fixedScriptWallet.BitGoPsb
|
|
|
68
68
|
* @param network - The network
|
|
69
69
|
* @returns A utxolib UtxoPsbt
|
|
70
70
|
*/
|
|
71
|
-
export declare function
|
|
71
|
+
export declare function toPsbtToUtxolibPsbt(wasmPsbt: fixedScriptWallet.BitGoPsbt | utxolib.bitgo.UtxoPsbt, coinName: CoinName): utxolib.bitgo.UtxoPsbt;
|
|
72
72
|
/**
|
|
73
73
|
* Create a backup key recovery PSBT.
|
|
74
74
|
*
|
|
@@ -78,7 +78,7 @@ export declare function wasmPsbtToUtxolibPsbt(wasmPsbt: fixedScriptWallet.BitGoP
|
|
|
78
78
|
* @param options - Options for creating the PSBT
|
|
79
79
|
* @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')
|
|
80
80
|
*/
|
|
81
|
-
export declare function createBackupKeyRecoveryPsbt(
|
|
82
|
-
export declare function getRecoveryAmount(psbt: utxolib.bitgo.UtxoPsbt, address: string): bigint;
|
|
81
|
+
export declare function createBackupKeyRecoveryPsbt(coinName: CoinName, rootWalletKeys: RootWalletKeys, unspents: WalletUnspent<bigint>[], options: CreateBackupKeyRecoveryPsbtOptions, backend?: PsbtBackend): utxolib.bitgo.UtxoPsbt | fixedScriptWallet.BitGoPsbt;
|
|
82
|
+
export declare function getRecoveryAmount(psbt: utxolib.bitgo.UtxoPsbt | fixedScriptWallet.BitGoPsbt, walletKeys: RootWalletKeys, address: string): bigint;
|
|
83
83
|
export {};
|
|
84
84
|
//# sourceMappingURL=psbt.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"psbt.d.ts","sourceRoot":"","sources":["../../../../src/recovery/psbt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EAAE,iBAAiB,EAAE,aAAa,
|
|
1
|
+
{"version":3,"file":"psbt.d.ts","sourceRoot":"","sources":["../../../../src/recovery/psbt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EAAE,QAAQ,EAAE,iBAAiB,EAAE,aAAa,EAA0B,MAAM,kBAAkB,CAAC;AAItG,KAAK,cAAc,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC;AACnD,KAAK,aAAa,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,IAAI,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;AAI3F,KAAK,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC;AAEzC;;;;GAIG;AACH,MAAM,MAAM,WAAW,GAAG,WAAW,GAAG,SAAS,CAAC;AAElD;;GAEG;AACH,wBAAgB,cAAc,CAAC,KAAK,EAAE,SAAS,GAAG,OAAO,CAIxD;AAED;;GAEG;AACH,wBAAgB,aAAa,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,aAAa,CAAC,WAAW,CAMjF;AAmBD,UAAU,kCAAkC;IAC1C,YAAY,EAAE,MAAM,CAAC;IACrB,mBAAmB,EAAE,MAAM,CAAC;IAC5B,qBAAqB,EAAE,MAAM,CAAC;IAC9B,4BAA4B,EAAE,MAAM,GAAG,SAAS,CAAC;IACjD,kFAAkF;IAClF,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAqED;;GAEG;AACH,MAAM,WAAW,0BAA0B;IACzC,kFAAkF;IAClF,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED;;;;;;;;GAQG;AACH,wBAAgB,mBAAmB,CACjC,QAAQ,EAAE,QAAQ,EAClB,cAAc,EAAE,cAAc,EAC9B,OAAO,CAAC,EAAE,0BAA0B,GACnC,iBAAiB,CAAC,SAAS,CAU7B;AAED;;;;;;;GAOG;AACH,wBAAgB,yBAAyB,CACvC,QAAQ,EAAE,iBAAiB,CAAC,SAAS,EACrC,QAAQ,EAAE,aAAa,CAAC,MAAM,CAAC,EAAE,EACjC,cAAc,EAAE,cAAc,GAC7B,IAAI,CAwBN;AAED;;;;;;;;GAQG;AACH,wBAAgB,mBAAmB,CACjC,QAAQ,EAAE,iBAAiB,CAAC,SAAS,EACrC,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,MAAM,EACb,QAAQ,EAAE,QAAQ,GACjB,MAAM,CAGR;AAED;;;;;;GAMG;AACH,wBAAgB,mBAAmB,CACjC,QAAQ,EAAE,iBAAiB,CAAC,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC9D,QAAQ,EAAE,QAAQ,GACjB,OAAO,CAAC,KAAK,CAAC,QAAQ,CAMxB;AA6CD;;;;;;;;GAQG;AACH,wBAAgB,2BAA2B,CACzC,QAAQ,EAAE,QAAQ,EAClB,cAAc,EAAE,cAAc,EAC9B,QAAQ,EAAE,aAAa,CAAC,MAAM,CAAC,EAAE,EACjC,OAAO,EAAE,kCAAkC,EAC3C,OAAO,GAAE,WAAyB,GACjC,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,iBAAiB,CAAC,SAAS,CAUtD;AAED,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,iBAAiB,CAAC,SAAS,EAC1D,UAAU,EAAE,cAAc,EAC1B,OAAO,EAAE,MAAM,GACd,MAAM,CAkBR"}
|
|
@@ -38,12 +38,13 @@ exports.toNetworkName = toNetworkName;
|
|
|
38
38
|
exports.createEmptyWasmPsbt = createEmptyWasmPsbt;
|
|
39
39
|
exports.addWalletInputsToWasmPsbt = addWalletInputsToWasmPsbt;
|
|
40
40
|
exports.addOutputToWasmPsbt = addOutputToWasmPsbt;
|
|
41
|
-
exports.
|
|
41
|
+
exports.toPsbtToUtxolibPsbt = toPsbtToUtxolibPsbt;
|
|
42
42
|
exports.createBackupKeyRecoveryPsbt = createBackupKeyRecoveryPsbt;
|
|
43
43
|
exports.getRecoveryAmount = getRecoveryAmount;
|
|
44
44
|
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
45
45
|
const unspents_1 = require("@bitgo-beta/unspents");
|
|
46
46
|
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
47
|
+
const names_1 = require("../names");
|
|
47
48
|
const { chainCodesP2tr, chainCodesP2trMusig2 } = utxolib.bitgo;
|
|
48
49
|
/**
|
|
49
50
|
* Check if a chain code is for a taproot script type
|
|
@@ -77,7 +78,8 @@ class InsufficientFundsError extends Error {
|
|
|
77
78
|
/**
|
|
78
79
|
* Create a backup key recovery PSBT using utxolib (legacy implementation)
|
|
79
80
|
*/
|
|
80
|
-
function createBackupKeyRecoveryPsbtUtxolib(
|
|
81
|
+
function createBackupKeyRecoveryPsbtUtxolib(coinName, rootWalletKeys, unspents, options) {
|
|
82
|
+
const network = (0, names_1.getNetworkFromCoinName)(coinName);
|
|
81
83
|
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
82
84
|
const psbt = utxolib.bitgo.createPsbtForNetwork({ network });
|
|
83
85
|
utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);
|
|
@@ -108,8 +110,8 @@ function createBackupKeyRecoveryPsbtUtxolib(network, rootWalletKeys, unspents, o
|
|
|
108
110
|
/**
|
|
109
111
|
* Check if the network is a Zcash network
|
|
110
112
|
*/
|
|
111
|
-
function
|
|
112
|
-
return
|
|
113
|
+
function isZcash(coinName) {
|
|
114
|
+
return coinName === 'zec' || coinName === 'tzec';
|
|
113
115
|
}
|
|
114
116
|
/**
|
|
115
117
|
* Default block heights for Zcash networks if not provided.
|
|
@@ -117,8 +119,8 @@ function isZcashNetwork(networkName) {
|
|
|
117
119
|
* TODO(BTC-2901): get the height from blockchair API instead of hardcoding.
|
|
118
120
|
*/
|
|
119
121
|
const ZCASH_DEFAULT_BLOCK_HEIGHTS = {
|
|
120
|
-
|
|
121
|
-
|
|
122
|
+
zec: 3146400,
|
|
123
|
+
tzec: 3536500,
|
|
122
124
|
};
|
|
123
125
|
/**
|
|
124
126
|
* Create an empty wasm-utxo BitGoPsbt for a given network.
|
|
@@ -129,16 +131,15 @@ const ZCASH_DEFAULT_BLOCK_HEIGHTS = {
|
|
|
129
131
|
* @param options - Optional settings (e.g., blockHeight for Zcash)
|
|
130
132
|
* @returns A wasm-utxo BitGoPsbt instance
|
|
131
133
|
*/
|
|
132
|
-
function createEmptyWasmPsbt(
|
|
133
|
-
|
|
134
|
-
if (isZcashNetwork(networkName)) {
|
|
134
|
+
function createEmptyWasmPsbt(coinName, rootWalletKeys, options) {
|
|
135
|
+
if (isZcash(coinName)) {
|
|
135
136
|
// For Zcash, use ZcashBitGoPsbt which requires block height to determine consensus branch ID
|
|
136
|
-
const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[
|
|
137
|
-
return wasm_utxo_1.fixedScriptWallet.ZcashBitGoPsbt.createEmpty(
|
|
137
|
+
const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[coinName];
|
|
138
|
+
return wasm_utxo_1.fixedScriptWallet.ZcashBitGoPsbt.createEmpty(coinName, rootWalletKeys, {
|
|
138
139
|
blockHeight,
|
|
139
140
|
});
|
|
140
141
|
}
|
|
141
|
-
return wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty(
|
|
142
|
+
return wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty(coinName, rootWalletKeys);
|
|
142
143
|
}
|
|
143
144
|
/**
|
|
144
145
|
* Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.
|
|
@@ -176,9 +177,9 @@ function addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys) {
|
|
|
176
177
|
* @param network - The network (used to convert address to script)
|
|
177
178
|
* @returns The output index
|
|
178
179
|
*/
|
|
179
|
-
function addOutputToWasmPsbt(wasmPsbt, address, value,
|
|
180
|
-
const script =
|
|
181
|
-
return wasmPsbt.addOutput({ script
|
|
180
|
+
function addOutputToWasmPsbt(wasmPsbt, address, value, coinName) {
|
|
181
|
+
const script = wasm_utxo_1.address.toOutputScriptWithCoin(address, coinName);
|
|
182
|
+
return wasmPsbt.addOutput({ script, value });
|
|
182
183
|
}
|
|
183
184
|
/**
|
|
184
185
|
* Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.
|
|
@@ -187,23 +188,25 @@ function addOutputToWasmPsbt(wasmPsbt, address, value, network) {
|
|
|
187
188
|
* @param network - The network
|
|
188
189
|
* @returns A utxolib UtxoPsbt
|
|
189
190
|
*/
|
|
190
|
-
function
|
|
191
|
-
|
|
191
|
+
function toPsbtToUtxolibPsbt(wasmPsbt, coinName) {
|
|
192
|
+
if (wasmPsbt instanceof wasm_utxo_1.fixedScriptWallet.BitGoPsbt) {
|
|
193
|
+
const network = (0, names_1.getNetworkFromCoinName)(coinName);
|
|
194
|
+
return utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);
|
|
195
|
+
}
|
|
196
|
+
return wasmPsbt;
|
|
192
197
|
}
|
|
193
198
|
/**
|
|
194
199
|
* Create a backup key recovery PSBT using wasm-utxo
|
|
195
200
|
*/
|
|
196
|
-
function createBackupKeyRecoveryPsbtWasm(
|
|
201
|
+
function createBackupKeyRecoveryPsbtWasm(coinName, rootWalletKeys, unspents, options) {
|
|
197
202
|
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
198
203
|
// Create PSBT with wasm-utxo and add wallet inputs using shared utilities
|
|
199
|
-
const wasmPsbt = createEmptyWasmPsbt(
|
|
204
|
+
const wasmPsbt = createEmptyWasmPsbt(coinName, rootWalletKeys, { blockHeight: options.blockHeight });
|
|
200
205
|
addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys);
|
|
201
206
|
// Calculate dimensions using wasm-utxo Dimensions
|
|
202
|
-
|
|
203
|
-
let dimensions = wasm_utxo_1.fixedScriptWallet.Dimensions.fromPsbt(wasmPsbt).plus(wasm_utxo_1.fixedScriptWallet.Dimensions.fromOutput(new Uint8Array(recoveryOutputScript)));
|
|
207
|
+
let dimensions = wasm_utxo_1.fixedScriptWallet.Dimensions.fromPsbt(wasmPsbt).plus(wasm_utxo_1.fixedScriptWallet.Dimensions.fromOutput(recoveryDestination, coinName));
|
|
204
208
|
if (keyRecoveryServiceFeeAddress) {
|
|
205
|
-
|
|
206
|
-
dimensions = dimensions.plus(wasm_utxo_1.fixedScriptWallet.Dimensions.fromOutput(new Uint8Array(krsOutputScript)));
|
|
209
|
+
dimensions = dimensions.plus(wasm_utxo_1.fixedScriptWallet.Dimensions.fromOutput(keyRecoveryServiceFeeAddress, coinName));
|
|
207
210
|
}
|
|
208
211
|
const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);
|
|
209
212
|
const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');
|
|
@@ -212,12 +215,12 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
212
215
|
throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);
|
|
213
216
|
}
|
|
214
217
|
// Add outputs to wasm PSBT
|
|
215
|
-
addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount,
|
|
218
|
+
addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount, coinName);
|
|
216
219
|
if (keyRecoveryServiceFeeAddress) {
|
|
217
|
-
addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee,
|
|
220
|
+
addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee, coinName);
|
|
218
221
|
}
|
|
219
222
|
// Convert to utxolib PSBT for signing and return
|
|
220
|
-
return
|
|
223
|
+
return wasmPsbt;
|
|
221
224
|
}
|
|
222
225
|
/**
|
|
223
226
|
* Create a backup key recovery PSBT.
|
|
@@ -228,23 +231,34 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
228
231
|
* @param options - Options for creating the PSBT
|
|
229
232
|
* @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')
|
|
230
233
|
*/
|
|
231
|
-
function createBackupKeyRecoveryPsbt(
|
|
234
|
+
function createBackupKeyRecoveryPsbt(coinName, rootWalletKeys, unspents, options, backend = 'wasm-utxo') {
|
|
232
235
|
if (options.keyRecoveryServiceFee > 0 && !options.keyRecoveryServiceFeeAddress) {
|
|
233
236
|
throw new Error('keyRecoveryServiceFeeAddress is required when keyRecoveryServiceFee is provided');
|
|
234
237
|
}
|
|
235
238
|
if (backend === 'wasm-utxo') {
|
|
236
|
-
return createBackupKeyRecoveryPsbtWasm(
|
|
239
|
+
return createBackupKeyRecoveryPsbtWasm(coinName, rootWalletKeys, unspents, options);
|
|
237
240
|
}
|
|
238
241
|
else {
|
|
239
|
-
return createBackupKeyRecoveryPsbtUtxolib(
|
|
242
|
+
return createBackupKeyRecoveryPsbtUtxolib(coinName, rootWalletKeys, unspents, options);
|
|
240
243
|
}
|
|
241
244
|
}
|
|
242
|
-
function getRecoveryAmount(psbt, address) {
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
245
|
+
function getRecoveryAmount(psbt, walletKeys, address) {
|
|
246
|
+
if (psbt instanceof utxolib.bitgo.UtxoPsbt) {
|
|
247
|
+
const recoveryOutputScript = utxolib.address.toOutputScript(address, psbt.network);
|
|
248
|
+
const output = psbt.txOutputs.find((o) => o.script.equals(recoveryOutputScript));
|
|
249
|
+
if (!output) {
|
|
250
|
+
throw new Error(`Recovery destination output not found in PSBT: ${address}`);
|
|
251
|
+
}
|
|
252
|
+
return output.value;
|
|
253
|
+
}
|
|
254
|
+
if (psbt instanceof wasm_utxo_1.fixedScriptWallet.BitGoPsbt) {
|
|
255
|
+
const parsedOutputs = psbt.parseOutputsWithWalletKeys(walletKeys);
|
|
256
|
+
const recoveryOutput = parsedOutputs.find((o) => o.address === address);
|
|
257
|
+
if (!recoveryOutput) {
|
|
258
|
+
throw new Error(`Recovery destination output not found in PSBT: ${address}`);
|
|
259
|
+
}
|
|
260
|
+
return recoveryOutput.value;
|
|
247
261
|
}
|
|
248
|
-
|
|
262
|
+
throw new Error('Invalid PSBT type');
|
|
249
263
|
}
|
|
250
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../../../src/recovery/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Dimensions } from '@bitgo-beta/unspents';\nimport { fixedScriptWallet, utxolibCompat } from '@bitgo/wasm-utxo';\n\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype WalletUnspent<TNumber extends number | bigint> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nconst { chainCodesP2tr, chainCodesP2trMusig2 } = utxolib.bitgo;\n\ntype ChainCode = utxolib.bitgo.ChainCode;\n\n/**\n * Backend to use for PSBT creation.\n * - 'wasm-utxo': Use wasm-utxo for PSBT creation (default)\n * - 'utxolib': Use utxolib for PSBT creation (legacy)\n */\nexport type PsbtBackend = 'wasm-utxo' | 'utxolib';\n\n/**\n * Check if a chain code is for a taproot script type\n */\nexport function isTaprootChain(chain: ChainCode): boolean {\n  return (\n    (chainCodesP2tr as readonly number[]).includes(chain) || (chainCodesP2trMusig2 as readonly number[]).includes(chain)\n  );\n}\n\n/**\n * Convert utxolib Network to wasm-utxo network name\n */\nexport function toNetworkName(network: utxolib.Network): utxolibCompat.UtxolibName {\n  const networkName = utxolib.getNetworkName(network);\n  if (!networkName) {\n    throw new Error(`Invalid network`);\n  }\n  return networkName;\n}\n\nclass InsufficientFundsError extends Error {\n  constructor(\n    public totalInputAmount: bigint,\n    public approximateFee: bigint,\n    public krsFee: bigint,\n    public recoveryAmount: bigint\n  ) {\n    super(\n      `This wallet's balance is too low to pay the fees specified by the KRS provider.` +\n        `Existing balance on wallet: ${totalInputAmount.toString()}. ` +\n        `Estimated network fee for the recovery transaction: ${approximateFee.toString()}` +\n        `KRS fee to pay: ${krsFee.toString()}. ` +\n        `After deducting fees, your total recoverable balance is ${recoveryAmount.toString()}`\n    );\n  }\n}\n\ninterface CreateBackupKeyRecoveryPsbtOptions {\n  feeRateSatVB: number;\n  recoveryDestination: string;\n  keyRecoveryServiceFee: bigint;\n  keyRecoveryServiceFeeAddress: string | undefined;\n  /** Block height for Zcash networks (required to determine consensus branch ID) */\n  blockHeight?: number;\n}\n\n/**\n * Create a backup key recovery PSBT using utxolib (legacy implementation)\n */\nfunction createBackupKeyRecoveryPsbtUtxolib(\n  network: utxolib.Network,\n  rootWalletKeys: RootWalletKeys,\n  unspents: WalletUnspent<bigint>[],\n  options: CreateBackupKeyRecoveryPsbtOptions\n): utxolib.bitgo.UtxoPsbt {\n  const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;\n\n  const psbt = utxolib.bitgo.createPsbtForNetwork({ network });\n  utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);\n  unspents.forEach((unspent) => {\n    utxolib.bitgo.addWalletUnspentToPsbt(psbt, unspent, rootWalletKeys, 'user', 'backup');\n  });\n\n  let dimensions = Dimensions.fromPsbt(psbt).plus(\n    Dimensions.fromOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network) })\n  );\n\n  if (keyRecoveryServiceFeeAddress) {\n    dimensions = dimensions.plus(\n      Dimensions.fromOutput({\n        script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),\n      })\n    );\n  }\n\n  const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);\n  const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');\n  const recoveryAmount = totalInputAmount - approximateFee - keyRecoveryServiceFee;\n\n  if (recoveryAmount < BigInt(0)) {\n    throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);\n  }\n\n  psbt.addOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network), value: recoveryAmount });\n\n  if (keyRecoveryServiceFeeAddress) {\n    psbt.addOutput({\n      script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),\n      value: keyRecoveryServiceFee,\n    });\n  }\n\n  return psbt;\n}\n\n/**\n * Check if the network is a Zcash network\n */\nfunction isZcashNetwork(networkName: utxolibCompat.UtxolibName): boolean {\n  return networkName === 'zcash' || networkName === 'zcashTest';\n}\n\n/**\n * Default block heights for Zcash networks if not provided.\n * These should be set to a height after the latest network upgrade.\n * TODO(BTC-2901): get the height from blockchair API instead of hardcoding.\n */\nconst ZCASH_DEFAULT_BLOCK_HEIGHTS: Record<string, number> = {\n  zcash: 3146400,\n  zcashTest: 3536500,\n};\n\n/**\n * Options for creating an empty wasm-utxo PSBT\n */\nexport interface CreateEmptyWasmPsbtOptions {\n  /** Block height for Zcash networks (required to determine consensus branch ID) */\n  blockHeight?: number;\n}\n\n/**\n * Create an empty wasm-utxo BitGoPsbt for a given network.\n * Handles Zcash networks specially by using ZcashBitGoPsbt.\n *\n * @param network - The network for the PSBT\n * @param rootWalletKeys - The wallet keys\n * @param options - Optional settings (e.g., blockHeight for Zcash)\n * @returns A wasm-utxo BitGoPsbt instance\n */\nexport function createEmptyWasmPsbt(\n  network: utxolib.Network,\n  rootWalletKeys: RootWalletKeys,\n  options?: CreateEmptyWasmPsbtOptions\n): fixedScriptWallet.BitGoPsbt {\n  const networkName = toNetworkName(network);\n\n  if (isZcashNetwork(networkName)) {\n    // For Zcash, use ZcashBitGoPsbt which requires block height to determine consensus branch ID\n    const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[networkName];\n    return fixedScriptWallet.ZcashBitGoPsbt.createEmpty(networkName as 'zcash' | 'zcashTest', rootWalletKeys, {\n      blockHeight,\n    });\n  }\n\n  return fixedScriptWallet.BitGoPsbt.createEmpty(networkName, rootWalletKeys);\n}\n\n/**\n * Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.\n * Handles taproot inputs by setting the appropriate signPath.\n *\n * @param wasmPsbt - The wasm-utxo BitGoPsbt to add inputs to\n * @param unspents - The wallet unspents to add as inputs\n * @param rootWalletKeys - The wallet keys\n */\nexport function addWalletInputsToWasmPsbt(\n  wasmPsbt: fixedScriptWallet.BitGoPsbt,\n  unspents: WalletUnspent<bigint>[],\n  rootWalletKeys: RootWalletKeys\n): void {\n  unspents.forEach((unspent) => {\n    const { txid, vout } = utxolib.bitgo.parseOutputId(unspent.id);\n    const signPath: fixedScriptWallet.SignPath | undefined = isTaprootChain(unspent.chain)\n      ? { signer: 'user', cosigner: 'backup' }\n      : undefined;\n\n    // prevTx may be added dynamically in backupKeyRecovery for non-segwit inputs\n    const prevTx = (unspent as WalletUnspent<bigint> & { prevTx?: Buffer }).prevTx;\n\n    wasmPsbt.addWalletInput(\n      {\n        txid,\n        vout,\n        value: unspent.value,\n        prevTx: prevTx,\n      },\n      rootWalletKeys,\n      {\n        scriptId: { chain: unspent.chain, index: unspent.index },\n        signPath,\n      }\n    );\n  });\n}\n\n/**\n * Add an output to a wasm-utxo BitGoPsbt.\n *\n * @param wasmPsbt - The wasm-utxo BitGoPsbt to add the output to\n * @param address - The destination address\n * @param value - The output value in satoshis\n * @param network - The network (used to convert address to script)\n * @returns The output index\n */\nexport function addOutputToWasmPsbt(\n  wasmPsbt: fixedScriptWallet.BitGoPsbt,\n  address: string,\n  value: bigint,\n  network: utxolib.Network\n): number {\n  const script = utxolib.address.toOutputScript(address, network);\n  return wasmPsbt.addOutput({ script: new Uint8Array(script), value });\n}\n\n/**\n * Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.\n *\n * @param wasmPsbt - The wasm-utxo BitGoPsbt to convert\n * @param network - The network\n * @returns A utxolib UtxoPsbt\n */\nexport function wasmPsbtToUtxolibPsbt(\n  wasmPsbt: fixedScriptWallet.BitGoPsbt,\n  network: utxolib.Network\n): utxolib.bitgo.UtxoPsbt {\n  return utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);\n}\n\n/**\n * Create a backup key recovery PSBT using wasm-utxo\n */\nfunction createBackupKeyRecoveryPsbtWasm(\n  network: utxolib.Network,\n  rootWalletKeys: RootWalletKeys,\n  unspents: WalletUnspent<bigint>[],\n  options: CreateBackupKeyRecoveryPsbtOptions\n): utxolib.bitgo.UtxoPsbt {\n  const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;\n\n  // Create PSBT with wasm-utxo and add wallet inputs using shared utilities\n  const wasmPsbt = createEmptyWasmPsbt(network, rootWalletKeys, { blockHeight: options.blockHeight });\n  addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys);\n\n  // Calculate dimensions using wasm-utxo Dimensions\n  const recoveryOutputScript = utxolib.address.toOutputScript(recoveryDestination, network);\n  let dimensions = fixedScriptWallet.Dimensions.fromPsbt(wasmPsbt).plus(\n    fixedScriptWallet.Dimensions.fromOutput(new Uint8Array(recoveryOutputScript))\n  );\n\n  if (keyRecoveryServiceFeeAddress) {\n    const krsOutputScript = utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network);\n    dimensions = dimensions.plus(fixedScriptWallet.Dimensions.fromOutput(new Uint8Array(krsOutputScript)));\n  }\n\n  const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);\n  const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');\n  const recoveryAmount = totalInputAmount - approximateFee - keyRecoveryServiceFee;\n\n  if (recoveryAmount < BigInt(0)) {\n    throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);\n  }\n\n  // Add outputs to wasm PSBT\n  addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount, network);\n\n  if (keyRecoveryServiceFeeAddress) {\n    addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee, network);\n  }\n\n  // Convert to utxolib PSBT for signing and return\n  return wasmPsbtToUtxolibPsbt(wasmPsbt, network);\n}\n\n/**\n * Create a backup key recovery PSBT.\n *\n * @param network - The network for the PSBT\n * @param rootWalletKeys - The wallet keys\n * @param unspents - The unspents to include in the PSBT\n * @param options - Options for creating the PSBT\n * @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')\n */\nexport function createBackupKeyRecoveryPsbt(\n  network: utxolib.Network,\n  rootWalletKeys: RootWalletKeys,\n  unspents: WalletUnspent<bigint>[],\n  options: CreateBackupKeyRecoveryPsbtOptions,\n  backend: PsbtBackend = 'wasm-utxo'\n): utxolib.bitgo.UtxoPsbt {\n  if (options.keyRecoveryServiceFee > 0 && !options.keyRecoveryServiceFeeAddress) {\n    throw new Error('keyRecoveryServiceFeeAddress is required when keyRecoveryServiceFee is provided');\n  }\n\n  if (backend === 'wasm-utxo') {\n    return createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, options);\n  } else {\n    return createBackupKeyRecoveryPsbtUtxolib(network, rootWalletKeys, unspents, options);\n  }\n}\n\nexport function getRecoveryAmount(psbt: utxolib.bitgo.UtxoPsbt, address: string): bigint {\n  const recoveryOutputScript = utxolib.address.toOutputScript(address, psbt.network);\n  const output = psbt.txOutputs.find((o) => o.script.equals(recoveryOutputScript));\n  if (!output) {\n    throw new Error(`Recovery destination output not found in PSBT: ${address}`);\n  }\n  return output.value;\n}\n"]}
|
|
264
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../../../src/recovery/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Dimensions } from '@bitgo-beta/unspents';\nimport { CoinName, fixedScriptWallet, utxolibCompat, address as wasmAddress } from '@bitgo/wasm-utxo';\n\nimport { getNetworkFromCoinName } from '../names';\n\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype WalletUnspent<TNumber extends number | bigint> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nconst { chainCodesP2tr, chainCodesP2trMusig2 } = utxolib.bitgo;\n\ntype ChainCode = utxolib.bitgo.ChainCode;\n\n/**\n * Backend to use for PSBT creation.\n * - 'wasm-utxo': Use wasm-utxo for PSBT creation (default)\n * - 'utxolib': Use utxolib for PSBT creation (legacy)\n */\nexport type PsbtBackend = 'wasm-utxo' | 'utxolib';\n\n/**\n * Check if a chain code is for a taproot script type\n */\nexport function isTaprootChain(chain: ChainCode): boolean {\n  return (\n    (chainCodesP2tr as readonly number[]).includes(chain) || (chainCodesP2trMusig2 as readonly number[]).includes(chain)\n  );\n}\n\n/**\n * Convert utxolib Network to wasm-utxo network name\n */\nexport function toNetworkName(network: utxolib.Network): utxolibCompat.UtxolibName {\n  const networkName = utxolib.getNetworkName(network);\n  if (!networkName) {\n    throw new Error(`Invalid network`);\n  }\n  return networkName;\n}\n\nclass InsufficientFundsError extends Error {\n  constructor(\n    public totalInputAmount: bigint,\n    public approximateFee: bigint,\n    public krsFee: bigint,\n    public recoveryAmount: bigint\n  ) {\n    super(\n      `This wallet's balance is too low to pay the fees specified by the KRS provider.` +\n        `Existing balance on wallet: ${totalInputAmount.toString()}. ` +\n        `Estimated network fee for the recovery transaction: ${approximateFee.toString()}` +\n        `KRS fee to pay: ${krsFee.toString()}. ` +\n        `After deducting fees, your total recoverable balance is ${recoveryAmount.toString()}`\n    );\n  }\n}\n\ninterface CreateBackupKeyRecoveryPsbtOptions {\n  feeRateSatVB: number;\n  recoveryDestination: string;\n  keyRecoveryServiceFee: bigint;\n  keyRecoveryServiceFeeAddress: string | undefined;\n  /** Block height for Zcash networks (required to determine consensus branch ID) */\n  blockHeight?: number;\n}\n\n/**\n * Create a backup key recovery PSBT using utxolib (legacy implementation)\n */\nfunction createBackupKeyRecoveryPsbtUtxolib(\n  coinName: CoinName,\n  rootWalletKeys: RootWalletKeys,\n  unspents: WalletUnspent<bigint>[],\n  options: CreateBackupKeyRecoveryPsbtOptions\n): utxolib.bitgo.UtxoPsbt {\n  const network = getNetworkFromCoinName(coinName);\n  const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;\n\n  const psbt = utxolib.bitgo.createPsbtForNetwork({ network });\n  utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);\n  unspents.forEach((unspent) => {\n    utxolib.bitgo.addWalletUnspentToPsbt(psbt, unspent, rootWalletKeys, 'user', 'backup');\n  });\n\n  let dimensions = Dimensions.fromPsbt(psbt).plus(\n    Dimensions.fromOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network) })\n  );\n\n  if (keyRecoveryServiceFeeAddress) {\n    dimensions = dimensions.plus(\n      Dimensions.fromOutput({\n        script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),\n      })\n    );\n  }\n\n  const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);\n  const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');\n  const recoveryAmount = totalInputAmount - approximateFee - keyRecoveryServiceFee;\n\n  if (recoveryAmount < BigInt(0)) {\n    throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);\n  }\n\n  psbt.addOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network), value: recoveryAmount });\n\n  if (keyRecoveryServiceFeeAddress) {\n    psbt.addOutput({\n      script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),\n      value: keyRecoveryServiceFee,\n    });\n  }\n\n  return psbt;\n}\n\n/**\n * Check if the network is a Zcash network\n */\nfunction isZcash(coinName: CoinName): coinName is 'zec' | 'tzec' {\n  return coinName === 'zec' || coinName === 'tzec';\n}\n\n/**\n * Default block heights for Zcash networks if not provided.\n * These should be set to a height after the latest network upgrade.\n * TODO(BTC-2901): get the height from blockchair API instead of hardcoding.\n */\nconst ZCASH_DEFAULT_BLOCK_HEIGHTS: Record<'zec' | 'tzec', number> = {\n  zec: 3146400,\n  tzec: 3536500,\n};\n\n/**\n * Options for creating an empty wasm-utxo PSBT\n */\nexport interface CreateEmptyWasmPsbtOptions {\n  /** Block height for Zcash networks (required to determine consensus branch ID) */\n  blockHeight?: number;\n}\n\n/**\n * Create an empty wasm-utxo BitGoPsbt for a given network.\n * Handles Zcash networks specially by using ZcashBitGoPsbt.\n *\n * @param network - The network for the PSBT\n * @param rootWalletKeys - The wallet keys\n * @param options - Optional settings (e.g., blockHeight for Zcash)\n * @returns A wasm-utxo BitGoPsbt instance\n */\nexport function createEmptyWasmPsbt(\n  coinName: CoinName,\n  rootWalletKeys: RootWalletKeys,\n  options?: CreateEmptyWasmPsbtOptions\n): fixedScriptWallet.BitGoPsbt {\n  if (isZcash(coinName)) {\n    // For Zcash, use ZcashBitGoPsbt which requires block height to determine consensus branch ID\n    const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[coinName];\n    return fixedScriptWallet.ZcashBitGoPsbt.createEmpty(coinName, rootWalletKeys, {\n      blockHeight,\n    });\n  }\n\n  return fixedScriptWallet.BitGoPsbt.createEmpty(coinName, rootWalletKeys);\n}\n\n/**\n * Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.\n * Handles taproot inputs by setting the appropriate signPath.\n *\n * @param wasmPsbt - The wasm-utxo BitGoPsbt to add inputs to\n * @param unspents - The wallet unspents to add as inputs\n * @param rootWalletKeys - The wallet keys\n */\nexport function addWalletInputsToWasmPsbt(\n  wasmPsbt: fixedScriptWallet.BitGoPsbt,\n  unspents: WalletUnspent<bigint>[],\n  rootWalletKeys: RootWalletKeys\n): void {\n  unspents.forEach((unspent) => {\n    const { txid, vout } = utxolib.bitgo.parseOutputId(unspent.id);\n    const signPath: fixedScriptWallet.SignPath | undefined = isTaprootChain(unspent.chain)\n      ? { signer: 'user', cosigner: 'backup' }\n      : undefined;\n\n    // prevTx may be added dynamically in backupKeyRecovery for non-segwit inputs\n    const prevTx = (unspent as WalletUnspent<bigint> & { prevTx?: Buffer }).prevTx;\n\n    wasmPsbt.addWalletInput(\n      {\n        txid,\n        vout,\n        value: unspent.value,\n        prevTx: prevTx,\n      },\n      rootWalletKeys,\n      {\n        scriptId: { chain: unspent.chain, index: unspent.index },\n        signPath,\n      }\n    );\n  });\n}\n\n/**\n * Add an output to a wasm-utxo BitGoPsbt.\n *\n * @param wasmPsbt - The wasm-utxo BitGoPsbt to add the output to\n * @param address - The destination address\n * @param value - The output value in satoshis\n * @param network - The network (used to convert address to script)\n * @returns The output index\n */\nexport function addOutputToWasmPsbt(\n  wasmPsbt: fixedScriptWallet.BitGoPsbt,\n  address: string,\n  value: bigint,\n  coinName: CoinName\n): number {\n  const script = wasmAddress.toOutputScriptWithCoin(address, coinName);\n  return wasmPsbt.addOutput({ script, value });\n}\n\n/**\n * Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.\n *\n * @param wasmPsbt - The wasm-utxo BitGoPsbt to convert\n * @param network - The network\n * @returns A utxolib UtxoPsbt\n */\nexport function toPsbtToUtxolibPsbt(\n  wasmPsbt: fixedScriptWallet.BitGoPsbt | utxolib.bitgo.UtxoPsbt,\n  coinName: CoinName\n): utxolib.bitgo.UtxoPsbt {\n  if (wasmPsbt instanceof fixedScriptWallet.BitGoPsbt) {\n    const network = getNetworkFromCoinName(coinName);\n    return utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);\n  }\n  return wasmPsbt;\n}\n\n/**\n * Create a backup key recovery PSBT using wasm-utxo\n */\nfunction createBackupKeyRecoveryPsbtWasm(\n  coinName: CoinName,\n  rootWalletKeys: RootWalletKeys,\n  unspents: WalletUnspent<bigint>[],\n  options: CreateBackupKeyRecoveryPsbtOptions\n): fixedScriptWallet.BitGoPsbt {\n  const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;\n\n  // Create PSBT with wasm-utxo and add wallet inputs using shared utilities\n  const wasmPsbt = createEmptyWasmPsbt(coinName, rootWalletKeys, { blockHeight: options.blockHeight });\n  addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys);\n\n  // Calculate dimensions using wasm-utxo Dimensions\n  let dimensions = fixedScriptWallet.Dimensions.fromPsbt(wasmPsbt).plus(\n    fixedScriptWallet.Dimensions.fromOutput(recoveryDestination, coinName)\n  );\n\n  if (keyRecoveryServiceFeeAddress) {\n    dimensions = dimensions.plus(fixedScriptWallet.Dimensions.fromOutput(keyRecoveryServiceFeeAddress, coinName));\n  }\n\n  const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);\n  const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');\n  const recoveryAmount = totalInputAmount - approximateFee - keyRecoveryServiceFee;\n\n  if (recoveryAmount < BigInt(0)) {\n    throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);\n  }\n\n  // Add outputs to wasm PSBT\n  addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount, coinName);\n\n  if (keyRecoveryServiceFeeAddress) {\n    addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee, coinName);\n  }\n\n  // Convert to utxolib PSBT for signing and return\n  return wasmPsbt;\n}\n\n/**\n * Create a backup key recovery PSBT.\n *\n * @param network - The network for the PSBT\n * @param rootWalletKeys - The wallet keys\n * @param unspents - The unspents to include in the PSBT\n * @param options - Options for creating the PSBT\n * @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')\n */\nexport function createBackupKeyRecoveryPsbt(\n  coinName: CoinName,\n  rootWalletKeys: RootWalletKeys,\n  unspents: WalletUnspent<bigint>[],\n  options: CreateBackupKeyRecoveryPsbtOptions,\n  backend: PsbtBackend = 'wasm-utxo'\n): utxolib.bitgo.UtxoPsbt | fixedScriptWallet.BitGoPsbt {\n  if (options.keyRecoveryServiceFee > 0 && !options.keyRecoveryServiceFeeAddress) {\n    throw new Error('keyRecoveryServiceFeeAddress is required when keyRecoveryServiceFee is provided');\n  }\n\n  if (backend === 'wasm-utxo') {\n    return createBackupKeyRecoveryPsbtWasm(coinName, rootWalletKeys, unspents, options);\n  } else {\n    return createBackupKeyRecoveryPsbtUtxolib(coinName, rootWalletKeys, unspents, options);\n  }\n}\n\nexport function getRecoveryAmount(\n  psbt: utxolib.bitgo.UtxoPsbt | fixedScriptWallet.BitGoPsbt,\n  walletKeys: RootWalletKeys,\n  address: string\n): bigint {\n  if (psbt instanceof utxolib.bitgo.UtxoPsbt) {\n    const recoveryOutputScript = utxolib.address.toOutputScript(address, psbt.network);\n    const output = psbt.txOutputs.find((o) => o.script.equals(recoveryOutputScript));\n    if (!output) {\n      throw new Error(`Recovery destination output not found in PSBT: ${address}`);\n    }\n    return output.value;\n  }\n  if (psbt instanceof fixedScriptWallet.BitGoPsbt) {\n    const parsedOutputs = psbt.parseOutputsWithWalletKeys(walletKeys);\n    const recoveryOutput = parsedOutputs.find((o) => o.address === address);\n    if (!recoveryOutput) {\n      throw new Error(`Recovery destination output not found in PSBT: ${address}`);\n    }\n    return recoveryOutput.value;\n  }\n  throw new Error('Invalid PSBT type');\n}\n"]}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
export declare function toTNumber(value: number | bigint, amountType: 'number'): number;
|
|
2
|
+
export declare function toTNumber(value: number | bigint, amountType: 'bigint'): bigint;
|
|
3
|
+
export declare function toTNumber(value: number | bigint, amountType: 'number' | 'bigint'): number | bigint;
|
|
4
|
+
//# sourceMappingURL=tnumber.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tnumber.d.ts","sourceRoot":"","sources":["../../../src/tnumber.ts"],"names":[],"mappings":"AAAA,wBAAgB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,EAAE,UAAU,EAAE,QAAQ,GAAG,MAAM,CAAC;AAChF,wBAAgB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,EAAE,UAAU,EAAE,QAAQ,GAAG,MAAM,CAAC;AAChF,wBAAgB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,EAAE,UAAU,EAAE,QAAQ,GAAG,QAAQ,GAAG,MAAM,GAAG,MAAM,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.toTNumber = toTNumber;
|
|
4
|
+
function toTNumber(value, amountType) {
|
|
5
|
+
switch (amountType) {
|
|
6
|
+
case 'number':
|
|
7
|
+
return Number(value);
|
|
8
|
+
case 'bigint':
|
|
9
|
+
return BigInt(value);
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG51bWJlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy90bnVtYmVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBR0EsOEJBT0M7QUFQRCxTQUFnQixTQUFTLENBQUMsS0FBc0IsRUFBRSxVQUErQjtJQUMvRSxRQUFRLFVBQVUsRUFBRSxDQUFDO1FBQ25CLEtBQUssUUFBUTtZQUNYLE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3ZCLEtBQUssUUFBUTtZQUNYLE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3pCLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIHRvVE51bWJlcih2YWx1ZTogbnVtYmVyIHwgYmlnaW50LCBhbW91bnRUeXBlOiAnbnVtYmVyJyk6IG51bWJlcjtcbmV4cG9ydCBmdW5jdGlvbiB0b1ROdW1iZXIodmFsdWU6IG51bWJlciB8IGJpZ2ludCwgYW1vdW50VHlwZTogJ2JpZ2ludCcpOiBiaWdpbnQ7XG5leHBvcnQgZnVuY3Rpb24gdG9UTnVtYmVyKHZhbHVlOiBudW1iZXIgfCBiaWdpbnQsIGFtb3VudFR5cGU6ICdudW1iZXInIHwgJ2JpZ2ludCcpOiBudW1iZXIgfCBiaWdpbnQ7XG5leHBvcnQgZnVuY3Rpb24gdG9UTnVtYmVyKHZhbHVlOiBudW1iZXIgfCBiaWdpbnQsIGFtb3VudFR5cGU6ICdudW1iZXInIHwgJ2JpZ2ludCcpOiBudW1iZXIgfCBiaWdpbnQge1xuICBzd2l0Y2ggKGFtb3VudFR5cGUpIHtcbiAgICBjYXNlICdudW1iZXInOlxuICAgICAgcmV0dXJuIE51bWJlcih2YWx1ZSk7XG4gICAgY2FzZSAnYmlnaW50JzpcbiAgICAgIHJldHVybiBCaWdJbnQodmFsdWUpO1xuICB9XG59XG4iXX0=
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
-
import type { PsbtParsedScriptType } from './
|
|
2
|
+
import type { PsbtParsedScriptType } from './signPsbtUtxolib';
|
|
3
3
|
type Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;
|
|
4
4
|
export declare class InputSigningError<TNumber extends number | bigint = number> extends Error {
|
|
5
5
|
inputIndex: number;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"SigningError.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/SigningError.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"SigningError.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/SigningError.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AAE9D,KAAK,OAAO,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,IAAI,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;AAExF,qBAAa,iBAAiB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,KAAK;IAe3E,UAAU,EAAE,MAAM;IAClB,SAAS,EAAE,oBAAoB,GAAG,IAAI;IACtC,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE;IAC1C,MAAM,EAAE,KAAK,GAAG,MAAM;IAjB/B,MAAM,CAAC,qBAAqB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,oBAAoB,GAAG,IAAI,EAAE,qCAAqC;IAC7E,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,GACzC,iBAAiB,CAAC,OAAO,CAAC;gBAUpB,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,oBAAoB,GAAG,IAAI,EAAE,qCAAqC;IAC7E,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,EAC1C,MAAM,EAAE,KAAK,GAAG,MAAM;CAIhC;AAED,qBAAa,uBAAuB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,KAAK;gBAC9E,UAAU,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE,EAAE,WAAW,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE;CAMhG"}
|
|
@@ -23,4 +23,4 @@ class TransactionSigningError extends Error {
|
|
|
23
23
|
}
|
|
24
24
|
}
|
|
25
25
|
exports.TransactionSigningError = TransactionSigningError;
|
|
26
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2lnbmluZ0Vycm9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL3RyYW5zYWN0aW9uL2ZpeGVkU2NyaXB0L1NpZ25pbmdFcnJvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFNQSxNQUFhLGlCQUE0RCxTQUFRLEtBQUs7SUFDcEYsTUFBTSxDQUFDLHFCQUFxQixDQUMxQixVQUFrQixFQUNsQixTQUFzQyxFQUFFLHFDQUFxQztJQUM3RSxPQUEwQztRQUUxQyxPQUFPLElBQUksaUJBQWlCLENBQzFCLFVBQVUsRUFDVixTQUFTLEVBQ1QsT0FBTyxFQUNQLHVEQUF1RCxDQUN4RCxDQUFDO0lBQ0osQ0FBQztJQUVELFlBQ1MsVUFBa0IsRUFDbEIsU0FBc0MsRUFBRSxxQ0FBcUM7SUFDN0UsT0FBMEMsRUFDMUMsTUFBc0I7UUFFN0IsS0FBSyxDQUFDLDBCQUEwQixVQUFVLFVBQVUsU0FBUyxjQUFjLE9BQU8sQ0FBQyxFQUFFLEtBQUssTUFBTSxFQUFFLENBQUMsQ0FBQztRQUw3RixlQUFVLEdBQVYsVUFBVSxDQUFRO1FBQ2xCLGNBQVMsR0FBVCxTQUFTLENBQTZCO1FBQ3RDLFlBQU8sR0FBUCxPQUFPLENBQW1DO1FBQzFDLFdBQU0sR0FBTixNQUFNLENBQWdCO0lBRy9CLENBQUM7Q0FDRjtBQXRCRCw4Q0FzQkM7QUFFRCxNQUFhLHVCQUFrRSxTQUFRLEtBQUs7SUFDMUYsWUFBWSxVQUF3QyxFQUFFLFdBQXlDO1FBQzdGLEtBQUssQ0FDSCwyQkFBMkIsVUFBVSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSztZQUNsRCw2QkFBNkIsV0FBVyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsd0JBQXdCLENBQzdFLENBQUM7SUFDSixDQUFDO0NBQ0Y7QUFQRCwwREFPQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuXG5pbXBvcnQgdHlwZSB7IFBzYnRQYXJzZWRTY3JpcHRUeXBlIH0gZnJvbSAnLi9zaWduUHNidFV0eG9saWInO1xuXG50eXBlIFVuc3BlbnQ8VE51bWJlciBleHRlbmRzIG51bWJlciB8IGJpZ2ludCA9IG51bWJlcj4gPSB1dHhvbGliLmJpdGdvLlVuc3BlbnQ8VE51bWJlcj47XG5cbmV4cG9ydCBjbGFzcyBJbnB1dFNpZ25pbmdFcnJvcjxUTnVtYmVyIGV4dGVuZHMgbnVtYmVyIHwgYmlnaW50ID0gbnVtYmVyPiBleHRlbmRzIEVycm9yIHtcbiAgc3RhdGljIGV4cGVjdGVkV2FsbGV0VW5zcGVudDxUTnVtYmVyIGV4dGVuZHMgbnVtYmVyIHwgYmlnaW50PihcbiAgICBpbnB1dEluZGV4OiBudW1iZXIsXG4gICAgaW5wdXRUeXBlOiBQc2J0UGFyc2VkU2NyaXB0VHlwZSB8IG51bGwsIC8vIG51bGwgZm9yIGxlZ2FjeSB0cmFuc2FjdGlvbiBmb3JtYXRcbiAgICB1bnNwZW50OiBVbnNwZW50PFROdW1iZXI+IHwgeyBpZDogc3RyaW5nIH1cbiAgKTogSW5wdXRTaWduaW5nRXJyb3I8VE51bWJlcj4ge1xuICAgIHJldHVybiBuZXcgSW5wdXRTaWduaW5nRXJyb3IoXG4gICAgICBpbnB1dEluZGV4LFxuICAgICAgaW5wdXRUeXBlLFxuICAgICAgdW5zcGVudCxcbiAgICAgIGBub3QgYSB3YWxsZXQgdW5zcGVudCwgbm90IGEgcmVwbGF5IHByb3RlY3Rpb24gdW5zcGVudGBcbiAgICApO1xuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIGlucHV0SW5kZXg6IG51bWJlcixcbiAgICBwdWJsaWMgaW5wdXRUeXBlOiBQc2J0UGFyc2VkU2NyaXB0VHlwZSB8IG51bGwsIC8vIG51bGwgZm9yIGxlZ2FjeSB0cmFuc2FjdGlvbiBmb3JtYXRcbiAgICBwdWJsaWMgdW5zcGVudDogVW5zcGVudDxUTnVtYmVyPiB8IHsgaWQ6IHN0cmluZyB9LFxuICAgIHB1YmxpYyByZWFzb246IEVycm9yIHwgc3RyaW5nXG4gICkge1xuICAgIHN1cGVyKGBzaWduaW5nIGVycm9yIGF0IGlucHV0ICR7aW5wdXRJbmRleH06IHR5cGU9JHtpbnB1dFR5cGV9IHVuc3BlbnRJZD0ke3Vuc3BlbnQuaWR9OiAke3JlYXNvbn1gKTtcbiAgfVxufVxuXG5leHBvcnQgY2xhc3MgVHJhbnNhY3Rpb25TaWduaW5nRXJyb3I8VE51bWJlciBleHRlbmRzIG51bWJlciB8IGJpZ2ludCA9IG51bWJlcj4gZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHNpZ25FcnJvcnM6IElucHV0U2lnbmluZ0Vycm9yPFROdW1iZXI+W10sIHZlcmlmeUVycm9yOiBJbnB1dFNpZ25pbmdFcnJvcjxUTnVtYmVyPltdKSB7XG4gICAgc3VwZXIoXG4gICAgICBgc2lnbiBlcnJvcnMgYXQgaW5wdXRzOiBbJHtzaWduRXJyb3JzLmpvaW4oJywnKX1dLCBgICtcbiAgICAgICAgYHZlcmlmeSBlcnJvcnMgYXQgaW5wdXRzOiBbJHt2ZXJpZnlFcnJvci5qb2luKCcsJyl9XSwgc2VlIGxvZyBmb3IgZGV0YWlsc2BcbiAgICApO1xuICB9XG59XG4iXX0=
|
|
@@ -7,24 +7,15 @@ export type PsbtParsedScriptType = 'p2sh' | 'p2wsh' | 'p2shP2wsh' | 'p2shP2pk' |
|
|
|
7
7
|
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
8
8
|
* failed.
|
|
9
9
|
*
|
|
10
|
-
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
11
|
-
*
|
|
12
10
|
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
13
11
|
* using TransactionBuilder
|
|
14
12
|
*
|
|
15
13
|
* @param psbt
|
|
16
14
|
* @param signerKeychain
|
|
17
|
-
* @param isLastSignature
|
|
18
15
|
*/
|
|
19
|
-
export declare function signAndVerifyPsbt(psbt: utxolib.bitgo.UtxoPsbt, signerKeychain: utxolib.BIP32Interface
|
|
20
|
-
|
|
21
|
-
allowNonSegwitSigningWithoutPrevTx, }: {
|
|
22
|
-
isLastSignature: boolean;
|
|
23
|
-
allowNonSegwitSigningWithoutPrevTx?: boolean;
|
|
24
|
-
}): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>;
|
|
25
|
-
export declare function signPsbtWithMusig2Participant(coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>, tx: utxolib.bitgo.UtxoPsbt, signerKeychain: BIP32Interface | undefined, params: {
|
|
26
|
-
isLastSignature: boolean;
|
|
16
|
+
export declare function signAndVerifyPsbt(psbt: utxolib.bitgo.UtxoPsbt, signerKeychain: utxolib.BIP32Interface): utxolib.bitgo.UtxoPsbt;
|
|
17
|
+
export declare function signPsbtWithMusig2ParticipantUtxolib(coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>, tx: utxolib.bitgo.UtxoPsbt, signerKeychain: BIP32Interface | undefined, params: {
|
|
27
18
|
signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
|
|
28
19
|
walletId: string | undefined;
|
|
29
|
-
}): Promise<utxolib.bitgo.UtxoPsbt
|
|
30
|
-
//# sourceMappingURL=
|
|
20
|
+
}): Promise<utxolib.bitgo.UtxoPsbt>;
|
|
21
|
+
//# sourceMappingURL=signPsbtUtxolib.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"signPsbtUtxolib.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signPsbtUtxolib.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAKvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAI7C,MAAM,MAAM,oBAAoB,GAC5B,MAAM,GACN,OAAO,GACP,WAAW,GACX,UAAU,GACV,qBAAqB,GACrB,wBAAwB,GAExB,YAAY,GACZ,sBAAsB,GACtB,mBAAmB,CAAC;AAExB;;;;;;;;;;GAUG;AACH,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,cAAc,EAAE,OAAO,CAAC,cAAc,GACrC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAwDxB;AAYD,wBAAsB,oCAAoC,CACxD,IAAI,EAAE,iBAAiB,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,EAC/C,EAAE,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC1B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,MAAM,EAAE;IACN,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;CAC9B,GACA,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CA6CjC"}
|
|
@@ -0,0 +1,157 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.signAndVerifyPsbt = signAndVerifyPsbt;
|
|
40
|
+
exports.signPsbtWithMusig2ParticipantUtxolib = signPsbtWithMusig2ParticipantUtxolib;
|
|
41
|
+
const assert_1 = __importDefault(require("assert"));
|
|
42
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
43
|
+
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
44
|
+
const debug_1 = __importDefault(require("debug"));
|
|
45
|
+
const SigningError_1 = require("./SigningError");
|
|
46
|
+
const debug = (0, debug_1.default)('bitgo:v2:utxo');
|
|
47
|
+
/**
|
|
48
|
+
* Sign all inputs of a psbt and verify signatures after signing.
|
|
49
|
+
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
50
|
+
* failed.
|
|
51
|
+
*
|
|
52
|
+
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
53
|
+
* using TransactionBuilder
|
|
54
|
+
*
|
|
55
|
+
* @param psbt
|
|
56
|
+
* @param signerKeychain
|
|
57
|
+
*/
|
|
58
|
+
function signAndVerifyPsbt(psbt, signerKeychain) {
|
|
59
|
+
const txInputs = psbt.txInputs;
|
|
60
|
+
const outputIds = [];
|
|
61
|
+
const scriptTypes = [];
|
|
62
|
+
const signErrors = psbt.data.inputs
|
|
63
|
+
.map((input, inputIndex) => {
|
|
64
|
+
const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
|
|
65
|
+
outputIds.push(outputId);
|
|
66
|
+
const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
|
|
67
|
+
scriptTypes.push(scriptType);
|
|
68
|
+
if (scriptType === 'p2shP2pk') {
|
|
69
|
+
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
try {
|
|
73
|
+
psbt.signInputHD(inputIndex, signerKeychain);
|
|
74
|
+
debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
|
|
75
|
+
}
|
|
76
|
+
catch (e) {
|
|
77
|
+
return new SigningError_1.InputSigningError(inputIndex, scriptType, { id: outputId }, e);
|
|
78
|
+
}
|
|
79
|
+
})
|
|
80
|
+
.filter((e) => e !== undefined);
|
|
81
|
+
const verifyErrors = psbt.data.inputs
|
|
82
|
+
.map((input, inputIndex) => {
|
|
83
|
+
const scriptType = scriptTypes[inputIndex];
|
|
84
|
+
if (scriptType === 'p2shP2pk') {
|
|
85
|
+
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
const outputId = outputIds[inputIndex];
|
|
89
|
+
try {
|
|
90
|
+
if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {
|
|
91
|
+
return new SigningError_1.InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
catch (e) {
|
|
95
|
+
debug('Invalid signature');
|
|
96
|
+
return new SigningError_1.InputSigningError(inputIndex, scriptType, { id: outputId }, e);
|
|
97
|
+
}
|
|
98
|
+
})
|
|
99
|
+
.filter((e) => e !== undefined);
|
|
100
|
+
if (signErrors.length || verifyErrors.length) {
|
|
101
|
+
throw new SigningError_1.TransactionSigningError(signErrors, verifyErrors);
|
|
102
|
+
}
|
|
103
|
+
return psbt;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Key Value: Unsigned tx id => PSBT
|
|
107
|
+
* It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.
|
|
108
|
+
* Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.
|
|
109
|
+
* For more info, check SignTransactionOptions.signingStep
|
|
110
|
+
*
|
|
111
|
+
* TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.
|
|
112
|
+
*/
|
|
113
|
+
const PSBT_CACHE = new Map();
|
|
114
|
+
async function signPsbtWithMusig2ParticipantUtxolib(coin, tx, signerKeychain, params) {
|
|
115
|
+
if (utxo_lib_1.bitgo.isTransactionWithKeyPathSpendInput(tx)) {
|
|
116
|
+
switch (params.signingStep) {
|
|
117
|
+
case 'signerNonce':
|
|
118
|
+
(0, assert_1.default)(signerKeychain);
|
|
119
|
+
tx.setAllInputsMusig2NonceHD(signerKeychain);
|
|
120
|
+
PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);
|
|
121
|
+
return tx;
|
|
122
|
+
case 'cosignerNonce':
|
|
123
|
+
(0, assert_1.default)(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
|
|
124
|
+
return await coin.getMusig2Nonces(tx, params.walletId);
|
|
125
|
+
case 'signerSignature':
|
|
126
|
+
const txId = tx.getUnsignedTx().getId();
|
|
127
|
+
const psbt = PSBT_CACHE.get(txId);
|
|
128
|
+
(0, assert_1.default)(psbt, `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).
|
|
129
|
+
This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`);
|
|
130
|
+
PSBT_CACHE.delete(txId);
|
|
131
|
+
tx = psbt.combine(tx);
|
|
132
|
+
break;
|
|
133
|
+
default:
|
|
134
|
+
// this instance is not an external signer
|
|
135
|
+
(0, assert_1.default)(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
|
|
136
|
+
(0, assert_1.default)(signerKeychain);
|
|
137
|
+
tx.setAllInputsMusig2NonceHD(signerKeychain);
|
|
138
|
+
const response = await coin.getMusig2Nonces(tx, params.walletId);
|
|
139
|
+
tx = tx.combine(response);
|
|
140
|
+
break;
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
switch (params.signingStep) {
|
|
145
|
+
case 'signerNonce':
|
|
146
|
+
case 'cosignerNonce':
|
|
147
|
+
/**
|
|
148
|
+
* In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
|
|
149
|
+
* Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
|
|
150
|
+
*/
|
|
151
|
+
return tx;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
(0, assert_1.default)(signerKeychain);
|
|
155
|
+
return signAndVerifyPsbt(tx, signerKeychain);
|
|
156
|
+
}
|
|
157
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signPsbtUtxolib.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signPsbtUtxolib.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport debugLib from 'debug';\n\nimport { InputSigningError, TransactionSigningError } from './SigningError';\nimport { Musig2Participant } from './musig2';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nexport type PsbtParsedScriptType =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend'\n  // wasm-utxo types\n  | 'p2trLegacy'\n  | 'p2trMusig2ScriptPath'\n  | 'p2trMusig2KeyPath';\n\n/**\n * Sign all inputs of a psbt and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of\n * using TransactionBuilder\n *\n * @param psbt\n * @param signerKeychain\n */\nexport function signAndVerifyPsbt(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: utxolib.BIP32Interface\n): utxolib.bitgo.UtxoPsbt {\n  const txInputs = psbt.txInputs;\n  const outputIds: string[] = [];\n  const scriptTypes: PsbtParsedScriptType[] = [];\n\n  const signErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex: number) => {\n      const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));\n      outputIds.push(outputId);\n\n      const { scriptType } = utxolib.bitgo.parsePsbtInput(input);\n      scriptTypes.push(scriptType);\n\n      if (scriptType === 'p2shP2pk') {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);\n        return;\n      }\n\n      try {\n        psbt.signInputHD(inputIndex, signerKeychain);\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, scriptType, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  const verifyErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex) => {\n      const scriptType = scriptTypes[inputIndex];\n      if (scriptType === 'p2shP2pk') {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          psbt.data.inputs.length\n        );\n        return;\n      }\n\n      const outputId = outputIds[inputIndex];\n      try {\n        if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {\n          return new InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, scriptType, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return psbt;\n}\n\n/**\n * Key Value: Unsigned tx id => PSBT\n * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.\n * Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.\n * For more info, check SignTransactionOptions.signingStep\n *\n * TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.\n */\nconst PSBT_CACHE = new Map<string, utxolib.bitgo.UtxoPsbt>();\n\nexport async function signPsbtWithMusig2ParticipantUtxolib(\n  coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>,\n  tx: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: BIP32Interface | undefined,\n  params: {\n    signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;\n    walletId: string | undefined;\n  }\n): Promise<utxolib.bitgo.UtxoPsbt> {\n  if (bitgo.isTransactionWithKeyPathSpendInput(tx)) {\n    switch (params.signingStep) {\n      case 'signerNonce':\n        assert(signerKeychain);\n        tx.setAllInputsMusig2NonceHD(signerKeychain);\n        PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);\n        return tx;\n      case 'cosignerNonce':\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        return await coin.getMusig2Nonces(tx, params.walletId);\n      case 'signerSignature':\n        const txId = tx.getUnsignedTx().getId();\n        const psbt = PSBT_CACHE.get(txId);\n        assert(\n          psbt,\n          `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).\n            This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`\n        );\n        PSBT_CACHE.delete(txId);\n        tx = psbt.combine(tx);\n        break;\n      default:\n        // this instance is not an external signer\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        assert(signerKeychain);\n        tx.setAllInputsMusig2NonceHD(signerKeychain);\n        const response = await coin.getMusig2Nonces(tx, params.walletId);\n        tx = tx.combine(response);\n        break;\n    }\n  } else {\n    switch (params.signingStep) {\n      case 'signerNonce':\n      case 'cosignerNonce':\n        /**\n         * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n         * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n         */\n        return tx;\n    }\n  }\n\n  assert(signerKeychain);\n  return signAndVerifyPsbt(tx, signerKeychain);\n}\n"]}
|