@bitgo-beta/abstract-utxo 1.6.1-alpha.446 → 1.6.1-alpha.448
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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"}
|
|
@@ -4,6 +4,7 @@ import { Dimensions } from '@bitgo-beta/unspents';
|
|
|
4
4
|
import { Wallet } from '@bitgo-beta/sdk-core';
|
|
5
5
|
import { decrypt } from '@bitgo-beta/sdk-api';
|
|
6
6
|
import { signAndVerifyPsbt } from '../transaction/fixedScript/signPsbt';
|
|
7
|
+
import { createEmptyWasmPsbt, addWalletInputsToWasmPsbt, addOutputToWasmPsbt, wasmPsbtToUtxolibPsbt, } from './psbt';
|
|
7
8
|
const { unspentSum } = utxolib.bitgo;
|
|
8
9
|
export async function getWallet(bitgo, coin, walletId) {
|
|
9
10
|
try {
|
|
@@ -234,7 +235,7 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
234
235
|
return getPrv(decrypt(passphrase, encryptedPrv));
|
|
235
236
|
}
|
|
236
237
|
/**
|
|
237
|
-
* Create a sweep transaction for cross-chain recovery using PSBT
|
|
238
|
+
* Create a sweep transaction for cross-chain recovery using PSBT (utxolib implementation)
|
|
238
239
|
* @param network
|
|
239
240
|
* @param walletKeys
|
|
240
241
|
* @param unspents
|
|
@@ -242,7 +243,7 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
242
243
|
* @param feeRateSatVB
|
|
243
244
|
* @return unsigned PSBT
|
|
244
245
|
*/
|
|
245
|
-
function
|
|
246
|
+
function createSweepTransactionUtxolib(network, walletKeys, unspents, targetAddress, feeRateSatVB) {
|
|
246
247
|
const inputValue = unspentSum(unspents.map((u) => ({ ...u, value: BigInt(u.value) })), 'bigint');
|
|
247
248
|
const vsize = Dimensions.fromUnspents(unspents, {
|
|
248
249
|
p2tr: { scriptPathLevel: 1 },
|
|
@@ -260,6 +261,50 @@ function createSweepTransaction(network, walletKeys, unspents, targetAddress, fe
|
|
|
260
261
|
psbt.addOutput({ script: recoveryOutputScript, value: inputValue - fee });
|
|
261
262
|
return psbt;
|
|
262
263
|
}
|
|
264
|
+
/**
|
|
265
|
+
* Create a sweep transaction for cross-chain recovery using wasm-utxo
|
|
266
|
+
* @param network
|
|
267
|
+
* @param walletKeys
|
|
268
|
+
* @param unspents
|
|
269
|
+
* @param targetAddress
|
|
270
|
+
* @param feeRateSatVB
|
|
271
|
+
* @return unsigned PSBT
|
|
272
|
+
*/
|
|
273
|
+
function createSweepTransactionWasm(network, walletKeys, unspents, targetAddress, feeRateSatVB) {
|
|
274
|
+
const inputValue = unspentSum(unspents.map((u) => ({ ...u, value: BigInt(u.value) })), 'bigint');
|
|
275
|
+
// Create PSBT with wasm-utxo and add wallet inputs using shared utilities
|
|
276
|
+
const unspentsBigint = unspents.map((u) => ({ ...u, value: BigInt(u.value) }));
|
|
277
|
+
const wasmPsbt = createEmptyWasmPsbt(network, walletKeys);
|
|
278
|
+
addWalletInputsToWasmPsbt(wasmPsbt, unspentsBigint, walletKeys);
|
|
279
|
+
// Convert to utxolib PSBT temporarily for dimension calculation
|
|
280
|
+
const tempPsbt = wasmPsbtToUtxolibPsbt(wasmPsbt, network);
|
|
281
|
+
const vsize = Dimensions.fromPsbt(tempPsbt)
|
|
282
|
+
.plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))
|
|
283
|
+
.getVSize();
|
|
284
|
+
const fee = BigInt(Math.round(vsize * feeRateSatVB));
|
|
285
|
+
// Add output to wasm PSBT
|
|
286
|
+
addOutputToWasmPsbt(wasmPsbt, targetAddress, inputValue - fee, network);
|
|
287
|
+
// Convert to utxolib PSBT for signing and return
|
|
288
|
+
return wasmPsbtToUtxolibPsbt(wasmPsbt, network);
|
|
289
|
+
}
|
|
290
|
+
/**
|
|
291
|
+
* Create a sweep transaction for cross-chain recovery using PSBT
|
|
292
|
+
* @param network
|
|
293
|
+
* @param walletKeys
|
|
294
|
+
* @param unspents
|
|
295
|
+
* @param targetAddress
|
|
296
|
+
* @param feeRateSatVB
|
|
297
|
+
* @param backend - Which backend to use for PSBT creation (default: 'wasm-utxo')
|
|
298
|
+
* @return unsigned PSBT
|
|
299
|
+
*/
|
|
300
|
+
function createSweepTransaction(network, walletKeys, unspents, targetAddress, feeRateSatVB, backend = 'wasm-utxo') {
|
|
301
|
+
if (backend === 'wasm-utxo') {
|
|
302
|
+
return createSweepTransactionWasm(network, walletKeys, unspents, targetAddress, feeRateSatVB);
|
|
303
|
+
}
|
|
304
|
+
else {
|
|
305
|
+
return createSweepTransactionUtxolib(network, walletKeys, unspents, targetAddress, feeRateSatVB);
|
|
306
|
+
}
|
|
307
|
+
}
|
|
263
308
|
/**
|
|
264
309
|
* Recover wallet deposits that were received on the wrong blockchain
|
|
265
310
|
* (for instance bitcoin deposits that were received for a litecoin wallet).
|
|
@@ -278,7 +323,9 @@ export async function recoverCrossChain(bitgo, params) {
|
|
|
278
323
|
const prv = params.xprv || params.walletPassphrase ? await getPrv(params.xprv, params.walletPassphrase, wallet) : undefined;
|
|
279
324
|
const feeRateSatVB = await getFeeRateSatVB(params.sourceCoin);
|
|
280
325
|
// Create PSBT for both signed and unsigned recovery
|
|
281
|
-
|
|
326
|
+
// Use wasm-utxo for testnet coins only, utxolib for mainnet
|
|
327
|
+
const backend = utxolib.isTestnet(params.sourceCoin.network) ? 'wasm-utxo' : 'utxolib';
|
|
328
|
+
const psbt = createSweepTransaction(params.sourceCoin.network, walletKeys, walletUnspents, params.recoveryAddress, feeRateSatVB, backend);
|
|
282
329
|
// For unsigned recovery, return unsigned PSBT hex
|
|
283
330
|
if (!prv) {
|
|
284
331
|
return {
|
|
@@ -300,4 +347,4 @@ export async function recoverCrossChain(bitgo, params) {
|
|
|
300
347
|
recoveryAmount,
|
|
301
348
|
};
|
|
302
349
|
}
|
|
303
|
-
//# 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"]}
|
|
350
|
+
//# 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"}
|
|
@@ -5,13 +5,13 @@ const { chainCodesP2tr, chainCodesP2trMusig2 } = utxolib.bitgo;
|
|
|
5
5
|
/**
|
|
6
6
|
* Check if a chain code is for a taproot script type
|
|
7
7
|
*/
|
|
8
|
-
function isTaprootChain(chain) {
|
|
8
|
+
export function isTaprootChain(chain) {
|
|
9
9
|
return (chainCodesP2tr.includes(chain) || chainCodesP2trMusig2.includes(chain));
|
|
10
10
|
}
|
|
11
11
|
/**
|
|
12
12
|
* Convert utxolib Network to wasm-utxo network name
|
|
13
13
|
*/
|
|
14
|
-
function toNetworkName(network) {
|
|
14
|
+
export function toNetworkName(network) {
|
|
15
15
|
const networkName = utxolib.getNetworkName(network);
|
|
16
16
|
if (!networkName) {
|
|
17
17
|
throw new Error(`Invalid network`);
|
|
@@ -78,24 +78,34 @@ const ZCASH_DEFAULT_BLOCK_HEIGHTS = {
|
|
|
78
78
|
zcashTest: 3536500,
|
|
79
79
|
};
|
|
80
80
|
/**
|
|
81
|
-
* Create
|
|
81
|
+
* Create an empty wasm-utxo BitGoPsbt for a given network.
|
|
82
|
+
* Handles Zcash networks specially by using ZcashBitGoPsbt.
|
|
83
|
+
*
|
|
84
|
+
* @param network - The network for the PSBT
|
|
85
|
+
* @param rootWalletKeys - The wallet keys
|
|
86
|
+
* @param options - Optional settings (e.g., blockHeight for Zcash)
|
|
87
|
+
* @returns A wasm-utxo BitGoPsbt instance
|
|
82
88
|
*/
|
|
83
|
-
function
|
|
84
|
-
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
89
|
+
export function createEmptyWasmPsbt(network, rootWalletKeys, options) {
|
|
85
90
|
const networkName = toNetworkName(network);
|
|
86
|
-
// Create PSBT with wasm-utxo and add wallet inputs
|
|
87
|
-
// wasm-utxo's RootWalletKeys.from() accepts utxolib's RootWalletKeys format (IWalletKeys interface)
|
|
88
|
-
let wasmPsbt;
|
|
89
91
|
if (isZcashNetwork(networkName)) {
|
|
90
92
|
// For Zcash, use ZcashBitGoPsbt which requires block height to determine consensus branch ID
|
|
91
|
-
const blockHeight = options
|
|
92
|
-
|
|
93
|
+
const blockHeight = options?.blockHeight ?? ZCASH_DEFAULT_BLOCK_HEIGHTS[networkName];
|
|
94
|
+
return fixedScriptWallet.ZcashBitGoPsbt.createEmpty(networkName, rootWalletKeys, {
|
|
93
95
|
blockHeight,
|
|
94
96
|
});
|
|
95
97
|
}
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
98
|
+
return fixedScriptWallet.BitGoPsbt.createEmpty(networkName, rootWalletKeys);
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Add wallet inputs from unspents to a wasm-utxo BitGoPsbt.
|
|
102
|
+
* Handles taproot inputs by setting the appropriate signPath.
|
|
103
|
+
*
|
|
104
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to add inputs to
|
|
105
|
+
* @param unspents - The wallet unspents to add as inputs
|
|
106
|
+
* @param rootWalletKeys - The wallet keys
|
|
107
|
+
*/
|
|
108
|
+
export function addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys) {
|
|
99
109
|
unspents.forEach((unspent) => {
|
|
100
110
|
const { txid, vout } = utxolib.bitgo.parseOutputId(unspent.id);
|
|
101
111
|
const signPath = isTaprootChain(unspent.chain)
|
|
@@ -113,9 +123,41 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
113
123
|
signPath,
|
|
114
124
|
});
|
|
115
125
|
});
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Add an output to a wasm-utxo BitGoPsbt.
|
|
129
|
+
*
|
|
130
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to add the output to
|
|
131
|
+
* @param address - The destination address
|
|
132
|
+
* @param value - The output value in satoshis
|
|
133
|
+
* @param network - The network (used to convert address to script)
|
|
134
|
+
* @returns The output index
|
|
135
|
+
*/
|
|
136
|
+
export function addOutputToWasmPsbt(wasmPsbt, address, value, network) {
|
|
137
|
+
const script = utxolib.address.toOutputScript(address, network);
|
|
138
|
+
return wasmPsbt.addOutput({ script: new Uint8Array(script), value });
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Convert a wasm-utxo BitGoPsbt to a utxolib UtxoPsbt.
|
|
142
|
+
*
|
|
143
|
+
* @param wasmPsbt - The wasm-utxo BitGoPsbt to convert
|
|
144
|
+
* @param network - The network
|
|
145
|
+
* @returns A utxolib UtxoPsbt
|
|
146
|
+
*/
|
|
147
|
+
export function wasmPsbtToUtxolibPsbt(wasmPsbt, network) {
|
|
148
|
+
return utxolib.bitgo.createPsbtFromBuffer(Buffer.from(wasmPsbt.serialize()), network);
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Create a backup key recovery PSBT using wasm-utxo
|
|
152
|
+
*/
|
|
153
|
+
function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, options) {
|
|
154
|
+
const { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress } = options;
|
|
155
|
+
// Create PSBT with wasm-utxo and add wallet inputs using shared utilities
|
|
156
|
+
const wasmPsbt = createEmptyWasmPsbt(network, rootWalletKeys, { blockHeight: options.blockHeight });
|
|
157
|
+
addWalletInputsToWasmPsbt(wasmPsbt, unspents, rootWalletKeys);
|
|
158
|
+
// Convert to utxolib PSBT temporarily for dimension calculation
|
|
159
|
+
const tempPsbt = wasmPsbtToUtxolibPsbt(wasmPsbt, network);
|
|
160
|
+
let dimensions = Dimensions.fromPsbt(tempPsbt).plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network) }));
|
|
119
161
|
if (keyRecoveryServiceFeeAddress) {
|
|
120
162
|
dimensions = dimensions.plus(Dimensions.fromOutput({
|
|
121
163
|
script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),
|
|
@@ -127,14 +169,13 @@ function createBackupKeyRecoveryPsbtWasm(network, rootWalletKeys, unspents, opti
|
|
|
127
169
|
if (recoveryAmount < BigInt(0)) {
|
|
128
170
|
throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);
|
|
129
171
|
}
|
|
130
|
-
|
|
172
|
+
// Add outputs to wasm PSBT
|
|
173
|
+
addOutputToWasmPsbt(wasmPsbt, recoveryDestination, recoveryAmount, network);
|
|
131
174
|
if (keyRecoveryServiceFeeAddress) {
|
|
132
|
-
|
|
133
|
-
script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),
|
|
134
|
-
value: keyRecoveryServiceFee,
|
|
135
|
-
});
|
|
175
|
+
addOutputToWasmPsbt(wasmPsbt, keyRecoveryServiceFeeAddress, keyRecoveryServiceFee, network);
|
|
136
176
|
}
|
|
137
|
-
return
|
|
177
|
+
// Convert to utxolib PSBT for signing and return
|
|
178
|
+
return wasmPsbtToUtxolibPsbt(wasmPsbt, network);
|
|
138
179
|
}
|
|
139
180
|
/**
|
|
140
181
|
* Create a backup key recovery PSBT.
|
|
@@ -164,4 +205,4 @@ export function getRecoveryAmount(psbt, address) {
|
|
|
164
205
|
}
|
|
165
206
|
return output.value;
|
|
166
207
|
}
|
|
167
|
-
//# 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"]}
|
|
208
|
+
//# 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"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@bitgo-beta/abstract-utxo",
|
|
3
|
-
"version": "1.6.1-alpha.
|
|
3
|
+
"version": "1.6.1-alpha.448",
|
|
4
4
|
"description": "BitGo SDK coin library for UTXO base implementation",
|
|
5
5
|
"main": "./dist/cjs/src/index.js",
|
|
6
6
|
"module": "./dist/esm/index.js",
|
|
@@ -60,15 +60,15 @@
|
|
|
60
60
|
]
|
|
61
61
|
},
|
|
62
62
|
"dependencies": {
|
|
63
|
-
"@bitgo-beta/blockapis": "1.3.3-alpha.
|
|
64
|
-
"@bitgo-beta/sdk-api": "1.6.1-alpha.
|
|
65
|
-
"@bitgo-beta/sdk-core": "2.4.1-alpha.
|
|
66
|
-
"@bitgo-beta/secp256k1": "1.0.1-alpha.
|
|
67
|
-
"@bitgo-beta/unspents": "0.11.3-alpha.
|
|
68
|
-
"@bitgo-beta/utxo-core": "1.0.1-alpha.
|
|
69
|
-
"@bitgo-beta/utxo-lib": "4.0.1-alpha.
|
|
70
|
-
"@bitgo-beta/utxo-ord": "1.1.3-alpha.
|
|
71
|
-
"@bitgo/wasm-utxo": "1.
|
|
63
|
+
"@bitgo-beta/blockapis": "1.3.3-alpha.449",
|
|
64
|
+
"@bitgo-beta/sdk-api": "1.6.1-alpha.448",
|
|
65
|
+
"@bitgo-beta/sdk-core": "2.4.1-alpha.448",
|
|
66
|
+
"@bitgo-beta/secp256k1": "1.0.1-alpha.406",
|
|
67
|
+
"@bitgo-beta/unspents": "0.11.3-alpha.449",
|
|
68
|
+
"@bitgo-beta/utxo-core": "1.0.1-alpha.179",
|
|
69
|
+
"@bitgo-beta/utxo-lib": "4.0.1-alpha.449",
|
|
70
|
+
"@bitgo-beta/utxo-ord": "1.1.3-alpha.403",
|
|
71
|
+
"@bitgo/wasm-utxo": "1.19.0",
|
|
72
72
|
"@types/lodash": "^4.14.121",
|
|
73
73
|
"@types/superagent": "4.1.15",
|
|
74
74
|
"bignumber.js": "^9.0.2",
|
|
@@ -82,5 +82,5 @@
|
|
|
82
82
|
"@bitgo-beta/sdk-test": "^9.1.20",
|
|
83
83
|
"mocha": "^10.2.0"
|
|
84
84
|
},
|
|
85
|
-
"gitHead": "
|
|
85
|
+
"gitHead": "9a3df919e67ff6d212d0f8f9970be93a182a6b60"
|
|
86
86
|
}
|