@bitgo-beta/abstract-utxo 1.6.1-alpha.430 → 1.6.1-alpha.432
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/backupKeyRecovery.d.ts +5 -4
- package/dist/cjs/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/backupKeyRecovery.js +34 -52
- package/dist/cjs/src/recovery/crossChainRecovery.d.ts +3 -3
- package/dist/cjs/src/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/crossChainRecovery.js +31 -96
- package/dist/cjs/src/recovery/psbt.d.ts +12 -0
- package/dist/cjs/src/recovery/psbt.d.ts.map +1 -0
- package/dist/cjs/src/recovery/psbt.js +92 -0
- package/dist/cjs/test/unit/recovery/backupKeyRecovery.js +17 -7
- package/dist/cjs/test/unit/recovery/crossChainRecovery.js +33 -44
- package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -1
- package/dist/cjs/test/unit/transaction/fixedScript/util.js +2 -8
- package/dist/cjs/test/unit/transaction.js +7 -3
- package/dist/cjs/tsconfig.tsbuildinfo +1 -1
- package/dist/esm/recovery/backupKeyRecovery.d.ts +5 -4
- package/dist/esm/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/backupKeyRecovery.js +33 -52
- package/dist/esm/recovery/crossChainRecovery.d.ts +3 -3
- package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/crossChainRecovery.js +31 -96
- package/dist/esm/recovery/psbt.d.ts +12 -0
- package/dist/esm/recovery/psbt.d.ts.map +1 -0
- package/dist/esm/recovery/psbt.js +55 -0
- package/package.json +12 -12
|
@@ -3,8 +3,8 @@ import { bip32 } from '@bitgo-beta/secp256k1';
|
|
|
3
3
|
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
|
-
import {
|
|
7
|
-
const { unspentSum
|
|
6
|
+
import { signAndVerifyPsbt } from '../transaction/fixedScript/signPsbt';
|
|
7
|
+
const { unspentSum } = utxolib.bitgo;
|
|
8
8
|
export async function getWallet(bitgo, coin, walletId) {
|
|
9
9
|
try {
|
|
10
10
|
return await coin.wallets().get({ id: walletId });
|
|
@@ -234,90 +234,31 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
234
234
|
return getPrv(decrypt(passphrase, encryptedPrv));
|
|
235
235
|
}
|
|
236
236
|
/**
|
|
237
|
+
* Create a sweep transaction for cross-chain recovery using PSBT
|
|
237
238
|
* @param network
|
|
239
|
+
* @param walletKeys
|
|
238
240
|
* @param unspents
|
|
239
241
|
* @param targetAddress
|
|
240
242
|
* @param feeRateSatVB
|
|
241
|
-
* @
|
|
242
|
-
* @param amountType
|
|
243
|
-
* @return transaction spending full input amount to targetAddress
|
|
243
|
+
* @return unsigned PSBT
|
|
244
244
|
*/
|
|
245
|
-
function createSweepTransaction(network, unspents, targetAddress, feeRateSatVB
|
|
246
|
-
const inputValue = unspentSum(unspents,
|
|
245
|
+
function createSweepTransaction(network, walletKeys, unspents, targetAddress, feeRateSatVB) {
|
|
246
|
+
const inputValue = unspentSum(unspents.map((u) => ({ ...u, value: BigInt(u.value) })), 'bigint');
|
|
247
247
|
const vsize = Dimensions.fromUnspents(unspents, {
|
|
248
248
|
p2tr: { scriptPathLevel: 1 },
|
|
249
249
|
p2trMusig2: { scriptPathLevel: undefined },
|
|
250
250
|
})
|
|
251
251
|
.plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))
|
|
252
252
|
.getVSize();
|
|
253
|
-
const fee = vsize * feeRateSatVB;
|
|
254
|
-
const
|
|
255
|
-
|
|
253
|
+
const fee = BigInt(Math.round(vsize * feeRateSatVB));
|
|
254
|
+
const psbt = utxolib.bitgo.createPsbtForNetwork({ network });
|
|
255
|
+
utxolib.bitgo.addXpubsToPsbt(psbt, walletKeys);
|
|
256
256
|
unspents.forEach((unspent) => {
|
|
257
|
-
utxolib.bitgo.
|
|
258
|
-
});
|
|
259
|
-
let transaction = transactionBuilder.buildIncomplete();
|
|
260
|
-
if (signer) {
|
|
261
|
-
transaction = signAndVerifyWalletTransaction(transactionBuilder, unspents, signer, {
|
|
262
|
-
isLastSignature: false,
|
|
263
|
-
});
|
|
264
|
-
}
|
|
265
|
-
return transaction;
|
|
266
|
-
}
|
|
267
|
-
function getTxInfo(transaction, unspents, walletId, walletKeys, amountType = 'number') {
|
|
268
|
-
const inputAmount = utxolib.bitgo.unspentSum(unspents, amountType);
|
|
269
|
-
const outputAmount = utxolib.bitgo.toTNumber(transaction.outs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0)), amountType);
|
|
270
|
-
const outputs = transaction.outs.map((o) => ({
|
|
271
|
-
address: utxolib.address.fromOutputScript(o.script, transaction.network),
|
|
272
|
-
valueString: o.value.toString(),
|
|
273
|
-
change: false,
|
|
274
|
-
}));
|
|
275
|
-
const inputs = unspents.map((u) => {
|
|
276
|
-
// NOTE:
|
|
277
|
-
// The `redeemScript` and `walletScript` properties are required for legacy versions of BitGoJS
|
|
278
|
-
// which might require these scripts for signing. The Wallet Recovery Wizard (WRW) can create
|
|
279
|
-
// unsigned prebuilds that are submitted to BitGoJS instances which are not necessarily the same
|
|
280
|
-
// version.
|
|
281
|
-
const addressKeys = walletKeys.deriveForChainAndIndex(u.chain, u.index);
|
|
282
|
-
const scriptType = scriptTypeForChain(u.chain);
|
|
283
|
-
const { redeemScript, witnessScript } = outputScripts.createOutputScript2of3(addressKeys.publicKeys, scriptType);
|
|
284
|
-
return {
|
|
285
|
-
...u,
|
|
286
|
-
wallet: walletId,
|
|
287
|
-
fromWallet: walletId,
|
|
288
|
-
redeemScript: redeemScript?.toString('hex'),
|
|
289
|
-
witnessScript: witnessScript?.toString('hex'),
|
|
290
|
-
};
|
|
257
|
+
utxolib.bitgo.addWalletUnspentToPsbt(psbt, { ...unspent, value: BigInt(unspent.value) }, walletKeys, 'user', 'backup', { skipNonWitnessUtxo: true });
|
|
291
258
|
});
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
minerFee: inputAmount - outputAmount,
|
|
296
|
-
spendAmount: outputAmount,
|
|
297
|
-
inputs,
|
|
298
|
-
unspents: inputs,
|
|
299
|
-
outputs,
|
|
300
|
-
externalOutputs: outputs,
|
|
301
|
-
changeOutputs: [],
|
|
302
|
-
payGoFee: 0,
|
|
303
|
-
} /* cast to TransactionInfo to allow extra fields may be required by legacy consumers of this data */;
|
|
304
|
-
}
|
|
305
|
-
function getFeeInfo(transaction, unspents, amountType = 'number') {
|
|
306
|
-
const vsize = Dimensions.fromUnspents(unspents, {
|
|
307
|
-
p2tr: { scriptPathLevel: 1 },
|
|
308
|
-
p2trMusig2: { scriptPathLevel: undefined },
|
|
309
|
-
})
|
|
310
|
-
.plus(Dimensions.fromOutputs(transaction.outs))
|
|
311
|
-
.getVSize();
|
|
312
|
-
const inputAmount = utxolib.bitgo.unspentSum(unspents, amountType);
|
|
313
|
-
const outputAmount = transaction.outs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0));
|
|
314
|
-
const fee = Number(BigInt(inputAmount) - outputAmount);
|
|
315
|
-
return {
|
|
316
|
-
size: vsize,
|
|
317
|
-
fee,
|
|
318
|
-
feeRate: fee / vsize,
|
|
319
|
-
payGoFee: 0,
|
|
320
|
-
};
|
|
259
|
+
const recoveryOutputScript = utxolib.address.toOutputScript(targetAddress, network);
|
|
260
|
+
psbt.addOutput({ script: recoveryOutputScript, value: inputValue - fee });
|
|
261
|
+
return psbt;
|
|
321
262
|
}
|
|
322
263
|
/**
|
|
323
264
|
* Recover wallet deposits that were received on the wrong blockchain
|
|
@@ -335,34 +276,28 @@ export async function recoverCrossChain(bitgo, params) {
|
|
|
335
276
|
const walletUnspents = await toWalletUnspents(params.sourceCoin, params.recoveryCoin, unspents, wallet);
|
|
336
277
|
const walletKeys = await getWalletKeys(params.recoveryCoin, wallet);
|
|
337
278
|
const prv = params.xprv || params.walletPassphrase ? await getPrv(params.xprv, params.walletPassphrase, wallet) : undefined;
|
|
338
|
-
const signer = prv
|
|
339
|
-
? new utxolib.bitgo.WalletUnspentSigner(walletKeys, prv, walletKeys.bitgo)
|
|
340
|
-
: undefined;
|
|
341
279
|
const feeRateSatVB = await getFeeRateSatVB(params.sourceCoin);
|
|
342
|
-
|
|
343
|
-
const
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
if (prv) {
|
|
280
|
+
// Create PSBT for both signed and unsigned recovery
|
|
281
|
+
const psbt = createSweepTransaction(params.sourceCoin.network, walletKeys, walletUnspents, params.recoveryAddress, feeRateSatVB);
|
|
282
|
+
// For unsigned recovery, return unsigned PSBT hex
|
|
283
|
+
if (!prv) {
|
|
347
284
|
return {
|
|
348
|
-
|
|
285
|
+
txHex: psbt.toHex(),
|
|
349
286
|
walletId: params.walletId,
|
|
350
|
-
txHex,
|
|
351
|
-
txInfo,
|
|
352
|
-
sourceCoin: params.sourceCoin.getChain(),
|
|
353
|
-
recoveryCoin: params.recoveryCoin.getChain(),
|
|
354
|
-
recoveryAmount,
|
|
355
|
-
};
|
|
356
|
-
}
|
|
357
|
-
else {
|
|
358
|
-
return {
|
|
359
|
-
txHex,
|
|
360
|
-
txInfo,
|
|
361
|
-
walletId: params.walletId,
|
|
362
|
-
feeInfo: getFeeInfo(transaction, walletUnspents, params.sourceCoin.amountType),
|
|
363
287
|
address: params.recoveryAddress,
|
|
364
288
|
coin: params.sourceCoin.getChain(),
|
|
365
289
|
};
|
|
366
290
|
}
|
|
291
|
+
// For signed recovery, sign the PSBT with user key and return half-signed PSBT
|
|
292
|
+
signAndVerifyPsbt(psbt, prv, { isLastSignature: false });
|
|
293
|
+
const recoveryAmount = utxolib.bitgo.toTNumber(psbt.txOutputs[0].value, params.sourceCoin.amountType);
|
|
294
|
+
return {
|
|
295
|
+
version: wallet instanceof Wallet ? 2 : 1,
|
|
296
|
+
walletId: params.walletId,
|
|
297
|
+
txHex: psbt.toHex(),
|
|
298
|
+
sourceCoin: params.sourceCoin.getChain(),
|
|
299
|
+
recoveryCoin: params.recoveryCoin.getChain(),
|
|
300
|
+
recoveryAmount,
|
|
301
|
+
};
|
|
367
302
|
}
|
|
368
|
-
//# 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 { signAndVerifyWalletTransaction } from '../transaction/fixedScript/signLegacyTransaction';\n\nconst { unspentSum, scriptTypeForChain, outputScripts } = 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>;\ntype WalletUnspentLegacy<TNumber extends number | bigint = number> = utxolib.bitgo.WalletUnspentLegacy<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 * @param network\n * @param unspents\n * @param targetAddress\n * @param feeRateSatVB\n * @param signer - if set, sign transaction\n * @param amountType\n * @return transaction spending full input amount to targetAddress\n */\nfunction createSweepTransaction<TNumber extends number | bigint = number>(\n  network: utxolib.Network,\n  unspents: WalletUnspent<TNumber>[],\n  targetAddress: string,\n  feeRateSatVB: number,\n  signer?: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  amountType: 'number' | 'bigint' = 'number'\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const inputValue = unspentSum<TNumber>(unspents, amountType);\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 = vsize * feeRateSatVB;\n\n  const transactionBuilder = utxolib.bitgo.createTransactionBuilderForNetwork<TNumber>(network);\n  transactionBuilder.addOutput(\n    targetAddress,\n    utxolib.bitgo.toTNumber<TNumber>(BigInt(inputValue) - BigInt(fee), amountType)\n  );\n  unspents.forEach((unspent) => {\n    utxolib.bitgo.addToTransactionBuilder(transactionBuilder, unspent);\n  });\n  let transaction = transactionBuilder.buildIncomplete();\n  if (signer) {\n    transaction = signAndVerifyWalletTransaction<TNumber>(transactionBuilder, unspents, signer, {\n      isLastSignature: false,\n    });\n  }\n  return transaction;\n}\n\nfunction getTxInfo<TNumber extends number | bigint = number>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber>,\n  unspents: WalletUnspent<TNumber>[],\n  walletId: string,\n  walletKeys: RootWalletKeys,\n  amountType: 'number' | 'bigint' = 'number'\n): TransactionInfo<TNumber> {\n  const inputAmount = utxolib.bitgo.unspentSum<TNumber>(unspents, amountType);\n  const outputAmount = utxolib.bitgo.toTNumber<TNumber>(\n    transaction.outs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0)),\n    amountType\n  );\n  const outputs = transaction.outs.map((o) => ({\n    address: utxolib.address.fromOutputScript(o.script, transaction.network),\n    valueString: o.value.toString(),\n    change: false,\n  }));\n  const inputs = unspents.map((u) => {\n    // NOTE:\n    // The `redeemScript` and `walletScript` properties are required for legacy versions of BitGoJS\n    // which might require these scripts for signing. The Wallet Recovery Wizard (WRW) can create\n    // unsigned prebuilds that are submitted to BitGoJS instances which are not necessarily the same\n    // version.\n    const addressKeys = walletKeys.deriveForChainAndIndex(u.chain, u.index);\n    const scriptType = scriptTypeForChain(u.chain);\n    const { redeemScript, witnessScript } = outputScripts.createOutputScript2of3(addressKeys.publicKeys, scriptType);\n\n    return {\n      ...u,\n      wallet: walletId,\n      fromWallet: walletId,\n      redeemScript: redeemScript?.toString('hex'),\n      witnessScript: witnessScript?.toString('hex'),\n    } as WalletUnspentLegacy<TNumber>;\n  });\n  return {\n    inputAmount,\n    outputAmount,\n    minerFee: inputAmount - outputAmount,\n    spendAmount: outputAmount,\n    inputs,\n    unspents: inputs,\n    outputs,\n    externalOutputs: outputs,\n    changeOutputs: [],\n    payGoFee: 0,\n  } /* cast to TransactionInfo to allow extra fields may be required by legacy consumers of this data */ as TransactionInfo<TNumber>;\n}\n\nfunction getFeeInfo<TNumber extends number | bigint = number>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber>,\n  unspents: WalletUnspent<TNumber>[],\n  amountType: 'number' | 'bigint' = 'number'\n): FeeInfo {\n  const vsize = Dimensions.fromUnspents(unspents, {\n    p2tr: { scriptPathLevel: 1 },\n    p2trMusig2: { scriptPathLevel: undefined },\n  })\n    .plus(Dimensions.fromOutputs(transaction.outs))\n    .getVSize();\n  const inputAmount = utxolib.bitgo.unspentSum<TNumber>(unspents, amountType);\n  const outputAmount = transaction.outs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0));\n  const fee = Number(BigInt(inputAmount) - outputAmount);\n  return {\n    size: vsize,\n    fee,\n    feeRate: fee / vsize,\n    payGoFee: 0,\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 signer = prv\n    ? new utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>(walletKeys, prv, walletKeys.bitgo)\n    : undefined;\n  const feeRateSatVB = await getFeeRateSatVB(params.sourceCoin);\n  const transaction = createSweepTransaction<TNumber>(\n    params.sourceCoin.network,\n    walletUnspents,\n    params.recoveryAddress,\n    feeRateSatVB,\n    signer,\n    params.sourceCoin.amountType\n  );\n  const recoveryAmount = transaction.outs[0].value;\n  const txHex = transaction.toBuffer().toString('hex');\n  const txInfo = getTxInfo<TNumber>(\n    transaction,\n    walletUnspents,\n    params.walletId,\n    walletKeys,\n    params.sourceCoin.amountType\n  );\n  if (prv) {\n    return {\n      version: wallet instanceof Wallet ? 2 : 1,\n      walletId: params.walletId,\n      txHex,\n      txInfo,\n      sourceCoin: params.sourceCoin.getChain(),\n      recoveryCoin: params.recoveryCoin.getChain(),\n      recoveryAmount,\n    };\n  } else {\n    return {\n      txHex,\n      txInfo,\n      walletId: params.walletId,\n      feeInfo: getFeeInfo(transaction, walletUnspents, params.sourceCoin.amountType),\n      address: params.recoveryAddress,\n      coin: params.sourceCoin.getChain(),\n    };\n  }\n}\n"]}
|
|
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"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
+
type RootWalletKeys = utxolib.bitgo.RootWalletKeys;
|
|
3
|
+
type WalletUnspent<TNumber extends number | bigint> = utxolib.bitgo.WalletUnspent<TNumber>;
|
|
4
|
+
export declare function createBackupKeyRecoveryPsbt(network: utxolib.Network, rootWalletKeys: RootWalletKeys, unspents: WalletUnspent<bigint>[], { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress, }: {
|
|
5
|
+
feeRateSatVB: number;
|
|
6
|
+
recoveryDestination: string;
|
|
7
|
+
keyRecoveryServiceFee: bigint;
|
|
8
|
+
keyRecoveryServiceFeeAddress: string | undefined;
|
|
9
|
+
}): utxolib.bitgo.UtxoPsbt;
|
|
10
|
+
export declare function getRecoveryAmount(psbt: utxolib.bitgo.UtxoPsbt, address: string): bigint;
|
|
11
|
+
export {};
|
|
12
|
+
//# sourceMappingURL=psbt.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"psbt.d.ts","sourceRoot":"","sources":["../../../src/recovery/psbt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAGhD,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;AAmB3F,wBAAgB,2BAA2B,CACzC,OAAO,EAAE,OAAO,CAAC,OAAO,EACxB,cAAc,EAAE,cAAc,EAC9B,QAAQ,EAAE,aAAa,CAAC,MAAM,CAAC,EAAE,EACjC,EACE,YAAY,EACZ,mBAAmB,EACnB,qBAAqB,EACrB,4BAA4B,GAC7B,EAAE;IACD,YAAY,EAAE,MAAM,CAAC;IACrB,mBAAmB,EAAE,MAAM,CAAC;IAC5B,qBAAqB,EAAE,MAAM,CAAC;IAC9B,4BAA4B,EAAE,MAAM,GAAG,SAAS,CAAC;CAClD,GACA,OAAO,CAAC,KAAK,CAAC,QAAQ,CA4CxB;AAED,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,CAOvF"}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
+
import { Dimensions } from '@bitgo-beta/unspents';
|
|
3
|
+
class InsufficientFundsError extends Error {
|
|
4
|
+
constructor(totalInputAmount, approximateFee, krsFee, recoveryAmount) {
|
|
5
|
+
super(`This wallet's balance is too low to pay the fees specified by the KRS provider.` +
|
|
6
|
+
`Existing balance on wallet: ${totalInputAmount.toString()}. ` +
|
|
7
|
+
`Estimated network fee for the recovery transaction: ${approximateFee.toString()}` +
|
|
8
|
+
`KRS fee to pay: ${krsFee.toString()}. ` +
|
|
9
|
+
`After deducting fees, your total recoverable balance is ${recoveryAmount.toString()}`);
|
|
10
|
+
this.totalInputAmount = totalInputAmount;
|
|
11
|
+
this.approximateFee = approximateFee;
|
|
12
|
+
this.krsFee = krsFee;
|
|
13
|
+
this.recoveryAmount = recoveryAmount;
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
export function createBackupKeyRecoveryPsbt(network, rootWalletKeys, unspents, { feeRateSatVB, recoveryDestination, keyRecoveryServiceFee, keyRecoveryServiceFeeAddress, }) {
|
|
17
|
+
if (keyRecoveryServiceFee > 0 && !keyRecoveryServiceFeeAddress) {
|
|
18
|
+
throw new Error('keyRecoveryServiceFeeAddress is required when keyRecoveryServiceFee is provided');
|
|
19
|
+
}
|
|
20
|
+
const psbt = utxolib.bitgo.createPsbtForNetwork({ network });
|
|
21
|
+
utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);
|
|
22
|
+
unspents.forEach((unspent) => {
|
|
23
|
+
utxolib.bitgo.addWalletUnspentToPsbt(psbt, unspent, rootWalletKeys, 'user', 'backup');
|
|
24
|
+
});
|
|
25
|
+
let dimensions = Dimensions.fromPsbt(psbt).plus(Dimensions.fromOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network) }));
|
|
26
|
+
if (keyRecoveryServiceFeeAddress) {
|
|
27
|
+
dimensions = dimensions.plus(Dimensions.fromOutput({
|
|
28
|
+
script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),
|
|
29
|
+
}));
|
|
30
|
+
}
|
|
31
|
+
const approximateFee = BigInt(dimensions.getVSize() * feeRateSatVB);
|
|
32
|
+
const totalInputAmount = utxolib.bitgo.unspentSum(unspents, 'bigint');
|
|
33
|
+
const recoveryAmount = totalInputAmount - approximateFee - keyRecoveryServiceFee;
|
|
34
|
+
// FIXME(BTC-2650): we should check for dust limit here instead
|
|
35
|
+
if (recoveryAmount < BigInt(0)) {
|
|
36
|
+
throw new InsufficientFundsError(totalInputAmount, approximateFee, keyRecoveryServiceFee, recoveryAmount);
|
|
37
|
+
}
|
|
38
|
+
psbt.addOutput({ script: utxolib.address.toOutputScript(recoveryDestination, network), value: recoveryAmount });
|
|
39
|
+
if (keyRecoveryServiceFeeAddress) {
|
|
40
|
+
psbt.addOutput({
|
|
41
|
+
script: utxolib.address.toOutputScript(keyRecoveryServiceFeeAddress, network),
|
|
42
|
+
value: keyRecoveryServiceFee,
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
return psbt;
|
|
46
|
+
}
|
|
47
|
+
export function getRecoveryAmount(psbt, address) {
|
|
48
|
+
const recoveryOutputScript = utxolib.address.toOutputScript(address, psbt.network);
|
|
49
|
+
const output = psbt.txOutputs.find((o) => o.script.equals(recoveryOutputScript));
|
|
50
|
+
if (!output) {
|
|
51
|
+
throw new Error(`Recovery destination output not found in PSBT: ${address}`);
|
|
52
|
+
}
|
|
53
|
+
return output.value;
|
|
54
|
+
}
|
|
55
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHNidC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9yZWNvdmVyeS9wc2J0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sc0JBQXNCLENBQUM7QUFDaEQsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBS2xELE1BQU0sc0JBQXVCLFNBQVEsS0FBSztJQUN4QyxZQUNTLGdCQUF3QixFQUN4QixjQUFzQixFQUN0QixNQUFjLEVBQ2QsY0FBc0I7UUFFN0IsS0FBSyxDQUNILGlGQUFpRjtZQUMvRSwrQkFBK0IsZ0JBQWdCLENBQUMsUUFBUSxFQUFFLElBQUk7WUFDOUQsdURBQXVELGNBQWMsQ0FBQyxRQUFRLEVBQUUsRUFBRTtZQUNsRixtQkFBbUIsTUFBTSxDQUFDLFFBQVEsRUFBRSxJQUFJO1lBQ3hDLDJEQUEyRCxjQUFjLENBQUMsUUFBUSxFQUFFLEVBQUUsQ0FDekYsQ0FBQztRQVhLLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBUTtRQUN4QixtQkFBYyxHQUFkLGNBQWMsQ0FBUTtRQUN0QixXQUFNLEdBQU4sTUFBTSxDQUFRO1FBQ2QsbUJBQWMsR0FBZCxjQUFjLENBQVE7SUFTL0IsQ0FBQztDQUNGO0FBRUQsTUFBTSxVQUFVLDJCQUEyQixDQUN6QyxPQUF3QixFQUN4QixjQUE4QixFQUM5QixRQUFpQyxFQUNqQyxFQUNFLFlBQVksRUFDWixtQkFBbUIsRUFDbkIscUJBQXFCLEVBQ3JCLDRCQUE0QixHQU03QjtJQUVELElBQUkscUJBQXFCLEdBQUcsQ0FBQyxJQUFJLENBQUMsNEJBQTRCLEVBQUUsQ0FBQztRQUMvRCxNQUFNLElBQUksS0FBSyxDQUFDLGlGQUFpRixDQUFDLENBQUM7SUFDckcsQ0FBQztJQUVELE1BQU0sSUFBSSxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsb0JBQW9CLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQzdELE9BQU8sQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxjQUFjLENBQUMsQ0FBQztJQUNuRCxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUU7UUFDM0IsT0FBTyxDQUFDLEtBQUssQ0FBQyxzQkFBc0IsQ0FBQyxJQUFJLEVBQUUsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDeEYsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLFVBQVUsR0FBRyxVQUFVLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxDQUFDLElBQUksQ0FDN0MsVUFBVSxDQUFDLFVBQVUsQ0FBQyxFQUFFLE1BQU0sRUFBRSxPQUFPLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQyxtQkFBbUIsRUFBRSxPQUFPLENBQUMsRUFBRSxDQUFDLENBQ2hHLENBQUM7SUFFRixJQUFJLDRCQUE0QixFQUFFLENBQUM7UUFDakMsVUFBVSxHQUFHLFVBQVUsQ0FBQyxJQUFJLENBQzFCLFVBQVUsQ0FBQyxVQUFVLENBQUM7WUFDcEIsTUFBTSxFQUFFLE9BQU8sQ0FBQyxPQUFPLENBQUMsY0FBYyxDQUFDLDRCQUE0QixFQUFFLE9BQU8sQ0FBQztTQUM5RSxDQUFDLENBQ0gsQ0FBQztJQUNKLENBQUM7SUFFRCxNQUFNLGNBQWMsR0FBRyxNQUFNLENBQUMsVUFBVSxDQUFDLFFBQVEsRUFBRSxHQUFHLFlBQVksQ0FBQyxDQUFDO0lBRXBFLE1BQU0sZ0JBQWdCLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsUUFBUSxFQUFFLFFBQVEsQ0FBQyxDQUFDO0lBRXRFLE1BQU0sY0FBYyxHQUFHLGdCQUFnQixHQUFHLGNBQWMsR0FBRyxxQkFBcUIsQ0FBQztJQUVqRiwrREFBK0Q7SUFDL0QsSUFBSSxjQUFjLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDL0IsTUFBTSxJQUFJLHNCQUFzQixDQUFDLGdCQUFnQixFQUFFLGNBQWMsRUFBRSxxQkFBcUIsRUFBRSxjQUFjLENBQUMsQ0FBQztJQUM1RyxDQUFDO0lBRUQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxFQUFFLE1BQU0sRUFBRSxPQUFPLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQyxtQkFBbUIsRUFBRSxPQUFPLENBQUMsRUFBRSxLQUFLLEVBQUUsY0FBYyxFQUFFLENBQUMsQ0FBQztJQUVoSCxJQUFJLDRCQUE0QixFQUFFLENBQUM7UUFDakMsSUFBSSxDQUFDLFNBQVMsQ0FBQztZQUNiLE1BQU0sRUFBRSxPQUFPLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQyw0QkFBNEIsRUFBRSxPQUFPLENBQUM7WUFDN0UsS0FBSyxFQUFFLHFCQUFxQjtTQUM3QixDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQsT0FBTyxJQUFJLENBQUM7QUFDZCxDQUFDO0FBRUQsTUFBTSxVQUFVLGlCQUFpQixDQUFDLElBQTRCLEVBQUUsT0FBZTtJQUM3RSxNQUFNLG9CQUFvQixHQUFHLE9BQU8sQ0FBQyxPQUFPLENBQUMsY0FBYyxDQUFDLE9BQU8sRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDbkYsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLG9CQUFvQixDQUFDLENBQUMsQ0FBQztJQUNqRixJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDWixNQUFNLElBQUksS0FBSyxDQUFDLGtEQUFrRCxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQy9FLENBQUM7SUFDRCxPQUFPLE1BQU0sQ0FBQyxLQUFLLENBQUM7QUFDdEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuaW1wb3J0IHsgRGltZW5zaW9ucyB9IGZyb20gJ0BiaXRnby1iZXRhL3Vuc3BlbnRzJztcblxudHlwZSBSb290V2FsbGV0S2V5cyA9IHV0eG9saWIuYml0Z28uUm9vdFdhbGxldEtleXM7XG50eXBlIFdhbGxldFVuc3BlbnQ8VE51bWJlciBleHRlbmRzIG51bWJlciB8IGJpZ2ludD4gPSB1dHhvbGliLmJpdGdvLldhbGxldFVuc3BlbnQ8VE51bWJlcj47XG5cbmNsYXNzIEluc3VmZmljaWVudEZ1bmRzRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyB0b3RhbElucHV0QW1vdW50OiBiaWdpbnQsXG4gICAgcHVibGljIGFwcHJveGltYXRlRmVlOiBiaWdpbnQsXG4gICAgcHVibGljIGtyc0ZlZTogYmlnaW50LFxuICAgIHB1YmxpYyByZWNvdmVyeUFtb3VudDogYmlnaW50XG4gICkge1xuICAgIHN1cGVyKFxuICAgICAgYFRoaXMgd2FsbGV0J3MgYmFsYW5jZSBpcyB0b28gbG93IHRvIHBheSB0aGUgZmVlcyBzcGVjaWZpZWQgYnkgdGhlIEtSUyBwcm92aWRlci5gICtcbiAgICAgICAgYEV4aXN0aW5nIGJhbGFuY2Ugb24gd2FsbGV0OiAke3RvdGFsSW5wdXRBbW91bnQudG9TdHJpbmcoKX0uIGAgK1xuICAgICAgICBgRXN0aW1hdGVkIG5ldHdvcmsgZmVlIGZvciB0aGUgcmVjb3ZlcnkgdHJhbnNhY3Rpb246ICR7YXBwcm94aW1hdGVGZWUudG9TdHJpbmcoKX1gICtcbiAgICAgICAgYEtSUyBmZWUgdG8gcGF5OiAke2tyc0ZlZS50b1N0cmluZygpfS4gYCArXG4gICAgICAgIGBBZnRlciBkZWR1Y3RpbmcgZmVlcywgeW91ciB0b3RhbCByZWNvdmVyYWJsZSBiYWxhbmNlIGlzICR7cmVjb3ZlcnlBbW91bnQudG9TdHJpbmcoKX1gXG4gICAgKTtcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlQmFja3VwS2V5UmVjb3ZlcnlQc2J0KFxuICBuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmssXG4gIHJvb3RXYWxsZXRLZXlzOiBSb290V2FsbGV0S2V5cyxcbiAgdW5zcGVudHM6IFdhbGxldFVuc3BlbnQ8YmlnaW50PltdLFxuICB7XG4gICAgZmVlUmF0ZVNhdFZCLFxuICAgIHJlY292ZXJ5RGVzdGluYXRpb24sXG4gICAga2V5UmVjb3ZlcnlTZXJ2aWNlRmVlLFxuICAgIGtleVJlY292ZXJ5U2VydmljZUZlZUFkZHJlc3MsXG4gIH06IHtcbiAgICBmZWVSYXRlU2F0VkI6IG51bWJlcjtcbiAgICByZWNvdmVyeURlc3RpbmF0aW9uOiBzdHJpbmc7XG4gICAga2V5UmVjb3ZlcnlTZXJ2aWNlRmVlOiBiaWdpbnQ7XG4gICAga2V5UmVjb3ZlcnlTZXJ2aWNlRmVlQWRkcmVzczogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICB9XG4pOiB1dHhvbGliLmJpdGdvLlV0eG9Qc2J0IHtcbiAgaWYgKGtleVJlY292ZXJ5U2VydmljZUZlZSA+IDAgJiYgIWtleVJlY292ZXJ5U2VydmljZUZlZUFkZHJlc3MpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ2tleVJlY292ZXJ5U2VydmljZUZlZUFkZHJlc3MgaXMgcmVxdWlyZWQgd2hlbiBrZXlSZWNvdmVyeVNlcnZpY2VGZWUgaXMgcHJvdmlkZWQnKTtcbiAgfVxuXG4gIGNvbnN0IHBzYnQgPSB1dHhvbGliLmJpdGdvLmNyZWF0ZVBzYnRGb3JOZXR3b3JrKHsgbmV0d29yayB9KTtcbiAgdXR4b2xpYi5iaXRnby5hZGRYcHVic1RvUHNidChwc2J0LCByb290V2FsbGV0S2V5cyk7XG4gIHVuc3BlbnRzLmZvckVhY2goKHVuc3BlbnQpID0+IHtcbiAgICB1dHhvbGliLmJpdGdvLmFkZFdhbGxldFVuc3BlbnRUb1BzYnQocHNidCwgdW5zcGVudCwgcm9vdFdhbGxldEtleXMsICd1c2VyJywgJ2JhY2t1cCcpO1xuICB9KTtcblxuICBsZXQgZGltZW5zaW9ucyA9IERpbWVuc2lvbnMuZnJvbVBzYnQocHNidCkucGx1cyhcbiAgICBEaW1lbnNpb25zLmZyb21PdXRwdXQoeyBzY3JpcHQ6IHV0eG9saWIuYWRkcmVzcy50b091dHB1dFNjcmlwdChyZWNvdmVyeURlc3RpbmF0aW9uLCBuZXR3b3JrKSB9KVxuICApO1xuXG4gIGlmIChrZXlSZWNvdmVyeVNlcnZpY2VGZWVBZGRyZXNzKSB7XG4gICAgZGltZW5zaW9ucyA9IGRpbWVuc2lvbnMucGx1cyhcbiAgICAgIERpbWVuc2lvbnMuZnJvbU91dHB1dCh7XG4gICAgICAgIHNjcmlwdDogdXR4b2xpYi5hZGRyZXNzLnRvT3V0cHV0U2NyaXB0KGtleVJlY292ZXJ5U2VydmljZUZlZUFkZHJlc3MsIG5ldHdvcmspLFxuICAgICAgfSlcbiAgICApO1xuICB9XG5cbiAgY29uc3QgYXBwcm94aW1hdGVGZWUgPSBCaWdJbnQoZGltZW5zaW9ucy5nZXRWU2l6ZSgpICogZmVlUmF0ZVNhdFZCKTtcblxuICBjb25zdCB0b3RhbElucHV0QW1vdW50ID0gdXR4b2xpYi5iaXRnby51bnNwZW50U3VtKHVuc3BlbnRzLCAnYmlnaW50Jyk7XG5cbiAgY29uc3QgcmVjb3ZlcnlBbW91bnQgPSB0b3RhbElucHV0QW1vdW50IC0gYXBwcm94aW1hdGVGZWUgLSBrZXlSZWNvdmVyeVNlcnZpY2VGZWU7XG5cbiAgLy8gRklYTUUoQlRDLTI2NTApOiB3ZSBzaG91bGQgY2hlY2sgZm9yIGR1c3QgbGltaXQgaGVyZSBpbnN0ZWFkXG4gIGlmIChyZWNvdmVyeUFtb3VudCA8IEJpZ0ludCgwKSkge1xuICAgIHRocm93IG5ldyBJbnN1ZmZpY2llbnRGdW5kc0Vycm9yKHRvdGFsSW5wdXRBbW91bnQsIGFwcHJveGltYXRlRmVlLCBrZXlSZWNvdmVyeVNlcnZpY2VGZWUsIHJlY292ZXJ5QW1vdW50KTtcbiAgfVxuXG4gIHBzYnQuYWRkT3V0cHV0KHsgc2NyaXB0OiB1dHhvbGliLmFkZHJlc3MudG9PdXRwdXRTY3JpcHQocmVjb3ZlcnlEZXN0aW5hdGlvbiwgbmV0d29yayksIHZhbHVlOiByZWNvdmVyeUFtb3VudCB9KTtcblxuICBpZiAoa2V5UmVjb3ZlcnlTZXJ2aWNlRmVlQWRkcmVzcykge1xuICAgIHBzYnQuYWRkT3V0cHV0KHtcbiAgICAgIHNjcmlwdDogdXR4b2xpYi5hZGRyZXNzLnRvT3V0cHV0U2NyaXB0KGtleVJlY292ZXJ5U2VydmljZUZlZUFkZHJlc3MsIG5ldHdvcmspLFxuICAgICAgdmFsdWU6IGtleVJlY292ZXJ5U2VydmljZUZlZSxcbiAgICB9KTtcbiAgfVxuXG4gIHJldHVybiBwc2J0O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0UmVjb3ZlcnlBbW91bnQocHNidDogdXR4b2xpYi5iaXRnby5VdHhvUHNidCwgYWRkcmVzczogc3RyaW5nKTogYmlnaW50IHtcbiAgY29uc3QgcmVjb3ZlcnlPdXRwdXRTY3JpcHQgPSB1dHhvbGliLmFkZHJlc3MudG9PdXRwdXRTY3JpcHQoYWRkcmVzcywgcHNidC5uZXR3b3JrKTtcbiAgY29uc3Qgb3V0cHV0ID0gcHNidC50eE91dHB1dHMuZmluZCgobykgPT4gby5zY3JpcHQuZXF1YWxzKHJlY292ZXJ5T3V0cHV0U2NyaXB0KSk7XG4gIGlmICghb3V0cHV0KSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBSZWNvdmVyeSBkZXN0aW5hdGlvbiBvdXRwdXQgbm90IGZvdW5kIGluIFBTQlQ6ICR7YWRkcmVzc31gKTtcbiAgfVxuICByZXR1cm4gb3V0cHV0LnZhbHVlO1xufVxuIl19
|
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.432",
|
|
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.433",
|
|
64
|
+
"@bitgo-beta/sdk-api": "1.6.1-alpha.432",
|
|
65
|
+
"@bitgo-beta/sdk-core": "2.4.1-alpha.432",
|
|
66
|
+
"@bitgo-beta/secp256k1": "1.0.1-alpha.390",
|
|
67
|
+
"@bitgo-beta/unspents": "0.11.3-alpha.433",
|
|
68
|
+
"@bitgo-beta/utxo-core": "1.0.1-alpha.163",
|
|
69
|
+
"@bitgo-beta/utxo-lib": "4.0.1-alpha.433",
|
|
70
|
+
"@bitgo-beta/utxo-ord": "1.1.3-alpha.387",
|
|
71
|
+
"@bitgo/wasm-utxo": "1.15.0",
|
|
72
72
|
"@types/lodash": "^4.14.121",
|
|
73
73
|
"@types/superagent": "4.1.15",
|
|
74
74
|
"bignumber.js": "^9.0.2",
|
|
@@ -79,8 +79,8 @@
|
|
|
79
79
|
"superagent": "^9.0.1"
|
|
80
80
|
},
|
|
81
81
|
"devDependencies": {
|
|
82
|
-
"@bitgo-beta/sdk-test": "^9.1.
|
|
82
|
+
"@bitgo-beta/sdk-test": "^9.1.19",
|
|
83
83
|
"mocha": "^10.2.0"
|
|
84
84
|
},
|
|
85
|
-
"gitHead": "
|
|
85
|
+
"gitHead": "b75f7d198d66a10d68a7581409fd248db8703fac"
|
|
86
86
|
}
|