@bitgo-beta/abstract-utxo 1.6.1-alpha.431 → 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.
@@ -42,6 +42,7 @@ const sinon = __importStar(require("sinon"));
42
42
  const nock = require("nock");
43
43
  const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
44
44
  const sdk_core_1 = require("@bitgo-beta/sdk-core");
45
+ const unspents_1 = require("@bitgo-beta/unspents");
45
46
  const src_1 = require("../../../src");
46
47
  const util_1 = require("../util");
47
48
  const mock_1 = require("./mock");
@@ -91,6 +92,7 @@ const exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(util_1.keychains, [
91
92
  ]);
92
93
  const keysFullSignedRecoveryExotic = getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase);
93
94
  function run(coin, scriptTypes, walletKeys, params, tags = []) {
95
+ const defaultFeeRateSatB = 100;
94
96
  describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {
95
97
  const externalWallet = (0, util_1.getWalletKeys)('external');
96
98
  const recoveryDestination = (0, util_1.getWalletAddress)(coin.network, externalWallet);
@@ -138,6 +140,7 @@ function run(coin, scriptTypes, walletKeys, params, tags = []) {
138
140
  krsProvider: params.krsProvider,
139
141
  ...params.keys,
140
142
  recoveryProvider: new mock_1.MockRecoveryProvider(mockedApiUnspents),
143
+ feeRate: defaultFeeRateSatB,
141
144
  });
142
145
  const txHex = recovery.transactionHex ?? recovery.txHex;
143
146
  const isPsbt = utxolib.bitgo.isPsbt(txHex);
@@ -151,6 +154,19 @@ function run(coin, scriptTypes, walletKeys, params, tags = []) {
151
154
  const recoveryProvider = new mock_1.MockRecoveryProvider(mockedApiUnspents);
152
155
  (await recoveryProvider.getUnspentsForAddresses(mockedApiUnspents.map((u) => u.address))).length.should.eql(mockedApiUnspents.length);
153
156
  });
