@bitgo-beta/abstract-utxo 1.6.1-alpha.447 → 1.6.1-alpha.449
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/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/crossChainRecovery.js +51 -4
- package/dist/cjs/src/recovery/psbt.d.ts +54 -0
- package/dist/cjs/src/recovery/psbt.d.ts.map +1 -1
- package/dist/cjs/src/recovery/psbt.js +68 -21
- package/dist/cjs/tsconfig.tsbuildinfo +1 -1
- package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/crossChainRecovery.js +51 -4
- package/dist/esm/recovery/psbt.d.ts +54 -0
- package/dist/esm/recovery/psbt.d.ts.map +1 -1
- package/dist/esm/recovery/psbt.js +64 -23
- package/package.json +11 -11
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"crossChainRecovery.d.ts","sourceRoot":"","sources":["../../../../src/recovery/crossChainRecovery.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAGhD,OAAO,EAAE,SAAS,EAAE,OAAO,EAA4B,MAAM,sBAAsB,CAAC;AAGpF,OAAO,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;
|
|
1
|
+
{"version":3,"file":"crossChainRecovery.d.ts","sourceRoot":"","sources":["../../../../src/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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"}
|
|
@@ -44,6 +44,7 @@ const unspents_1 = require("@bitgo-beta/unspents");
|
|
|
44
44
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
45
45
|
const sdk_api_1 = require("@bitgo-beta/sdk-api");
|
|
46
46
|
const signPsbt_1 = require("../transaction/fixedScript/signPsbt");
|
|
47
|
+
const psbt_1 = require("./psbt");
|
|
47
48
|
const { unspentSum } = utxolib.bitgo;
|
|
48
49
|
async function getWallet(bitgo, coin, walletId) {
|
|
49
50
|
try {
|
|
@@ -274,7 +275,7 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
274
275
|
return getPrv((0, sdk_api_1.decrypt)(passphrase, encryptedPrv));
|
|
275
276
|
}
|
|
276
277
|
/**
|
|
277
|
-
* Create a sweep transaction for cross-chain recovery using PSBT
|
|
278
|
+
* Create a sweep transaction for cross-chain recovery using PSBT (utxolib implementation)
|
|
278
279
|
* @param network
|
|
279
280
|
* @param walletKeys
|
|
280
281
|
* @param unspents
|
|
@@ -282,7 +283,7 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
282
283
|
* @param feeRateSatVB
|
|
283
284
|
* @return unsigned PSBT
|
|
284
285
|
*/
|
|
285
|
-
function
|
|
286
|
+
function createSweepTransactionUtxolib(network, walletKeys, unspents, targetAddress, feeRateSatVB) {
|
|
286
287
|
const inputValue = unspentSum(unspents.map((u) => ({ ...u, value: BigInt(u.value) })), 'bigint');
|
|
287
288
|
const vsize = unspents_1.Dimensions.fromUnspents(unspents, {
|
|
288
289
|
p2tr: { scriptPathLevel: 1 },
|
|
@@ -300,6 +301,50 @@ function createSweepTransaction(network, walletKeys, unspents, targetAddress, fe
|
|
|
300
301
|
psbt.addOutput({ script: recoveryOutputScript, value: inputValue - fee });
|
|
301
302
|
return psbt;
|
|
302
303
|
}
|
|
304
|
+
/**
|
|
305
|
+
* Create a sweep transaction for cross-chain recovery using wasm-utxo
|
|
306
|
+
* @param network
|
|
307
|
+
* @param walletKeys
|
|
308
|
+
* @param unspents
|
|
309
|
+
* @param targetAddress
|
|
310
|
+
* @param feeRateSatVB
|
|
311
|
+
* @return unsigned PSBT
|
|
312
|
+
*/
|
|
313
|
+
function createSweepTransactionWasm(network, walletKeys, unspents, targetAddress, feeRateSatVB) {
|
|
314
|
+
const inputValue = unspentSum(unspents.map((u) => ({ ...u, value: BigInt(u.value) })), 'bigint');
|
|
315
|
+
// Create PSBT with wasm-utxo and add wallet inputs using shared utilities
|
|
316
|
+
const unspentsBigint = unspents.map((u) => ({ ...u, value: BigInt(u.value) }));
|
|
317
|
+
const wasmPsbt = (0, psbt_1.createEmptyWasmPsbt)(network, walletKeys);
|
|
318
|
+
(0, psbt_1.addWalletInputsToWasmPsbt)(wasmPsbt, unspentsBigint, walletKeys);
|
|
319
|
+
// Convert to utxolib PSBT temporarily for dimension calculation
|
|
320
|
+
const tempPsbt = (0, psbt_1.wasmPsbtToUtxolibPsbt)(wasmPsbt, network);
|
|
321
|
+
const vsize = unspents_1.Dimensions.fromPsbt(tempPsbt)
|
|
322
|
+
.plus(unspents_1.Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))
|
|
323
|
+
.getVSize();
|
|
324
|
+
const fee = BigInt(Math.round(vsize * feeRateSatVB));
|
|
325
|
+
// Add output to wasm PSBT
|
|
326
|
+
(0, psbt_1.addOutputToWasmPsbt)(wasmPsbt, targetAddress, inputValue - fee, network);
|
|
327
|
+
// Convert to utxolib PSBT for signing and return
|
|
328
|
+
return (0, psbt_1.wasmPsbtToUtxolibPsbt)(wasmPsbt, network);
|
|
329
|
+
}
|
|
330
|
+
/**
|
|
331
|
+
* Create a sweep transaction for cross-chain recovery using PSBT
|
|
332
|
+
* @param network
|
|
333
|
+
* @param walletKeys
|
|
334
|
+
* @param unspents
|
|
335
|
+
* @param targetAddress
|
|
336
|
+
* @param feeRateSatVB
|
|
337
|
+
* @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')
|
|
338
|
+
* @return unsigned PSBT
|
|
339
|
+
*/
|
|
340
|
+
function createSweepTransaction(network, walletKeys, unspents, targetAddress, feeRateSatVB, backend = 'wasm-utxo') {
|
|
341
|
+
if (backend === 'wasm-utxo') {
|
|
342
|
+
return createSweepTransactionWasm(network, walletKeys, unspents, targetAddress, feeRateSatVB);
|
|
343
|
+
}
|
|
344
|
+
else {
|
|
345
|
+
return createSweepTransactionUtxolib(network, walletKeys, unspents, targetAddress, feeRateSatVB);
|
|
346
|
+
}
|
|
347
|
+
}
|
|
303
348
|
/**
|
|
304
349
|
* Recover wallet deposits that were received on the wrong blockchain
|
|
305
350
|
* (for instance bitcoin deposits that were received for a litecoin wallet).
|
|
@@ -318,7 +363,9 @@ async function recoverCrossChain(bitgo, params) {
|
|
|
318
363
|
const prv = params.xprv || params.walletPassphrase ? await getPrv(params.xprv, params.walletPassphrase, wallet) : undefined;
|
|
319
364
|
const feeRateSatVB = await getFeeRateSatVB(params.sourceCoin);
|
|
320
365
|
// Create PSBT for both signed and unsigned recovery
|
|
321
|
-
|
|
366
|
+
// Use wasm-utxo for testnet coins only, utxolib for mainnet
|
|
367
|
+
const backend = utxolib.isTestnet(params.sourceCoin.network) ? 'wasm-utxo' : 'utxolib';
|
|
368
|
+
const psbt = createSweepTransaction(params.sourceCoin.network, walletKeys, walletUnspents, params.recoveryAddress, feeRateSatVB, backend);
|
|
322
369
|
// For unsigned recovery, return unsigned PSBT hex
|
|
323
370
|
if (!prv) {
|
|
324
371
|
return {
|
|
@@ -340,4 +387,4 @@ async function recoverCrossChain(bitgo, params) {
|
|
|
340
387
|
recoveryAmount,
|
|
341
388
|
};
|
|
342
389
|
}
|
|
343
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../../src/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface, bip32 } from '@bitgo-beta/secp256k1';\nimport { Dimensions } from '@bitgo-beta/unspents';\nimport { BitGoBase, IWallet, Keychain, Triple, Wallet } from '@bitgo-beta/sdk-core';\nimport { decrypt } from '@bitgo-beta/sdk-api';\n\nimport { AbstractUtxoCoin, TransactionInfo } from '../abstractUtxoCoin';\nimport { signAndVerifyPsbt } from '../transaction/fixedScript/signPsbt';\n\nconst { unspentSum } = utxolib.bitgo;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\ntype WalletUnspent<TNumber extends number | bigint = number> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nexport interface BuildRecoveryTransactionOptions {\n  wallet: string;\n  faultyTxId: string;\n  recoveryAddress: string;\n}\n\ntype FeeInfo = {\n  size: number;\n  feeRate: number;\n  fee: number;\n  payGoFee: number;\n};\n\nexport interface CrossChainRecoveryUnsigned<TNumber extends number | bigint = number> {\n  txHex: string;\n  txInfo?: TransactionInfo<TNumber>;\n  walletId: string;\n  feeInfo?: FeeInfo;\n  address: string;\n  coin: string;\n}\n\nexport interface CrossChainRecoverySigned<TNumber extends number | bigint = number> {\n  version: 1 | 2;\n  txHex: string;\n  txInfo?: TransactionInfo<TNumber>;\n  walletId: string;\n  sourceCoin: string;\n  recoveryCoin: string;\n  recoveryAddress?: string;\n  recoveryAmount?: TNumber;\n}\n\ntype WalletV1 = {\n  keychains: { xpub: string }[];\n  address({ address }: { address: string }): Promise<{ chain: number; index: number }>;\n  getEncryptedUserKeychain(): Promise<{ encryptedXprv: string }>;\n};\n\nexport async function getWallet(\n  bitgo: BitGoBase,\n  coin: AbstractUtxoCoin,\n  walletId: string\n): Promise<IWallet | WalletV1> {\n  try {\n    return await coin.wallets().get({ id: walletId });\n  } catch (e) {\n    // TODO: BG-46364 handle errors more gracefully\n    // The v2 endpoint coin.wallets().get() may throw 404 or 400 errors, but this should not prevent us from searching for the walletId in v1 wallets.\n    if (e.status >= 500) {\n      throw e;\n    }\n  }\n\n  try {\n    return await bitgo.wallets().get({ id: walletId });\n  } catch (e) {\n    throw new Error(`could not get wallet ${walletId} from v1 or v2: ${e.toString()}`);\n  }\n}\n\n/**\n * @param recoveryCoin\n * @param wallet\n * @return wallet pubkeys\n */\nexport async function getWalletKeys(\n  recoveryCoin: AbstractUtxoCoin,\n  wallet: IWallet | WalletV1\n): Promise<RootWalletKeys> {\n  let xpubs: Triple<string>;\n\n  if (wallet instanceof Wallet) {\n    const keychains = (await recoveryCoin.keychains().getKeysForSigning({ wallet })) as unknown as Keychain[];\n    if (keychains.length !== 3) {\n      throw new Error(`expected triple got ${keychains.length}`);\n    }\n    xpubs = keychains.map((k) => k.pub) as Triple<string>;\n  } else {\n    xpubs = (wallet as WalletV1).keychains.map((k) => k.xpub) as Triple<string>;\n  }\n\n  return new utxolib.bitgo.RootWalletKeys(xpubs.map((k) => bip32.fromBase58(k)) as Triple<BIP32Interface>);\n}\n\nexport async function isWalletAddress(wallet: IWallet | WalletV1, address: string): Promise<boolean> {\n  try {\n    let addressData;\n    if (wallet instanceof Wallet) {\n      addressData = await wallet.getAddress({ address });\n    } else {\n      addressData = await (wallet as WalletV1).address({ address });\n    }\n\n    return addressData !== undefined;\n  } catch (e) {\n    return false;\n  }\n}\n\n/**\n * Convert a Litecoin P2SH address from M... format (scriptHash 0x32) to the legacy 3... format (scriptHash 0x05).\n * This is needed for cross-chain recovery when LTC was sent to a BTC address, because the BTC wallet\n * stores addresses in the 3... format while the LTC blockchain returns addresses in M... format.\n *\n * @param address - LTC address to convert\n * @param network - The Litecoin network\n * @returns The address in legacy 3... format, or the original address if it's not a P2SH address\n */\nexport function convertLtcAddressToLegacyFormat(address: string, network: utxolib.Network): string {\n  try {\n    // Try to decode as bech32 - these don't need conversion\n    utxolib.address.fromBech32(address);\n    return address;\n  } catch (e) {\n    // Not bech32, continue to base58\n  }\n\n  try {\n    const decoded = utxolib.address.fromBase58Check(address, network);\n    // Only convert P2SH addresses (scriptHash), not P2PKH (pubKeyHash)\n    if (decoded.version === network.scriptHash) {\n      // Convert to legacy format using Bitcoin's scriptHash (0x05)\n      const legacyScriptHash = utxolib.networks.bitcoin.scriptHash;\n      return utxolib.address.toBase58Check(decoded.hash, legacyScriptHash, network);\n    }\n    // P2PKH or other - return unchanged\n    return address;\n  } catch (e) {\n    // If decoding fails, return the original address\n    return address;\n  }\n}\n\n/**\n * @param coin\n * @param txid\n * @param amountType\n * @param wallet\n * @param apiKey - a blockchair api key\n * @return all unspents for transaction outputs, including outputs from other transactions\n */\nasync function getAllRecoveryOutputs<TNumber extends number | bigint = number>(\n  coin: AbstractUtxoCoin,\n  txid: string,\n  amountType: 'number' | 'bigint' = 'number',\n  wallet: IWallet | WalletV1,\n  apiKey?: string\n): Promise<Unspent<TNumber>[]> {\n  const api = coin.getRecoveryProvider(apiKey);\n  const tx = await api.getTransactionIO(txid);\n  const walletAddresses = (\n    await Promise.all(\n      tx.outputs.map(async (output) => {\n        // For some coins (bch) we need to convert the address to legacy format since the api returns the address\n        // in non legacy format. However, we want to keep the address in the same format as the response since we\n        // are going to hit the API again to fetch address unspents.\n        const canonicalAddress = coin.canonicalAddress(output.address);\n        let isWalletOwned = await isWalletAddress(wallet, canonicalAddress);\n\n        // For LTC cross-chain recovery: if the address isn't found, try the legacy format.\n        // When LTC is sent to a BTC address, the LTC blockchain returns M... addresses\n        // but the BTC wallet stores addresses in 3... format.\n        if (!isWalletOwned && coin.getFamily() === 'ltc') {\n          const legacyAddress = convertLtcAddressToLegacyFormat(output.address, coin.network);\n          if (legacyAddress !== output.address) {\n            isWalletOwned = await isWalletAddress(wallet, legacyAddress);\n          }\n        }\n\n        return isWalletOwned ? output.address : null;\n      })\n    )\n  ).filter((address) => address !== null);\n\n  const unspents = await api.getUnspentsForAddresses(walletAddresses as string[]);\n  if (unspents.length === 0) {\n    throw new Error(`No recovery unspents found.`);\n  }\n  // the api may return cashaddr's instead of legacy for BCH and BCHA\n  // downstream processes's only expect legacy addresses\n  return unspents.map((recoveryOutput) => {\n    return {\n      ...recoveryOutput,\n      address: coin.canonicalAddress(recoveryOutput.address),\n      value: utxolib.bitgo.toTNumber<TNumber>(BigInt(recoveryOutput.value), amountType),\n    };\n  });\n}\n\n/**\n * Data required for address and signature derivation\n */\ntype ScriptId = {\n  chain: number;\n  index: number;\n};\n\nasync function getScriptId(coin: AbstractUtxoCoin, wallet: IWallet | WalletV1, script: Buffer): Promise<ScriptId> {\n  const address = utxolib.address.fromOutputScript(script, coin.network);\n  let addressData: { chain: number; index: number };\n  if (wallet instanceof Wallet) {\n    addressData = await wallet.getAddress({ address });\n  } else {\n    addressData = await (wallet as WalletV1).address({ address });\n  }\n  if (typeof addressData.chain === 'number' && typeof addressData.index === 'number') {\n    return { chain: addressData.chain, index: addressData.index };\n  }\n\n  throw new Error(`invalid address data: ${JSON.stringify(addressData)}`);\n}\n\n/**\n * Lookup address data from unspents on sourceCoin in address database of recoveryCoin.\n * Return full walletUnspents including scriptId in sourceCoin format.\n *\n * @param sourceCoin\n * @param recoveryCoin\n * @param unspents\n * @param wallet\n * @return walletUnspents\n */\nasync function toWalletUnspents<TNumber extends number | bigint = number>(\n  sourceCoin: AbstractUtxoCoin,\n  recoveryCoin: AbstractUtxoCoin,\n  unspents: Unspent<TNumber>[],\n  wallet: IWallet | WalletV1\n): Promise<WalletUnspent<TNumber>[]> {\n  const addresses = new Set(unspents.map((u) => u.address));\n  const walletUnspents: WalletUnspent<TNumber>[] = [];\n\n  for (const address of addresses) {\n    let scriptId;\n    try {\n      scriptId = await getScriptId(recoveryCoin, wallet, utxolib.address.toOutputScript(address, sourceCoin.network));\n    } catch (e) {\n      console.error(`error getting scriptId for ${address}:`, e);\n      continue;\n    }\n    const filteredUnspents = unspents\n      .filter((u) => u.address === address)\n      .map((u) => ({\n        ...u,\n        ...scriptId,\n      }));\n    walletUnspents.push(...filteredUnspents);\n  }\n\n  return walletUnspents;\n}\n\n/**\n * @param coin\n * @return feeRate for transaction\n */\nasync function getFeeRateSatVB(coin: AbstractUtxoCoin): Promise<number> {\n  // TODO: use feeRate API\n  const feeRate = {\n    bch: 20,\n    tbch: 20,\n    bcha: 20,\n    tbcha: 20,\n    bsv: 20,\n    tbsv: 20,\n    btc: 80,\n    tbtc: 80,\n    tbtcsig: 80,\n    tbtc4: 80,\n    tbtcbgsig: 80,\n    ltc: 100,\n    tltc: 100,\n    doge: 1000,\n    tdoge: 1000,\n  }[coin.getChain()];\n\n  if (!feeRate) {\n    throw new Error(`no feeRate for ${coin.getChain()}`);\n  }\n\n  return feeRate;\n}\n\n/**\n * @param xprv\n * @param passphrase\n * @param wallet\n * @return signing key\n */\nasync function getPrv(xprv?: string, passphrase?: string, wallet?: IWallet | WalletV1): Promise<BIP32Interface> {\n  if (xprv) {\n    const key = bip32.fromBase58(xprv);\n    if (key.isNeutered()) {\n      throw new Error(`not a private key`);\n    }\n    return key;\n  }\n\n  if (!wallet || !passphrase) {\n    throw new Error(`no xprv given: need wallet and passphrase to continue`);\n  }\n\n  let encryptedPrv: string;\n  if (wallet instanceof Wallet) {\n    encryptedPrv = (await wallet.getEncryptedUserKeychain()).encryptedPrv;\n  } else {\n    encryptedPrv = (await (wallet as WalletV1).getEncryptedUserKeychain()).encryptedXprv;\n  }\n\n  return getPrv(decrypt(passphrase, encryptedPrv));\n}\n\n/**\n * Create a sweep transaction for cross-chain recovery using PSBT\n * @param network\n * @param walletKeys\n * @param unspents\n * @param targetAddress\n * @param feeRateSatVB\n * @return unsigned PSBT\n */\nfunction createSweepTransaction<TNumber extends number | bigint = number>(\n  network: utxolib.Network,\n  walletKeys: RootWalletKeys,\n  unspents: WalletUnspent<TNumber>[],\n  targetAddress: string,\n  feeRateSatVB: number\n): utxolib.bitgo.UtxoPsbt {\n  const inputValue = unspentSum<bigint>(\n    unspents.map((u) => ({ ...u, value: BigInt(u.value) })),\n    'bigint'\n  );\n  const vsize = Dimensions.fromUnspents(unspents, {\n    p2tr: { scriptPathLevel: 1 },\n    p2trMusig2: { scriptPathLevel: undefined },\n  })\n    .plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))\n    .getVSize();\n  const fee = BigInt(Math.round(vsize * feeRateSatVB));\n\n  const psbt = utxolib.bitgo.createPsbtForNetwork({ network });\n  utxolib.bitgo.addXpubsToPsbt(psbt, walletKeys);\n\n  unspents.forEach((unspent) => {\n    utxolib.bitgo.addWalletUnspentToPsbt(\n      psbt,\n      { ...unspent, value: BigInt(unspent.value) },\n      walletKeys,\n      'user',\n      'backup',\n      { skipNonWitnessUtxo: true }\n    );\n  });\n\n  const recoveryOutputScript = utxolib.address.toOutputScript(targetAddress, network);\n  psbt.addOutput({ script: recoveryOutputScript, value: inputValue - fee });\n\n  return psbt;\n}\n\ntype RecoverParams = {\n  /** Wallet ID (can be v1 wallet or v2 wallet) */\n  walletId: string;\n  /** Coin to create the transaction for */\n  sourceCoin: AbstractUtxoCoin;\n  /** Coin that wallet keys were set up for */\n  recoveryCoin: AbstractUtxoCoin;\n  /** Source coin transaction to recover outputs from (sourceCoin) */\n  txid: string;\n  /** Source coin address to send the funds to */\n  recoveryAddress: string;\n  /** If set, decrypts private key and signs transaction */\n  walletPassphrase?: string;\n  /** If set, signs transaction */\n  xprv?: string;\n  /** for utxo coins other than [BTC,TBTC] this is a Block Chair api key **/\n  apiKey?: string;\n};\n\n/**\n * Recover wallet deposits that were received on the wrong blockchain\n * (for instance bitcoin deposits that were received for a litecoin wallet).\n *\n * Fetches the unspent data from BitGo's public blockchain API and the script data from the user's\n * wallet.\n *\n * @param {BitGoBase} bitgo\n * @param {RecoverParams} params\n */\nexport async function recoverCrossChain<TNumber extends number | bigint = number>(\n  bitgo: BitGoBase,\n  params: RecoverParams\n): Promise<CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>> {\n  const wallet = await getWallet(bitgo, params.recoveryCoin, params.walletId);\n  const unspents = await getAllRecoveryOutputs<TNumber>(\n    params.sourceCoin,\n    params.txid,\n    params.sourceCoin.amountType,\n    wallet,\n    params.apiKey\n  );\n  const walletUnspents = await toWalletUnspents<TNumber>(params.sourceCoin, params.recoveryCoin, unspents, wallet);\n  const walletKeys = await getWalletKeys(params.recoveryCoin, wallet);\n  const prv =\n    params.xprv || params.walletPassphrase ? await getPrv(params.xprv, params.walletPassphrase, wallet) : undefined;\n  const feeRateSatVB = await getFeeRateSatVB(params.sourceCoin);\n\n  // Create PSBT for both signed and unsigned recovery\n  const psbt = createSweepTransaction<TNumber>(\n    params.sourceCoin.network,\n    walletKeys,\n    walletUnspents,\n    params.recoveryAddress,\n    feeRateSatVB\n  );\n\n  // For unsigned recovery, return unsigned PSBT hex\n  if (!prv) {\n    return {\n      txHex: psbt.toHex(),\n      walletId: params.walletId,\n      address: params.recoveryAddress,\n      coin: params.sourceCoin.getChain(),\n    };\n  }\n\n  // For signed recovery, sign the PSBT with user key and return half-signed PSBT\n  signAndVerifyPsbt(psbt, prv, { isLastSignature: false });\n  const recoveryAmount = utxolib.bitgo.toTNumber<TNumber>(psbt.txOutputs[0].value, params.sourceCoin.amountType);\n\n  return {\n    version: wallet instanceof Wallet ? 2 : 1,\n    walletId: params.walletId,\n    txHex: psbt.toHex(),\n    sourceCoin: params.sourceCoin.getChain(),\n    recoveryCoin: params.recoveryCoin.getChain(),\n    recoveryAmount,\n  };\n}\n"]}
|
|
390
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../../src/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface, bip32 } from '@bitgo-beta/secp256k1';\nimport { Dimensions } from '@bitgo-beta/unspents';\nimport { BitGoBase, IWallet, Keychain, Triple, Wallet } from '@bitgo-beta/sdk-core';\nimport { decrypt } from '@bitgo-beta/sdk-api';\n\nimport { AbstractUtxoCoin, TransactionInfo } from '../abstractUtxoCoin';\nimport { signAndVerifyPsbt } from '../transaction/fixedScript/signPsbt';\n\nimport {\n  PsbtBackend,\n  createEmptyWasmPsbt,\n  addWalletInputsToWasmPsbt,\n  addOutputToWasmPsbt,\n  wasmPsbtToUtxolibPsbt,\n} from './psbt';\n\nconst { unspentSum } = utxolib.bitgo;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\ntype WalletUnspent<TNumber extends number | bigint = number> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nexport interface BuildRecoveryTransactionOptions {\n  wallet: string;\n  faultyTxId: string;\n  recoveryAddress: string;\n}\n\ntype FeeInfo = {\n  size: number;\n  feeRate: number;\n  fee: number;\n  payGoFee: number;\n};\n\nexport interface CrossChainRecoveryUnsigned<TNumber extends number | bigint = number> {\n  txHex: string;\n  txInfo?: TransactionInfo<TNumber>;\n  walletId: string;\n  feeInfo?: FeeInfo;\n  address: string;\n  coin: string;\n}\n\nexport interface CrossChainRecoverySigned<TNumber extends number | bigint = number> {\n  version: 1 | 2;\n  txHex: string;\n  txInfo?: TransactionInfo<TNumber>;\n  walletId: string;\n  sourceCoin: string;\n  recoveryCoin: string;\n  recoveryAddress?: string;\n  recoveryAmount?: TNumber;\n}\n\ntype WalletV1 = {\n  keychains: { xpub: string }[];\n  address({ address }: { address: string }): Promise<{ chain: number; index: number }>;\n  getEncryptedUserKeychain(): Promise<{ encryptedXprv: string }>;\n};\n\nexport async function getWallet(\n  bitgo: BitGoBase,\n  coin: AbstractUtxoCoin,\n  walletId: string\n): Promise<IWallet | WalletV1> {\n  try {\n    return await coin.wallets().get({ id: walletId });\n  } catch (e) {\n    // TODO: BG-46364 handle errors more gracefully\n    // The v2 endpoint coin.wallets().get() may throw 404 or 400 errors, but this should not prevent us from searching for the walletId in v1 wallets.\n    if (e.status >= 500) {\n      throw e;\n    }\n  }\n\n  try {\n    return await bitgo.wallets().get({ id: walletId });\n  } catch (e) {\n    throw new Error(`could not get wallet ${walletId} from v1 or v2: ${e.toString()}`);\n  }\n}\n\n/**\n * @param recoveryCoin\n * @param wallet\n * @return wallet pubkeys\n */\nexport async function getWalletKeys(\n  recoveryCoin: AbstractUtxoCoin,\n  wallet: IWallet | WalletV1\n): Promise<RootWalletKeys> {\n  let xpubs: Triple<string>;\n\n  if (wallet instanceof Wallet) {\n    const keychains = (await recoveryCoin.keychains().getKeysForSigning({ wallet })) as unknown as Keychain[];\n    if (keychains.length !== 3) {\n      throw new Error(`expected triple got ${keychains.length}`);\n    }\n    xpubs = keychains.map((k) => k.pub) as Triple<string>;\n  } else {\n    xpubs = (wallet as WalletV1).keychains.map((k) => k.xpub) as Triple<string>;\n  }\n\n  return new utxolib.bitgo.RootWalletKeys(xpubs.map((k) => bip32.fromBase58(k)) as Triple<BIP32Interface>);\n}\n\nexport async function isWalletAddress(wallet: IWallet | WalletV1, address: string): Promise<boolean> {\n  try {\n    let addressData;\n    if (wallet instanceof Wallet) {\n      addressData = await wallet.getAddress({ address });\n    } else {\n      addressData = await (wallet as WalletV1).address({ address });\n    }\n\n    return addressData !== undefined;\n  } catch (e) {\n    return false;\n  }\n}\n\n/**\n * Convert a Litecoin P2SH address from M... format (scriptHash 0x32) to the legacy 3... format (scriptHash 0x05).\n * This is needed for cross-chain recovery when LTC was sent to a BTC address, because the BTC wallet\n * stores addresses in the 3... format while the LTC blockchain returns addresses in M... format.\n *\n * @param address - LTC address to convert\n * @param network - The Litecoin network\n * @returns The address in legacy 3... format, or the original address if it's not a P2SH address\n */\nexport function convertLtcAddressToLegacyFormat(address: string, network: utxolib.Network): string {\n  try {\n    // Try to decode as bech32 - these don't need conversion\n    utxolib.address.fromBech32(address);\n    return address;\n  } catch (e) {\n    // Not bech32, continue to base58\n  }\n\n  try {\n    const decoded = utxolib.address.fromBase58Check(address, network);\n    // Only convert P2SH addresses (scriptHash), not P2PKH (pubKeyHash)\n    if (decoded.version === network.scriptHash) {\n      // Convert to legacy format using Bitcoin's scriptHash (0x05)\n      const legacyScriptHash = utxolib.networks.bitcoin.scriptHash;\n      return utxolib.address.toBase58Check(decoded.hash, legacyScriptHash, network);\n    }\n    // P2PKH or other - return unchanged\n    return address;\n  } catch (e) {\n    // If decoding fails, return the original address\n    return address;\n  }\n}\n\n/**\n * @param coin\n * @param txid\n * @param amountType\n * @param wallet\n * @param apiKey - a blockchair api key\n * @return all unspents for transaction outputs, including outputs from other transactions\n */\nasync function getAllRecoveryOutputs<TNumber extends number | bigint = number>(\n  coin: AbstractUtxoCoin,\n  txid: string,\n  amountType: 'number' | 'bigint' = 'number',\n  wallet: IWallet | WalletV1,\n  apiKey?: string\n): Promise<Unspent<TNumber>[]> {\n  const api = coin.getRecoveryProvider(apiKey);\n  const tx = await api.getTransactionIO(txid);\n  const walletAddresses = (\n    await Promise.all(\n      tx.outputs.map(async (output) => {\n        // For some coins (bch) we need to convert the address to legacy format since the api returns the address\n        // in non legacy format. However, we want to keep the address in the same format as the response since we\n        // are going to hit the API again to fetch address unspents.\n        const canonicalAddress = coin.canonicalAddress(output.address);\n        let isWalletOwned = await isWalletAddress(wallet, canonicalAddress);\n\n        // For LTC cross-chain recovery: if the address isn't found, try the legacy format.\n        // When LTC is sent to a BTC address, the LTC blockchain returns M... addresses\n        // but the BTC wallet stores addresses in 3... format.\n        if (!isWalletOwned && coin.getFamily() === 'ltc') {\n          const legacyAddress = convertLtcAddressToLegacyFormat(output.address, coin.network);\n          if (legacyAddress !== output.address) {\n            isWalletOwned = await isWalletAddress(wallet, legacyAddress);\n          }\n        }\n\n        return isWalletOwned ? output.address : null;\n      })\n    )\n  ).filter((address) => address !== null);\n\n  const unspents = await api.getUnspentsForAddresses(walletAddresses as string[]);\n  if (unspents.length === 0) {\n    throw new Error(`No recovery unspents found.`);\n  }\n  // the api may return cashaddr's instead of legacy for BCH and BCHA\n  // downstream processes's only expect legacy addresses\n  return unspents.map((recoveryOutput) => {\n    return {\n      ...recoveryOutput,\n      address: coin.canonicalAddress(recoveryOutput.address),\n      value: utxolib.bitgo.toTNumber<TNumber>(BigInt(recoveryOutput.value), amountType),\n    };\n  });\n}\n\n/**\n * Data required for address and signature derivation\n */\ntype ScriptId = {\n  chain: number;\n  index: number;\n};\n\nasync function getScriptId(coin: AbstractUtxoCoin, wallet: IWallet | WalletV1, script: Buffer): Promise<ScriptId> {\n  const address = utxolib.address.fromOutputScript(script, coin.network);\n  let addressData: { chain: number; index: number };\n  if (wallet instanceof Wallet) {\n    addressData = await wallet.getAddress({ address });\n  } else {\n    addressData = await (wallet as WalletV1).address({ address });\n  }\n  if (typeof addressData.chain === 'number' && typeof addressData.index === 'number') {\n    return { chain: addressData.chain, index: addressData.index };\n  }\n\n  throw new Error(`invalid address data: ${JSON.stringify(addressData)}`);\n}\n\n/**\n * Lookup address data from unspents on sourceCoin in address database of recoveryCoin.\n * Return full walletUnspents including scriptId in sourceCoin format.\n *\n * @param sourceCoin\n * @param recoveryCoin\n * @param unspents\n * @param wallet\n * @return walletUnspents\n */\nasync function toWalletUnspents<TNumber extends number | bigint = number>(\n  sourceCoin: AbstractUtxoCoin,\n  recoveryCoin: AbstractUtxoCoin,\n  unspents: Unspent<TNumber>[],\n  wallet: IWallet | WalletV1\n): Promise<WalletUnspent<TNumber>[]> {\n  const addresses = new Set(unspents.map((u) => u.address));\n  const walletUnspents: WalletUnspent<TNumber>[] = [];\n\n  for (const address of addresses) {\n    let scriptId;\n    try {\n      scriptId = await getScriptId(recoveryCoin, wallet, utxolib.address.toOutputScript(address, sourceCoin.network));\n    } catch (e) {\n      console.error(`error getting scriptId for ${address}:`, e);\n      continue;\n    }\n    const filteredUnspents = unspents\n      .filter((u) => u.address === address)\n      .map((u) => ({\n        ...u,\n        ...scriptId,\n      }));\n    walletUnspents.push(...filteredUnspents);\n  }\n\n  return walletUnspents;\n}\n\n/**\n * @param coin\n * @return feeRate for transaction\n */\nasync function getFeeRateSatVB(coin: AbstractUtxoCoin): Promise<number> {\n  // TODO: use feeRate API\n  const feeRate = {\n    bch: 20,\n    tbch: 20,\n    bcha: 20,\n    tbcha: 20,\n    bsv: 20,\n    tbsv: 20,\n    btc: 80,\n    tbtc: 80,\n    tbtcsig: 80,\n    tbtc4: 80,\n    tbtcbgsig: 80,\n    ltc: 100,\n    tltc: 100,\n    doge: 1000,\n    tdoge: 1000,\n  }[coin.getChain()];\n\n  if (!feeRate) {\n    throw new Error(`no feeRate for ${coin.getChain()}`);\n  }\n\n  return feeRate;\n}\n\n/**\n * @param xprv\n * @param passphrase\n * @param wallet\n * @return signing key\n */\nasync function getPrv(xprv?: string, passphrase?: string, wallet?: IWallet | WalletV1): Promise<BIP32Interface> {\n  if (xprv) {\n    const key = bip32.fromBase58(xprv);\n    if (key.isNeutered()) {\n      throw new Error(`not a private key`);\n    }\n    return key;\n  }\n\n  if (!wallet || !passphrase) {\n    throw new Error(`no xprv given: need wallet and passphrase to continue`);\n  }\n\n  let encryptedPrv: string;\n  if (wallet instanceof Wallet) {\n    encryptedPrv = (await wallet.getEncryptedUserKeychain()).encryptedPrv;\n  } else {\n    encryptedPrv = (await (wallet as WalletV1).getEncryptedUserKeychain()).encryptedXprv;\n  }\n\n  return getPrv(decrypt(passphrase, encryptedPrv));\n}\n\n/**\n * Create a sweep transaction for cross-chain recovery using PSBT (utxolib implementation)\n * @param network\n * @param walletKeys\n * @param unspents\n * @param targetAddress\n * @param feeRateSatVB\n * @return unsigned PSBT\n */\nfunction createSweepTransactionUtxolib<TNumber extends number | bigint = number>(\n  network: utxolib.Network,\n  walletKeys: RootWalletKeys,\n  unspents: WalletUnspent<TNumber>[],\n  targetAddress: string,\n  feeRateSatVB: number\n): utxolib.bitgo.UtxoPsbt {\n  const inputValue = unspentSum<bigint>(\n    unspents.map((u) => ({ ...u, value: BigInt(u.value) })),\n    'bigint'\n  );\n  const vsize = Dimensions.fromUnspents(unspents, {\n    p2tr: { scriptPathLevel: 1 },\n    p2trMusig2: { scriptPathLevel: undefined },\n  })\n    .plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))\n    .getVSize();\n  const fee = BigInt(Math.round(vsize * feeRateSatVB));\n\n  const psbt = utxolib.bitgo.createPsbtForNetwork({ network });\n  utxolib.bitgo.addXpubsToPsbt(psbt, walletKeys);\n\n  unspents.forEach((unspent) => {\n    utxolib.bitgo.addWalletUnspentToPsbt(\n      psbt,\n      { ...unspent, value: BigInt(unspent.value) },\n      walletKeys,\n      'user',\n      'backup',\n      { skipNonWitnessUtxo: true }\n    );\n  });\n\n  const recoveryOutputScript = utxolib.address.toOutputScript(targetAddress, network);\n  psbt.addOutput({ script: recoveryOutputScript, value: inputValue - fee });\n\n  return psbt;\n}\n\n/**\n * Create a sweep transaction for cross-chain recovery using wasm-utxo\n * @param network\n * @param walletKeys\n * @param unspents\n * @param targetAddress\n * @param feeRateSatVB\n * @return unsigned PSBT\n */\nfunction createSweepTransactionWasm<TNumber extends number | bigint = number>(\n  network: utxolib.Network,\n  walletKeys: RootWalletKeys,\n  unspents: WalletUnspent<TNumber>[],\n  targetAddress: string,\n  feeRateSatVB: number\n): utxolib.bitgo.UtxoPsbt {\n  const inputValue = unspentSum<bigint>(\n    unspents.map((u) => ({ ...u, value: BigInt(u.value) })),\n    'bigint'\n  );\n\n  // Create PSBT with wasm-utxo and add wallet inputs using shared utilities\n  const unspentsBigint = unspents.map((u) => ({ ...u, value: BigInt(u.value) }));\n  const wasmPsbt = createEmptyWasmPsbt(network, walletKeys);\n  addWalletInputsToWasmPsbt(wasmPsbt, unspentsBigint, walletKeys);\n\n  // Convert to utxolib PSBT temporarily for dimension calculation\n  const tempPsbt = wasmPsbtToUtxolibPsbt(wasmPsbt, network);\n  const vsize = Dimensions.fromPsbt(tempPsbt)\n    .plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))\n    .getVSize();\n  const fee = BigInt(Math.round(vsize * feeRateSatVB));\n\n  // Add output to wasm PSBT\n  addOutputToWasmPsbt(wasmPsbt, targetAddress, inputValue - fee, network);\n\n  // Convert to utxolib PSBT for signing and return\n  return wasmPsbtToUtxolibPsbt(wasmPsbt, network);\n}\n\n/**\n * Create a sweep transaction for cross-chain recovery using PSBT\n * @param network\n * @param walletKeys\n * @param unspents\n * @param targetAddress\n * @param feeRateSatVB\n * @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')\n * @return unsigned PSBT\n */\nfunction createSweepTransaction<TNumber extends number | bigint = number>(\n  network: utxolib.Network,\n  walletKeys: RootWalletKeys,\n  unspents: WalletUnspent<TNumber>[],\n  targetAddress: string,\n  feeRateSatVB: number,\n  backend: PsbtBackend = 'wasm-utxo'\n): utxolib.bitgo.UtxoPsbt {\n  if (backend === 'wasm-utxo') {\n    return createSweepTransactionWasm(network, walletKeys, unspents, targetAddress, feeRateSatVB);\n  } else {\n    return createSweepTransactionUtxolib(network, walletKeys, unspents, targetAddress, feeRateSatVB);\n  }\n}\n\ntype RecoverParams = {\n  /** Wallet ID (can be v1 wallet or v2 wallet) */\n  walletId: string;\n  /** Coin to create the transaction for */\n  sourceCoin: AbstractUtxoCoin;\n  /** Coin that wallet keys were set up for */\n  recoveryCoin: AbstractUtxoCoin;\n  /** Source coin transaction to recover outputs from (sourceCoin) */\n  txid: string;\n  /** Source coin address to send the funds to */\n  recoveryAddress: string;\n  /** If set, decrypts private key and signs transaction */\n  walletPassphrase?: string;\n  /** If set, signs transaction */\n  xprv?: string;\n  /** for utxo coins other than [BTC,TBTC] this is a Block Chair api key **/\n  apiKey?: string;\n};\n\n/**\n * Recover wallet deposits that were received on the wrong blockchain\n * (for instance bitcoin deposits that were received for a litecoin wallet).\n *\n * Fetches the unspent data from BitGo's public blockchain API and the script data from the user's\n * wallet.\n *\n * @param {BitGoBase} bitgo\n * @param {RecoverParams} params\n */\nexport async function recoverCrossChain<TNumber extends number | bigint = number>(\n  bitgo: BitGoBase,\n  params: RecoverParams\n): Promise<CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>> {\n  const wallet = await getWallet(bitgo, params.recoveryCoin, params.walletId);\n  const unspents = await getAllRecoveryOutputs<TNumber>(\n    params.sourceCoin,\n    params.txid,\n    params.sourceCoin.amountType,\n    wallet,\n    params.apiKey\n  );\n  const walletUnspents = await toWalletUnspents<TNumber>(params.sourceCoin, params.recoveryCoin, unspents, wallet);\n  const walletKeys = await getWalletKeys(params.recoveryCoin, wallet);\n  const prv =\n    params.xprv || params.walletPassphrase ? await getPrv(params.xprv, params.walletPassphrase, wallet) : undefined;\n  const feeRateSatVB = await getFeeRateSatVB(params.sourceCoin);\n\n  // Create PSBT for both signed and unsigned recovery\n  // Use wasm-utxo for testnet coins only, utxolib for mainnet\n  const backend: PsbtBackend = utxolib.isTestnet(params.sourceCoin.network) ? 'wasm-utxo' : 'utxolib';\n  const psbt = createSweepTransaction<TNumber>(\n    params.sourceCoin.network,\n    walletKeys,\n    walletUnspents,\n    params.recoveryAddress,\n    feeRateSatVB,\n    backend\n  );\n\n  // For unsigned recovery, return unsigned PSBT hex\n  if (!prv) {\n    return {\n      txHex: psbt.toHex(),\n      walletId: params.walletId,\n      address: params.recoveryAddress,\n      coin: params.sourceCoin.getChain(),\n    };\n  }\n\n  // For signed recovery, sign the PSBT with user key and return half-signed PSBT\n  signAndVerifyPsbt(psbt, prv, { isLastSignature: false });\n  const recoveryAmount = utxolib.bitgo.toTNumber<TNumber>(psbt.txOutputs[0].value, params.sourceCoin.amountType);\n\n  return {\n    version: wallet instanceof Wallet ? 2 : 1,\n    walletId: params.walletId,\n    txHex: psbt.toHex(),\n    sourceCoin: params.sourceCoin.getChain(),\n    recoveryCoin: params.recoveryCoin.getChain(),\n    recoveryAmount,\n  };\n}\n"]}
|
|
@@ -1,12 +1,22 @@
|
|
|
1
1
|
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
+
import { fixedScriptWallet, utxolibCompat } from '@bitgo/wasm-utxo';
|
|
2
3
|
type RootWalletKeys = utxolib.bitgo.RootWalletKeys;
|
|
3
4
|
type WalletUnspent<TNumber extends number | bigint> = utxolib.bitgo.WalletUnspent<TNumber>;
|
|
5
|
+
type ChainCode = utxolib.bitgo.ChainCode;
|
|
4
6
|
/**
|
|
5
7
|
* Backend to use for PSBT creation.
|
|
6
8
|
* - 'wasm-utxo': Use wasm-utxo for PSBT creation (default)
|
|
7
9
|
* - 'utxolib': Use utxolib for PSBT creation (legacy)
|
|
8
10
|
*/
|
|
9
11
|
export type PsbtBackend = 'wasm-utxo' | 'utxolib';
|
|
12
|
+
/**
|
|
13
|
+
* Check if a chain code is for a taproot script type
|
|
14
|
+
*/
|
|
15
|
+
export declare function isTaprootChain(chain: ChainCode): boolean;
|
|
16
|
+
/**
|
|
17
|
+
* Convert utxolib Network to wasm-utxo network name
|
|
18
|
+
*/
|
|
19
|
+
export declare function toNetworkName(network: utxolib.Network): utxolibCompat.UtxolibName;
|
|
10
20
|
interface CreateBackupKeyRecoveryPsbtOptions {
|
|
11
21
|
feeRateSatVB: number;
|
|
12
22
|
recoveryDestination: string;
|
|
@@ -15,6 +25,50 @@ interface CreateBackupKeyRecoveryPsbtOptions {
|
|
|
15
25
|
/** Block height for Zcash networks (required to determine consensus branch ID) */
|
|
16
26
|
blockHeight?: number;
|
|
17
27
|
}
|
|
28
|
+
/**
|
|
29
|
+
* Options for creating an empty wasm-utxo PSBT
|
|
30
|
+
*/
|
|
31
|
+
export interface CreateEmptyWasmPsbtOptions {
|
|
32
|
+
/** Block height for Zcash networks (required to determine consensus branch ID) */
|
|
33
|
+
blockHeight?: number;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Create an empty wasm-utxo BitGoPsbt for a given network.
|
|
37
|
+
* Handles Zcash networks specially by using ZcashBitGoPsbt.
|
|
38
|
+
*
|
|
39
|
+
* @param network - The network for the PSBT
|
|
40
|
+
* @param rootWalletKeys - The wallet keys
|
|
41
|
+
* @param options - Optional settings (e.g., blockHeight for Zcash)
|
|
42
|
+
* @returns A wasm-utxo BitGoPsbt instance
|
|
43
|
+
*/
|
|
44
|
+
export declare function createEmptyWasmPsbt(network: utxolib.Network, rootWalletKeys: RootWalletKeys, options?: CreateEmptyWasmPsbtOptions): fixedScriptWallet.BitGoPsbt;
|
|
45
|
+
/**
|
|
46
|
+
* Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.
|
|
47
|
+
* Handles taproot inputs by setting the appropriate signPath.
|
|
48
|
+
*
|
|
49
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to add inputs to
|
|
50
|
+
* @param unspents - The wallet unspents to add as inputs
|
|
51
|
+
* @param rootWalletKeys - The wallet keys
|
|
52
|
+
*/
|
|
53
|
+
export declare function addWalletInputsToWasmPsbt(wasmPsbt: fixedScriptWallet.BitGoPsbt, unspents: WalletUnspent<bigint>[], rootWalletKeys: RootWalletKeys): void;
|
|
54
|
+
/**
|
|
55
|
+
* Add an output to a wasm-utxo BitGoPsbt.
|
|
56
|
+
*
|
|
57
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to add the output to
|
|
58
|
+
* @param address - The destination address
|
|
59
|
+
* @param value - The output value in satoshis
|
|
60
|
+
* @param network - The network (used to convert address to script)
|
|
61
|
+
* @returns The output index
|
|
62
|
+
*/
|
|
63
|
+
export declare function addOutputToWasmPsbt(wasmPsbt: fixedScriptWallet.BitGoPsbt, address: string, value: bigint, network: utxolib.Network): number;
|
|
64
|
+
/**
|
|
65
|
+
* Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.
|
|
66
|
+
*
|
|
67
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to convert
|
|
68
|
+
* @param network - The network
|
|
69
|
+
* @returns A utxolib UtxoPsbt
|
|
70
|
+
*/
|
|
71
|
+
export declare function wasmPsbtToUtxolibPsbt(wasmPsbt: fixedScriptWallet.BitGoPsbt, network: utxolib.Network): utxolib.bitgo.UtxoPsbt;
|
|
18
72
|
/**
|
|
19
73
|
* Create a backup key recovery PSBT.
|
|
20
74
|
*
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"psbt.d.ts","sourceRoot":"","sources":["../../../../src/recovery/psbt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;
|
|
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,EAAE,MAAM,kBAAkB,CAAC;AAEpE,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;AAoED;;GAEG;AACH,MAAM,WAAW,0BAA0B;IACzC,kFAAkF;IAClF,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED;;;;;;;;GAQG;AACH,wBAAgB,mBAAmB,CACjC,OAAO,EAAE,OAAO,CAAC,OAAO,EACxB,cAAc,EAAE,cAAc,EAC9B,OAAO,CAAC,EAAE,0BAA0B,GACnC,iBAAiB,CAAC,SAAS,CAY7B;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,OAAO,EAAE,OAAO,CAAC,OAAO,GACvB,MAAM,CAGR;AAED;;;;;;GAMG;AACH,wBAAgB,qBAAqB,CACnC,QAAQ,EAAE,iBAAiB,CAAC,SAAS,EACrC,OAAO,EAAE,OAAO,CAAC,OAAO,GACvB,OAAO,CAAC,KAAK,CAAC,QAAQ,CAExB;AAkDD;;;;;;;;GAQG;AACH,wBAAgB,2BAA2B,CACzC,OAAO,EAAE,OAAO,CAAC,OAAO,EACxB,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,CAUxB;AAED,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,CAOvF"}
|
|
@@ -33,6 +33,12 @@ var __importStar = (this && this.__importStar) || (function () {
|
|
|
33
33
|
};
|
|
34
34
|
})();
|
|
35
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.isTaprootChain = isTaprootChain;
|
|
37
|
+
exports.toNetworkName = toNetworkName;
|
|
38
|
+
exports.createEmptyWasmPsbt = createEmptyWasmPsbt;
|
|
39
|
+
exports.addWalletInputsToWasmPsbt = addWalletInputsToWasmPsbt;
|
|
40
|
+
exports.addOutputToWasmPsbt = addOutputToWasmPsbt;
|
|
41
|
+
exports.wasmPsbtToUtxolibPsbt = wasmPsbtToUtxolibPsbt;
|
|
36
42
|
exports.createBackupKeyRecoveryPsbt = createBackupKeyRecoveryPsbt;
|
|
37
43
|
exports.getRecoveryAmount = getRecoveryAmount;
|
|
38
44
|
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
@@ -115,24 +121,34 @@ const ZCASH_DEFAULT_BLOCK_HEIGHTS = {
|
|
|
115
121
|
zcashTest: 3536500,
|
|
116
122
|
};
|
|
117
123
|
/**
|
|
118
|
-
* Create
|
|
124
|
+
* Create an empty wasm-utxo BitGoPsbt for a given network.
|
|
125
|
+
* Handles Zcash networks specially by using ZcashBitGoPsbt.
|
|
126
|
+
*
|
|
127
|
+
* @param network - The network for the PSBT
|
|
128
|
+
* @param rootWalletKeys - The wallet keys
|
|
129
|
+
* @param options - Optional settings (e.g., blockHeight for Zcash)
|
|
130
|
+
* @returns A wasm-utxo BitGoPsbt instance
|
|
119
131
|
*/
|
|
120
|
-
function
|
|
121
|
-
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
132
|
+
function createEmptyWasmPsbt(network, rootWalletKeys, options) {
|
|
122
133
|
const networkName = toNetworkName(network);
|
|
123
|
-
// Create PSBT with wasm-utxo and add wallet inputs
|
|
124
|
-
// wasm-utxo's RootWalletKeys.from() accepts utxolib's RootWalletKeys format (IWalletKeys interface)
|
|
125
|
-
let wasmPsbt;
|
|
126
134
|
if (isZcashNetwork(networkName)) {
|
|
127
135
|
// For Zcash, use ZcashBitGoPsbt which requires block height to determine consensus branch ID
|
|
128
|
-
const blockHeight = options
|
|
129
|
-
|
|
136
|
+
const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[networkName];
|
|
137
|
+
return wasm_utxo_1.fixedScriptWallet.ZcashBitGoPsbt.createEmpty(networkName, rootWalletKeys, {
|
|
130
138
|
blockHeight,
|
|
131
139
|
});
|
|
132
140
|
}
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
141
|
+
return wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty(networkName, rootWalletKeys);
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.
|
|
145
|
+
* Handles taproot inputs by setting the appropriate signPath.
|
|
146
|
+
*
|
|
147
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to add inputs to
|
|
148
|
+
* @param unspents - The wallet unspents to add as inputs
|
|
149
|
+
* @param rootWalletKeys - The wallet keys
|
|
150
|
+
*/
|
|
151
|
+
function addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys) {
|
|
136
152
|
unspents.forEach((unspent) => {
|
|
137
153
|
const { txid, vout } = utxolib.bitgo.parseOutputId(unspent.id);
|
|
138
154
|
const signPath = isTaprootChain(unspent.chain)
|
|
@@ -150,9 +166,41 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
150
166
|
signPath,
|
|
151
167
|
});
|
|
152
168
|
});
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Add an output to a wasm-utxo BitGoPsbt.
|
|
172
|
+
*
|
|
173
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to add the output to
|
|
174
|
+
* @param address - The destination address
|
|
175
|
+
* @param value - The output value in satoshis
|
|
176
|
+
* @param network - The network (used to convert address to script)
|
|
177
|
+
* @returns The output index
|
|
178
|
+
*/
|
|
179
|
+
function addOutputToWasmPsbt(wasmPsbt, address, value, network) {
|
|
180
|
+
const script = utxolib.address.toOutputScript(address, network);
|
|
181
|
+
return wasmPsbt.addOutput({ script: new Uint8Array(script), value });
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.
|
|
185
|
+
*
|
|
186
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to convert
|
|
187
|
+
* @param network - The network
|
|
188
|
+
* @returns A utxolib UtxoPsbt
|
|
189
|
+
*/
|
|
190
|
+
function wasmPsbtToUtxolibPsbt(wasmPsbt, network) {
|
|
191
|
+
return utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Create a backup key recovery PSBT using wasm-utxo
|
|
195
|
+
*/
|
|
196
|
+
function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, options) {
|
|
197
|
+
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
198
|
+
// Create PSBT with wasm-utxo and add wallet inputs using shared utilities
|
|
199
|
+
const wasmPsbt = createEmptyWasmPsbt(network, rootWalletKeys, { blockHeight: options.blockHeight });
|
|
200
|
+
addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys);
|
|
201
|
+
// Convert to utxolib PSBT temporarily for dimension calculation
|
|
202
|
+
const tempPsbt = wasmPsbtToUtxolibPsbt(wasmPsbt, network);
|
|
203
|
+
let dimensions = unspents_1.Dimensions.fromPsbt(tempPsbt).plus(unspents_1.Dimensions.fromOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network) }));
|
|
156
204
|
if (keyRecoveryServiceFeeAddress) {
|
|
157
205
|
dimensions = dimensions.plus(unspents_1.Dimensions.fromOutput({
|
|
158
206
|
script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),
|
|
@@ -164,14 +212,13 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
164
212
|
if (recoveryAmount < BigInt(0)) {
|
|
165
213
|
throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);
|
|
166
214
|
}
|
|
167
|
-
|
|
215
|
+
// Add outputs to wasm PSBT
|
|
216
|
+
addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount, network);
|
|
168
217
|
if (keyRecoveryServiceFeeAddress) {
|
|
169
|
-
|
|
170
|
-
script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),
|
|
171
|
-
value: keyRecoveryServiceFee,
|
|
172
|
-
});
|
|
218
|
+
addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee, network);
|
|
173
219
|
}
|
|
174
|
-
return
|
|
220
|
+
// Convert to utxolib PSBT for signing and return
|
|
221
|
+
return wasmPsbtToUtxolibPsbt(wasmPsbt, network);
|
|
175
222
|
}
|
|
176
223
|
/**
|
|
177
224
|
* Create a backup key recovery PSBT.
|
|
@@ -201,4 +248,4 @@ function getRecoveryAmount(psbt, address) {
|
|
|
201
248
|
}
|
|
202
249
|
return output.value;
|
|
203
250
|
}
|
|
204
|
-
//# 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 */\nfunction 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 */\nfunction 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 * 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  const networkName = toNetworkName(network);\n\n  // Create PSBT with wasm-utxo and add wallet inputs\n  // wasm-utxo's RootWalletKeys.from() accepts utxolib's RootWalletKeys format (IWalletKeys interface)\n  let wasmPsbt: fixedScriptWallet.BitGoPsbt;\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    wasmPsbt = fixedScriptWallet.ZcashBitGoPsbt.createEmpty(networkName as 'zcash' | 'zcashTest', rootWalletKeys, {\n      blockHeight,\n    });\n  } else {\n    wasmPsbt = fixedScriptWallet.BitGoPsbt.createEmpty(networkName, rootWalletKeys);\n  }\n\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  // Convert wasm-utxo PSBT to utxolib PSBT for dimension calculation and output addition\n  const psbt = utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);\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 * 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"]}
|
|
251
|
+
//# 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  // Convert to utxolib PSBT temporarily for dimension calculation\n  const tempPsbt = wasmPsbtToUtxolibPsbt(wasmPsbt, network);\n  let dimensions = Dimensions.fromPsbt(tempPsbt).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  // 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"]}
|