@bitgo-beta/abstract-utxo 1.6.1-alpha.23 → 1.6.1-alpha.231
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/CHANGELOG.md +853 -0
- package/dist/src/abstractUtxoCoin.d.ts +146 -34
- package/dist/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/src/abstractUtxoCoin.js +350 -187
- package/dist/src/descriptor/NamedDescriptor.d.ts +7 -0
- package/dist/src/descriptor/NamedDescriptor.d.ts.map +1 -0
- package/dist/src/descriptor/NamedDescriptor.js +9 -0
- package/dist/src/descriptor/assertDescriptorWalletAddress.d.ts +4 -0
- package/dist/src/descriptor/assertDescriptorWalletAddress.d.ts.map +1 -0
- package/dist/src/descriptor/assertDescriptorWalletAddress.js +37 -0
- package/dist/src/descriptor/index.d.ts +4 -0
- package/dist/src/descriptor/index.d.ts.map +1 -0
- package/dist/src/descriptor/index.js +11 -0
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +8 -2
- package/dist/src/parseOutput.d.ts.map +1 -1
- package/dist/src/parseOutput.js +38 -1
- package/dist/src/recovery/RecoveryProvider.d.ts +1 -1
- package/dist/src/recovery/RecoveryProvider.d.ts.map +1 -1
- package/dist/src/recovery/RecoveryProvider.js +1 -2
- package/dist/src/recovery/backupKeyRecovery.d.ts +39 -11
- package/dist/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/src/recovery/backupKeyRecovery.js +124 -83
- package/dist/src/recovery/baseApi.d.ts +2 -2
- package/dist/src/recovery/baseApi.d.ts.map +1 -1
- package/dist/src/recovery/crossChainRecovery.d.ts +12 -3
- package/dist/src/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/src/recovery/crossChainRecovery.js +50 -10
- package/dist/src/recovery/index.d.ts +0 -1
- package/dist/src/recovery/index.d.ts.map +1 -1
- package/dist/src/recovery/index.js +6 -3
- package/dist/src/recovery/mempoolApi.d.ts.map +1 -1
- package/dist/src/recovery/mempoolApi.js +6 -3
- package/dist/src/sign.d.ts +29 -5
- package/dist/src/sign.d.ts.map +1 -1
- package/dist/src/sign.js +73 -7
- package/dist/src/transaction.d.ts +36 -0
- package/dist/src/transaction.d.ts.map +1 -0
- package/dist/src/transaction.js +278 -0
- package/dist/tsconfig.tsbuildinfo +1 -7906
- package/package.json +12 -10
- package/dist/src/recovery/smartbitApi.d.ts +0 -11
- package/dist/src/recovery/smartbitApi.d.ts.map +0 -1
- package/dist/src/recovery/smartbitApi.js +0 -36
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.recoverCrossChain = exports.getWallet = void 0;
|
|
3
|
+
exports.recoverCrossChain = exports.isWalletAddress = exports.getWalletKeys = exports.getWallet = void 0;
|
|
4
4
|
/**
|
|
5
5
|
* @prettier
|
|
6
6
|
*/
|
|
@@ -50,17 +50,46 @@ async function getWalletKeys(recoveryCoin, wallet) {
|
|
|
50
50
|
}
|
|
51
51
|
return new utxolib.bitgo.RootWalletKeys(xpubs.map((k) => utxo_lib_1.bip32.fromBase58(k)));
|
|
52
52
|
}
|
|
53
|
+
exports.getWalletKeys = getWalletKeys;
|
|
54
|
+
async function isWalletAddress(wallet, address) {
|
|
55
|
+
try {
|
|
56
|
+
let addressData;
|
|
57
|
+
if (wallet instanceof sdk_core_1.Wallet) {
|
|
58
|
+
addressData = await wallet.getAddress({ address });
|
|
59
|
+
}
|
|
60
|
+
else {
|
|
61
|
+
addressData = await wallet.address({ address });
|
|
62
|
+
}
|
|
63
|
+
return addressData !== undefined;
|
|
64
|
+
}
|
|
65
|
+
catch (e) {
|
|
66
|
+
return false;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
exports.isWalletAddress = isWalletAddress;
|
|
53
70
|
/**
|
|
54
71
|
* @param coin
|
|
55
72
|
* @param txid
|
|
73
|
+
* @param amountType
|
|
74
|
+
* @param wallet
|
|
56
75
|
* @param apiKey - a blockchair api key
|
|
57
76
|
* @return all unspents for transaction outputs, including outputs from other transactions
|
|
58
77
|
*/
|
|
59
|
-
async function getAllRecoveryOutputs(coin, txid, amountType = 'number', apiKey) {
|
|
78
|
+
async function getAllRecoveryOutputs(coin, txid, amountType = 'number', wallet, apiKey) {
|
|
60
79
|
const api = coin.getRecoveryProvider(apiKey);
|
|
61
80
|
const tx = await api.getTransactionIO(txid);
|
|
62
|
-
const
|
|
63
|
-
|
|
81
|
+
const walletAddresses = (await Promise.all(tx.outputs.map(async (output) => {
|
|
82
|
+
// For some coins (bch) we need to convert the address to legacy format since the api returns the address
|
|
83
|
+
// in non legacy format. However, we want to keep the address in the same format as the response since we
|
|
84
|
+
// are going to hit the API again to fetch address unspents.
|
|
85
|
+
const canonicalAddress = coin.canonicalAddress(output.address);
|
|
86
|
+
const isWalletOwned = await isWalletAddress(wallet, canonicalAddress);
|
|
87
|
+
return isWalletOwned ? output.address : null;
|
|
88
|
+
}))).filter((address) => address !== null);
|
|
89
|
+
const unspents = await api.getUnspentsForAddresses(walletAddresses);
|
|
90
|
+
if (unspents.length === 0) {
|
|
91
|
+
throw new Error(`No recovery unspents found.`);
|
|
92
|
+
}
|
|
64
93
|
// the api may return cashaddr's instead of legacy for BCH and BCHA
|
|
65
94
|
// downstream processes's only expect legacy addresses
|
|
66
95
|
return unspents.map((recoveryOutput) => {
|
|
@@ -129,6 +158,9 @@ async function getFeeRateSatVB(coin) {
|
|
|
129
158
|
tbsv: 20,
|
|
130
159
|
btc: 80,
|
|
131
160
|
tbtc: 80,
|
|
161
|
+
tbtcsig: 80,
|
|
162
|
+
tbtc4: 80,
|
|
163
|
+
tbtcbgsig: 80,
|
|
132
164
|
ltc: 100,
|
|
133
165
|
tltc: 100,
|
|
134
166
|
doge: 1000,
|
|
@@ -163,7 +195,7 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
163
195
|
else {
|
|
164
196
|
encryptedPrv = (await wallet.getEncryptedUserKeychain()).encryptedXprv;
|
|
165
197
|
}
|
|
166
|
-
return getPrv(sdk_api_1.decrypt(passphrase, encryptedPrv));
|
|
198
|
+
return getPrv((0, sdk_api_1.decrypt)(passphrase, encryptedPrv));
|
|
167
199
|
}
|
|
168
200
|
/**
|
|
169
201
|
* @param network
|
|
@@ -176,7 +208,10 @@ async function getPrv(xprv, passphrase, wallet) {
|
|
|
176
208
|
*/
|
|
177
209
|
function createSweepTransaction(network, unspents, targetAddress, feeRateSatVB, signer, amountType = 'number') {
|
|
178
210
|
const inputValue = unspentSum(unspents, amountType);
|
|
179
|
-
const vsize = unspents_1.Dimensions.fromUnspents(unspents
|
|
211
|
+
const vsize = unspents_1.Dimensions.fromUnspents(unspents, {
|
|
212
|
+
p2tr: { scriptPathLevel: 1 },
|
|
213
|
+
p2trMusig2: { scriptPathLevel: undefined },
|
|
214
|
+
})
|
|
180
215
|
.plus(unspents_1.Dimensions.fromOutput({ script: utxolib.address.toOutputScript(targetAddress, network) }))
|
|
181
216
|
.getVSize();
|
|
182
217
|
const fee = vsize * feeRateSatVB;
|
|
@@ -187,7 +222,7 @@ function createSweepTransaction(network, unspents, targetAddress, feeRateSatVB,
|
|
|
187
222
|
});
|
|
188
223
|
let transaction = transactionBuilder.buildIncomplete();
|
|
189
224
|
if (signer) {
|
|
190
|
-
transaction = sign_1.signAndVerifyWalletTransaction(transactionBuilder, unspents, signer, {
|
|
225
|
+
transaction = (0, sign_1.signAndVerifyWalletTransaction)(transactionBuilder, unspents, signer, {
|
|
191
226
|
isLastSignature: false,
|
|
192
227
|
});
|
|
193
228
|
}
|
|
@@ -232,7 +267,12 @@ function getTxInfo(transaction, unspents, walletId, walletKeys, amountType = 'nu
|
|
|
232
267
|
} /* cast to TransactionInfo to allow extra fields may be required by legacy consumers of this data */;
|
|
233
268
|
}
|
|
234
269
|
function getFeeInfo(transaction, unspents, amountType = 'number') {
|
|
235
|
-
const vsize = unspents_1.Dimensions.fromUnspents(unspents
|
|
270
|
+
const vsize = unspents_1.Dimensions.fromUnspents(unspents, {
|
|
271
|
+
p2tr: { scriptPathLevel: 1 },
|
|
272
|
+
p2trMusig2: { scriptPathLevel: undefined },
|
|
273
|
+
})
|
|
274
|
+
.plus(unspents_1.Dimensions.fromOutputs(transaction.outs))
|
|
275
|
+
.getVSize();
|
|
236
276
|
const inputAmount = utxolib.bitgo.unspentSum(unspents, amountType);
|
|
237
277
|
const outputAmount = transaction.outs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0));
|
|
238
278
|
const fee = Number(BigInt(inputAmount) - outputAmount);
|
|
@@ -255,7 +295,7 @@ function getFeeInfo(transaction, unspents, amountType = 'number') {
|
|
|
255
295
|
*/
|
|
256
296
|
async function recoverCrossChain(bitgo, params) {
|
|
257
297
|
const wallet = await getWallet(bitgo, params.recoveryCoin, params.walletId);
|
|
258
|
-
const unspents = await getAllRecoveryOutputs(params.sourceCoin, params.txid, params.sourceCoin.amountType, params.apiKey);
|
|
298
|
+
const unspents = await getAllRecoveryOutputs(params.sourceCoin, params.txid, params.sourceCoin.amountType, wallet, params.apiKey);
|
|
259
299
|
const walletUnspents = await toWalletUnspents(params.sourceCoin, params.recoveryCoin, unspents, wallet);
|
|
260
300
|
const walletKeys = await getWalletKeys(params.recoveryCoin, wallet);
|
|
261
301
|
const prv = params.xprv || params.walletPassphrase ? await getPrv(params.xprv, params.walletPassphrase, wallet) : undefined;
|
|
@@ -290,4 +330,4 @@ async function recoverCrossChain(bitgo, params) {
|
|
|
290
330
|
}
|
|
291
331
|
}
|
|
292
332
|
exports.recoverCrossChain = recoverCrossChain;
|
|
293
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../src/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as Bluebird from 'bluebird';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip32, BIP32Interface } from '@bitgo-beta/utxo-lib';\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\nimport { Dimensions } from '@bitgo-beta/unspents';\n\nimport { BitGoBase, IWallet, Keychain, Triple, Wallet } from '@bitgo-beta/sdk-core';\nimport { AbstractUtxoCoin, TransactionInfo } from '../abstractUtxoCoin';\n\nimport { decrypt } from '@bitgo-beta/sdk-api';\nimport { signAndVerifyWalletTransaction } from '../sign';\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 */\nasync function getWalletKeys(recoveryCoin: AbstractUtxoCoin, wallet: IWallet | WalletV1): 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\n/**\n * @param coin\n * @param txid\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  apiKey?: string\n): Promise<Unspent<TNumber>[]> {\n  const api = coin.getRecoveryProvider(apiKey);\n  const tx = await api.getTransactionIO(txid);\n  const addresses = tx.outputs.map((output) => output.address);\n  const unspents = await api.getUnspentsForAddresses(addresses);\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  return (\n    await Bluebird.mapSeries(addresses, async (address): Promise<WalletUnspent<TNumber>[]> => {\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        return [];\n      }\n      return unspents\n        .filter((u) => u.address === address)\n        .map((u) => ({\n          ...u,\n          ...scriptId,\n        }));\n    })\n  ).flat();\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    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    .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).plus(Dimensions.fromOutputs(transaction.outs)).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    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"]}
|
|
333
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../src/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as Bluebird from 'bluebird';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip32, BIP32Interface } from '@bitgo-beta/utxo-lib';\n\nconst { unspentSum, scriptTypeForChain, outputScripts } = utxolib.bitgo;\nexport type 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\nimport { Dimensions } from '@bitgo-beta/unspents';\n\nimport { BitGoBase, IWallet, Keychain, Triple, Wallet } from '@bitgo-beta/sdk-core';\nimport { AbstractUtxoCoin, TransactionInfo } from '../abstractUtxoCoin';\n\nimport { decrypt } from '@bitgo-beta/sdk-api';\nimport { signAndVerifyWalletTransaction } from '../sign';\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 * @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        const isWalletOwned = await isWalletAddress(wallet, canonicalAddress);\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  return (\n    await Bluebird.mapSeries(addresses, async (address): Promise<WalletUnspent<TNumber>[]> => {\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        return [];\n      }\n      return unspents\n        .filter((u) => u.address === address)\n        .map((u) => ({\n          ...u,\n          ...scriptId,\n        }));\n    })\n  ).flat();\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"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/recovery/index.ts"],"names":[],"mappings":"AAAA,cAAc,oBAAoB,CAAC;AACnC,cAAc,qBAAqB,CAAC;AACpC,cAAc,WAAW,CAAC;AAC1B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,sBAAsB,CAAC;AACrC,cAAc,cAAc,CAAC
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/recovery/index.ts"],"names":[],"mappings":"AAAA,cAAc,oBAAoB,CAAC;AACnC,cAAc,qBAAqB,CAAC;AACpC,cAAc,WAAW,CAAC;AAC1B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,sBAAsB,CAAC;AACrC,cAAc,cAAc,CAAC"}
|
|
@@ -1,7 +1,11 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
3
|
if (k2 === undefined) k2 = k;
|
|
4
|
-
Object.
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
5
9
|
}) : (function(o, m, k, k2) {
|
|
6
10
|
if (k2 === undefined) k2 = k;
|
|
7
11
|
o[k2] = m[k];
|
|
@@ -16,5 +20,4 @@ __exportStar(require("./baseApi"), exports);
|
|
|
16
20
|
__exportStar(require("./coingeckoApi"), exports);
|
|
17
21
|
__exportStar(require("./crossChainRecovery"), exports);
|
|
18
22
|
__exportStar(require("./mempoolApi"), exports);
|
|
19
|
-
|
|
20
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcmVjb3ZlcnkvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBQUEscURBQW1DO0FBQ25DLHNEQUFvQztBQUNwQyw0Q0FBMEI7QUFDMUIsaURBQStCO0FBQy9CLHVEQUFxQztBQUNyQywrQ0FBNkI7QUFDN0IsZ0RBQThCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9SZWNvdmVyeVByb3ZpZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vYmFja3VwS2V5UmVjb3ZlcnknO1xuZXhwb3J0ICogZnJvbSAnLi9iYXNlQXBpJztcbmV4cG9ydCAqIGZyb20gJy4vY29pbmdlY2tvQXBpJztcbmV4cG9ydCAqIGZyb20gJy4vY3Jvc3NDaGFpblJlY292ZXJ5JztcbmV4cG9ydCAqIGZyb20gJy4vbWVtcG9vbEFwaSc7XG5leHBvcnQgKiBmcm9tICcuL3NtYXJ0Yml0QXBpJztcbiJdfQ==
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcmVjb3ZlcnkvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHFEQUFtQztBQUNuQyxzREFBb0M7QUFDcEMsNENBQTBCO0FBQzFCLGlEQUErQjtBQUMvQix1REFBcUM7QUFDckMsK0NBQTZCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9SZWNvdmVyeVByb3ZpZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vYmFja3VwS2V5UmVjb3ZlcnknO1xuZXhwb3J0ICogZnJvbSAnLi9iYXNlQXBpJztcbmV4cG9ydCAqIGZyb20gJy4vY29pbmdlY2tvQXBpJztcbmV4cG9ydCAqIGZyb20gJy4vY3Jvc3NDaGFpblJlY292ZXJ5JztcbmV4cG9ydCAqIGZyb20gJy4vbWVtcG9vbEFwaSc7XG4iXX0=
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mempoolApi.d.ts","sourceRoot":"","sources":["../../../src/recovery/mempoolApi.ts"],"names":[],"mappings":"AAKA,OAAO,EAA0B,OAAO,EAAE,MAAM,WAAW,CAAC;AAE5D,qBAAa,UAAW,SAAQ,OAAO;IACrC,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,MAAM,GAAG,UAAU;gBAUhC,OAAO,EAAE,MAAM;IAIrB,sBAAsB,IAAI,OAAO,CAAC,MAAM,CAAC;
|
|
1
|
+
{"version":3,"file":"mempoolApi.d.ts","sourceRoot":"","sources":["../../../src/recovery/mempoolApi.ts"],"names":[],"mappings":"AAKA,OAAO,EAA0B,OAAO,EAAE,MAAM,WAAW,CAAC;AAE5D,qBAAa,UAAW,SAAQ,OAAO;IACrC,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,MAAM,GAAG,UAAU;gBAUhC,OAAO,EAAE,MAAM;IAIrB,sBAAsB,IAAI,OAAO,CAAC,MAAM,CAAC;CAWhD"}
|
|
@@ -22,12 +22,15 @@ class MempoolApi extends baseApi_1.BaseApi {
|
|
|
22
22
|
async getRecoveryFeePerBytes() {
|
|
23
23
|
const res = await this.get('/fees/recommended');
|
|
24
24
|
return res.map((body) => {
|
|
25
|
-
if (body.
|
|
26
|
-
return body.
|
|
25
|
+
if (body.fastestFee && _.isInteger(body.fastestFee)) {
|
|
26
|
+
return body.fastestFee;
|
|
27
|
+
}
|
|
28
|
+
else if (body.hourFee && _.isInteger(body.hourFee)) {
|
|
29
|
+
return body.hourFee;
|
|
27
30
|
}
|
|
28
31
|
throw new Error('unexpected response');
|
|
29
32
|
});
|
|
30
33
|
}
|
|
31
34
|
}
|
|
32
35
|
exports.MempoolApi = MempoolApi;
|
|
33
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
36
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVtcG9vbEFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9yZWNvdmVyeS9tZW1wb29sQXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBOztHQUVHO0FBQ0gsNEJBQTRCO0FBRTVCLHVDQUE0RDtBQUU1RCxNQUFhLFVBQVcsU0FBUSxpQkFBTztJQUNyQyxNQUFNLENBQUMsT0FBTyxDQUFDLFFBQWdCO1FBQzdCLFFBQVEsUUFBUSxFQUFFO1lBQ2hCLEtBQUssS0FBSyxDQUFDO1lBQ1gsS0FBSyxNQUFNO2dCQUNULGtEQUFrRDtnQkFDbEQsT0FBTyxJQUFJLFVBQVUsQ0FBQyw4QkFBOEIsQ0FBQyxDQUFDO1NBQ3pEO1FBQ0QsTUFBTSxJQUFJLGdDQUFzQixDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQzdDLENBQUM7SUFFRCxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ2pCLENBQUM7SUFFRCxLQUFLLENBQUMsc0JBQXNCO1FBQzFCLE1BQU0sR0FBRyxHQUFHLE1BQU0sSUFBSSxDQUFDLEdBQUcsQ0FBTSxtQkFBbUIsQ0FBQyxDQUFDO1FBQ3JELE9BQU8sR0FBRyxDQUFDLEdBQUcsQ0FBQyxDQUFDLElBQUksRUFBRSxFQUFFO1lBQ3RCLElBQUksSUFBSSxDQUFDLFVBQVUsSUFBSSxDQUFDLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsRUFBRTtnQkFDbkQsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDO2FBQ3hCO2lCQUFNLElBQUksSUFBSSxDQUFDLE9BQU8sSUFBSSxDQUFDLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsRUFBRTtnQkFDcEQsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDO2FBQ3JCO1lBQ0QsTUFBTSxJQUFJLEtBQUssQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDO1FBQ3pDLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBMUJELGdDQTBCQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQHByZXR0aWVyXG4gKi9cbmltcG9ydCAqIGFzIF8gZnJvbSAnbG9kYXNoJztcblxuaW1wb3J0IHsgQXBpTm90SW1wbGVtZW50ZWRFcnJvciwgQmFzZUFwaSB9IGZyb20gJy4vYmFzZUFwaSc7XG5cbmV4cG9ydCBjbGFzcyBNZW1wb29sQXBpIGV4dGVuZHMgQmFzZUFwaSB7XG4gIHN0YXRpYyBmb3JDb2luKGNvaW5OYW1lOiBzdHJpbmcpOiBNZW1wb29sQXBpIHtcbiAgICBzd2l0Y2ggKGNvaW5OYW1lKSB7XG4gICAgICBjYXNlICdidGMnOlxuICAgICAgY2FzZSAndGJ0Yyc6XG4gICAgICAgIC8vIEZJWE1FOiB0YnRjIGlzIGVuYWJsZWQgaGVyZSBmb3IgbGVnYWN5IHJlYXNvbnM7XG4gICAgICAgIHJldHVybiBuZXcgTWVtcG9vbEFwaSgnaHR0cHM6Ly9tZW1wb29sLnNwYWNlL2FwaS92MScpO1xuICAgIH1cbiAgICB0aHJvdyBuZXcgQXBpTm90SW1wbGVtZW50ZWRFcnJvcihjb2luTmFtZSk7XG4gIH1cblxuICBjb25zdHJ1Y3RvcihiYXNlVXJsOiBzdHJpbmcpIHtcbiAgICBzdXBlcihiYXNlVXJsKTtcbiAgfVxuXG4gIGFzeW5jIGdldFJlY292ZXJ5RmVlUGVyQnl0ZXMoKTogUHJvbWlzZTxudW1iZXI+IHtcbiAgICBjb25zdCByZXMgPSBhd2FpdCB0aGlzLmdldDxhbnk+KCcvZmVlcy9yZWNvbW1lbmRlZCcpO1xuICAgIHJldHVybiByZXMubWFwKChib2R5KSA9PiB7XG4gICAgICBpZiAoYm9keS5mYXN0ZXN0RmVlICYmIF8uaXNJbnRlZ2VyKGJvZHkuZmFzdGVzdEZlZSkpIHtcbiAgICAgICAgcmV0dXJuIGJvZHkuZmFzdGVzdEZlZTtcbiAgICAgIH0gZWxzZSBpZiAoYm9keS5ob3VyRmVlICYmIF8uaXNJbnRlZ2VyKGJvZHkuaG91ckZlZSkpIHtcbiAgICAgICAgcmV0dXJuIGJvZHkuaG91ckZlZTtcbiAgICAgIH1cbiAgICAgIHRocm93IG5ldyBFcnJvcigndW5leHBlY3RlZCByZXNwb25zZScpO1xuICAgIH0pO1xuICB9XG59XG4iXX0=
|
package/dist/src/sign.d.ts
CHANGED
|
@@ -2,18 +2,42 @@
|
|
|
2
2
|
* @prettier
|
|
3
3
|
*/
|
|
4
4
|
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
5
|
-
|
|
6
|
-
|
|
5
|
+
type Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;
|
|
6
|
+
type RootWalletKeys = utxolib.bitgo.RootWalletKeys;
|
|
7
7
|
export declare class InputSigningError<TNumber extends number | bigint = number> extends Error {
|
|
8
8
|
inputIndex: number;
|
|
9
|
-
unspent: Unspent<TNumber
|
|
9
|
+
unspent: Unspent<TNumber> | {
|
|
10
|
+
id: string;
|
|
11
|
+
};
|
|
10
12
|
reason: Error | string;
|
|
11
|
-
static expectedWalletUnspent<TNumber extends number | bigint>(inputIndex: number, unspent: Unspent<TNumber>
|
|
12
|
-
|
|
13
|
+
static expectedWalletUnspent<TNumber extends number | bigint>(inputIndex: number, unspent: Unspent<TNumber> | {
|
|
14
|
+
id: string;
|
|
15
|
+
}): InputSigningError<TNumber>;
|
|
16
|
+
constructor(inputIndex: number, unspent: Unspent<TNumber> | {
|
|
17
|
+
id: string;
|
|
18
|
+
}, reason: Error | string);
|
|
13
19
|
}
|
|
14
20
|
export declare class TransactionSigningError<TNumber extends number | bigint = number> extends Error {
|
|
15
21
|
constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]);
|
|
16
22
|
}
|
|
23
|
+
/**
|
|
24
|
+
* Sign all inputs of a psbt and verify signatures after signing.
|
|
25
|
+
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
26
|
+
* failed.
|
|
27
|
+
*
|
|
28
|
+
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
29
|
+
*
|
|
30
|
+
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
31
|
+
* using TransactionBuilder
|
|
32
|
+
*
|
|
33
|
+
* @param psbt
|
|
34
|
+
* @param signerKeychain
|
|
35
|
+
* @param isLastSignature
|
|
36
|
+
*/
|
|
37
|
+
export declare function signAndVerifyPsbt(psbt: utxolib.bitgo.UtxoPsbt, signerKeychain: utxolib.BIP32Interface, { isLastSignature, allowNonSegwitSigningWithoutPrevTx, }: {
|
|
38
|
+
isLastSignature: boolean;
|
|
39
|
+
allowNonSegwitSigningWithoutPrevTx?: boolean;
|
|
40
|
+
}): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>;
|
|
17
41
|
/**
|
|
18
42
|
* Sign all inputs of a wallet transaction and verify signatures after signing.
|
|
19
43
|
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
package/dist/src/sign.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"sign.d.ts","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,
|
|
1
|
+
{"version":3,"file":"sign.d.ts","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,KAAK,OAAO,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,IAAI,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;AACxF,KAAK,cAAc,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC;AAgBnD,qBAAa,iBAAiB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,KAAK;IAS3E,UAAU,EAAE,MAAM;IAClB,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE;IAC1C,MAAM,EAAE,KAAK,GAAG,MAAM;IAV/B,MAAM,CAAC,qBAAqB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,GACzC,iBAAiB,CAAC,OAAO,CAAC;gBAKpB,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,EAC1C,MAAM,EAAE,KAAK,GAAG,MAAM;CAIhC;AAED,qBAAa,uBAAuB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,KAAK;gBAC9E,UAAU,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE,EAAE,WAAW,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE;CAMhG;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,cAAc,EAAE,OAAO,CAAC,cAAc,EACtC,EACE,eAAe,EACf,kCAAkC,GACnC,EAAE;IAAE,eAAe,EAAE,OAAO,CAAC;IAAC,kCAAkC,CAAC,EAAE,OAAO,CAAA;CAAE,GAC5E,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,CAuEhE;AAED;;;;;;;;;GASG;AACH,wBAAgB,8BAA8B,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC5E,WAAW,EAAE,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,GAAG,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,OAAO,CAAC,EACnG,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC,EAAE,EAC5B,YAAY,EAAE,OAAO,CAAC,KAAK,CAAC,mBAAmB,CAAC,cAAc,CAAC,EAC/D,EAAE,eAAe,EAAE,EAAE;IAAE,eAAe,EAAE,OAAO,CAAA;CAAE,GAChD,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,CAqExC"}
|
package/dist/src/sign.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.signAndVerifyWalletTransaction = exports.TransactionSigningError = exports.InputSigningError = void 0;
|
|
3
|
+
exports.signAndVerifyWalletTransaction = exports.signAndVerifyPsbt = exports.TransactionSigningError = exports.InputSigningError = void 0;
|
|
4
4
|
/**
|
|
5
5
|
* @prettier
|
|
6
6
|
*/
|
|
@@ -10,15 +10,15 @@ const debugLib = require("debug");
|
|
|
10
10
|
const replayProtection_1 = require("./replayProtection");
|
|
11
11
|
const debug = debugLib('bitgo:v2:utxo');
|
|
12
12
|
class InputSigningError extends Error {
|
|
13
|
+
static expectedWalletUnspent(inputIndex, unspent) {
|
|
14
|
+
return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);
|
|
15
|
+
}
|
|
13
16
|
constructor(inputIndex, unspent, reason) {
|
|
14
17
|
super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);
|
|
15
18
|
this.inputIndex = inputIndex;
|
|
16
19
|
this.unspent = unspent;
|
|
17
20
|
this.reason = reason;
|
|
18
21
|
}
|
|
19
|
-
static expectedWalletUnspent(inputIndex, unspent) {
|
|
20
|
-
return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);
|
|
21
|
-
}
|
|
22
22
|
}
|
|
23
23
|
exports.InputSigningError = InputSigningError;
|
|
24
24
|
class TransactionSigningError extends Error {
|
|
@@ -28,6 +28,72 @@ class TransactionSigningError extends Error {
|
|
|
28
28
|
}
|
|
29
29
|
}
|
|
30
30
|
exports.TransactionSigningError = TransactionSigningError;
|
|
31
|
+
/**
|
|
32
|
+
* Sign all inputs of a psbt and verify signatures after signing.
|
|
33
|
+
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
34
|
+
* failed.
|
|
35
|
+
*
|
|
36
|
+
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
37
|
+
*
|
|
38
|
+
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
39
|
+
* using TransactionBuilder
|
|
40
|
+
*
|
|
41
|
+
* @param psbt
|
|
42
|
+
* @param signerKeychain
|
|
43
|
+
* @param isLastSignature
|
|
44
|
+
*/
|
|
45
|
+
function signAndVerifyPsbt(psbt, signerKeychain, { isLastSignature, allowNonSegwitSigningWithoutPrevTx, }) {
|
|
46
|
+
const txInputs = psbt.txInputs;
|
|
47
|
+
const outputIds = [];
|
|
48
|
+
const scriptTypes = [];
|
|
49
|
+
const signErrors = psbt.data.inputs
|
|
50
|
+
.map((input, inputIndex) => {
|
|
51
|
+
const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
|
|
52
|
+
outputIds.push(outputId);
|
|
53
|
+
const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
|
|
54
|
+
scriptTypes.push(scriptType);
|
|
55
|
+
if (scriptType === 'p2shP2pk') {
|
|
56
|
+
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
try {
|
|
60
|
+
utxolib.bitgo.withUnsafeNonSegwit(psbt, () => psbt.signInputHD(inputIndex, signerKeychain), !!allowNonSegwitSigningWithoutPrevTx);
|
|
61
|
+
debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
|
|
62
|
+
}
|
|
63
|
+
catch (e) {
|
|
64
|
+
return new InputSigningError(inputIndex, { id: outputId }, e);
|
|
65
|
+
}
|
|
66
|
+
})
|
|
67
|
+
.filter((e) => e !== undefined);
|
|
68
|
+
const verifyErrors = psbt.data.inputs
|
|
69
|
+
.map((input, inputIndex) => {
|
|
70
|
+
const scriptType = scriptTypes[inputIndex];
|
|
71
|
+
if (scriptType === 'p2shP2pk') {
|
|
72
|
+
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
const outputId = outputIds[inputIndex];
|
|
76
|
+
try {
|
|
77
|
+
if (!utxolib.bitgo.withUnsafeNonSegwit(psbt, () => psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain), !!allowNonSegwitSigningWithoutPrevTx)) {
|
|
78
|
+
return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
catch (e) {
|
|
82
|
+
debug('Invalid signature');
|
|
83
|
+
return new InputSigningError(inputIndex, { id: outputId }, e);
|
|
84
|
+
}
|
|
85
|
+
})
|
|
86
|
+
.filter((e) => e !== undefined);
|
|
87
|
+
if (signErrors.length || verifyErrors.length) {
|
|
88
|
+
throw new TransactionSigningError(signErrors, verifyErrors);
|
|
89
|
+
}
|
|
90
|
+
if (isLastSignature) {
|
|
91
|
+
psbt.finalizeAllInputs();
|
|
92
|
+
return psbt.extractTransaction();
|
|
93
|
+
}
|
|
94
|
+
return psbt;
|
|
95
|
+
}
|
|
96
|
+
exports.signAndVerifyPsbt = signAndVerifyPsbt;
|
|
31
97
|
/**
|
|
32
98
|
* Sign all inputs of a wallet transaction and verify signatures after signing.
|
|
33
99
|
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
@@ -56,7 +122,7 @@ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { i
|
|
|
56
122
|
}
|
|
57
123
|
const signErrors = unspents
|
|
58
124
|
.map((unspent, inputIndex) => {
|
|
59
|
-
if (replayProtection_1.isReplayProtectionUnspent(unspent, network)) {
|
|
125
|
+
if ((0, replayProtection_1.isReplayProtectionUnspent)(unspent, network)) {
|
|
60
126
|
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);
|
|
61
127
|
return;
|
|
62
128
|
}
|
|
@@ -76,7 +142,7 @@ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { i
|
|
|
76
142
|
const verifyErrors = signedTransaction.ins
|
|
77
143
|
.map((input, inputIndex) => {
|
|
78
144
|
const unspent = unspents[inputIndex];
|
|
79
|
-
if (replayProtection_1.isReplayProtectionUnspent(unspent, network)) {
|
|
145
|
+
if ((0, replayProtection_1.isReplayProtectionUnspent)(unspent, network)) {
|
|
80
146
|
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, unspents.length);
|
|
81
147
|
return;
|
|
82
148
|
}
|
|
@@ -101,4 +167,4 @@ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { i
|
|
|
101
167
|
return signedTransaction;
|
|
102
168
|
}
|
|
103
169
|
exports.signAndVerifyWalletTransaction = signAndVerifyWalletTransaction;
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign.js","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\nimport * as debugLib from 'debug';\n\nimport { isReplayProtectionUnspent } from './replayProtection';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nexport class InputSigningError<TNumber extends number | bigint = number> extends Error {\n  static expectedWalletUnspent<TNumber extends number | bigint>(\n    inputIndex: number,\n    unspent: Unspent<TNumber>\n  ): InputSigningError<TNumber> {\n    return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);\n  }\n\n  constructor(public inputIndex: number, public unspent: Unspent<TNumber>, public reason: Error | string) {\n    super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);\n  }\n}\n\nexport class TransactionSigningError<TNumber extends number | bigint = number> extends Error {\n  constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]) {\n    super(\n      `sign errors at inputs: [${signErrors.join(',')}], ` +\n        `verify errors at inputs: [${verifyError.join(',')}], see log for details`\n    );\n  }\n}\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  { isLastSignature }: { isLastSignature: boolean }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n"]}
|
|
170
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign.js","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\nimport * as debugLib from 'debug';\n\nimport { isReplayProtectionUnspent } from './replayProtection';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\ntype PsbtParsedScriptTypes =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend';\n\nexport class InputSigningError<TNumber extends number | bigint = number> extends Error {\n  static expectedWalletUnspent<TNumber extends number | bigint>(\n    inputIndex: number,\n    unspent: Unspent<TNumber> | { id: string }\n  ): InputSigningError<TNumber> {\n    return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);\n  }\n\n  constructor(\n    public inputIndex: number,\n    public unspent: Unspent<TNumber> | { id: string },\n    public reason: Error | string\n  ) {\n    super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);\n  }\n}\n\nexport class TransactionSigningError<TNumber extends number | bigint = number> extends Error {\n  constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]) {\n    super(\n      `sign errors at inputs: [${signErrors.join(',')}], ` +\n        `verify errors at inputs: [${verifyError.join(',')}], see log for details`\n    );\n  }\n}\n\n/**\n * Sign all inputs of a psbt and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * If it is the last signature, finalize and extract the transaction from the psbt.\n *\n * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of\n * using TransactionBuilder\n *\n * @param psbt\n * @param signerKeychain\n * @param isLastSignature\n */\nexport function signAndVerifyPsbt(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: utxolib.BIP32Interface,\n  {\n    isLastSignature,\n    allowNonSegwitSigningWithoutPrevTx,\n  }: { isLastSignature: boolean; allowNonSegwitSigningWithoutPrevTx?: boolean }\n): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint> {\n  const txInputs = psbt.txInputs;\n  const outputIds: string[] = [];\n  const scriptTypes: PsbtParsedScriptTypes[] = [];\n\n  const signErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex: number) => {\n      const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));\n      outputIds.push(outputId);\n\n      const { scriptType } = utxolib.bitgo.parsePsbtInput(input);\n      scriptTypes.push(scriptType);\n\n      if (scriptType === 'p2shP2pk') {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);\n        return;\n      }\n\n      try {\n        utxolib.bitgo.withUnsafeNonSegwit(\n          psbt,\n          () => psbt.signInputHD(inputIndex, signerKeychain),\n          !!allowNonSegwitSigningWithoutPrevTx\n        );\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  const verifyErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex) => {\n      const scriptType = scriptTypes[inputIndex];\n      if (scriptType === 'p2shP2pk') {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          psbt.data.inputs.length\n        );\n        return;\n      }\n\n      const outputId = outputIds[inputIndex];\n      try {\n        if (\n          !utxolib.bitgo.withUnsafeNonSegwit(\n            psbt,\n            () => psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain),\n            !!allowNonSegwitSigningWithoutPrevTx\n          )\n        ) {\n          return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  if (isLastSignature) {\n    psbt.finalizeAllInputs();\n    return psbt.extractTransaction();\n  }\n\n  return psbt;\n}\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  { isLastSignature }: { isLastSignature: boolean }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n"]}
|