157
+ it('has expected fee rate', function () {
158
+ if (!(recoveryTx instanceof utxolib.bitgo.UtxoPsbt)) {
159
+ this.skip();
160
+ }
161
+ const inputSum = utxolib.bitgo.unspentSum(recoverUnspents, 'bigint');
162
+ const outputSum = recoveryTx.txOutputs.reduce((sum, o) => sum + o.value, BigInt(0));
163
+ const fee = inputSum - outputSum;
164
+ const vsize = unspents_1.Dimensions.fromPsbt(recoveryTx).getVSize();
165
+ const feeRateSatB = Number(fee) / vsize;
166
+ const diff = Math.abs(feeRateSatB - defaultFeeRateSatB) / defaultFeeRateSatB;
167
+ // within 1%
168
+ assert_1.default.strictEqual(diff < 0.01, true, `expected fee rate ${defaultFeeRateSatB} but got ${feeRateSatB}`);
169
+ });
154
170
  it('matches fixture', async function () {
155
171
  const fixtureCoin = (0, util_1.getNormalTestnetCoin)(coin);
156
172
  const fixtureRecovery = { ...recovery };
@@ -261,12 +277,6 @@ function runWithScriptTypes(scriptTypes, { runKeyProviderTests = true } = {}) {
261
277
  hasUserSignature: true,
262
278
  hasBackupSignature: true,
263
279
  }, ['fullSignedRecovery', ...scriptTypes]);
264
- run(coin, scriptTypes, walletKeys, {
265
- keys: keysFullSignedRecovery,
266
- hasUserSignature: true,
267
- hasBackupSignature: true,
268
- feeRate: 2,
269
- }, ['fullSignedRecovery', 'fixedFeeRate', ...scriptTypes]);
270
280
  run(coin, scriptTypes, exoticWalletKeys, {
271
281
  keys: keysFullSignedRecoveryExotic,
272
282
  userKeyPath: exoticUserKeyPath,
@@ -283,4 +293,4 @@ describe('Backup Key Recovery', function () {
283
293
  // taproot compatible coins
284
294
  runWithScriptTypes(['p2tr', 'p2trMusig2'], { runKeyProviderTests: false });
285
295
  });
286
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backupKeyRecovery.js","sourceRoot":"","sources":["../../../../../test/unit/recovery/backupKeyRecovery.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,oDAA4B;AAE5B,kBAAgB;AAEhB,6CAA+B;AAC/B,6BAA8B;AAE9B,8DAAgD;AAChD,mDAAoE;AAEpE,sCAMsB;AACtB,kCAYiB;AAEjB,iCAA8C;AAE9C,MAAM,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;AAKnC,MAAM,MAAM,GAAG,EAAE,YAAY,EAAZ,uBAAY,EAAE,CAAC;AAEhC,IAAI,CAAC,iBAAiB,EAAE,CAAC;AAEzB,SAAS,cAAc,CAAC,CAA2B;IACjD,MAAM,MAAM,GAAG,EAAE,GAAG,uBAAY,EAAE,CAAC;IACnC,MAAM,CAAC;QACL,CAAC,CAAC,MAAM,CAAC,CAAC;IACZ,CAAC,CAAC,CAAC;IACH,KAAK,CAAC;QACJ,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE;YACxC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QAChB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,gBAAgB,GAAG,KAAK,CAAC;AAQ/B,SAAS,YAAY,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EAAE,QAAgB;IAC5F,SAAS,MAAM,CAAC,CAAiB;QAC/B,OAAO,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,IAAA,sBAAe,EAAC,QAAQ,EAAE,IAAA,uBAAgB,EAAC,CAAC,CAAC,CAAC,CAAC;IACxF,CAAC;IACD,OAAO;QACL,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC;QACxB,SAAS,EAAE,MAAM,CAAC,SAAS,CAAC;QAC5B,QAAQ,EAAE,MAAM,CAAC,QAAQ,CAAC;KAC3B,CAAC;AACJ,CAAC;AAED,SAAS,uBAAuB,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EAAE,QAAgB;IACvG,OAAO,YAAY,CAAC,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,SAAS,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC;AACjG,CAAC;AAED,SAAS,4BAA4B,CACnC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EACtD,QAAgB;IAEhB,OAAO,YAAY,CAAC,CAAC,OAAO,EAAE,SAAS,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC;AACtF,CAAC;AAED,SAAS,4BAA4B,CACnC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EACtD,QAAgB;IAEhB,OAAO,YAAY,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC;AAC3E,CAAC;AAED,MAAM,UAAU,GAAG,IAAA,2BAAoB,GAAE,CAAC;AAC1C,MAAM,iBAAiB,GAAG,uBAAuB,CAAC,UAAU,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AACvF,MAAM,sBAAsB,GAAG,4BAA4B,CAAC,UAAU,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AACjG,MAAM,sBAAsB,GAAG,4BAA4B,CAAC,UAAU,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AAEjG,MAAM,iBAAiB,GAAG,OAAO,CAAC;AAClC,MAAM,gBAAgB,GAAG,IAAI,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,gBAAS,EAAE;IACnE,iBAAiB;IACjB,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,aAAa;IAC1C,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,aAAa;CAC3C,CAAC,CAAC;AACH,MAAM,4BAA4B,GAAG,4BAA4B,CAAC,gBAAgB,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AAE7G,SAAS,GAAG,CACV,IAAsB,EACtB,WAA6B,EAC7B,UAA0B,EAC1B,MAQC,EACD,OAAiB,EAAE;IAEnB,QAAQ,CAAC,wBAAwB,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,GAAG,IAAI,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE;QAC5F,MAAM,cAAc,GAAG,IAAA,oBAAa,EAAC,UAAU,CAAC,CAAC;QACjD,MAAM,mBAAmB,GAAG,IAAA,uBAAgB,EAAC,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;QAE3E,IAAI,yBAAiC,CAAC;QACtC,IAAI,QAA6F,CAAC;QAClG,IAAI,UAAmF,CAAC;QAExF,uGAAuG;QACvG,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;QACxE,MAAM,CAAC,MAAM,EAAE;YACb,KAAK,CAAC,IAAI,CAAC,kBAAY,CAAC,SAAS,EAAE,aAAa,CAAC,CAAC,QAAQ,CAAC,KAAM,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;QAEH,cAAc,CAAC,UAAU,MAAc;YACrC,MAAM,kBAAkB,GAAG,EAAE,GAAG,MAAM,CAAC,YAAY,EAAE,CAAC;YACtD,kBAAkB,CAAC,GAAG,CAAC,cAAc,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YAC3D,kBAAkB,CAAC,SAAS,CAAC,cAAc,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YACjE,yBAAyB,GAAG,IAAA,uBAAgB,EAAC,IAAI,CAAC,OAAO,EAAE,cAAc,EAAE,CAAC,EAAE,GAAG,CAAC,CAAC;YACnF,kBAAkB,CAAC,SAAS,CAAC,YAAY,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,EAAE,yBAAyB,EAAE,CAAC;YAC7F,MAAM,CAAC,YAAY,GAAG,kBAAkB,CAAC;QAC3C,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC;YACJ,KAAK,CAAC,OAAO,EAAE,CAAC;QAClB,CAAC,CAAC,CAAC;QAEH,IAAI,eAAgD,CAAC;QACrD,IAAI,iBAAkD,CAAC;QAEvD,MAAM,CAAC,sBAAsB,EAAE,KAAK;YAClC,IAAI,CAAC,OAAO,CAAC,KAAM,CAAC,CAAC;YACrB,eAAe,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC,UAAU,EAAE,KAAK,EAAE,EAAE,CAAC;gBAC3D,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;gBAC1G,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;gBAC1G,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;aAC3G,CAAC,CAAC;YAEH,qHAAqH;YACrH,gDAAgD;YAChD,iBAAiB;gBACf,IAAI,CAAC,QAAQ,EAAE,KAAK,KAAK,IAAI,IAAI,CAAC,QAAQ,EAAE,KAAK,MAAM;oBACrD,CAAC,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;oBAC7G,CAAC,CAAC,eAAe,CAAC;YAEtB,gBAAM,CAAC,WAAW,CAAC,iBAAiB,CAAC,MAAM,EAAE,eAAe,CAAC,MAAM,CAAC,CAAC;YACrE,QAAQ,GAAG,MAAM,IAAA,uBAAiB,EAAC,IAAI,EAAE,mBAAY,EAAE;gBACrD,gBAAgB;gBAChB,mBAAmB;gBACnB,IAAI,EAAE,CAAC;gBACP,kBAAkB,EAAE,EAAE;gBACtB,WAAW,EAAE,MAAM,CAAC,WAAW;gBAC/B,WAAW,EAAE,MAAM,CAAC,WAAW;gBAC/B,GAAG,MAAM,CAAC,IAAI;gBACd,gBAAgB,EAAE,IAAI,2BAAoB,CAAC,iBAAiB,CAAC;aAC9D,CAAC,CAAC;YACH,MAAM,KAAK,GACR,QAA4C,CAAC,cAAc,IAAK,QAAwC,CAAC,KAAK,CAAC;YAClH,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC3C,UAAU,GAAG,MAAM;gBACjB,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC;gBACtD,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,wBAAwB,CAAC,KAAe,EAAE,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;YAC3F,QAAQ,CAAC,IAAI;gBACX,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QAC3G,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qCAAqC,EAAE,KAAK;YAC7C,MAAM,gBAAgB,GAAG,IAAI,2BAAoB,CAAC,iBAAiB,CAAC,CAAC;YACrE,CAAC,MAAM,gBAAgB,CAAC,uBAAuB,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CACzG,iBAAiB,CAAC,MAAM,CACzB,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iBAAiB,EAAE,KAAK;YACzB,MAAM,WAAW,GAAG,IAAA,2BAAoB,EAAC,IAAI,CAAC,CAAC;YAC/C,MAAM,eAAe,GAAG,EAAE,GAAG,QAAQ,EAAE,CAAC;YACxC,IAAI,eAAe,CAAC,IAAI,EAAE,CAAC;gBACzB,eAAe,CAAC,IAAI,GAAG,WAAW,CAAC,QAAQ,EAAE,CAAC;YAChD,CAAC;YAED,IAAA,sBAAe,EACb,eAAe,EACf,MAAM,IAAA,iBAAU,EACd,WAAW,EACX,8BAA8B,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,EACzG,QAAQ,CACT,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0BAA0B,EAAE;YAC7B,CAAC,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CACxG,eAAe,CAAC,MAAM,CACvB,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,SAAS,mBAAmB,CAC1B,EAA2E,EAC3E,OAAuB,EACvB,WAAmB;YAEnB,IAAI,EAAE,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;gBACzC,SAAS,QAAQ,CAAC,EAA0B,EAAE,UAAkB;oBAC9D,IAAI,CAAC;wBACH,OAAO,EAAE,CAAC,2BAA2B,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;oBAC7D,CAAC;oBAAC,OAAO,CAAC,EAAE,CAAC;wBACX,IAAI,CAAC,CAAC,OAAO,KAAK,2BAA2B,EAAE,CAAC;4BAC9C,OAAO,KAAK,CAAC;wBACf,CAAC;wBACD,MAAM,CAAC,CAAC;oBACV,CAAC;gBACH,CAAC;gBACD,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;oBAC3C,QAAQ,CAAC,EAAE,EAAE,UAAU,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;gBACrD,CAAC,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,MAAM,WAAW,GAAG,eAAe;qBAChC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;qBACrC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC;gBACpF,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;oBACnC,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,CAAkB,CAAC;oBAC7D,MAAM,EAAE,SAAS,EAAE,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,iBAAiB,CAAC,OAAO,EAAE,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;oBAC9G,MAAM,UAAU,GAAG,OAAO,CAAC,KAAK;yBAC7B,yBAAyB,CACxB,EAAE,EACF,UAAU,EACV,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,EACvD,EAAE,SAAS,EAAE,EACb,WAAW,CACZ;yBACA,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC;oBAC3C,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;gBAC5C,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,EAAE,CAAC,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,iBAAiB,EAAE;YACnE,mBAAmB,CAAC,UAAU,EAAE,UAAU,CAAC,IAAI,EAAE,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,mBAAmB,EAAE;YACvE,mBAAmB,CAAC,UAAU,EAAE,UAAU,CAAC,MAAM,EAAE,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACxF,CAAC,CAAC,CAAC;QAEH,IAAI,MAAM,CAAC,gBAAgB,IAAI,MAAM,CAAC,kBAAkB,EAAE,CAAC;YACzD,EAAE,CAAC,+BAA+B,EAAE;gBAClC,IAAI,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,eAAe,EAAE,CAAC;oBACxD,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;wBAC/B,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC;wBACzD,QAAQ,MAAM,CAAC,UAAU,EAAE,CAAC;4BAC1B,KAAK,MAAM,CAAC;4BACZ,KAAK,WAAW,CAAC;4BACjB,KAAK,OAAO,CAAC;4BACb,KAAK,wBAAwB;gCAC3B,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,CAAC,EAAE,EAAE;oCACzC,IAAI,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,SAAS,CAAC,EAAE,CAAC;wCACpD,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,EAAE,CAAC,CAAC;oCACzD,CAAC;gCACH,CAAC,CAAC,CAAC;gCACH,MAAM;4BACR;gCACE,MAAM,IAAI,KAAK,CAAC,yBAAyB,MAAM,CAAC,UAAU,EAAE,CAAC,CAAC;wBAClE,CAAC;oBACH,CAAC,CAAC,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,IAAI,EAAE,CAAC;gBACd,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;QAED,EAAE,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,8BAA8B,EAAE;YAC5E,MAAM,IAAI,GAAG,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC;YAC9G,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAC1B,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC,CAAC,MAAM,EAAE,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;YACxG,eAAe;iBACZ,QAAQ,CAAC,yBAAyB,CAAC;iBACnC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,IAAI,MAAM,CAAC,WAAW,KAAK,WAAW,CAAC,CAAC;YAC3E,eAAe,CAAC,QAAQ,CAAC,mBAAmB,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACjE,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,kBAAkB,CACzB,WAA6B,EAC7B,EAAE,mBAAmB,GAAG,IAAI,KAAwC,EAAE;IAEtE,gBAAS;SACN,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAC,CAAC;SAC7E,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;QAChB,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;YACE,IAAI,EAAE,iBAAiB;YACvB,gBAAgB,EAAE,KAAK;YACvB,kBAAkB,EAAE,KAAK;SAC1B,EACD,CAAC,kBAAkB,EAAE,GAAG,WAAW,CAAC,CACrC,CAAC;QAEF,IAAI,mBAAmB,EAAE,CAAC;YACxB,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,EAAE;gBAC3C,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;oBACE,IAAI,EAAE,sBAAsB;oBAC5B,WAAW,EAAE,WAAW;oBACxB,gBAAgB,EAAE,IAAI;oBACtB,kBAAkB,EAAE,KAAK;oBACzB,YAAY,EAAE,KAAK;iBACpB,EACD,CAAC,oBAAoB,EAAE,GAAG,WAAW,CAAC,CACvC,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC;QAED,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;YACE,IAAI,EAAE,sBAAsB;YAC5B,gBAAgB,EAAE,IAAI;YACtB,kBAAkB,EAAE,IAAI;SACzB,EACD,CAAC,oBAAoB,EAAE,GAAG,WAAW,CAAC,CACvC,CAAC;QAEF,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;YACE,IAAI,EAAE,sBAAsB;YAC5B,gBAAgB,EAAE,IAAI;YACtB,kBAAkB,EAAE,IAAI;YACxB,OAAO,EAAE,CAAC;SACX,EACD,CAAC,oBAAoB,EAAE,cAAc,EAAE,GAAG,WAAW,CAAC,CACvD,CAAC;QAEF,GAAG,CACD,IAAI,EACJ,WAAW,EACX,gBAAgB,EAChB;YACE,IAAI,EAAE,4BAA4B;YAClC,WAAW,EAAE,iBAAiB;YAC9B,gBAAgB,EAAE,IAAI;YACtB,kBAAkB,EAAE,IAAI;SACzB,EACD,CAAC,oBAAoB,EAAE,mBAAmB,EAAE,GAAG,WAAW,CAAC,CAC5D,CAAC;IACJ,CAAC,CAAC,CAAC;AACP,CAAC;AAED,QAAQ,CAAC,qBAAqB,EAAE;IAC9B,4BAA4B;IAC5B,kBAAkB,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;IAE7B,0BAA0B;IAC1B,kBAAkB,CAAC,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC,CAAC;IAE3C,2BAA2B;IAC3B,kBAAkB,CAAC,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,EAAE,mBAAmB,EAAE,KAAK,EAAE,CAAC,CAAC;AAC7E,CAAC,CAAC,CAAC","sourcesContent":["import assert from 'assert';\n\nimport 'should';\nimport * as mocha from 'mocha';\nimport * as sinon from 'sinon';\nimport nock = require('nock');\nimport { BIP32Interface } from '@bitgo-beta/utxo-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Config, krsProviders, Triple } from '@bitgo-beta/sdk-core';\n\nimport {\n  AbstractUtxoCoin,\n  backupKeyRecovery,\n  BackupKeyRecoveryTransansaction,\n  CoingeckoApi,\n  FormattedOfflineVaultTxInfo,\n} from '../../../src';\nimport {\n  defaultBitGo,\n  encryptKeychain,\n  getDefaultWalletKeys,\n  getFixture,\n  getNormalTestnetCoin,\n  getWalletAddress,\n  getWalletKeys,\n  keychains,\n  shouldEqualJSON,\n  toKeychainBase58,\n  utxoCoins,\n} from '../util';\n\nimport { MockRecoveryProvider } from './mock';\n\nconst { toOutput } = utxolib.bitgo;\ntype WalletUnspent = utxolib.bitgo.WalletUnspent<bigint>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\n\nconst config = { krsProviders };\n\nnock.disableNetConnect();\n\nfunction configOverride(f: (config: Config) => void) {\n  const backup = { ...krsProviders };\n  before(function () {\n    f(config);\n  });\n  after(function () {\n    Object.entries(backup).forEach(([k, v]) => {\n      config[k] = v;\n    });\n  });\n}\n\nconst walletPassphrase = 'lol';\n\ntype NamedKeys = {\n  userKey: string;\n  backupKey: string;\n  bitgoKey: string;\n};\n\nfunction getNamedKeys([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  function encode(k: BIP32Interface): string {\n    return k.isNeutered() ? k.toBase58() : encryptKeychain(password, toKeychainBase58(k));\n  }\n  return {\n    userKey: encode(userKey),\n    backupKey: encode(backupKey),\n    bitgoKey: encode(bitgoKey),\n  };\n}\n\nfunction getKeysForUnsignedSweep([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  return getNamedKeys([userKey.neutered(), backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForKeyRecoveryService(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForFullSignedRecovery(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey, bitgoKey.neutered()], password);\n}\n\nconst walletKeys = getDefaultWalletKeys();\nconst keysUnsignedSweep = getKeysForUnsignedSweep(walletKeys.triple, walletPassphrase);\nconst keysKeyRecoveryService = getKeysForKeyRecoveryService(walletKeys.triple, walletPassphrase);\nconst keysFullSignedRecovery = getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase);\n\nconst exoticUserKeyPath = '99/99';\nconst exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(keychains, [\n  exoticUserKeyPath,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n]);\nconst keysFullSignedRecoveryExotic = getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase);\n\nfunction run(\n  coin: AbstractUtxoCoin,\n  scriptTypes: ScriptType2Of3[],\n  walletKeys: RootWalletKeys,\n  params: {\n    keys: NamedKeys;\n    userKeyPath?: string;\n    krsProvider?: string;\n    hasUserSignature: boolean;\n    hasBackupSignature: boolean;\n    hasKrsOutput?: boolean;\n    feeRate?: number;\n  },\n  tags: string[] = []\n) {\n  describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {\n    const externalWallet = getWalletKeys('external');\n    const recoveryDestination = getWalletAddress(coin.network, externalWallet);\n\n    let keyRecoveryServiceAddress: string;\n    let recovery: (BackupKeyRecoveryTransansaction | FormattedOfflineVaultTxInfo) & { txid?: string };\n    let recoveryTx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt;\n\n    // 1e8 * 9e7 < 9.007e15 but 2e8 * 9e7 > 9.007e15 to test both code paths in queryBlockchainUnspentsPath\n    const valueMul = coin.amountType === 'bigint' ? BigInt(9e7) : BigInt(1);\n    before('mock', function () {\n      sinon.stub(CoingeckoApi.prototype, 'getUSDPrice').resolves(69_420);\n    });\n\n    configOverride(function (config: Config) {\n      const configKrsProviders = { ...config.krsProviders };\n      configKrsProviders.dai.supportedCoins = [coin.getFamily()];\n      configKrsProviders.keyternal.supportedCoins = [coin.getFamily()];\n      keyRecoveryServiceAddress = getWalletAddress(coin.network, externalWallet, 0, 100);\n      configKrsProviders.keyternal.feeAddresses = { [coin.getChain()]: keyRecoveryServiceAddress };\n      config.krsProviders = configKrsProviders;\n    });\n\n    after(function () {\n      sinon.restore();\n    });\n\n    let recoverUnspents: utxolib.bitgo.Unspent<bigint>[];\n    let mockedApiUnspents: utxolib.bitgo.Unspent<bigint>[];\n\n    before('create recovery data', async function () {\n      this.timeout(10_000);\n      recoverUnspents = scriptTypes.flatMap((scriptType, index) => [\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(3e8) * valueMul }, index, coin.network, walletKeys),\n      ]);\n\n      // If the coin is bch, convert the mocked unspent address to cashaddr format since that is the format that blockchair\n      // returns on the /dashboards/addresses response\n      mockedApiUnspents =\n        coin.getChain() === 'bch' || coin.getChain() === 'bcha'\n          ? recoverUnspents.map((u) => ({ ...u, address: coin.canonicalAddress(u.address, 'cashaddr').split(':')[1] }))\n          : recoverUnspents;\n\n      assert.strictEqual(mockedApiUnspents.length, recoverUnspents.length);\n      recovery = await backupKeyRecovery(coin, defaultBitGo, {\n        walletPassphrase,\n        recoveryDestination,\n        scan: 5,\n        ignoreAddressTypes: [],\n        userKeyPath: params.userKeyPath,\n        krsProvider: params.krsProvider,\n        ...params.keys,\n        recoveryProvider: new MockRecoveryProvider(mockedApiUnspents),\n      });\n      const txHex =\n        (recovery as BackupKeyRecoveryTransansaction).transactionHex ?? (recovery as FormattedOfflineVaultTxInfo).txHex;\n      const isPsbt = utxolib.bitgo.isPsbt(txHex);\n      recoveryTx = isPsbt\n        ? utxolib.bitgo.createPsbtFromHex(txHex, coin.network)\n        : utxolib.bitgo.createTransactionFromHex(txHex as string, coin.network, coin.amountType);\n      recovery.txid =\n        recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().getId() : recoveryTx.getId();\n    });\n\n    it('has correct recovery provider mocks', async function () {\n      const recoveryProvider = new MockRecoveryProvider(mockedApiUnspents);\n      (await recoveryProvider.getUnspentsForAddresses(mockedApiUnspents.map((u) => u.address))).length.should.eql(\n        mockedApiUnspents.length\n      );\n    });\n\n    it('matches fixture', async function () {\n      const fixtureCoin = getNormalTestnetCoin(coin);\n      const fixtureRecovery = { ...recovery };\n      if (fixtureRecovery.coin) {\n        fixtureRecovery.coin = fixtureCoin.getChain();\n      }\n\n      shouldEqualJSON(\n        fixtureRecovery,\n        await getFixture(\n          fixtureCoin,\n          `recovery/backupKeyRecovery-${(params.krsProvider ? tags.concat([params.krsProvider]) : tags).join('-')}`,\n          recovery\n        )\n      );\n    });\n\n    it('has expected input count', function () {\n      (recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.data.inputs : recoveryTx.ins).length.should.eql(\n        recoverUnspents.length\n      );\n    });\n\n    function checkInputsSignedBy(\n      tx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt,\n      rootKey: BIP32Interface,\n      expectCount: number\n    ) {\n      if (tx instanceof utxolib.bitgo.UtxoPsbt) {\n        function validate(tx: utxolib.bitgo.UtxoPsbt, inputIndex: number) {\n          try {\n            return tx.validateSignaturesOfInputHD(inputIndex, rootKey);\n          } catch (e) {\n            if (e.message === 'No signatures to validate') {\n              return false;\n            }\n            throw e;\n          }\n        }\n        tx.data.inputs.forEach((input, inputIndex) => {\n          validate(tx, inputIndex).should.eql(!!expectCount);\n        });\n      } else {\n        const prevOutputs = recoverUnspents\n          .map((u) => toOutput(u, coin.network))\n          .map((v) => ({ ...v, value: utxolib.bitgo.toTNumber(v.value, coin.amountType) }));\n        tx.ins.forEach((input, inputIndex) => {\n          const unspent = recoverUnspents[inputIndex] as WalletUnspent;\n          const { publicKey } = rootKey.derivePath(walletKeys.getDerivationPath(rootKey, unspent.chain, unspent.index));\n          const signatures = utxolib.bitgo\n            .getSignatureVerifications(\n              tx,\n              inputIndex,\n              utxolib.bitgo.toTNumber(unspent.value, coin.amountType),\n              { publicKey },\n              prevOutputs\n            )\n            .filter((s) => s.signedBy !== undefined);\n          signatures.length.should.eql(expectCount);\n        });\n      }\n    }\n\n    it((params.hasUserSignature ? 'has' : 'has no') + ' user signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.user, params.hasUserSignature ? 1 : 0);\n    });\n\n    it((params.hasBackupSignature ? 'has' : 'has no') + ' backup signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.backup, params.hasBackupSignature ? 1 : 0);\n    });\n\n    if (params.hasUserSignature && params.hasBackupSignature) {\n      it('has no placeholder signatures', function (this: mocha.Context) {\n        if (recoveryTx instanceof utxolib.bitgo.UtxoTransaction) {\n          recoveryTx.ins.forEach((input) => {\n            const parsed = utxolib.bitgo.parseSignatureScript(input);\n            switch (parsed.scriptType) {\n              case 'p2sh':\n              case 'p2shP2wsh':\n              case 'p2wsh':\n              case 'taprootScriptPathSpend':\n                parsed.signatures.forEach((signature, i) => {\n                  if (utxolib.bitgo.isPlaceholderSignature(signature)) {\n                    throw new Error(`placeholder signature at index ${i}`);\n                  }\n                });\n                break;\n              default:\n                throw new Error(`unexpected scriptType ${parsed.scriptType}`);\n            }\n          });\n        } else {\n          this.skip();\n        }\n      });\n    }\n\n    it((params.hasKrsOutput ? 'has' : 'has no') + ' key recovery service output', function () {\n      const outs = recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().outs : recoveryTx.outs;\n      outs.length.should.eql(1);\n      const outputAddresses = outs.map((o) => utxolib.address.fromOutputScript(o.script, recoveryTx.network));\n      outputAddresses\n        .includes(keyRecoveryServiceAddress)\n        .should.eql(!!params.hasKrsOutput && params.krsProvider === 'keyternal');\n      outputAddresses.includes(recoveryDestination).should.eql(true);\n    });\n  });\n}\n\nfunction runWithScriptTypes(\n  scriptTypes: ScriptType2Of3[],\n  { runKeyProviderTests = true }: { runKeyProviderTests?: boolean } = {}\n) {\n  utxoCoins\n    .filter((coin) => scriptTypes.every((type) => coin.supportsAddressType(type)))\n    .forEach((coin) => {\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysUnsignedSweep,\n          hasUserSignature: false,\n          hasBackupSignature: false,\n        },\n        ['unsignedRecovery', ...scriptTypes]\n      );\n\n      if (runKeyProviderTests) {\n        ['dai', 'keyternal'].forEach((krsProvider) => {\n          run(\n            coin,\n            scriptTypes,\n            walletKeys,\n            {\n              keys: keysKeyRecoveryService,\n              krsProvider: krsProvider,\n              hasUserSignature: true,\n              hasBackupSignature: false,\n              hasKrsOutput: false,\n            },\n            ['keyRecoveryService', ...scriptTypes]\n          );\n        });\n      }\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n          feeRate: 2,\n        },\n        ['fullSignedRecovery', 'fixedFeeRate', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        exoticWalletKeys,\n        {\n          keys: keysFullSignedRecoveryExotic,\n          userKeyPath: exoticUserKeyPath,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', 'customUserKeyPath', ...scriptTypes]\n      );\n    });\n}\n\ndescribe('Backup Key Recovery', function () {\n  // compatible with all coins\n  runWithScriptTypes(['p2sh']);\n\n  // segwit compatible coins\n  runWithScriptTypes(['p2shP2wsh', 'p2wsh']);\n\n  // taproot compatible coins\n  runWithScriptTypes(['p2tr', 'p2trMusig2'], { runKeyProviderTests: false });\n});\n"]}
296
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backupKeyRecovery.js","sourceRoot":"","sources":["../../../../../test/unit/recovery/backupKeyRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport 'should';\nimport * as mocha from 'mocha';\nimport * as sinon from 'sinon';\nimport nock = require('nock');\nimport { BIP32Interface } from '@bitgo-beta/utxo-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Config, krsProviders, Triple } from '@bitgo-beta/sdk-core';\nimport { Dimensions } from '@bitgo-beta/unspents';\n\nimport {\n  AbstractUtxoCoin,\n  backupKeyRecovery,\n  BackupKeyRecoveryTransansaction,\n  CoingeckoApi,\n  FormattedOfflineVaultTxInfo,\n} from '../../../src';\nimport {\n  defaultBitGo,\n  encryptKeychain,\n  getDefaultWalletKeys,\n  getFixture,\n  getNormalTestnetCoin,\n  getWalletAddress,\n  getWalletKeys,\n  keychains,\n  shouldEqualJSON,\n  toKeychainBase58,\n  utxoCoins,\n} from '../util';\n\nimport { MockRecoveryProvider } from './mock';\n\nconst { toOutput } = utxolib.bitgo;\ntype WalletUnspent = utxolib.bitgo.WalletUnspent<bigint>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\n\nconst config = { krsProviders };\n\nnock.disableNetConnect();\n\nfunction configOverride(f: (config: Config) => void) {\n  const backup = { ...krsProviders };\n  before(function () {\n    f(config);\n  });\n  after(function () {\n    Object.entries(backup).forEach(([k, v]) => {\n      config[k] = v;\n    });\n  });\n}\n\nconst walletPassphrase = 'lol';\n\ntype NamedKeys = {\n  userKey: string;\n  backupKey: string;\n  bitgoKey: string;\n};\n\nfunction getNamedKeys([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  function encode(k: BIP32Interface): string {\n    return k.isNeutered() ? k.toBase58() : encryptKeychain(password, toKeychainBase58(k));\n  }\n  return {\n    userKey: encode(userKey),\n    backupKey: encode(backupKey),\n    bitgoKey: encode(bitgoKey),\n  };\n}\n\nfunction getKeysForUnsignedSweep([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  return getNamedKeys([userKey.neutered(), backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForKeyRecoveryService(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForFullSignedRecovery(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey, bitgoKey.neutered()], password);\n}\n\nconst walletKeys = getDefaultWalletKeys();\nconst keysUnsignedSweep = getKeysForUnsignedSweep(walletKeys.triple, walletPassphrase);\nconst keysKeyRecoveryService = getKeysForKeyRecoveryService(walletKeys.triple, walletPassphrase);\nconst keysFullSignedRecovery = getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase);\n\nconst exoticUserKeyPath = '99/99';\nconst exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(keychains, [\n  exoticUserKeyPath,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n]);\nconst keysFullSignedRecoveryExotic = getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase);\n\nfunction run(\n  coin: AbstractUtxoCoin,\n  scriptTypes: ScriptType2Of3[],\n  walletKeys: RootWalletKeys,\n  params: {\n    keys: NamedKeys;\n    userKeyPath?: string;\n    krsProvider?: string;\n    hasUserSignature: boolean;\n    hasBackupSignature: boolean;\n    hasKrsOutput?: boolean;\n  },\n  tags: string[] = []\n) {\n  const defaultFeeRateSatB = 100;\n\n  describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {\n    const externalWallet = getWalletKeys('external');\n    const recoveryDestination = getWalletAddress(coin.network, externalWallet);\n\n    let keyRecoveryServiceAddress: string;\n    let recovery: (BackupKeyRecoveryTransansaction | FormattedOfflineVaultTxInfo) & { txid?: string };\n    let recoveryTx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt;\n\n    // 1e8 * 9e7 < 9.007e15 but 2e8 * 9e7 > 9.007e15 to test both code paths in queryBlockchainUnspentsPath\n    const valueMul = coin.amountType === 'bigint' ? BigInt(9e7) : BigInt(1);\n    before('mock', function () {\n      sinon.stub(CoingeckoApi.prototype, 'getUSDPrice').resolves(69_420);\n    });\n\n    configOverride(function (config: Config) {\n      const configKrsProviders = { ...config.krsProviders };\n      configKrsProviders.dai.supportedCoins = [coin.getFamily()];\n      configKrsProviders.keyternal.supportedCoins = [coin.getFamily()];\n      keyRecoveryServiceAddress = getWalletAddress(coin.network, externalWallet, 0, 100);\n      configKrsProviders.keyternal.feeAddresses = { [coin.getChain()]: keyRecoveryServiceAddress };\n      config.krsProviders = configKrsProviders;\n    });\n\n    after(function () {\n      sinon.restore();\n    });\n\n    let recoverUnspents: utxolib.bitgo.Unspent<bigint>[];\n    let mockedApiUnspents: utxolib.bitgo.Unspent<bigint>[];\n\n    before('create recovery data', async function () {\n      this.timeout(10_000);\n      recoverUnspents = scriptTypes.flatMap((scriptType, index) => [\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(3e8) * valueMul }, index, coin.network, walletKeys),\n      ]);\n\n      // If the coin is bch, convert the mocked unspent address to cashaddr format since that is the format that blockchair\n      // returns on the /dashboards/addresses response\n      mockedApiUnspents =\n        coin.getChain() === 'bch' || coin.getChain() === 'bcha'\n          ? recoverUnspents.map((u) => ({ ...u, address: coin.canonicalAddress(u.address, 'cashaddr').split(':')[1] }))\n          : recoverUnspents;\n\n      assert.strictEqual(mockedApiUnspents.length, recoverUnspents.length);\n      recovery = await backupKeyRecovery(coin, defaultBitGo, {\n        walletPassphrase,\n        recoveryDestination,\n        scan: 5,\n        ignoreAddressTypes: [],\n        userKeyPath: params.userKeyPath,\n        krsProvider: params.krsProvider,\n        ...params.keys,\n        recoveryProvider: new MockRecoveryProvider(mockedApiUnspents),\n        feeRate: defaultFeeRateSatB,\n      });\n      const txHex =\n        (recovery as BackupKeyRecoveryTransansaction).transactionHex ?? (recovery as FormattedOfflineVaultTxInfo).txHex;\n      const isPsbt = utxolib.bitgo.isPsbt(txHex);\n      recoveryTx = isPsbt\n        ? utxolib.bitgo.createPsbtFromHex(txHex, coin.network)\n        : utxolib.bitgo.createTransactionFromHex(txHex as string, coin.network, coin.amountType);\n      recovery.txid =\n        recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().getId() : recoveryTx.getId();\n    });\n\n    it('has correct recovery provider mocks', async function () {\n      const recoveryProvider = new MockRecoveryProvider(mockedApiUnspents);\n      (await recoveryProvider.getUnspentsForAddresses(mockedApiUnspents.map((u) => u.address))).length.should.eql(\n        mockedApiUnspents.length\n      );\n    });\n\n    it('has expected fee rate', function () {\n      if (!(recoveryTx instanceof utxolib.bitgo.UtxoPsbt)) {\n        this.skip();\n      }\n      const inputSum = utxolib.bitgo.unspentSum(recoverUnspents, 'bigint');\n      const outputSum = recoveryTx.txOutputs.reduce((sum, o) => sum + o.value, BigInt(0));\n      const fee = inputSum - outputSum;\n      const vsize = Dimensions.fromPsbt(recoveryTx).getVSize();\n      const feeRateSatB = Number(fee) / vsize;\n      const diff = Math.abs(feeRateSatB - defaultFeeRateSatB) / defaultFeeRateSatB;\n      // within 1%\n      assert.strictEqual(diff < 0.01, true, `expected fee rate ${defaultFeeRateSatB} but got ${feeRateSatB}`);\n    });\n\n    it('matches fixture', async function () {\n      const fixtureCoin = getNormalTestnetCoin(coin);\n      const fixtureRecovery = { ...recovery };\n      if (fixtureRecovery.coin) {\n        fixtureRecovery.coin = fixtureCoin.getChain();\n      }\n\n      shouldEqualJSON(\n        fixtureRecovery,\n        await getFixture(\n          fixtureCoin,\n          `recovery/backupKeyRecovery-${(params.krsProvider ? tags.concat([params.krsProvider]) : tags).join('-')}`,\n          recovery\n        )\n      );\n    });\n\n    it('has expected input count', function () {\n      (recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.data.inputs : recoveryTx.ins).length.should.eql(\n        recoverUnspents.length\n      );\n    });\n\n    function checkInputsSignedBy(\n      tx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt,\n      rootKey: BIP32Interface,\n      expectCount: number\n    ) {\n      if (tx instanceof utxolib.bitgo.UtxoPsbt) {\n        function validate(tx: utxolib.bitgo.UtxoPsbt, inputIndex: number) {\n          try {\n            return tx.validateSignaturesOfInputHD(inputIndex, rootKey);\n          } catch (e) {\n            if (e.message === 'No signatures to validate') {\n              return false;\n            }\n            throw e;\n          }\n        }\n        tx.data.inputs.forEach((input, inputIndex) => {\n          validate(tx, inputIndex).should.eql(!!expectCount);\n        });\n      } else {\n        const prevOutputs = recoverUnspents\n          .map((u) => toOutput(u, coin.network))\n          .map((v) => ({ ...v, value: utxolib.bitgo.toTNumber(v.value, coin.amountType) }));\n        tx.ins.forEach((input, inputIndex) => {\n          const unspent = recoverUnspents[inputIndex] as WalletUnspent;\n          const { publicKey } = rootKey.derivePath(walletKeys.getDerivationPath(rootKey, unspent.chain, unspent.index));\n          const signatures = utxolib.bitgo\n            .getSignatureVerifications(\n              tx,\n              inputIndex,\n              utxolib.bitgo.toTNumber(unspent.value, coin.amountType),\n              { publicKey },\n              prevOutputs\n            )\n            .filter((s) => s.signedBy !== undefined);\n          signatures.length.should.eql(expectCount);\n        });\n      }\n    }\n\n    it((params.hasUserSignature ? 'has' : 'has no') + ' user signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.user, params.hasUserSignature ? 1 : 0);\n    });\n\n    it((params.hasBackupSignature ? 'has' : 'has no') + ' backup signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.backup, params.hasBackupSignature ? 1 : 0);\n    });\n\n    if (params.hasUserSignature && params.hasBackupSignature) {\n      it('has no placeholder signatures', function (this: mocha.Context) {\n        if (recoveryTx instanceof utxolib.bitgo.UtxoTransaction) {\n          recoveryTx.ins.forEach((input) => {\n            const parsed = utxolib.bitgo.parseSignatureScript(input);\n            switch (parsed.scriptType) {\n              case 'p2sh':\n              case 'p2shP2wsh':\n              case 'p2wsh':\n              case 'taprootScriptPathSpend':\n                parsed.signatures.forEach((signature, i) => {\n                  if (utxolib.bitgo.isPlaceholderSignature(signature)) {\n                    throw new Error(`placeholder signature at index ${i}`);\n                  }\n                });\n                break;\n              default:\n                throw new Error(`unexpected scriptType ${parsed.scriptType}`);\n            }\n          });\n        } else {\n          this.skip();\n        }\n      });\n    }\n\n    it((params.hasKrsOutput ? 'has' : 'has no') + ' key recovery service output', function () {\n      const outs = recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().outs : recoveryTx.outs;\n      outs.length.should.eql(1);\n      const outputAddresses = outs.map((o) => utxolib.address.fromOutputScript(o.script, recoveryTx.network));\n      outputAddresses\n        .includes(keyRecoveryServiceAddress)\n        .should.eql(!!params.hasKrsOutput && params.krsProvider === 'keyternal');\n      outputAddresses.includes(recoveryDestination).should.eql(true);\n    });\n  });\n}\n\nfunction runWithScriptTypes(\n  scriptTypes: ScriptType2Of3[],\n  { runKeyProviderTests = true }: { runKeyProviderTests?: boolean } = {}\n) {\n  utxoCoins\n    .filter((coin) => scriptTypes.every((type) => coin.supportsAddressType(type)))\n    .forEach((coin) => {\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysUnsignedSweep,\n          hasUserSignature: false,\n          hasBackupSignature: false,\n        },\n        ['unsignedRecovery', ...scriptTypes]\n      );\n\n      if (runKeyProviderTests) {\n        ['dai', 'keyternal'].forEach((krsProvider) => {\n          run(\n            coin,\n            scriptTypes,\n            walletKeys,\n            {\n              keys: keysKeyRecoveryService,\n              krsProvider: krsProvider,\n              hasUserSignature: true,\n              hasBackupSignature: false,\n              hasKrsOutput: false,\n            },\n            ['keyRecoveryService', ...scriptTypes]\n          );\n        });\n      }\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        exoticWalletKeys,\n        {\n          keys: keysFullSignedRecoveryExotic,\n          userKeyPath: exoticUserKeyPath,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', 'customUserKeyPath', ...scriptTypes]\n      );\n    });\n}\n\ndescribe('Backup Key Recovery', function () {\n  // compatible with all coins\n  runWithScriptTypes(['p2sh']);\n\n  // segwit compatible coins\n  runWithScriptTypes(['p2shP2wsh', 'p2wsh']);\n\n  // taproot compatible coins\n  runWithScriptTypes(['p2tr', 'p2trMusig2'], { runKeyProviderTests: false });\n});\n"]}
@@ -147,39 +147,28 @@ function run(sourceCoin, recoveryCoin) {
147
147
  afterEach(function () {
148
148
  sinon.restore();
149
149
  });
150
- function testMatchFixture(name, getRecoveryResult) {
151
- it(`should match fixture (${name})`, async function () {
152
- const recovery = getRecoveryResult();
153
- let recoveryObj = {
154
- ...recovery,
155
- tx: (0, util_1.transactionHexToObj)(recovery.txHex, sourceCoin.network, sourceCoin.amountType),
156
- };
157
- if (sourceCoin.amountType === 'bigint') {
158
- recoveryObj = JSON.parse(JSON.stringify(recoveryObj, (k, v) => {
159
- if (typeof v === 'bigint') {
160
- return v.toString();
161
- }
162
- else {
163
- return v;
164
- }
165
- }));
166
- }
167
- (0, util_1.shouldEqualJSON)(recoveryObj, await (0, util_1.getFixture)(sourceCoin, `recovery/crossChainRecovery-${recoveryCoin.getChain()}-${name}`, recoveryObj));
168
- });
169
- }
170
- function checkRecoveryTransactionSignature(tx) {
171
- if (typeof tx === 'string') {
172
- tx = utxolib.bitgo.createTransactionFromBuffer(Buffer.from(tx, 'hex'), sourceCoin.network, {
173
- amountType: sourceCoin.amountType,
174
- });
150
+ async function matchFixture(name, recovery) {
151
+ let recoveryObj = { ...recovery };
152
+ if (sourceCoin.amountType === 'bigint') {
153
+ recoveryObj = JSON.parse(JSON.stringify(recoveryObj, (k, v) => {
154
+ if (typeof v === 'bigint') {
155
+ return v.toString();
156
+ }
157
+ else {
158
+ return v;
159
+ }
160
+ }));
175
161
  }
162
+ (0, util_1.shouldEqualJSON)(recoveryObj, await (0, util_1.getFixture)(sourceCoin, `recovery/crossChainRecovery-${recoveryCoin.getChain()}-${name}`, recoveryObj));
163
+ }
164
+ function checkRecoveryPsbtSignature(psbtHex) {
165
+ const psbt = utxolib.bitgo.createPsbtFromHex(psbtHex, sourceCoin.network);
176
166
  const unspents = getRecoveryUnspents();
177
- should.equal(tx.ins.length, unspents.length);
178
- tx.ins.forEach((input, i) => {
179
- assert.ok(typeof tx !== 'string');
180
- utxolib.bitgo
181
- .verifySignatureWithUnspent(tx, i, getRecoveryUnspents(), walletKeys)
182
- .should.eql([true, false, false]);
167
+ should.equal(psbt.data.inputs.length, unspents.length);
168
+ // Verify user key has signed each input (same pattern as backupKeyRecovery test)
169
+ psbt.data.inputs.forEach((input, i) => {
170
+ const userSigned = psbt.validateSignaturesOfInputHD(i, walletKeys.user);
171
+ userSigned.should.eql(true, `Input ${i} should be signed by user key`);
183
172
  });
184
173
  }
185
174
  it('should test signed cross chain recovery', async () => {
@@ -197,10 +186,10 @@ function run(sourceCoin, recoveryCoin) {
197
186
  xprv: util_1.keychainsBase58[0].prv,
198
187
  }));
199
188
  should.equal(getRecoveryProviderStub.callCount, 1);
200
- testMatchFixture('signed', () => signedRecovery);
201
- it('should have valid signatures for signed recovery', function () {
202
- checkRecoveryTransactionSignature(signedRecovery.txHex);
203
- });
189
+ // Verify fixture match
190
+ await matchFixture('signed', signedRecovery);
191
+ // Verify PSBT has valid signatures (user key signed)
192
+ checkRecoveryPsbtSignature(signedRecovery.txHex);
204
193
  });
205
194
  it('should test unsigned cross chain recovery', async () => {
206
195
  const getRecoveryProviderStub = sinon
@@ -217,15 +206,15 @@ function run(sourceCoin, recoveryCoin) {
217
206
  signed: false,
218
207
  }));
219
208
  should.equal(getRecoveryProviderStub.callCount, 1);
220
- testMatchFixture('unsigned', () => unsignedRecovery);
221
- it('should be signable for unsigned recovery', async function () {
222
- const signedTx = await sourceCoin.signTransaction({
223
- txPrebuild: unsignedRecovery,
224
- prv: util_1.keychainsBase58[0].prv,
225
- pubs: util_1.keychainsBase58.map((k) => k.pub),
226
- });
227
- checkRecoveryTransactionSignature(signedTx.txHex);
209
+ // Verify fixture match
210
+ await matchFixture('unsigned', unsignedRecovery);
211
+ // Verify the unsigned PSBT can be signed
212
+ const signedTx = await sourceCoin.signTransaction({
213
+ txPrebuild: unsignedRecovery,
214
+ prv: util_1.keychainsBase58[0].prv,
215
+ pubs: util_1.keychainsBase58.map((k) => k.pub),
228
216
  });
217
+ checkRecoveryPsbtSignature(signedTx.txHex);
229
218
  });
230
219
  });
231
220
  }
@@ -303,4 +292,4 @@ describe('convertLtcAddressToLegacyFormat', function () {
303
292
  assert.strictEqual(result, bech32Address);
304
293
  });
305
294
  });
