@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"}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
2
|
import { Dimensions } from '@bitgo-beta/unspents';
|
|
3
|
-
import { fixedScriptWallet } from '@bitgo/wasm-utxo';
|
|
3
|
+
import { fixedScriptWallet, address as wasmAddress } from '@bitgo/wasm-utxo';
|
|
4
|
+
import { getNetworkFromCoinName } from '../names';
|
|
4
5
|
const { chainCodesP2tr, chainCodesP2trMusig2 } = utxolib.bitgo;
|
|
5
6
|
/**
|
|
6
7
|
* Check if a chain code is for a taproot script type
|
|
@@ -34,7 +35,8 @@ class InsufficientFundsError extends Error {
|
|
|
34
35
|
/**
|
|
35
36
|
* Create a backup key recovery PSBT using utxolib (legacy implementation)
|
|
36
37
|
*/
|
|
37
|
-
function createBackupKeyRecoveryPsbtUtxolib(
|
|
38
|
+
function createBackupKeyRecoveryPsbtUtxolib(coinName, rootWalletKeys, unspents, options) {
|
|
39
|
+
const network = getNetworkFromCoinName(coinName);
|
|
38
40
|
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
39
41
|
const psbt = utxolib.bitgo.createPsbtForNetwork({ network });
|
|
40
42
|
utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);
|
|
@@ -65,8 +67,8 @@ function createBackupKeyRecoveryPsbtUtxolib(network, rootWalletKeys, unspents, o
|
|
|
65
67
|
/**
|
|
66
68
|
* Check if the network is a Zcash network
|
|
67
69
|
*/
|
|
68
|
-
function
|
|
69
|
-
return
|
|
70
|
+
function isZcash(coinName) {
|
|
71
|
+
return coinName === 'zec' || coinName === 'tzec';
|
|
70
72
|
}
|
|
71
73
|
/**
|
|
72
74
|
* Default block heights for Zcash networks if not provided.
|
|
@@ -74,8 +76,8 @@ function isZcashNetwork(networkName) {
|
|
|
74
76
|
* TODO(BTC-2901): get the height from blockchair API instead of hardcoding.
|
|
75
77
|
*/
|
|
76
78
|
const ZCASH_DEFAULT_BLOCK_HEIGHTS = {
|
|
77
|
-
|
|
78
|
-
|
|
79
|
+
zec: 3146400,
|
|
80
|
+
tzec: 3536500,
|
|
79
81
|
};
|
|
80
82
|
/**
|
|
81
83
|
* Create an empty wasm-utxo BitGoPsbt for a given network.
|
|
@@ -86,16 +88,15 @@ const ZCASH_DEFAULT_BLOCK_HEIGHTS = {
|
|
|
86
88
|
* @param options - Optional settings (e.g., blockHeight for Zcash)
|
|
87
89
|
* @returns A wasm-utxo BitGoPsbt instance
|
|
88
90
|
*/
|
|
89
|
-
export function createEmptyWasmPsbt(
|
|
90
|
-
|
|
91
|
-
if (isZcashNetwork(networkName)) {
|
|
91
|
+
export function createEmptyWasmPsbt(coinName, rootWalletKeys, options) {
|
|
92
|
+
if (isZcash(coinName)) {
|
|
92
93
|
// For Zcash, use ZcashBitGoPsbt which requires block height to determine consensus branch ID
|
|
93
|
-
const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[
|
|
94
|
-
return fixedScriptWallet.ZcashBitGoPsbt.createEmpty(
|
|
94
|
+
const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[coinName];
|
|
95
|
+
return fixedScriptWallet.ZcashBitGoPsbt.createEmpty(coinName, rootWalletKeys, {
|
|
95
96
|
blockHeight,
|
|
96
97
|
});
|
|
97
98
|
}
|
|
98
|
-
return fixedScriptWallet.BitGoPsbt.createEmpty(
|
|
99
|
+
return fixedScriptWallet.BitGoPsbt.createEmpty(coinName, rootWalletKeys);
|
|
99
100
|
}
|
|
100
101
|
/**
|
|
101
102
|
* Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.
|
|
@@ -133,9 +134,9 @@ export function addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys) {
|
|
|
133
134
|
* @param network - The network (used to convert address to script)
|
|
134
135
|
* @returns The output index
|
|
135
136
|
*/
|
|
136
|
-
export function addOutputToWasmPsbt(wasmPsbt, address, value,
|
|
137
|
-
const script =
|
|
138
|
-
return wasmPsbt.addOutput({ script
|
|
137
|
+
export function addOutputToWasmPsbt(wasmPsbt, address, value, coinName) {
|
|
138
|
+
const script = wasmAddress.toOutputScriptWithCoin(address, coinName);
|
|
139
|
+
return wasmPsbt.addOutput({ script, value });
|
|
139
140
|
}
|
|
140
141
|
/**
|
|
141
142
|
* Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.
|
|
@@ -144,23 +145,25 @@ export function addOutputToWasmPsbt(wasmPsbt, address, value, network) {
|
|
|
144
145
|
* @param network - The network
|
|
145
146
|
* @returns A utxolib UtxoPsbt
|
|
146
147
|
*/
|
|
147
|
-
export function
|
|
148
|
-
|
|
148
|
+
export function toPsbtToUtxolibPsbt(wasmPsbt, coinName) {
|
|
149
|
+
if (wasmPsbt instanceof fixedScriptWallet.BitGoPsbt) {
|
|
150
|
+
const network = getNetworkFromCoinName(coinName);
|
|
151
|
+
return utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);
|
|
152
|
+
}
|
|
153
|
+
return wasmPsbt;
|
|
149
154
|
}
|
|
150
155
|
/**
|
|
151
156
|
* Create a backup key recovery PSBT using wasm-utxo
|
|
152
157
|
*/
|
|
153
|
-
function createBackupKeyRecoveryPsbtWasm(
|
|
158
|
+
function createBackupKeyRecoveryPsbtWasm(coinName, rootWalletKeys, unspents, options) {
|
|
154
159
|
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
155
160
|
// Create PSBT with wasm-utxo and add wallet inputs using shared utilities
|
|
156
|
-
const wasmPsbt = createEmptyWasmPsbt(
|
|
161
|
+
const wasmPsbt = createEmptyWasmPsbt(coinName, rootWalletKeys, { blockHeight: options.blockHeight });
|
|
157
162
|
addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys);
|
|
158
163
|
// Calculate dimensions using wasm-utxo Dimensions
|
|
159
|
-
|
|
160
|
-
let dimensions = fixedScriptWallet.Dimensions.fromPsbt(wasmPsbt).plus(fixedScriptWallet.Dimensions.fromOutput(new Uint8Array(recoveryOutputScript)));
|
|
164
|
+
let dimensions = fixedScriptWallet.Dimensions.fromPsbt(wasmPsbt).plus(fixedScriptWallet.Dimensions.fromOutput(recoveryDestination, coinName));
|
|
161
165
|
if (keyRecoveryServiceFeeAddress) {
|
|
162
|
-
|
|
163
|
-
dimensions = dimensions.plus(fixedScriptWallet.Dimensions.fromOutput(new Uint8Array(krsOutputScript)));
|
|
166
|
+
dimensions = dimensions.plus(fixedScriptWallet.Dimensions.fromOutput(keyRecoveryServiceFeeAddress, coinName));
|
|
164
167
|
}
|
|
165
168
|
const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);
|
|
166
169
|
const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');
|
|
@@ -169,12 +172,12 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
169
172
|
throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);
|
|
170
173
|
}
|
|
171
174
|
// Add outputs to wasm PSBT
|
|
172
|
-
addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount,
|
|
175
|
+
addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount, coinName);
|
|
173
176
|
if (keyRecoveryServiceFeeAddress) {
|
|
174
|
-
addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee,
|
|
177
|
+
addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee, coinName);
|
|
175
178
|
}
|
|
176
179
|
// Convert to utxolib PSBT for signing and return
|
|
177
|
-
return
|
|
180
|
+
return wasmPsbt;
|
|
178
181
|
}
|
|
179
182
|
/**
|
|
180
183
|
* Create a backup key recovery PSBT.
|
|
@@ -185,23 +188,34 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
185
188
|
* @param options - Options for creating the PSBT
|
|
186
189
|
* @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')
|
|
187
190
|
*/
|
|
188
|
-
export function createBackupKeyRecoveryPsbt(
|
|
191
|
+
export function createBackupKeyRecoveryPsbt(coinName, rootWalletKeys, unspents, options, backend = 'wasm-utxo') {
|
|
189
192
|
if (options.keyRecoveryServiceFee > 0 && !options.keyRecoveryServiceFeeAddress) {
|
|
190
193
|
throw new Error('keyRecoveryServiceFeeAddress is required when keyRecoveryServiceFee is provided');
|
|
191
194
|
}
|
|
192
195
|
if (backend === 'wasm-utxo') {
|
|
193
|
-
return createBackupKeyRecoveryPsbtWasm(
|
|
196
|
+
return createBackupKeyRecoveryPsbtWasm(coinName, rootWalletKeys, unspents, options);
|
|
194
197
|
}
|
|
195
198
|
else {
|
|
196
|
-
return createBackupKeyRecoveryPsbtUtxolib(
|
|
199
|
+
return createBackupKeyRecoveryPsbtUtxolib(coinName, rootWalletKeys, unspents, options);
|
|
197
200
|
}
|
|
198
201
|
}
|
|
199
|
-
export function getRecoveryAmount(psbt, address) {
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
202
|
+
export function getRecoveryAmount(psbt, walletKeys, address) {
|
|
203
|
+
if (psbt instanceof utxolib.bitgo.UtxoPsbt) {
|
|
204
|
+
const recoveryOutputScript = utxolib.address.toOutputScript(address, psbt.network);
|
|
205
|
+
const output = psbt.txOutputs.find((o) => o.script.equals(recoveryOutputScript));
|
|
206
|
+
if (!output) {
|
|
207
|
+
throw new Error(`Recovery destination output not found in PSBT: ${address}`);
|
|
208
|
+
}
|
|
209
|
+
return output.value;
|
|
210
|
+
}
|
|
211
|
+
if (psbt instanceof fixedScriptWallet.BitGoPsbt) {
|
|
212
|
+
const parsedOutputs = psbt.parseOutputsWithWalletKeys(walletKeys);
|
|
213
|
+
const recoveryOutput = parsedOutputs.find((o) => o.address === address);
|
|
214
|
+
if (!recoveryOutput) {
|
|
215
|
+
throw new Error(`Recovery destination output not found in PSBT: ${address}`);
|
|
216
|
+
}
|
|
217
|
+
return recoveryOutput.value;
|
|
204
218
|
}
|
|
205
|
-
|
|
219
|
+
throw new Error('Invalid PSBT type');
|
|
206
220
|
}
|
|
207
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../../src/recovery/psbt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAClD,OAAO,EAAE,iBAAiB,EAAiB,MAAM,kBAAkB,CAAC;AAKpE,MAAM,EAAE,cAAc,EAAE,oBAAoB,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;AAW/D;;GAEG;AACH,MAAM,UAAU,cAAc,CAAC,KAAgB;IAC7C,OAAO,CACJ,cAAoC,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAK,oBAA0C,CAAC,QAAQ,CAAC,KAAK,CAAC,CACrH,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,aAAa,CAAC,OAAwB;IACpD,MAAM,WAAW,GAAG,OAAO,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;IACpD,IAAI,CAAC,WAAW,EAAE,CAAC;QACjB,MAAM,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAC;IACrC,CAAC;IACD,OAAO,WAAW,CAAC;AACrB,CAAC;AAED,MAAM,sBAAuB,SAAQ,KAAK;IACxC,YACS,gBAAwB,EACxB,cAAsB,EACtB,MAAc,EACd,cAAsB;QAE7B,KAAK,CACH,iFAAiF;YAC/E,+BAA+B,gBAAgB,CAAC,QAAQ,EAAE,IAAI;YAC9D,uDAAuD,cAAc,CAAC,QAAQ,EAAE,EAAE;YAClF,mBAAmB,MAAM,CAAC,QAAQ,EAAE,IAAI;YACxC,2DAA2D,cAAc,CAAC,QAAQ,EAAE,EAAE,CACzF,CAAC;QAXK,qBAAgB,GAAhB,gBAAgB,CAAQ;QACxB,mBAAc,GAAd,cAAc,CAAQ;QACtB,WAAM,GAAN,MAAM,CAAQ;QACd,mBAAc,GAAd,cAAc,CAAQ;IAS/B,CAAC;CACF;AAWD;;GAEG;AACH,SAAS,kCAAkC,CACzC,OAAwB,EACxB,cAA8B,EAC9B,QAAiC,EACjC,OAA2C;IAE3C,MAAM,EAAE,YAAY,EAAE,mBAAmB,EAAE,qBAAqB,EAAE,4BAA4B,EAAE,GAAG,OAAO,CAAC;IAE3G,MAAM,IAAI,GAAG,OAAO,CAAC,KAAK,CAAC,oBAAoB,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;IAC7D,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;IACnD,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,IAAI,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;IACxF,CAAC,CAAC,CAAC;IAEH,IAAI,UAAU,GAAG,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAI,CAC7C,UAAU,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,mBAAmB,EAAE,OAAO,CAAC,EAAE,CAAC,CAChG,CAAC;IAEF,IAAI,4BAA4B,EAAE,CAAC;QACjC,UAAU,GAAG,UAAU,CAAC,IAAI,CAC1B,UAAU,CAAC,UAAU,CAAC;YACpB,MAAM,EAAE,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,4BAA4B,EAAE,OAAO,CAAC;SAC9E,CAAC,CACH,CAAC;IACJ,CAAC;IAED,MAAM,cAAc,GAAG,MAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,GAAG,YAAY,CAAC,CAAC;IACpE,MAAM,gBAAgB,GAAG,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IACtE,MAAM,cAAc,GAAG,gBAAgB,GAAG,cAAc,GAAG,qBAAqB,CAAC;IAEjF,IAAI,cAAc,GAAG,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC;QAC/B,MAAM,IAAI,sBAAsB,CAAC,gBAAgB,EAAE,cAAc,EAAE,qBAAqB,EAAE,cAAc,CAAC,CAAC;IAC5G,CAAC;IAED,IAAI,CAAC,SAAS,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,mBAAmB,EAAE,OAAO,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAC;IAEhH,IAAI,4BAA4B,EAAE,CAAC;QACjC,IAAI,CAAC,SAAS,CAAC;YACb,MAAM,EAAE,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,4BAA4B,EAAE,OAAO,CAAC;YAC7E,KAAK,EAAE,qBAAqB;SAC7B,CAAC,CAAC;IACL,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;GAEG;AACH,SAAS,cAAc,CAAC,WAAsC;IAC5D,OAAO,WAAW,KAAK,OAAO,IAAI,WAAW,KAAK,WAAW,CAAC;AAChE,CAAC;AAED;;;;GAIG;AACH,MAAM,2BAA2B,GAA2B;IAC1D,KAAK,EAAE,OAAO;IACd,SAAS,EAAE,OAAO;CACnB,CAAC;AAUF;;;;;;;;GAQG;AACH,MAAM,UAAU,mBAAmB,CACjC,OAAwB,EACxB,cAA8B,EAC9B,OAAoC;IAEpC,MAAM,WAAW,GAAG,aAAa,CAAC,OAAO,CAAC,CAAC;IAE3C,IAAI,cAAc,CAAC,WAAW,CAAC,EAAE,CAAC;QAChC,6FAA6F;QAC7F,MAAM,WAAW,GAAG,OAAO,EAAE,WAAW,IAAI,2BAA2B,CAAC,WAAW,CAAC,CAAC;QACrF,OAAO,iBAAiB,CAAC,cAAc,CAAC,WAAW,CAAC,WAAoC,EAAE,cAAc,EAAE;YACxG,WAAW;SACZ,CAAC,CAAC;IACL,CAAC;IAED,OAAO,iBAAiB,CAAC,SAAS,CAAC,WAAW,CAAC,WAAW,EAAE,cAAc,CAAC,CAAC;AAC9E,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,yBAAyB,CACvC,QAAqC,EACrC,QAAiC,EACjC,cAA8B;IAE9B,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAC/D,MAAM,QAAQ,GAA2C,cAAc,CAAC,OAAO,CAAC,KAAK,CAAC;YACpF,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE;YACxC,CAAC,CAAC,SAAS,CAAC;QAEd,6EAA6E;QAC7E,MAAM,MAAM,GAAI,OAAuD,CAAC,MAAM,CAAC;QAE/E,QAAQ,CAAC,cAAc,CACrB;YACE,IAAI;YACJ,IAAI;YACJ,KAAK,EAAE,OAAO,CAAC,KAAK;YACpB,MAAM,EAAE,MAAM;SACf,EACD,cAAc,EACd;YACE,QAAQ,EAAE,EAAE,KAAK,EAAE,OAAO,CAAC,KAAK,EAAE,KAAK,EAAE,OAAO,CAAC,KAAK,EAAE;YACxD,QAAQ;SACT,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;GAQG;AACH,MAAM,UAAU,mBAAmB,CACjC,QAAqC,EACrC,OAAe,EACf,KAAa,EACb,OAAwB;IAExB,MAAM,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IAChE,OAAO,QAAQ,CAAC,SAAS,CAAC,EAAE,MAAM,EAAE,IAAI,UAAU,CAAC,MAAM,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;AACvE,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,qBAAqB,CACnC,QAAqC,EACrC,OAAwB;IAExB,OAAO,OAAO,CAAC,KAAK,CAAC,oBAAoB,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,EAAE,OAAO,CAAC,CAAC;AACxF,CAAC;AAED;;GAEG;AACH,SAAS,+BAA+B,CACtC,OAAwB,EACxB,cAA8B,EAC9B,QAAiC,EACjC,OAA2C;IAE3C,MAAM,EAAE,YAAY,EAAE,mBAAmB,EAAE,qBAAqB,EAAE,4BAA4B,EAAE,GAAG,OAAO,CAAC;IAE3G,0EAA0E;IAC1E,MAAM,QAAQ,GAAG,mBAAmB,CAAC,OAAO,EAAE,cAAc,EAAE,EAAE,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;IACpG,yBAAyB,CAAC,QAAQ,EAAE,QAAQ,EAAE,cAAc,CAAC,CAAC;IAE9D,kDAAkD;IAClD,MAAM,oBAAoB,GAAG,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,mBAAmB,EAAE,OAAO,CAAC,CAAC;IAC1F,IAAI,UAAU,GAAG,iBAAiB,CAAC,UAAU,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,IAAI,CACnE,iBAAiB,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,UAAU,CAAC,oBAAoB,CAAC,CAAC,CAC9E,CAAC;IAEF,IAAI,4BAA4B,EAAE,CAAC;QACjC,MAAM,eAAe,GAAG,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,4BAA4B,EAAE,OAAO,CAAC,CAAC;QAC9F,UAAU,GAAG,UAAU,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,UAAU,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;IACzG,CAAC;IAED,MAAM,cAAc,GAAG,MAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,GAAG,YAAY,CAAC,CAAC;IACpE,MAAM,gBAAgB,GAAG,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IACtE,MAAM,cAAc,GAAG,gBAAgB,GAAG,cAAc,GAAG,qBAAqB,CAAC;IAEjF,IAAI,cAAc,GAAG,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC;QAC/B,MAAM,IAAI,sBAAsB,CAAC,gBAAgB,EAAE,cAAc,EAAE,qBAAqB,EAAE,cAAc,CAAC,CAAC;IAC5G,CAAC;IAED,2BAA2B;IAC3B,mBAAmB,CAAC,QAAQ,EAAE,mBAAmB,EAAE,cAAc,EAAE,OAAO,CAAC,CAAC;IAE5E,IAAI,4BAA4B,EAAE,CAAC;QACjC,mBAAmB,CAAC,QAAQ,EAAE,4BAA4B,EAAE,qBAAqB,EAAE,OAAO,CAAC,CAAC;IAC9F,CAAC;IAED,iDAAiD;IACjD,OAAO,qBAAqB,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;AAClD,CAAC;AAED;;;;;;;;GAQG;AACH,MAAM,UAAU,2BAA2B,CACzC,OAAwB,EACxB,cAA8B,EAC9B,QAAiC,EACjC,OAA2C,EAC3C,UAAuB,WAAW;IAElC,IAAI,OAAO,CAAC,qBAAqB,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,4BAA4B,EAAE,CAAC;QAC/E,MAAM,IAAI,KAAK,CAAC,iFAAiF,CAAC,CAAC;IACrG,CAAC;IAED,IAAI,OAAO,KAAK,WAAW,EAAE,CAAC;QAC5B,OAAO,+BAA+B,CAAC,OAAO,EAAE,cAAc,EAAE,QAAQ,EAAE,OAAO,CAAC,CAAC;IACrF,CAAC;SAAM,CAAC;QACN,OAAO,kCAAkC,CAAC,OAAO,EAAE,cAAc,EAAE,QAAQ,EAAE,OAAO,CAAC,CAAC;IACxF,CAAC;AACH,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAC,IAA4B,EAAE,OAAe;IAC7E,MAAM,oBAAoB,GAAG,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IACnF,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,oBAAoB,CAAC,CAAC,CAAC;IACjF,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CAAC,kDAAkD,OAAO,EAAE,CAAC,CAAC;IAC/E,CAAC;IACD,OAAO,MAAM,CAAC,KAAK,CAAC;AACtB,CAAC","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"]}
|
|
221
|
+
//# 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,9 @@
|
|
|
1
|
+
export function toTNumber(value, amountType) {
|
|
2
|
+
switch (amountType) {
|
|
3
|
+
case 'number':
|
|
4
|
+
return Number(value);
|
|
5
|
+
case 'bigint':
|
|
6
|
+
return BigInt(value);
|
|
7
|
+
}
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG51bWJlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90bnVtYmVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUdBLE1BQU0sVUFBVSxTQUFTLENBQUMsS0FBc0IsRUFBRSxVQUErQjtJQUMvRSxRQUFRLFVBQVUsRUFBRSxDQUFDO1FBQ25CLEtBQUssUUFBUTtZQUNYLE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3ZCLEtBQUssUUFBUTtZQUNYLE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3pCLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIHRvVE51bWJlcih2YWx1ZTogbnVtYmVyIHwgYmlnaW50LCBhbW91bnRUeXBlOiAnbnVtYmVyJyk6IG51bWJlcjtcbmV4cG9ydCBmdW5jdGlvbiB0b1ROdW1iZXIodmFsdWU6IG51bWJlciB8IGJpZ2ludCwgYW1vdW50VHlwZTogJ2JpZ2ludCcpOiBiaWdpbnQ7XG5leHBvcnQgZnVuY3Rpb24gdG9UTnVtYmVyKHZhbHVlOiBudW1iZXIgfCBiaWdpbnQsIGFtb3VudFR5cGU6ICdudW1iZXInIHwgJ2JpZ2ludCcpOiBudW1iZXIgfCBiaWdpbnQ7XG5leHBvcnQgZnVuY3Rpb24gdG9UTnVtYmVyKHZhbHVlOiBudW1iZXIgfCBiaWdpbnQsIGFtb3VudFR5cGU6ICdudW1iZXInIHwgJ2JpZ2ludCcpOiBudW1iZXIgfCBiaWdpbnQge1xuICBzd2l0Y2ggKGFtb3VudFR5cGUpIHtcbiAgICBjYXNlICdudW1iZXInOlxuICAgICAgcmV0dXJuIE51bWJlcih2YWx1ZSk7XG4gICAgY2FzZSAnYmlnaW50JzpcbiAgICAgIHJldHVybiBCaWdJbnQodmFsdWUpO1xuICB9XG59XG4iXX0=
|
|
@@ -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"}
|
|
@@ -18,4 +18,4 @@ export class TransactionSigningError extends Error {
|
|
|
18
18
|
`verify errors at inputs: [${verifyError.join(',')}], see log for details`);
|
|
19
19
|
}
|
|
20
20
|
}
|
|
21
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2lnbmluZ0Vycm9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3RyYW5zYWN0aW9uL2ZpeGVkU2NyaXB0L1NpZ25pbmdFcnJvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFNQSxNQUFNLE9BQU8saUJBQTRELFNBQVEsS0FBSztJQUNwRixNQUFNLENBQUMscUJBQXFCLENBQzFCLFVBQWtCLEVBQ2xCLFNBQXNDLEVBQUUscUNBQXFDO0lBQzdFLE9BQTBDO1FBRTFDLE9BQU8sSUFBSSxpQkFBaUIsQ0FDMUIsVUFBVSxFQUNWLFNBQVMsRUFDVCxPQUFPLEVBQ1AsdURBQXVELENBQ3hELENBQUM7SUFDSixDQUFDO0lBRUQsWUFDUyxVQUFrQixFQUNsQixTQUFzQyxFQUFFLHFDQUFxQztJQUM3RSxPQUEwQyxFQUMxQyxNQUFzQjtRQUU3QixLQUFLLENBQUMsMEJBQTBCLFVBQVUsVUFBVSxTQUFTLGNBQWMsT0FBTyxDQUFDLEVBQUUsS0FBSyxNQUFNLEVBQUUsQ0FBQyxDQUFDO1FBTDdGLGVBQVUsR0FBVixVQUFVLENBQVE7UUFDbEIsY0FBUyxHQUFULFNBQVMsQ0FBNkI7UUFDdEMsWUFBTyxHQUFQLE9BQU8sQ0FBbUM7UUFDMUMsV0FBTSxHQUFOLE1BQU0sQ0FBZ0I7SUFHL0IsQ0FBQztDQUNGO0FBRUQsTUFBTSxPQUFPLHVCQUFrRSxTQUFRLEtBQUs7SUFDMUYsWUFBWSxVQUF3QyxFQUFFLFdBQXlDO1FBQzdGLEtBQUssQ0FDSCwyQkFBMkIsVUFBVSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSztZQUNsRCw2QkFBNkIsV0FBVyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsd0JBQXdCLENBQzdFLENBQUM7SUFDSixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyB1dHhvbGliIGZyb20gJ0BiaXRnby1iZXRhL3V0eG8tbGliJztcblxuaW1wb3J0IHR5cGUgeyBQc2J0UGFyc2VkU2NyaXB0VHlwZSB9IGZyb20gJy4vc2lnblBzYnRVdHhvbGliJztcblxudHlwZSBVbnNwZW50PFROdW1iZXIgZXh0ZW5kcyBudW1iZXIgfCBiaWdpbnQgPSBudW1iZXI+ID0gdXR4b2xpYi5iaXRnby5VbnNwZW50PFROdW1iZXI+O1xuXG5leHBvcnQgY2xhc3MgSW5wdXRTaWduaW5nRXJyb3I8VE51bWJlciBleHRlbmRzIG51bWJlciB8IGJpZ2ludCA9IG51bWJlcj4gZXh0ZW5kcyBFcnJvciB7XG4gIHN0YXRpYyBleHBlY3RlZFdhbGxldFVuc3BlbnQ8VE51bWJlciBleHRlbmRzIG51bWJlciB8IGJpZ2ludD4oXG4gICAgaW5wdXRJbmRleDogbnVtYmVyLFxuICAgIGlucHV0VHlwZTogUHNidFBhcnNlZFNjcmlwdFR5cGUgfCBudWxsLCAvLyBudWxsIGZvciBsZWdhY3kgdHJhbnNhY3Rpb24gZm9ybWF0XG4gICAgdW5zcGVudDogVW5zcGVudDxUTnVtYmVyPiB8IHsgaWQ6IHN0cmluZyB9XG4gICk6IElucHV0U2lnbmluZ0Vycm9yPFROdW1iZXI+IHtcbiAgICByZXR1cm4gbmV3IElucHV0U2lnbmluZ0Vycm9yKFxuICAgICAgaW5wdXRJbmRleCxcbiAgICAgIGlucHV0VHlwZSxcbiAgICAgIHVuc3BlbnQsXG4gICAgICBgbm90IGEgd2FsbGV0IHVuc3BlbnQsIG5vdCBhIHJlcGxheSBwcm90ZWN0aW9uIHVuc3BlbnRgXG4gICAgKTtcbiAgfVxuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBpbnB1dEluZGV4OiBudW1iZXIsXG4gICAgcHVibGljIGlucHV0VHlwZTogUHNidFBhcnNlZFNjcmlwdFR5cGUgfCBudWxsLCAvLyBudWxsIGZvciBsZWdhY3kgdHJhbnNhY3Rpb24gZm9ybWF0XG4gICAgcHVibGljIHVuc3BlbnQ6IFVuc3BlbnQ8VE51bWJlcj4gfCB7IGlkOiBzdHJpbmcgfSxcbiAgICBwdWJsaWMgcmVhc29uOiBFcnJvciB8IHN0cmluZ1xuICApIHtcbiAgICBzdXBlcihgc2lnbmluZyBlcnJvciBhdCBpbnB1dCAke2lucHV0SW5kZXh9OiB0eXBlPSR7aW5wdXRUeXBlfSB1bnNwZW50SWQ9JHt1bnNwZW50LmlkfTogJHtyZWFzb259YCk7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIFRyYW5zYWN0aW9uU2lnbmluZ0Vycm9yPFROdW1iZXIgZXh0ZW5kcyBudW1iZXIgfCBiaWdpbnQgPSBudW1iZXI+IGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihzaWduRXJyb3JzOiBJbnB1dFNpZ25pbmdFcnJvcjxUTnVtYmVyPltdLCB2ZXJpZnlFcnJvcjogSW5wdXRTaWduaW5nRXJyb3I8VE51bWJlcj5bXSkge1xuICAgIHN1cGVyKFxuICAgICAgYHNpZ24gZXJyb3JzIGF0IGlucHV0czogWyR7c2lnbkVycm9ycy5qb2luKCcsJyl9XSwgYCArXG4gICAgICAgIGB2ZXJpZnkgZXJyb3JzIGF0IGlucHV0czogWyR7dmVyaWZ5RXJyb3Iuam9pbignLCcpfV0sIHNlZSBsb2cgZm9yIGRldGFpbHNgXG4gICAgKTtcbiAgfVxufVxuIl19
|
|
@@ -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,117 @@
|
|
|
1
|
+
import assert from 'assert';
|
|
2
|
+
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
3
|
+
import { bitgo } from '@bitgo-beta/utxo-lib';
|
|
4
|
+
import debugLib from 'debug';
|
|
5
|
+
import { InputSigningError, TransactionSigningError } from './SigningError';
|
|
6
|
+
const debug = debugLib('bitgo:v2:utxo');
|
|
7
|
+
/**
|
|
8
|
+
* Sign all inputs of a psbt and verify signatures after signing.
|
|
9
|
+
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
10
|
+
* failed.
|
|
11
|
+
*
|
|
12
|
+
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
13
|
+
* using TransactionBuilder
|
|
14
|
+
*
|
|
15
|
+
* @param psbt
|
|
16
|
+
* @param signerKeychain
|
|
17
|
+
*/
|
|
18
|
+
export function signAndVerifyPsbt(psbt, signerKeychain) {
|
|
19
|
+
const txInputs = psbt.txInputs;
|
|
20
|
+
const outputIds = [];
|
|
21
|
+
const scriptTypes = [];
|
|
22
|
+
const signErrors = psbt.data.inputs
|
|
23
|
+
.map((input, inputIndex) => {
|
|
24
|
+
const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
|
|
25
|
+
outputIds.push(outputId);
|
|
26
|
+
const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
|
|
27
|
+
scriptTypes.push(scriptType);
|
|
28
|
+
if (scriptType === 'p2shP2pk') {
|
|
29
|
+
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
try {
|
|
33
|
+
psbt.signInputHD(inputIndex, signerKeychain);
|
|
34
|
+
debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
|
|
35
|
+
}
|
|
36
|
+
catch (e) {
|
|
37
|
+
return new InputSigningError(inputIndex, scriptType, { id: outputId }, e);
|
|
38
|
+
}
|
|
39
|
+
})
|
|
40
|
+
.filter((e) => e !== undefined);
|
|
41
|
+
const verifyErrors = psbt.data.inputs
|
|
42
|
+
.map((input, inputIndex) => {
|
|
43
|
+
const scriptType = scriptTypes[inputIndex];
|
|
44
|
+
if (scriptType === 'p2shP2pk') {
|
|
45
|
+
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
const outputId = outputIds[inputIndex];
|
|
49
|
+
try {
|
|
50
|
+
if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {
|
|
51
|
+
return new InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
catch (e) {
|
|
55
|
+
debug('Invalid signature');
|
|
56
|
+
return new InputSigningError(inputIndex, scriptType, { id: outputId }, e);
|
|
57
|
+
}
|
|
58
|
+
})
|
|
59
|
+
.filter((e) => e !== undefined);
|
|
60
|
+
if (signErrors.length || verifyErrors.length) {
|
|
61
|
+
throw new TransactionSigningError(signErrors, verifyErrors);
|
|
62
|
+
}
|
|
63
|
+
return psbt;
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Key Value: Unsigned tx id => PSBT
|
|
67
|
+
* It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.
|
|
68
|
+
* Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.
|
|
69
|
+
* For more info, check SignTransactionOptions.signingStep
|
|
70
|
+
*
|
|
71
|
+
* TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.
|
|
72
|
+
*/
|
|
73
|
+
const PSBT_CACHE = new Map();
|
|
74
|
+
export async function signPsbtWithMusig2ParticipantUtxolib(coin, tx, signerKeychain, params) {
|
|
75
|
+
if (bitgo.isTransactionWithKeyPathSpendInput(tx)) {
|
|
76
|
+
switch (params.signingStep) {
|
|
77
|
+
case 'signerNonce':
|
|
78
|
+
assert(signerKeychain);
|
|
79
|
+
tx.setAllInputsMusig2NonceHD(signerKeychain);
|
|
80
|
+
PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);
|
|
81
|
+
return tx;
|
|
82
|
+
case 'cosignerNonce':
|
|
83
|
+
assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
|
|
84
|
+
return await coin.getMusig2Nonces(tx, params.walletId);
|
|
85
|
+
case 'signerSignature':
|
|
86
|
+
const txId = tx.getUnsignedTx().getId();
|
|
87
|
+
const psbt = PSBT_CACHE.get(txId);
|
|
88
|
+
assert(psbt, `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).
|
|
89
|
+
This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`);
|
|
90
|
+
PSBT_CACHE.delete(txId);
|
|
91
|
+
tx = psbt.combine(tx);
|
|
92
|
+
break;
|
|
93
|
+
default:
|
|
94
|
+
// this instance is not an external signer
|
|
95
|
+
assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
|
|
96
|
+
assert(signerKeychain);
|
|
97
|
+
tx.setAllInputsMusig2NonceHD(signerKeychain);
|
|
98
|
+
const response = await coin.getMusig2Nonces(tx, params.walletId);
|
|
99
|
+
tx = tx.combine(response);
|
|
100
|
+
break;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
switch (params.signingStep) {
|
|
105
|
+
case 'signerNonce':
|
|
106
|
+
case 'cosignerNonce':
|
|
107
|
+
/**
|
|
108
|
+
* In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
|
|
109
|
+
* Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
|
|
110
|
+
*/
|
|
111
|
+
return tx;
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
assert(signerKeychain);
|
|
115
|
+
return signAndVerifyPsbt(tx, signerKeychain);
|
|
116
|
+
}
|
|
117
|
+
//# 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"]}
|
|
@@ -7,16 +7,11 @@ export type ReplayProtectionKeys = {
|
|
|
7
7
|
/**
|
|
8
8
|
* Sign all inputs of a PSBT and verify signatures after signing.
|
|
9
9
|
* Collects and logs signing errors and verification errors, throws error in the end if any of them failed.
|
|
10
|
-
*
|
|
11
|
-
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
12
10
|
*/
|
|
13
|
-
export declare function signAndVerifyPsbtWasm(tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface, rootWalletKeys: fixedScriptWallet.RootWalletKeys, replayProtection: ReplayProtectionKeys
|
|
14
|
-
isLastSignature: boolean;
|
|
15
|
-
}): fixedScriptWallet.BitGoPsbt | Buffer;
|
|
11
|
+
export declare function signAndVerifyPsbtWasm(tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface, rootWalletKeys: fixedScriptWallet.RootWalletKeys, replayProtection: ReplayProtectionKeys): fixedScriptWallet.BitGoPsbt;
|
|
16
12
|
export declare function signPsbtWithMusig2ParticipantWasm(coin: Musig2Participant<fixedScriptWallet.BitGoPsbt>, tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface | undefined, rootWalletKeys: fixedScriptWallet.RootWalletKeys, params: {
|
|
17
13
|
replayProtection: ReplayProtectionKeys;
|
|
18
|
-
isLastSignature: boolean;
|
|
19
14
|
signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
|
|
20
15
|
walletId: string | undefined;
|
|
21
|
-
}): Promise<fixedScriptWallet.BitGoPsbt
|
|
16
|
+
}): Promise<fixedScriptWallet.BitGoPsbt>;
|
|
22
17
|
//# sourceMappingURL=signPsbtWasm.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"signPsbtWasm.d.ts","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signPsbtWasm.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAS,MAAM,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AAGpE,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAE7C,MAAM,MAAM,oBAAoB,GAAG;IACjC,UAAU,EAAE,CAAC,UAAU,GAAG,MAAM,CAAC,EAAE,CAAC;CACrC,CAAC;AAmBF
|
|
1
|
+
{"version":3,"file":"signPsbtWasm.d.ts","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signPsbtWasm.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAS,MAAM,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AAGpE,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAE7C,MAAM,MAAM,oBAAoB,GAAG;IACjC,UAAU,EAAE,CAAC,UAAU,GAAG,MAAM,CAAC,EAAE,CAAC;CACrC,CAAC;AAmBF;;;GAGG;AACH,wBAAgB,qBAAqB,CACnC,EAAE,EAAE,iBAAiB,CAAC,SAAS,EAC/B,cAAc,EAAE,cAAc,EAC9B,cAAc,EAAE,iBAAiB,CAAC,cAAc,EAChD,gBAAgB,EAAE,oBAAoB,GACrC,iBAAiB,CAAC,SAAS,CA6C7B;AAOD,wBAAsB,iCAAiC,CACrD,IAAI,EAAE,iBAAiB,CAAC,iBAAiB,CAAC,SAAS,CAAC,EACpD,EAAE,EAAE,iBAAiB,CAAC,SAAS,EAC/B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,cAAc,EAAE,iBAAiB,CAAC,cAAc,EAChD,MAAM,EAAE;IACN,gBAAgB,EAAE,oBAAoB,CAAC;IACvC,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;CAC9B,GACA,OAAO,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAiDtC"}
|