306
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../../../test/unit/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport should = require('should');\nimport nock = require('nock');\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Triple } from '@bitgo-beta/sdk-core';\nimport { getSeed } from '@bitgo-beta/sdk-test';\nimport * as sinon from 'sinon';\n\nimport {\n  AbstractUtxoCoin,\n  CrossChainRecoverySigned,\n  CrossChainRecoveryUnsigned,\n  getWallet,\n  supportedCrossChainRecoveries,\n  generateAddress,\n  convertLtcAddressToLegacyFormat,\n} from '../../../src';\nimport {\n  getFixture,\n  keychainsBase58,\n  KeychainBase58,\n  mockUnspent,\n  shouldEqualJSON,\n  utxoCoins,\n  transactionHexToObj,\n  getDefaultWalletKeys,\n  defaultBitGo,\n  getUtxoCoin,\n} from '../util';\nimport { nockBitGo } from '../util/nockBitGo';\nimport { createFullSignedTransaction } from '../util/transaction';\nimport { getDefaultWalletUnspentSigner } from '../util/keychains';\n\nimport { MockCrossChainRecoveryProvider } from './mock';\n\ntype WalletUnspent<TNumber extends number | bigint = number> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nfunction getKeyId(k: KeychainBase58): string {\n  return getSeed(k.pub).toString('hex');\n}\n\nfunction nockWallet(coin: AbstractUtxoCoin, walletId: string, walletKeys: Triple<KeychainBase58>): nock.Scope[] {\n  return [\n    nockBitGo()\n      .get(`/api/v2/${coin.getChain()}/wallet/${walletId}`)\n      .reply(200, {\n        id: walletId,\n        coin: coin.getChain(),\n        label: 'crossChainRecovery',\n        keys: walletKeys.map((k) => getKeyId(k)),\n      })\n      .persist(),\n    ...walletKeys.map((k) =>\n      nockBitGo()\n        .get(`/api/v2/${coin.getChain()}/key/${getKeyId(k)}`)\n        .reply(200, k)\n        .persist()\n    ),\n  ];\n}\n\ntype Address = {\n  address: string;\n  chain: number;\n  index: number;\n};\n\nfunction nockWalletAddress(coin: AbstractUtxoCoin, walletId: string, address: Address): nock.Scope {\n  return nockBitGo()\n    .get(`/api/v2/${coin.getChain()}/wallet/${walletId}/address/${address.address}`)\n    .reply(200, {\n      address: address.address,\n      chain: address.chain,\n      index: address.index,\n      coin: coin.getChain(),\n      wallet: walletId,\n    })\n    .persist();\n}\n\n/**\n * Setup test for cross-chain recovery.\n *\n * Users can receive deposits on wallet addresses that are on a different chain.\n *\n * For instance, a user can receive litecoin on a bitcoin wallet.\n * This means that the litecoin blockchain has a transaction with outputs that are spendable\n * with keys that were originally created for a BitGo BTC wallet.\n * In this example, LTC is the \"source coin\" and BTC is the \"recovery coin\"\n * In cases like these we must use construct a transaction for litecoin network using keys of the\n * bitcoin wallet.\n *\n * @param sourceCoin - the coin to construct the transaction for\n * @param recoveryCoin - the coin the receiving wallet was set up for\n */\nfunction run<TNumber extends number | bigint = number>(sourceCoin: AbstractUtxoCoin, recoveryCoin: AbstractUtxoCoin) {\n  describe(`Cross-Chain Recovery [sourceCoin=${sourceCoin.getChain()} recoveryCoin=${recoveryCoin.getChain()}]`, function () {\n    const walletKeys = getDefaultWalletKeys();\n    const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';\n    // the address the accidental deposit went to, in both sourceCoin and addressCoin formats\n    const [depositAddressSourceCoin, depositAddressRecoveryCoin] = [sourceCoin, recoveryCoin].map((coin) => ({\n      address: generateAddress(coin.network, { keychains: keychainsBase58, chain: 0, index: 0 }),\n      chain: 0,\n      index: 0,\n    }));\n    const chain = 0;\n    const index = 1;\n    // the address where we want to recover our funds to\n    const recoveryAddress = generateAddress(sourceCoin.network, {\n      keychains: keychainsBase58,\n      chain,\n      index,\n    });\n    const nocks: nock.Scope[] = [];\n\n    let depositTx: utxolib.bitgo.UtxoTransaction<TNumber>;\n\n    function getDepositUnspents(): utxolib.bitgo.Unspent<TNumber>[] {\n      return [\n        mockUnspent<TNumber>(\n          sourceCoin.network,\n          walletKeys,\n          'p2sh',\n          0,\n          (sourceCoin.amountType === 'bigint' ? BigInt('10999999800000001') : 1e8) as TNumber\n        ),\n      ];\n    }\n\n    function getDepositTransaction(): utxolib.bitgo.UtxoTransaction<TNumber> {\n      return createFullSignedTransaction<TNumber>(\n        sourceCoin.network,\n        getDepositUnspents(),\n        depositAddressSourceCoin.address,\n        getDefaultWalletUnspentSigner()\n      );\n    }\n\n    before('prepare deposit tx', function () {\n      depositTx = getDepositTransaction();\n    });\n\n    function getRecoveryUnspents(): WalletUnspent<TNumber>[] {\n      return [\n        {\n          id: depositTx.getId(),\n          address: depositAddressSourceCoin.address,\n          chain: chain,\n          index: index,\n          value: depositTx.outs[0].value,\n        },\n      ];\n    }\n\n    before('setup nocks', function () {\n      nocks.push(...nockWallet(recoveryCoin, recoveryWalletId, keychainsBase58));\n      nocks.push(nockWalletAddress(recoveryCoin, recoveryWalletId, depositAddressRecoveryCoin));\n    });\n\n    after(function () {\n      nocks.forEach((n) => n.done());\n    });\n\n    after(function () {\n      nock.cleanAll();\n    });\n\n    afterEach(function () {\n      sinon.restore();\n    });\n\n    function testMatchFixture(\n      name: string,\n      getRecoveryResult: () => CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>\n    ) {\n      it(`should match fixture (${name})`, async function () {\n        const recovery = getRecoveryResult();\n        let recoveryObj = {\n          ...recovery,\n          tx: transactionHexToObj(recovery.txHex as string, sourceCoin.network, sourceCoin.amountType),\n        };\n        if (sourceCoin.amountType === 'bigint') {\n          recoveryObj = JSON.parse(\n            JSON.stringify(recoveryObj, (k, v) => {\n              if (typeof v === 'bigint') {\n                return v.toString();\n              } else {\n                return v;\n              }\n            })\n          );\n        }\n        shouldEqualJSON(\n          recoveryObj,\n          await getFixture(sourceCoin, `recovery/crossChainRecovery-${recoveryCoin.getChain()}-${name}`, recoveryObj)\n        );\n      });\n    }\n\n    function checkRecoveryTransactionSignature(tx: string | utxolib.bitgo.UtxoTransaction<TNumber>) {\n      if (typeof tx === 'string') {\n        tx = utxolib.bitgo.createTransactionFromBuffer<TNumber>(Buffer.from(tx, 'hex'), sourceCoin.network, {\n          amountType: sourceCoin.amountType,\n        });\n      }\n      const unspents = getRecoveryUnspents();\n      should.equal(tx.ins.length, unspents.length);\n      tx.ins.forEach((input, i) => {\n        assert.ok(typeof tx !== 'string');\n        utxolib.bitgo\n          .verifySignatureWithUnspent<TNumber>(tx, i, getRecoveryUnspents(), walletKeys)\n          .should.eql([true, false, false]);\n      });\n    }\n\n    it('should test signed cross chain recovery', async () => {\n      const getRecoveryProviderStub = sinon\n        .stub(AbstractUtxoCoin.prototype, 'getRecoveryProvider')\n        .returns(new MockCrossChainRecoveryProvider<TNumber>(sourceCoin, getDepositUnspents(), depositTx));\n      const params = {\n        recoveryCoin,\n        txid: depositTx.getId(),\n        recoveryAddress,\n        wallet: recoveryWalletId,\n      };\n      const signedRecovery = (await sourceCoin.recoverFromWrongChain<TNumber>({\n        ...params,\n        xprv: keychainsBase58[0].prv,\n      })) as CrossChainRecoverySigned<TNumber>;\n      should.equal(getRecoveryProviderStub.callCount, 1);\n\n      testMatchFixture('signed', () => signedRecovery);\n\n      it('should have valid signatures for signed recovery', function () {\n        checkRecoveryTransactionSignature(signedRecovery.txHex as string);\n      });\n    });\n\n    it('should test unsigned cross chain recovery', async () => {\n      const getRecoveryProviderStub = sinon\n        .stub(AbstractUtxoCoin.prototype, 'getRecoveryProvider')\n        .returns(new MockCrossChainRecoveryProvider<TNumber>(sourceCoin, getDepositUnspents(), depositTx));\n      const params = {\n        recoveryCoin,\n        txid: depositTx.getId(),\n        recoveryAddress,\n        wallet: recoveryWalletId,\n      };\n      const unsignedRecovery = (await sourceCoin.recoverFromWrongChain<TNumber>({\n        ...params,\n        signed: false,\n      })) as CrossChainRecoveryUnsigned<TNumber>;\n      should.equal(getRecoveryProviderStub.callCount, 1);\n\n      testMatchFixture('unsigned', () => unsignedRecovery);\n\n      it('should be signable for unsigned recovery', async function () {\n        const signedTx = await sourceCoin.signTransaction<TNumber>({\n          txPrebuild: unsignedRecovery,\n          prv: keychainsBase58[0].prv,\n          pubs: keychainsBase58.map((k) => k.pub) as Triple<string>,\n        });\n        checkRecoveryTransactionSignature((signedTx as { txHex: string }).txHex);\n      });\n    });\n  });\n}\n\nfunction isSupportedCrossChainRecovery(sourceCoin: AbstractUtxoCoin, recoveryCoin: AbstractUtxoCoin): boolean {\n  return supportedCrossChainRecoveries[sourceCoin.getFamily()]?.includes(recoveryCoin.getFamily());\n}\n\nutxoCoins.forEach((coin) => {\n  utxoCoins\n    .filter(\n      (otherCoin) =>\n        coin !== otherCoin &&\n        isSupportedCrossChainRecovery(coin, otherCoin) &&\n        ((utxolib.isMainnet(coin.network) && utxolib.isMainnet(otherCoin.network)) ||\n          (utxolib.isTestnet(coin.network) && utxolib.isTestnet(otherCoin.network)))\n    )\n    .forEach((otherCoin) => {\n      if (coin.amountType === 'bigint') {\n        run<bigint>(coin, otherCoin);\n      } else {\n        run(coin, otherCoin);\n      }\n    });\n});\n\ndescribe(`Cross-Chain Recovery getWallet`, async function () {\n  const bitgo = defaultBitGo;\n  const recoveryCoin = getUtxoCoin('btc');\n  const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';\n\n  it('should search v1 wallets if the v2 endpoint responds with a 4xx error', async function () {\n    const errorResponses = [400, 404];\n\n    for (const error of errorResponses) {\n      const nockV2Wallet = nockBitGo(bitgo)\n        .get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)\n        .reply(error);\n      const nockV1Wallet = nockBitGo(bitgo).get(`/api/v1/wallet/${recoveryWalletId}`).reply(error);\n      await assert.rejects(\n        () => getWallet(bitgo, recoveryCoin, recoveryWalletId),\n        Error(`could not get wallet ${recoveryWalletId} from v1 or v2: ApiResponseError: ${error}`)\n      );\n      nockV2Wallet.done();\n      nockV1Wallet.done();\n    }\n  });\n\n  it('should throw an error if the v2 endpoint responds with a 5xx error', async function () {\n    const errorResponses = [500];\n    for (const error of errorResponses) {\n      const nockV2Wallet = nockBitGo(bitgo)\n        .get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)\n        .reply(error);\n      await assert.rejects(() => getWallet(bitgo, recoveryCoin, recoveryWalletId), {\n        name: 'ApiResponseError',\n        status: 500,\n        result: {},\n        invalidToken: false,\n        needsOTP: false,\n      });\n      nockV2Wallet.done();\n    }\n  });\n});\n\ndescribe('convertLtcAddressToLegacyFormat', function () {\n  const ltcNetwork = utxolib.networks.litecoin;\n\n  it('should convert M... P2SH address to 3... legacy format', function () {\n    // These two addresses represent the same underlying script hash:\n    // - MNQ7zkgMsaV67rsjA3JuP59RC5wxRXpwgE is the LTC format (scriptHash 0x32)\n    // - 3GBygsGPvTdfKMbq4AKZZRu1sPMWPEsBfd is the BTC format (scriptHash 0x05)\n    const ltcAddress = 'MNQ7zkgMsaV67rsjA3JuP59RC5wxRXpwgE';\n    const expectedLegacyAddress = '3GBygsGPvTdfKMbq4AKZZRu1sPMWPEsBfd';\n\n    const legacyAddress = convertLtcAddressToLegacyFormat(ltcAddress, ltcNetwork);\n    assert.strictEqual(legacyAddress, expectedLegacyAddress);\n  });\n\n  it('should convert MD68PsdheKxcYsrVLyZRXgoSDLnB1MdVtE to legacy format', function () {\n    const address = 'MD68PsdheKxcYsrVLyZRXgoSDLnB1MdVtE';\n    const legacyAddress = convertLtcAddressToLegacyFormat(address, ltcNetwork);\n\n    // Should start with '3' (legacy BTC P2SH format)\n    assert.ok(legacyAddress.startsWith('3'), `Expected address to start with '3', got: ${legacyAddress}`);\n  });\n\n  it('should not modify bech32 addresses', function () {\n    const bech32Address = 'ltc1qgrl8zpndsklaa9swgd5vevyxmx5x63vcrl7dk4';\n    const result = convertLtcAddressToLegacyFormat(bech32Address, ltcNetwork);\n    assert.strictEqual(result, bech32Address);\n  });\n});\n"]}
295
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../../../test/unit/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport should = require('should');\nimport nock = require('nock');\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Triple } from '@bitgo-beta/sdk-core';\nimport { getSeed } from '@bitgo-beta/sdk-test';\nimport * as sinon from 'sinon';\n\nimport {\n  AbstractUtxoCoin,\n  CrossChainRecoverySigned,\n  CrossChainRecoveryUnsigned,\n  getWallet,\n  supportedCrossChainRecoveries,\n  generateAddress,\n  convertLtcAddressToLegacyFormat,\n} from '../../../src';\nimport {\n  getFixture,\n  keychainsBase58,\n  KeychainBase58,\n  mockUnspent,\n  shouldEqualJSON,\n  utxoCoins,\n  getDefaultWalletKeys,\n  defaultBitGo,\n  getUtxoCoin,\n} from '../util';\nimport { nockBitGo } from '../util/nockBitGo';\nimport { createFullSignedTransaction } from '../util/transaction';\nimport { getDefaultWalletUnspentSigner } from '../util/keychains';\n\nimport { MockCrossChainRecoveryProvider } from './mock';\n\ntype WalletUnspent<TNumber extends number | bigint = number> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nfunction getKeyId(k: KeychainBase58): string {\n  return getSeed(k.pub).toString('hex');\n}\n\nfunction nockWallet(coin: AbstractUtxoCoin, walletId: string, walletKeys: Triple<KeychainBase58>): nock.Scope[] {\n  return [\n    nockBitGo()\n      .get(`/api/v2/${coin.getChain()}/wallet/${walletId}`)\n      .reply(200, {\n        id: walletId,\n        coin: coin.getChain(),\n        label: 'crossChainRecovery',\n        keys: walletKeys.map((k) => getKeyId(k)),\n      })\n      .persist(),\n    ...walletKeys.map((k) =>\n      nockBitGo()\n        .get(`/api/v2/${coin.getChain()}/key/${getKeyId(k)}`)\n        .reply(200, k)\n        .persist()\n    ),\n  ];\n}\n\ntype Address = {\n  address: string;\n  chain: number;\n  index: number;\n};\n\nfunction nockWalletAddress(coin: AbstractUtxoCoin, walletId: string, address: Address): nock.Scope {\n  return nockBitGo()\n    .get(`/api/v2/${coin.getChain()}/wallet/${walletId}/address/${address.address}`)\n    .reply(200, {\n      address: address.address,\n      chain: address.chain,\n      index: address.index,\n      coin: coin.getChain(),\n      wallet: walletId,\n    })\n    .persist();\n}\n\n/**\n * Setup test for cross-chain recovery.\n *\n * Users can receive deposits on wallet addresses that are on a different chain.\n *\n * For instance, a user can receive litecoin on a bitcoin wallet.\n * This means that the litecoin blockchain has a transaction with outputs that are spendable\n * with keys that were originally created for a BitGo BTC wallet.\n * In this example, LTC is the \"source coin\" and BTC is the \"recovery coin\"\n * In cases like these we must use construct a transaction for litecoin network using keys of the\n * bitcoin wallet.\n *\n * @param sourceCoin - the coin to construct the transaction for\n * @param recoveryCoin - the coin the receiving wallet was set up for\n */\nfunction run<TNumber extends number | bigint = number>(sourceCoin: AbstractUtxoCoin, recoveryCoin: AbstractUtxoCoin) {\n  describe(`Cross-Chain Recovery [sourceCoin=${sourceCoin.getChain()} recoveryCoin=${recoveryCoin.getChain()}]`, function () {\n    const walletKeys = getDefaultWalletKeys();\n    const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';\n    // the address the accidental deposit went to, in both sourceCoin and addressCoin formats\n    const [depositAddressSourceCoin, depositAddressRecoveryCoin] = [sourceCoin, recoveryCoin].map((coin) => ({\n      address: generateAddress(coin.network, { keychains: keychainsBase58, chain: 0, index: 0 }),\n      chain: 0,\n      index: 0,\n    }));\n    const chain = 0;\n    const index = 1;\n    // the address where we want to recover our funds to\n    const recoveryAddress = generateAddress(sourceCoin.network, {\n      keychains: keychainsBase58,\n      chain,\n      index,\n    });\n    const nocks: nock.Scope[] = [];\n\n    let depositTx: utxolib.bitgo.UtxoTransaction<TNumber>;\n\n    function getDepositUnspents(): utxolib.bitgo.Unspent<TNumber>[] {\n      return [\n        mockUnspent<TNumber>(\n          sourceCoin.network,\n          walletKeys,\n          'p2sh',\n          0,\n          (sourceCoin.amountType === 'bigint' ? BigInt('10999999800000001') : 1e8) as TNumber\n        ),\n      ];\n    }\n\n    function getDepositTransaction(): utxolib.bitgo.UtxoTransaction<TNumber> {\n      return createFullSignedTransaction<TNumber>(\n        sourceCoin.network,\n        getDepositUnspents(),\n        depositAddressSourceCoin.address,\n        getDefaultWalletUnspentSigner()\n      );\n    }\n\n    before('prepare deposit tx', function () {\n      depositTx = getDepositTransaction();\n    });\n\n    function getRecoveryUnspents(): WalletUnspent<TNumber>[] {\n      return [\n        {\n          id: depositTx.getId(),\n          address: depositAddressSourceCoin.address,\n          chain: chain,\n          index: index,\n          value: depositTx.outs[0].value,\n        },\n      ];\n    }\n\n    before('setup nocks', function () {\n      nocks.push(...nockWallet(recoveryCoin, recoveryWalletId, keychainsBase58));\n      nocks.push(nockWalletAddress(recoveryCoin, recoveryWalletId, depositAddressRecoveryCoin));\n    });\n\n    after(function () {\n      nocks.forEach((n) => n.done());\n    });\n\n    after(function () {\n      nock.cleanAll();\n    });\n\n    afterEach(function () {\n      sinon.restore();\n    });\n\n    async function matchFixture(\n      name: string,\n      recovery: CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>\n    ) {\n      let recoveryObj: Record<string, unknown> = { ...recovery };\n      if (sourceCoin.amountType === 'bigint') {\n        recoveryObj = JSON.parse(\n          JSON.stringify(recoveryObj, (k, v) => {\n            if (typeof v === 'bigint') {\n              return v.toString();\n            } else {\n              return v;\n            }\n          })\n        );\n      }\n      shouldEqualJSON(\n        recoveryObj,\n        await getFixture(sourceCoin, `recovery/crossChainRecovery-${recoveryCoin.getChain()}-${name}`, recoveryObj)\n      );\n    }\n\n    function checkRecoveryPsbtSignature(psbtHex: string) {\n      const psbt = utxolib.bitgo.createPsbtFromHex(psbtHex, sourceCoin.network);\n      const unspents = getRecoveryUnspents();\n      should.equal(psbt.data.inputs.length, unspents.length);\n      // Verify user key has signed each input (same pattern as backupKeyRecovery test)\n      psbt.data.inputs.forEach((input, i) => {\n        const userSigned = psbt.validateSignaturesOfInputHD(i, walletKeys.user);\n        userSigned.should.eql(true, `Input ${i} should be signed by user key`);\n      });\n    }\n\n    it('should test signed cross chain recovery', async () => {\n      const getRecoveryProviderStub = sinon\n        .stub(AbstractUtxoCoin.prototype, 'getRecoveryProvider')\n        .returns(new MockCrossChainRecoveryProvider<TNumber>(sourceCoin, getDepositUnspents(), depositTx));\n      const params = {\n        recoveryCoin,\n        txid: depositTx.getId(),\n        recoveryAddress,\n        wallet: recoveryWalletId,\n      };\n      const signedRecovery = (await sourceCoin.recoverFromWrongChain<TNumber>({\n        ...params,\n        xprv: keychainsBase58[0].prv,\n      })) as CrossChainRecoverySigned<TNumber>;\n      should.equal(getRecoveryProviderStub.callCount, 1);\n\n      // Verify fixture match\n      await matchFixture('signed', signedRecovery);\n\n      // Verify PSBT has valid signatures (user key signed)\n      checkRecoveryPsbtSignature(signedRecovery.txHex as string);\n    });\n\n    it('should test unsigned cross chain recovery', async () => {\n      const getRecoveryProviderStub = sinon\n        .stub(AbstractUtxoCoin.prototype, 'getRecoveryProvider')\n        .returns(new MockCrossChainRecoveryProvider<TNumber>(sourceCoin, getDepositUnspents(), depositTx));\n      const params = {\n        recoveryCoin,\n        txid: depositTx.getId(),\n        recoveryAddress,\n        wallet: recoveryWalletId,\n      };\n      const unsignedRecovery = (await sourceCoin.recoverFromWrongChain<TNumber>({\n        ...params,\n        signed: false,\n      })) as CrossChainRecoveryUnsigned<TNumber>;\n      should.equal(getRecoveryProviderStub.callCount, 1);\n\n      // Verify fixture match\n      await matchFixture('unsigned', unsignedRecovery);\n\n      // Verify the unsigned PSBT can be signed\n      const signedTx = await sourceCoin.signTransaction<TNumber>({\n        txPrebuild: unsignedRecovery,\n        prv: keychainsBase58[0].prv,\n        pubs: keychainsBase58.map((k) => k.pub) as Triple<string>,\n      });\n      checkRecoveryPsbtSignature((signedTx as { txHex: string }).txHex);\n    });\n  });\n}\n\nfunction isSupportedCrossChainRecovery(sourceCoin: AbstractUtxoCoin, recoveryCoin: AbstractUtxoCoin): boolean {\n  return supportedCrossChainRecoveries[sourceCoin.getFamily()]?.includes(recoveryCoin.getFamily());\n}\n\nutxoCoins.forEach((coin) => {\n  utxoCoins\n    .filter(\n      (otherCoin) =>\n        coin !== otherCoin &&\n        isSupportedCrossChainRecovery(coin, otherCoin) &&\n        ((utxolib.isMainnet(coin.network) && utxolib.isMainnet(otherCoin.network)) ||\n          (utxolib.isTestnet(coin.network) && utxolib.isTestnet(otherCoin.network)))\n    )\n    .forEach((otherCoin) => {\n      if (coin.amountType === 'bigint') {\n        run<bigint>(coin, otherCoin);\n      } else {\n        run(coin, otherCoin);\n      }\n    });\n});\n\ndescribe(`Cross-Chain Recovery getWallet`, async function () {\n  const bitgo = defaultBitGo;\n  const recoveryCoin = getUtxoCoin('btc');\n  const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';\n\n  it('should search v1 wallets if the v2 endpoint responds with a 4xx error', async function () {\n    const errorResponses = [400, 404];\n\n    for (const error of errorResponses) {\n      const nockV2Wallet = nockBitGo(bitgo)\n        .get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)\n        .reply(error);\n      const nockV1Wallet = nockBitGo(bitgo).get(`/api/v1/wallet/${recoveryWalletId}`).reply(error);\n      await assert.rejects(\n        () => getWallet(bitgo, recoveryCoin, recoveryWalletId),\n        Error(`could not get wallet ${recoveryWalletId} from v1 or v2: ApiResponseError: ${error}`)\n      );\n      nockV2Wallet.done();\n      nockV1Wallet.done();\n    }\n  });\n\n  it('should throw an error if the v2 endpoint responds with a 5xx error', async function () {\n    const errorResponses = [500];\n    for (const error of errorResponses) {\n      const nockV2Wallet = nockBitGo(bitgo)\n        .get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)\n        .reply(error);\n      await assert.rejects(() => getWallet(bitgo, recoveryCoin, recoveryWalletId), {\n        name: 'ApiResponseError',\n        status: 500,\n        result: {},\n        invalidToken: false,\n        needsOTP: false,\n      });\n      nockV2Wallet.done();\n    }\n  });\n});\n\ndescribe('convertLtcAddressToLegacyFormat', function () {\n  const ltcNetwork = utxolib.networks.litecoin;\n\n  it('should convert M... P2SH address to 3... legacy format', function () {\n    // These two addresses represent the same underlying script hash:\n    // - MNQ7zkgMsaV67rsjA3JuP59RC5wxRXpwgE is the LTC format (scriptHash 0x32)\n    // - 3GBygsGPvTdfKMbq4AKZZRu1sPMWPEsBfd is the BTC format (scriptHash 0x05)\n    const ltcAddress = 'MNQ7zkgMsaV67rsjA3JuP59RC5wxRXpwgE';\n    const expectedLegacyAddress = '3GBygsGPvTdfKMbq4AKZZRu1sPMWPEsBfd';\n\n    const legacyAddress = convertLtcAddressToLegacyFormat(ltcAddress, ltcNetwork);\n    assert.strictEqual(legacyAddress, expectedLegacyAddress);\n  });\n\n  it('should convert MD68PsdheKxcYsrVLyZRXgoSDLnB1MdVtE to legacy format', function () {\n    const address = 'MD68PsdheKxcYsrVLyZRXgoSDLnB1MdVtE';\n    const legacyAddress = convertLtcAddressToLegacyFormat(address, ltcNetwork);\n\n    // Should start with '3' (legacy BTC P2SH format)\n    assert.ok(legacyAddress.startsWith('3'), `Expected address to start with '3', got: ${legacyAddress}`);\n  });\n\n  it('should not modify bech32 addresses', function () {\n    const bech32Address = 'ltc1qgrl8zpndsklaa9swgd5vevyxmx5x63vcrl7dk4';\n    const result = convertLtcAddressToLegacyFormat(bech32Address, ltcNetwork);\n    assert.strictEqual(result, bech32Address);\n  });\n});\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"util.d.ts","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/util.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,OAAO,CAQpE"}
1
+ {"version":3,"file":"util.d.ts","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/util.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,OAAO,CAEpE"}
@@ -36,12 +36,6 @@ Object.defineProperty(exports, "__esModule", { value: true });
36
36
  exports.hasWasmUtxoSupport = hasWasmUtxoSupport;
37
37
  const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
38
38
  function hasWasmUtxoSupport(network) {
39
- return ![
40
- utxolib.networks.bitcoincash,
41
- utxolib.networks.bitcoingold,
42
- utxolib.networks.bitcoinsv,
43
- utxolib.networks.ecash,
44
- utxolib.networks.zcash,
45
- ].includes(utxolib.getMainnet(network));
39
+ return utxolib.getMainnet(network) !== utxolib.networks.zcash;
46
40
  }
47
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Rlc3QvdW5pdC90cmFuc2FjdGlvbi9maXhlZFNjcmlwdC91dGlsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUEsZ0RBUUM7QUFWRCw4REFBZ0Q7QUFFaEQsU0FBZ0Isa0JBQWtCLENBQUMsT0FBd0I7SUFDekQsT0FBTyxDQUFDO1FBQ04sT0FBTyxDQUFDLFFBQVEsQ0FBQyxXQUFXO1FBQzVCLE9BQU8sQ0FBQyxRQUFRLENBQUMsV0FBVztRQUM1QixPQUFPLENBQUMsUUFBUSxDQUFDLFNBQVM7UUFDMUIsT0FBTyxDQUFDLFFBQVEsQ0FBQyxLQUFLO1FBQ3RCLE9BQU8sQ0FBQyxRQUFRLENBQUMsS0FBSztLQUN2QixDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7QUFDMUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuXG5leHBvcnQgZnVuY3Rpb24gaGFzV2FzbVV0eG9TdXBwb3J0KG5ldHdvcms6IHV0eG9saWIuTmV0d29yayk6IGJvb2xlYW4ge1xuICByZXR1cm4gIVtcbiAgICB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5jYXNoLFxuICAgIHV0eG9saWIubmV0d29ya3MuYml0Y29pbmdvbGQsXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luc3YsXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy5lY2FzaCxcbiAgICB1dHhvbGliLm5ldHdvcmtzLnpjYXNoLFxuICBdLmluY2x1ZGVzKHV0eG9saWIuZ2V0TWFpbm5ldChuZXR3b3JrKSk7XG59XG4iXX0=
41
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Rlc3QvdW5pdC90cmFuc2FjdGlvbi9maXhlZFNjcmlwdC91dGlsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUEsZ0RBRUM7QUFKRCw4REFBZ0Q7QUFFaEQsU0FBZ0Isa0JBQWtCLENBQUMsT0FBd0I7SUFDekQsT0FBTyxPQUFPLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxLQUFLLE9BQU8sQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDO0FBQ2hFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyB1dHhvbGliIGZyb20gJ0BiaXRnby1iZXRhL3V0eG8tbGliJztcblxuZXhwb3J0IGZ1bmN0aW9uIGhhc1dhc21VdHhvU3VwcG9ydChuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmspOiBib29sZWFuIHtcbiAgcmV0dXJuIHV0eG9saWIuZ2V0TWFpbm5ldChuZXR3b3JrKSAhPT0gdXR4b2xpYi5uZXR3b3Jrcy56Y2FzaDtcbn1cbiJdfQ==