@bitgo-beta/sdk-coin-iota 1.0.1-beta.32 → 1.0.1-beta.320

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.
Files changed (63) hide show
  1. package/README.md +1 -1
  2. package/dist/src/iota.d.ts +13 -2
  3. package/dist/src/iota.d.ts.map +1 -1
  4. package/dist/src/iota.js +142 -22
  5. package/dist/src/lib/constants.d.ts +10 -6
  6. package/dist/src/lib/constants.d.ts.map +1 -1
  7. package/dist/src/lib/constants.js +12 -8
  8. package/dist/src/lib/iface.d.ts +33 -1
  9. package/dist/src/lib/iface.d.ts.map +1 -1
  10. package/dist/src/lib/iface.js +1 -1
  11. package/dist/src/lib/index.d.ts +1 -0
  12. package/dist/src/lib/index.d.ts.map +1 -1
  13. package/dist/src/lib/index.js +4 -2
  14. package/dist/src/lib/transaction.d.ts +72 -5
  15. package/dist/src/lib/transaction.d.ts.map +1 -1
  16. package/dist/src/lib/transaction.js +270 -6
  17. package/dist/src/lib/transactionBuilder.d.ts +70 -9
  18. package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
  19. package/dist/src/lib/transactionBuilder.js +155 -14
  20. package/dist/src/lib/transactionBuilderFactory.d.ts +15 -2
  21. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
  22. package/dist/src/lib/transactionBuilderFactory.js +45 -5
  23. package/dist/src/lib/transferBuilder.d.ts +10 -2
  24. package/dist/src/lib/transferBuilder.d.ts.map +1 -1
  25. package/dist/src/lib/transferBuilder.js +45 -3
  26. package/dist/src/lib/transferTransaction.d.ts +28 -0
  27. package/dist/src/lib/transferTransaction.d.ts.map +1 -0
  28. package/dist/src/lib/transferTransaction.js +173 -0
  29. package/dist/src/lib/utils.d.ts +2 -0
  30. package/dist/src/lib/utils.d.ts.map +1 -1
  31. package/dist/src/lib/utils.js +28 -4
  32. package/dist/test/resources/iota.d.ts +35 -0
  33. package/dist/test/resources/iota.d.ts.map +1 -0
  34. package/dist/test/resources/iota.js +93 -0
  35. package/dist/test/unit/index.d.ts +2 -0
  36. package/dist/test/unit/index.d.ts.map +1 -0
  37. package/dist/test/unit/index.js +16 -0
  38. package/dist/test/unit/iota.d.ts +2 -0
  39. package/dist/test/unit/iota.d.ts.map +1 -0
  40. package/dist/test/unit/iota.js +603 -0
  41. package/dist/test/unit/keyPair.d.ts +2 -0
  42. package/dist/test/unit/keyPair.d.ts.map +1 -0
  43. package/dist/test/unit/keyPair.js +135 -0
  44. package/dist/test/unit/transactionBuilder/transactionBuilder.d.ts +2 -0
  45. package/dist/test/unit/transactionBuilder/transactionBuilder.d.ts.map +1 -0
  46. package/dist/test/unit/transactionBuilder/transactionBuilder.js +306 -0
  47. package/dist/test/unit/transactionBuilder/transactionBuilderFactory.d.ts +2 -0
  48. package/dist/test/unit/transactionBuilder/transactionBuilderFactory.d.ts.map +1 -0
  49. package/dist/test/unit/transactionBuilder/transactionBuilderFactory.js +243 -0
  50. package/dist/test/unit/transactionBuilder/transferBuilder.d.ts +2 -0
  51. package/dist/test/unit/transactionBuilder/transferBuilder.d.ts.map +1 -0
  52. package/dist/test/unit/transactionBuilder/transferBuilder.js +400 -0
  53. package/dist/test/unit/transferTransaction.d.ts +2 -0
  54. package/dist/test/unit/transferTransaction.d.ts.map +1 -0
  55. package/dist/test/unit/transferTransaction.js +465 -0
  56. package/dist/test/unit/utils.d.ts +2 -0
  57. package/dist/test/unit/utils.d.ts.map +1 -0
  58. package/dist/test/unit/utils.js +278 -0
  59. package/dist/tsconfig.tsbuildinfo +1 -0
  60. package/package.json +16 -10
  61. package/.eslintignore +0 -5
  62. package/.mocharc.yml +0 -8
  63. package/CHANGELOG.md +0 -0
@@ -0,0 +1,603 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
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);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ const should_1 = __importDefault(require("should"));
40
+ const sdk_test_1 = require("@bitgo-beta/sdk-test");
41
+ const sdk_api_1 = require("@bitgo-beta/sdk-api");
42
+ const src_1 = require("../../src");
43
+ const assert_1 = __importDefault(require("assert"));
44
+ const statics_1 = require("@bitgo-beta/statics");
45
+ const testData = __importStar(require("../resources/iota"));
46
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
47
+ describe('IOTA:', function () {
48
+ let bitgo;
49
+ let basecoin;
50
+ before(function () {
51
+ bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'mock' });
52
+ bitgo.safeRegister('iota', src_1.Iota.createInstance);
53
+ bitgo.safeRegister('tiota', src_1.Iota.createInstance);
54
+ bitgo.initializeTestVars();
55
+ basecoin = bitgo.coin('tiota');
56
+ });
57
+ it('should return the right info', function () {
58
+ const iota = bitgo.coin('iota');
59
+ const tiota = bitgo.coin('tiota');
60
+ const iotaStatics = statics_1.coins.get('iota');
61
+ const tiotaStatics = statics_1.coins.get('tiota');
62
+ iota.getChain().should.equal('iota');
63
+ iota.getFamily().should.equal('iota');
64
+ iota.getFullName().should.equal('Iota');
65
+ iota.getBaseFactor().should.equal(1e9);
66
+ tiota.getChain().should.equal('tiota');
67
+ tiota.getFamily().should.equal('iota');
68
+ tiota.getFullName().should.equal('Testnet Iota');
69
+ tiota.getBaseFactor().should.equal(1e9);
70
+ iotaStatics.gasTankLowBalanceAlertFactor.should.equal(80);
71
+ tiotaStatics.gasTankLowBalanceAlertFactor.should.equal(80);
72
+ iotaStatics.gasTankMinBalanceRecommendationFactor.should.equal(200);
73
+ tiotaStatics.gasTankMinBalanceRecommendationFactor.should.equal(200);
74
+ });
75
+ it('is valid pub', function () {
76
+ // with 0x prefix
77
+ basecoin.isValidPub('0x9b4e96086d111500259f9b38680b0509a405c1904da18976455a20c691d3bb07').should.equal(false);
78
+ // without 0x prefix
79
+ basecoin.isValidPub('9b4e96086d111500259f9b38680b0509a405c1904da18976455a20c691d3bb07').should.equal(true);
80
+ });
81
+ describe('Address Validation', () => {
82
+ let keychains;
83
+ let commonKeychain;
84
+ before(function () {
85
+ commonKeychain =
86
+ '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781';
87
+ keychains = [
88
+ {
89
+ id: '6424c353eaf78d000766e95949868468',
90
+ source: 'user',
91
+ type: 'tss',
92
+ commonKeychain: '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781',
93
+ encryptedPrv: '{"iv":"cZd5i7L4RxtwrALW2rK7UA==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"5zgoH1Bd3Fw=","ct":"9vVlnXFRtrM9FVEo+d2chbGHlM9lFZemueBuAs3BIkPo33Fo7jzwwNK/kIWkEyg+NmEBd5IaqAS157nvvvwzzsmMWlQdUz9qbmXNv3pg987cXFR08exS+4uhwP1YNOjJTRvRNcO9ZqHb46d4fmyJ/yC9/susCge7r/EsbaN5C3afv1dzybuq912FwaQElZLYYp5BICudFOMZ9k0UDMfKM/PMDkH7WexoGHr9GKq/bgCH2B39TZZyHKU6Uy47lXep2s6h0DrMwHOrnmiL3DZjOj88Ynvphlzxuo4eOlD2UHia2+nvIaISYs29Pr0DAvREutchvcBpExj1kWWPv7hQYrv8F0NAdatsbWl3w+xKyfiMKo1USlrwyJviypGtQtXOJyw0XPN0rv2+L5lW8BbjpzHfYYN13fJTedlGTFhhkzVtbbPAKE02kx7zCJcjYaiexdSTsrDLScYNT9/Jhdt27KpsooehwVohLfSKz4vbFfRu2MPZw3/+c/hfiJNgtz6esWbnxGrcE8U2IwPYCaK+Ghk4DcqWNIni59RI5B5kAsQOToII40qPN510uTgxBSPO7q7MHgkxdd4CqBq+ojr9j0P7oao8E5Y+CBDJrojDoCh1oCCDW9vo2dXlVcD8SIbw7U/9AfvEbA4xyE/5md1M7CIwLnWs2Ynv0YtaKoqhdS9x6FmHlMDhN/DKHinrwmowtrTT82fOkpO5g9saSmgU7Qy3gLt8t+VwdEyeFeQUKRSyci8qgqXQaZIg4+aXgaSOnlCFMtmB8ekYxEhTY5uzRfrNgS4s1QeqFBpNtUF+Ydi297pbVXnJoXAN+SVWd80GCx+yI2dpVC89k3rOWK9WeyqlnzuLJWp2RIOB9cdW8GFv/fN+QAJpYeVxOE4+nZDsKnsj8nKcg9t4Dlx1G6gLM1/Vq9YxNLbuzuRC0asUYvdMnoMvszmpm++TxndYisgNYscpZSoz7wvcazJNEPfhPVjEkd6tUUuN4GM35H0DmKCUQNT+a6B6hmHlTZvjxiyGAg5bY59hdjvJ+22QduazlEEC6LI3HrA7uK0TpplWzS1tCIFvTMUhj65DEZmNJ2+ZY9bQ4vsMf+DRR3OOG4t+DMlNfjOd3zNv3QoY95BjfWpryFwPzDq7bCP67JDsoj7j2TY5FRSrRkD77H0Ewlux2cWfjRTwcMHcdQxxuV0OP0aNjGDjybFN"}',
94
+ },
95
+ {
96
+ id: '6424c353eaf78d000766e96137d4404b',
97
+ source: 'backup',
98
+ type: 'tss',
99
+ commonKeychain: '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781',
100
+ encryptedPrv: '{"iv":"vi0dPef/Rx7kG/pRySQi6Q==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"9efhQsiEvVs=","ct":"Gw6atvf6gxKzsjtl3xseipO3rAxp1mAz7Yu1ihFsi5/lf2vMZegApgZx+pyILFS9KKLHbNF3U6WgSYdrr2t4vzdLsXkH1WIxfHS+cd2C5N59yADZDnPJBT6pv/IRvaYelP0Ck3nIYQ2hSMm8op+VOWC/SzHeh7slYDqwEHTGan0Wigfvk1yRd7CCJTaEAomnc/4eFi2NY3X3gt/3opy9IAgknnwUFohn96EWpEQ0F6pbzH/Z8VF6gF+DUcrrByAxExUPnHQZiFk3YHU/vVV4FxBU/mVAE8xBsBn5ul5e5SUMPfc7TBuJWv4BByTNg9xDShF/91Yx2nbfUm5d9QmM8lpKgzzQvcK8POAPk87gRCuKnsGh5vNS0UppkHc+ocfzRQlGA6jze7QyyQO0rMj5Ly8kWjwk2vISvKYHYS1NR7VU549UIXo7NXjatunKSc3+IreoRUHIshiaLg6hl+pxCCuc0qQ43V0mdIfCjTN8gkGWLNk8R7tAGPz9jyapQPcPEGHgEz0ATIi6yMNWCsibS2eLiE1uVEJONoM4lk6FPl3Q2CHbW2MeEbqjY8hbaw18mNb2xSBH/Fwpiial+Tvi2imqgnCO4ZpO9bllKftZPcQy0stN+eGBlb5ufyflKkDSiChHYroGjEpmiFicdde48cJszF52uKNnf1q67fA9/S2FAHQab3EXojxH2Gbk+kkV2h/TYKFFZSWC3vi4e8mO+vjMUcR0AdsgPFyEIz0SCGuba3CnTLNdEuZwsauAeHkx2vUTnRgJPVgNeeuXmsVG76Sy2ggJHuals0Hj8U2Xda0qO1RuFfoCWfss9wn6HGRwPPkhSB/8oNguAqmRVGKkd8Zwt3IvrTd9fk0/rFFDJKGz7WyNHkYgUmNiGcItD12v0jx7FZ52EJzl3Av1RyJUQK18+8EYPh3SGiU9dt7VX0aF0uo6JouKhOeldUvMP+AugQz8fUclwTQsbboVg27Yxo0DyATVwThW5a56R6Qf5ZiQJluFuzs5y98rq0S5q046lE6o3vVmJpEdwjeSCJoET5CL4nTgkXyWvhm4eB8u/e66l3o0qbaSx8q9YYmT9EpRcl5TP4ThLBKETYdzVvg4exjQfektMatk5EyUpEIhZPXh5vXpJZesdfO9LJ8zTaHBsBjDPU7cdNgQMbebpataRi8A0el2/IJXl+E+olgAz5zC4i2O1Q=="}',
101
+ },
102
+ {
103
+ id: '6424c353eaf78d000766e9510b125fba',
104
+ source: 'bitgo',
105
+ type: 'tss',
106
+ commonKeychain: '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781',
107
+ verifiedVssProof: true,
108
+ isBitGo: true,
109
+ },
110
+ ];
111
+ });
112
+ it('should return true when validating a well formatted address prefixed with 0x', async function () {
113
+ const address = '0xf941ae3cbe5645dccc15da8346b533f7f91f202089a5521653c062b2ff10b304';
114
+ basecoin.isValidAddress(address).should.equal(true);
115
+ });
116
+ it('should return false when validating an old address', async function () {
117
+ const address = '0x2959bfc3fdb7dc23fed8deba2fafb70f3e606a59';
118
+ basecoin.isValidAddress(address).should.equal(false);
119
+ });
120
+ it('should return false when validating an incorrectly formatted', async function () {
121
+ const address = 'wrongaddress';
122
+ basecoin.isValidAddress(address).should.equal(false);
123
+ });
124
+ it('should return true for isWalletAddress with valid address for index 4', async function () {
125
+ const newAddress = '0x3f4b2a95d9b696989814f02f899cee491b4d600b1b918e979caec307af4b8dfc';
126
+ const index = 4;
127
+ const params = { commonKeychain, address: newAddress, index, keychains };
128
+ (await basecoin.isWalletAddress(params)).should.equal(true);
129
+ });
130
+ it('should throw error for isWalletAddress when keychains is missing', async function () {
131
+ const address = '0x2959bfc3fdb7dc23fed8deba2fafb70f3e606a59';
132
+ const index = 0;
133
+ const params = { commonKeychain, address, index };
134
+ await assert_1.default.rejects(async () => basecoin.isWalletAddress(params));
135
+ });
136
+ it('should throw error for isWalletAddress when new address is invalid', async function () {
137
+ const wrongAddress = 'badAddress';
138
+ const index = 0;
139
+ const params = { commonKeychain, address: wrongAddress, index };
140
+ await assert_1.default.rejects(async () => basecoin.isWalletAddress(params), {
141
+ message: `invalid address: ${wrongAddress}`,
142
+ });
143
+ });
144
+ });
145
+ describe('Transaction Methods', () => {
146
+ let factory;
147
+ before(function () {
148
+ factory = new src_1.TransactionBuilderFactory(statics_1.coins.get('tiota'));
149
+ });
150
+ describe('explainTransaction', () => {
151
+ it('should throw error for missing txHex', async function () {
152
+ await assert_1.default.rejects(async () => await basecoin.explainTransaction({ txHex: '' }), /missing required tx prebuild property txHex/);
153
+ });
154
+ it('should throw error for invalid transaction', async function () {
155
+ await assert_1.default.rejects(async () => await basecoin.explainTransaction({ txHex: 'invalidTxHex' }), /Failed to rebuild transaction/);
156
+ });
157
+ it('should call explainTransaction on transaction object', async function () {
158
+ const txBuilder = factory.getTransferBuilder();
159
+ txBuilder.sender(testData.sender.address);
160
+ txBuilder.recipients(testData.recipients);
161
+ txBuilder.paymentObjects(testData.paymentObjects);
162
+ txBuilder.gasData(testData.gasData);
163
+ const tx = (await txBuilder.build());
164
+ const explanation = tx.explainTransaction();
165
+ explanation.should.have.property('id');
166
+ explanation.should.have.property('outputs');
167
+ explanation.should.have.property('outputAmount');
168
+ explanation.should.have.property('fee');
169
+ explanation.should.have.property('type');
170
+ explanation.type.should.equal(sdk_core_1.TransactionType.Send);
171
+ explanation.fee.fee.should.equal(testData.GAS_BUDGET.toString());
172
+ });
173
+ });
174
+ describe('verifyTransaction', () => {
175
+ it('should throw error for missing txHex', async function () {
176
+ await assert_1.default.rejects(async () => await basecoin.verifyTransaction({
177
+ txPrebuild: {},
178
+ txParams: { recipients: testData.recipients },
179
+ }), /missing required tx prebuild property txHex/);
180
+ });
181
+ it('should verify transaction with matching recipients using JSON', async function () {
182
+ // Build transaction and get JSON instead of broadcast format to avoid parsing issues
183
+ const txBuilder = factory.getTransferBuilder();
184
+ txBuilder.sender(testData.sender.address);
185
+ txBuilder.recipients(testData.recipients);
186
+ txBuilder.paymentObjects(testData.paymentObjects);
187
+ txBuilder.gasData(testData.gasData);
188
+ const tx = (await txBuilder.build());
189
+ const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');
190
+ should_1.default.equal(await basecoin.verifyTransaction({
191
+ txPrebuild: { txHex: txHex },
192
+ txParams: { recipients: testData.recipients },
193
+ }), true);
194
+ });
195
+ it('should verify transaction with recipients containing extra fields', async function () {
196
+ // The verification should only compare address, amount, and tokenName
197
+ // Extra fields should be ignored
198
+ const txBuilder = factory.getTransferBuilder();
199
+ txBuilder.sender(testData.sender.address);
200
+ txBuilder.recipients(testData.recipients);
201
+ txBuilder.paymentObjects(testData.paymentObjects);
202
+ txBuilder.gasData(testData.gasData);
203
+ const tx = (await txBuilder.build());
204
+ const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');
205
+ // Add extra fields to recipients that should be ignored during comparison
206
+ const recipientsWithExtraFields = testData.recipients.map((r) => ({
207
+ ...r,
208
+ extraField: 'should be ignored',
209
+ anotherField: 123,
210
+ }));
211
+ should_1.default.equal(await basecoin.verifyTransaction({
212
+ txPrebuild: { txHex: txHex },
213
+ txParams: { recipients: recipientsWithExtraFields },
214
+ }), true);
215
+ });
216
+ it('should detect mismatched recipients', async function () {
217
+ const txBuilder = factory.getTransferBuilder();
218
+ txBuilder.sender(testData.sender.address);
219
+ txBuilder.recipients(testData.recipients);
220
+ txBuilder.paymentObjects(testData.paymentObjects);
221
+ txBuilder.gasData(testData.gasData);
222
+ const tx = (await txBuilder.build());
223
+ const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');
224
+ // Create mismatched recipients with different addresses/amounts
225
+ const mismatchedRecipients = [
226
+ {
227
+ address: testData.addresses.validAddresses[2], // Different address
228
+ amount: '9999', // Different amount
229
+ },
230
+ ];
231
+ await assert_1.default.rejects(async () => await basecoin.verifyTransaction({
232
+ txPrebuild: { txHex: txHex },
233
+ txParams: { recipients: mismatchedRecipients },
234
+ }), /Tx recipients does not match with expected txParams recipients/);
235
+ });
236
+ it('should verify transaction without recipients parameter', async function () {
237
+ const txBuilder = factory.getTransferBuilder();
238
+ txBuilder.sender(testData.sender.address);
239
+ txBuilder.recipients(testData.recipients);
240
+ txBuilder.paymentObjects(testData.paymentObjects);
241
+ txBuilder.gasData(testData.gasData);
242
+ const tx = (await txBuilder.build());
243
+ const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');
244
+ should_1.default.equal(await basecoin.verifyTransaction({
245
+ txPrebuild: { txHex: txHex },
246
+ txParams: {},
247
+ }), true);
248
+ });
249
+ });
250
+ describe('parseTransaction', () => {
251
+ it('should throw error for invalid transaction', async function () {
252
+ await assert_1.default.rejects(async () => await basecoin.parseTransaction({ txHex: 'invalidTxHex' }), /Failed to rebuild transaction/);
253
+ });
254
+ it('should parse transaction using JSON format', async function () {
255
+ // Build a transaction
256
+ const txBuilder = factory.getTransferBuilder();
257
+ txBuilder.sender(testData.sender.address);
258
+ txBuilder.recipients(testData.recipients);
259
+ txBuilder.paymentObjects(testData.paymentObjects);
260
+ txBuilder.gasData(testData.gasData);
261
+ const tx = (await txBuilder.build());
262
+ // Get the transaction explanation (which is what parseTransaction returns internally)
263
+ const explanation = tx.explainTransaction();
264
+ // Verify the parsed data structure
265
+ explanation.should.have.property('id');
266
+ explanation.should.have.property('outputs');
267
+ explanation.should.have.property('outputAmount');
268
+ explanation.should.have.property('fee');
269
+ // Verify outputs match recipients
270
+ explanation.outputs.length.should.equal(testData.recipients.length);
271
+ explanation.outputs.forEach((output, index) => {
272
+ output.address.should.equal(testData.recipients[index].address);
273
+ output.amount.should.equal(testData.recipients[index].amount);
274
+ });
275
+ // Verify output amount
276
+ const totalAmount = testData.recipients.reduce((sum, r) => sum + Number(r.amount), 0);
277
+ explanation.outputAmount.should.equal(totalAmount.toString());
278
+ // Verify fee
279
+ explanation.fee.fee.should.equal(testData.GAS_BUDGET.toString());
280
+ });
281
+ it('should parse transaction with single recipient', async function () {
282
+ const singleRecipient = [testData.recipients[0]];
283
+ const txBuilder = factory.getTransferBuilder();
284
+ txBuilder.sender(testData.sender.address);
285
+ txBuilder.recipients(singleRecipient);
286
+ txBuilder.paymentObjects(testData.paymentObjects);
287
+ txBuilder.gasData(testData.gasData);
288
+ const tx = (await txBuilder.build());
289
+ const explanation = tx.explainTransaction();
290
+ explanation.outputs.length.should.equal(1);
291
+ explanation.outputs[0].address.should.equal(singleRecipient[0].address);
292
+ explanation.outputs[0].amount.should.equal(singleRecipient[0].amount);
293
+ explanation.outputAmount.should.equal(singleRecipient[0].amount);
294
+ });
295
+ it('should parse transaction with multiple recipients', async function () {
296
+ const multipleRecipients = [
297
+ { address: testData.addresses.validAddresses[0], amount: '1000' },
298
+ { address: testData.addresses.validAddresses[1], amount: '2000' },
299
+ { address: testData.addresses.validAddresses[2], amount: '3000' },
300
+ ];
301
+ const txBuilder = factory.getTransferBuilder();
302
+ txBuilder.sender(testData.sender.address);
303
+ txBuilder.recipients(multipleRecipients);
304
+ txBuilder.paymentObjects(testData.paymentObjects);
305
+ txBuilder.gasData(testData.gasData);
306
+ const tx = (await txBuilder.build());
307
+ const explanation = tx.explainTransaction();
308
+ explanation.outputs.length.should.equal(3);
309
+ explanation.outputAmount.should.equal('6000');
310
+ });
311
+ });
312
+ describe('getSignablePayload', () => {
313
+ it('should get signable payload from transaction directly', async function () {
314
+ const txBuilder = factory.getTransferBuilder();
315
+ txBuilder.sender(testData.sender.address);
316
+ txBuilder.recipients(testData.recipients);
317
+ txBuilder.paymentObjects(testData.paymentObjects);
318
+ txBuilder.gasData(testData.gasData);
319
+ const tx = (await txBuilder.build());
320
+ const signablePayload = tx.signablePayload;
321
+ signablePayload.should.be.instanceOf(Buffer);
322
+ signablePayload.length.should.equal(32); // Blake2b hash is 32 bytes
323
+ });
324
+ it('should throw error for invalid transaction', async function () {
325
+ await assert_1.default.rejects(async () => await basecoin.getSignablePayload('invalidTxBase64'), /Failed to rebuild transaction/);
326
+ });
327
+ it('should generate consistent signable payload for identical transactions', async function () {
328
+ // Build first transaction
329
+ const txBuilder1 = factory.getTransferBuilder();
330
+ txBuilder1.sender(testData.sender.address);
331
+ txBuilder1.recipients(testData.recipients);
332
+ txBuilder1.paymentObjects(testData.paymentObjects);
333
+ txBuilder1.gasData(testData.gasData);
334
+ const tx1 = (await txBuilder1.build());
335
+ const payload1 = tx1.signablePayload;
336
+ // Build second identical transaction
337
+ const txBuilder2 = factory.getTransferBuilder();
338
+ txBuilder2.sender(testData.sender.address);
339
+ txBuilder2.recipients(testData.recipients);
340
+ txBuilder2.paymentObjects(testData.paymentObjects);
341
+ txBuilder2.gasData(testData.gasData);
342
+ const tx2 = (await txBuilder2.build());
343
+ const payload2 = tx2.signablePayload;
344
+ // Payloads should be identical for identical transactions
345
+ payload1.toString('hex').should.equal(payload2.toString('hex'));
346
+ });
347
+ it('should generate different signable payloads for different transactions', async function () {
348
+ // Build first transaction
349
+ const txBuilder1 = factory.getTransferBuilder();
350
+ txBuilder1.sender(testData.sender.address);
351
+ txBuilder1.recipients(testData.recipients);
352
+ txBuilder1.paymentObjects(testData.paymentObjects);
353
+ txBuilder1.gasData(testData.gasData);
354
+ const tx1 = (await txBuilder1.build());
355
+ const payload1 = tx1.signablePayload;
356
+ // Build second transaction with different recipient
357
+ const differentRecipients = [{ address: testData.addresses.validAddresses[0], amount: '5000' }];
358
+ const txBuilder2 = factory.getTransferBuilder();
359
+ txBuilder2.sender(testData.sender.address);
360
+ txBuilder2.recipients(differentRecipients);
361
+ txBuilder2.paymentObjects(testData.paymentObjects);
362
+ txBuilder2.gasData(testData.gasData);
363
+ const tx2 = (await txBuilder2.build());
364
+ const payload2 = tx2.signablePayload;
365
+ // Payloads should be different for different transactions
366
+ payload1.toString('hex').should.not.equal(payload2.toString('hex'));
367
+ });
368
+ it('should throw error when getting payload from simulate transaction', async function () {
369
+ const txBuilder = factory.getTransferBuilder();
370
+ txBuilder.sender(testData.sender.address);
371
+ txBuilder.recipients(testData.recipients);
372
+ txBuilder.paymentObjects(testData.paymentObjects);
373
+ // Don't set gasData - this will be a simulate transaction
374
+ const tx = (await txBuilder.build());
375
+ should_1.default.equal(tx.isSimulateTx, true);
376
+ (0, should_1.default)(() => tx.signablePayload).throwError('Cannot sign a simulate tx');
377
+ });
378
+ });
379
+ describe('setCoinSpecificFieldsInIntent', () => {
380
+ it('should set unspents in intent', function () {
381
+ const intent = {};
382
+ const params = {
383
+ unspents: [
384
+ { objectId: '0x123', version: '1', digest: 'abc' },
385
+ { objectId: '0x456', version: '2', digest: 'def' },
386
+ ],
387
+ };
388
+ basecoin.setCoinSpecificFieldsInIntent(intent, params);
389
+ intent.should.have.property('unspents');
390
+ intent.unspents.should.deepEqual(params.unspents);
391
+ });
392
+ it('should handle empty unspents', function () {
393
+ const intent = {};
394
+ const params = { unspents: [] };
395
+ basecoin.setCoinSpecificFieldsInIntent(intent, params);
396
+ intent.should.have.property('unspents');
397
+ intent.unspents.should.deepEqual([]);
398
+ });
399
+ it('should handle undefined unspents', function () {
400
+ const intent = {};
401
+ const params = {};
402
+ basecoin.setCoinSpecificFieldsInIntent(intent, params);
403
+ intent.should.have.property('unspents');
404
+ (intent.unspents === undefined).should.be.true();
405
+ });
406
+ });
407
+ describe('Transaction with Gas Sponsor', () => {
408
+ it('should build transaction with gas sponsor', async function () {
409
+ const txBuilder = factory.getTransferBuilder();
410
+ txBuilder.sender(testData.sender.address);
411
+ txBuilder.recipients(testData.recipients);
412
+ txBuilder.paymentObjects(testData.paymentObjects);
413
+ txBuilder.gasData(testData.gasData);
414
+ txBuilder.gasSponsor(testData.gasSponsor.address);
415
+ const tx = (await txBuilder.build());
416
+ should_1.default.equal(tx.sender, testData.sender.address);
417
+ should_1.default.equal(tx.gasSponsor, testData.gasSponsor.address);
418
+ should_1.default.notEqual(tx.sender, tx.gasSponsor);
419
+ should_1.default.equal(tx.type, sdk_core_1.TransactionType.Send);
420
+ });
421
+ it('should handle transaction where sender and gas sponsor are same', async function () {
422
+ const txBuilder = factory.getTransferBuilder();
423
+ txBuilder.sender(testData.sender.address);
424
+ txBuilder.recipients(testData.recipients);
425
+ txBuilder.paymentObjects(testData.paymentObjects);
426
+ txBuilder.gasData(testData.gasData);
427
+ txBuilder.gasSponsor(testData.sender.address); // Same as sender
428
+ const tx = (await txBuilder.build());
429
+ should_1.default.equal(tx.sender, testData.sender.address);
430
+ should_1.default.equal(tx.gasSponsor, testData.sender.address);
431
+ });
432
+ });
433
+ describe('Transaction ID Consistency', () => {
434
+ it('should generate same ID for identical transactions', async function () {
435
+ const txBuilder1 = factory.getTransferBuilder();
436
+ txBuilder1.sender(testData.sender.address);
437
+ txBuilder1.recipients(testData.recipients);
438
+ txBuilder1.paymentObjects(testData.paymentObjects);
439
+ txBuilder1.gasData(testData.gasData);
440
+ const tx1 = (await txBuilder1.build());
441
+ const txBuilder2 = factory.getTransferBuilder();
442
+ txBuilder2.sender(testData.sender.address);
443
+ txBuilder2.recipients(testData.recipients);
444
+ txBuilder2.paymentObjects(testData.paymentObjects);
445
+ txBuilder2.gasData(testData.gasData);
446
+ const tx2 = (await txBuilder2.build());
447
+ should_1.default.equal(tx1.id, tx2.id);
448
+ });
449
+ it('should generate different IDs for different transactions', async function () {
450
+ const txBuilder1 = factory.getTransferBuilder();
451
+ txBuilder1.sender(testData.sender.address);
452
+ txBuilder1.recipients(testData.recipients);
453
+ txBuilder1.paymentObjects(testData.paymentObjects);
454
+ txBuilder1.gasData(testData.gasData);
455
+ const tx1 = (await txBuilder1.build());
456
+ const differentRecipients = [{ address: testData.addresses.validAddresses[0], amount: '9999' }];
457
+ const txBuilder2 = factory.getTransferBuilder();
458
+ txBuilder2.sender(testData.sender.address);
459
+ txBuilder2.recipients(differentRecipients);
460
+ txBuilder2.paymentObjects(testData.paymentObjects);
461
+ txBuilder2.gasData(testData.gasData);
462
+ const tx2 = (await txBuilder2.build());
463
+ should_1.default.notEqual(tx1.id, tx2.id);
464
+ });
465
+ });
466
+ describe('Gas Configuration Edge Cases', () => {
467
+ it('should handle minimum gas values', async function () {
468
+ const minGasData = {
469
+ gasBudget: 1000,
470
+ gasPrice: 100,
471
+ gasPaymentObjects: [testData.gasPaymentObjects[0]],
472
+ };
473
+ const txBuilder = factory.getTransferBuilder();
474
+ txBuilder.sender(testData.sender.address);
475
+ txBuilder.recipients(testData.recipients);
476
+ txBuilder.paymentObjects(testData.paymentObjects);
477
+ txBuilder.gasData(minGasData);
478
+ const tx = (await txBuilder.build());
479
+ should_1.default.equal(tx.gasBudget, 1000);
480
+ should_1.default.equal(tx.gasPrice, 100);
481
+ should_1.default.equal(tx.gasPaymentObjects?.length, 1);
482
+ });
483
+ it('should handle large gas values', async function () {
484
+ const largeGasData = {
485
+ gasBudget: 50000000000, // 50 billion
486
+ gasPrice: 100000,
487
+ gasPaymentObjects: testData.gasPaymentObjects,
488
+ };
489
+ const txBuilder = factory.getTransferBuilder();
490
+ txBuilder.sender(testData.sender.address);
491
+ txBuilder.recipients(testData.recipients);
492
+ txBuilder.paymentObjects(testData.paymentObjects);
493
+ txBuilder.gasData(largeGasData);
494
+ const tx = (await txBuilder.build());
495
+ should_1.default.equal(tx.gasBudget, 50000000000);
496
+ should_1.default.equal(tx.gasPrice, 100000);
497
+ });
498
+ it('should return gas fee from transaction', async function () {
499
+ const txBuilder = factory.getTransferBuilder();
500
+ txBuilder.sender(testData.sender.address);
501
+ txBuilder.recipients(testData.recipients);
502
+ txBuilder.paymentObjects(testData.paymentObjects);
503
+ txBuilder.gasData(testData.gasData);
504
+ const tx = (await txBuilder.build());
505
+ const fee = tx.getFee();
506
+ should_1.default.exist(fee);
507
+ should_1.default.equal(fee, testData.GAS_BUDGET.toString());
508
+ });
509
+ });
510
+ describe('Transaction State Management', () => {
511
+ it('should track simulate mode correctly', async function () {
512
+ // Build without gas data - should be simulate mode
513
+ const simulateBuilder = factory.getTransferBuilder();
514
+ simulateBuilder.sender(testData.sender.address);
515
+ simulateBuilder.recipients(testData.recipients);
516
+ simulateBuilder.paymentObjects(testData.paymentObjects);
517
+ const simulateTx = (await simulateBuilder.build());
518
+ should_1.default.equal(simulateTx.isSimulateTx, true);
519
+ // Build with gas data - should not be simulate mode
520
+ const realBuilder = factory.getTransferBuilder();
521
+ realBuilder.sender(testData.sender.address);
522
+ realBuilder.recipients(testData.recipients);
523
+ realBuilder.paymentObjects(testData.paymentObjects);
524
+ realBuilder.gasData(testData.gasData);
525
+ const realTx = (await realBuilder.build());
526
+ should_1.default.equal(realTx.isSimulateTx, false);
527
+ });
528
+ it('should handle canSign based on simulate mode', async function () {
529
+ // Simulate transaction cannot be signed
530
+ const simulateBuilder = factory.getTransferBuilder();
531
+ simulateBuilder.sender(testData.sender.address);
532
+ simulateBuilder.recipients(testData.recipients);
533
+ simulateBuilder.paymentObjects(testData.paymentObjects);
534
+ const simulateTx = (await simulateBuilder.build());
535
+ should_1.default.equal(simulateTx.canSign({}), false);
536
+ // Real transaction can be signed
537
+ const realBuilder = factory.getTransferBuilder();
538
+ realBuilder.sender(testData.sender.address);
539
+ realBuilder.recipients(testData.recipients);
540
+ realBuilder.paymentObjects(testData.paymentObjects);
541
+ realBuilder.gasData(testData.gasData);
542
+ const realTx = (await realBuilder.build());
543
+ should_1.default.equal(realTx.canSign({}), true);
544
+ });
545
+ it('should handle transaction type correctly', async function () {
546
+ const txBuilder = factory.getTransferBuilder();
547
+ txBuilder.sender(testData.sender.address);
548
+ txBuilder.recipients(testData.recipients);
549
+ txBuilder.paymentObjects(testData.paymentObjects);
550
+ txBuilder.gasData(testData.gasData);
551
+ const tx = (await txBuilder.build());
552
+ should_1.default.equal(tx.type, sdk_core_1.TransactionType.Send);
553
+ });
554
+ });
555
+ describe('Transaction Serialization Formats', () => {
556
+ it('should serialize to consistent broadcast format', async function () {
557
+ const txBuilder1 = factory.getTransferBuilder();
558
+ txBuilder1.sender(testData.sender.address);
559
+ txBuilder1.recipients(testData.recipients);
560
+ txBuilder1.paymentObjects(testData.paymentObjects);
561
+ txBuilder1.gasData(testData.gasData);
562
+ const tx1 = (await txBuilder1.build());
563
+ const broadcast1 = await tx1.toBroadcastFormat();
564
+ const txBuilder2 = factory.getTransferBuilder();
565
+ txBuilder2.sender(testData.sender.address);
566
+ txBuilder2.recipients(testData.recipients);
567
+ txBuilder2.paymentObjects(testData.paymentObjects);
568
+ txBuilder2.gasData(testData.gasData);
569
+ const tx2 = (await txBuilder2.build());
570
+ const broadcast2 = await tx2.toBroadcastFormat();
571
+ should_1.default.equal(broadcast1, broadcast2);
572
+ });
573
+ it('should produce valid base64 broadcast format', async function () {
574
+ const txBuilder = factory.getTransferBuilder();
575
+ txBuilder.sender(testData.sender.address);
576
+ txBuilder.recipients(testData.recipients);
577
+ txBuilder.paymentObjects(testData.paymentObjects);
578
+ txBuilder.gasData(testData.gasData);
579
+ const tx = (await txBuilder.build());
580
+ const broadcast = await tx.toBroadcastFormat();
581
+ // Check if it's valid base64
582
+ should_1.default.equal(typeof broadcast, 'string');
583
+ should_1.default.equal(/^[A-Za-z0-9+/]*={0,2}$/.test(broadcast), true);
584
+ should_1.default.equal(broadcast.length > 0, true);
585
+ // Should be able to decode
586
+ const decoded = Buffer.from(broadcast, 'base64');
587
+ should_1.default.equal(decoded.length > 0, true);
588
+ });
589
+ it('should maintain JSON serialization consistency', async function () {
590
+ const txBuilder = factory.getTransferBuilder();
591
+ txBuilder.sender(testData.sender.address);
592
+ txBuilder.recipients(testData.recipients);
593
+ txBuilder.paymentObjects(testData.paymentObjects);
594
+ txBuilder.gasData(testData.gasData);
595
+ const tx = (await txBuilder.build());
596
+ const json1 = tx.toJson();
597
+ const json2 = tx.toJson();
598
+ should_1.default.deepEqual(json1, json2);
599
+ });
600
+ });
601
+ });
602
+ });
603
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"iota.js","sourceRoot":"","sources":["../../../test/unit/iota.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,oDAA4B;AAC5B,mDAA+D;AAC/D,iDAA+C;AAC/C,mCAAiF;AACjF,oDAA4B;AAC5B,iDAAgE;AAChE,4DAA8C;AAC9C,mDAAuD;AAEvD,QAAQ,CAAC,OAAO,EAAE;IAChB,IAAI,KAAmB,CAAC;IACxB,IAAI,QAAQ,CAAC;IAEb,MAAM,CAAC;QACL,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,kBAAQ,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;QACtD,KAAK,CAAC,YAAY,CAAC,MAAM,EAAE,UAAI,CAAC,cAAc,CAAC,CAAC;QAChD,KAAK,CAAC,YAAY,CAAC,OAAO,EAAE,UAAI,CAAC,cAAc,CAAC,CAAC;QACjD,KAAK,CAAC,kBAAkB,EAAE,CAAC;QAC3B,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE;QACjC,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAChC,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAClC,MAAM,WAAW,GAAG,eAAK,CAAC,GAAG,CAAC,MAAM,CAAuB,CAAC;QAC5D,MAAM,YAAY,GAAG,eAAK,CAAC,GAAG,CAAC,OAAO,CAAuB,CAAC;QAE9D,IAAI,CAAC,QAAQ,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACrC,IAAI,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACtC,IAAI,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACxC,IAAI,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAEvC,KAAK,CAAC,QAAQ,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QACvC,KAAK,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACvC,KAAK,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QACjD,KAAK,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAExC,WAAW,CAAC,4BAA4B,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC1D,YAAY,CAAC,4BAA4B,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC3D,WAAW,CAAC,qCAAqC,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACpE,YAAY,CAAC,qCAAqC,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACvE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,cAAc,EAAE;QACjB,iBAAiB;QACjB,QAAQ,CAAC,UAAU,CAAC,oEAAoE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC9G,oBAAoB;QACpB,QAAQ,CAAC,UAAU,CAAC,kEAAkE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAC7G,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;QAClC,IAAI,SAAS,CAAC;QACd,IAAI,cAAc,CAAC;QAEnB,MAAM,CAAC;YACL,cAAc;gBACZ,kIAAkI,CAAC;YACrI,SAAS,GAAG;gBACV;oBACE,EAAE,EAAE,kCAAkC;oBACtC,MAAM,EAAE,MAAM;oBACd,IAAI,EAAE,KAAK;oBACX,cAAc,EACZ,kIAAkI;oBACpI,YAAY,EACV,g0CAAg0C;iBACn0C;gBACD;oBACE,EAAE,EAAE,kCAAkC;oBACtC,MAAM,EAAE,QAAQ;oBAChB,IAAI,EAAE,KAAK;oBACX,cAAc,EACZ,kIAAkI;oBACpI,YAAY,EACV,g0CAAg0C;iBACn0C;gBACD;oBACE,EAAE,EAAE,kCAAkC;oBACtC,MAAM,EAAE,OAAO;oBACf,IAAI,EAAE,KAAK;oBACX,cAAc,EACZ,kIAAkI;oBACpI,gBAAgB,EAAE,IAAI;oBACtB,OAAO,EAAE,IAAI;iBACd;aACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8EAA8E,EAAE,KAAK;YACtF,MAAM,OAAO,GAAG,oEAAoE,CAAC;YACrF,QAAQ,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oDAAoD,EAAE,KAAK;YAC5D,MAAM,OAAO,GAAG,4CAA4C,CAAC;YAC7D,QAAQ,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8DAA8D,EAAE,KAAK;YACtE,MAAM,OAAO,GAAG,cAAc,CAAC;YAC/B,QAAQ,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uEAAuE,EAAE,KAAK;YAC/E,MAAM,UAAU,GAAG,oEAAoE,CAAC;YACxF,MAAM,KAAK,GAAG,CAAC,CAAC;YAEhB,MAAM,MAAM,GAAG,EAAE,cAAc,EAAE,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC;YACzE,CAAC,MAAM,QAAQ,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kEAAkE,EAAE,KAAK;YAC1E,MAAM,OAAO,GAAG,4CAA4C,CAAC;YAC7D,MAAM,KAAK,GAAG,CAAC,CAAC;YAEhB,MAAM,MAAM,GAAG,EAAE,cAAc,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;YAClD,MAAM,gBAAM,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC,QAAQ,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oEAAoE,EAAE,KAAK;YAC5E,MAAM,YAAY,GAAG,YAAY,CAAC;YAClC,MAAM,KAAK,GAAG,CAAC,CAAC;YAEhB,MAAM,MAAM,GAAG,EAAE,cAAc,EAAE,OAAO,EAAE,YAAY,EAAE,KAAK,EAAE,CAAC;YAChE,MAAM,gBAAM,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC,QAAQ,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE;gBACjE,OAAO,EAAE,oBAAoB,YAAY,EAAE;aAC5C,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,IAAI,OAAkC,CAAC;QAEvC,MAAM,CAAC;YACL,OAAO,GAAG,IAAI,+BAAyB,CAAC,eAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;YAClC,EAAE,CAAC,sCAAsC,EAAE,KAAK;gBAC9C,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CAAC,MAAM,QAAQ,CAAC,kBAAkB,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,EAC5D,6CAA6C,CAC9C,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;gBACpD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CAAC,MAAM,QAAQ,CAAC,kBAAkB,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,EACxE,+BAA+B,CAChC,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,sDAAsD,EAAE,KAAK;gBAC9D,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,WAAW,GAAG,EAAE,CAAC,kBAAkB,EAAE,CAAC;gBAE5C,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBACvC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;gBAC5C,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACjD,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBACxC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBACzC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,0BAAe,CAAC,IAAI,CAAC,CAAC;gBACpD,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC,CAAC;YACnE,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;YACjC,EAAE,CAAC,sCAAsC,EAAE,KAAK;gBAC9C,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBAC/B,UAAU,EAAE,EAAE;oBACd,QAAQ,EAAE,EAAE,UAAU,EAAE,QAAQ,CAAC,UAAU,EAAE;iBAC9C,CAAC,EACJ,6CAA6C,CAC9C,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,+DAA+D,EAAE,KAAK;gBACvE,qFAAqF;gBACrF,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,iBAAiB,EAAE,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAClF,gBAAM,CAAC,KAAK,CACV,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBAC/B,UAAU,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE;oBAC5B,QAAQ,EAAE,EAAE,UAAU,EAAE,QAAQ,CAAC,UAAU,EAAE;iBAC9C,CAAC,EACF,IAAI,CACL,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,mEAAmE,EAAE,KAAK;gBAC3E,sEAAsE;gBACtE,iCAAiC;gBACjC,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,iBAAiB,EAAE,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAElF,0EAA0E;gBAC1E,MAAM,yBAAyB,GAAG,QAAQ,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBAChE,GAAG,CAAC;oBACJ,UAAU,EAAE,mBAAmB;oBAC/B,YAAY,EAAE,GAAG;iBAClB,CAAC,CAAC,CAAC;gBAEJ,gBAAM,CAAC,KAAK,CACV,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBAC/B,UAAU,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE;oBAC5B,QAAQ,EAAE,EAAE,UAAU,EAAE,yBAAyB,EAAE;iBACpD,CAAC,EACF,IAAI,CACL,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,qCAAqC,EAAE,KAAK;gBAC7C,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,iBAAiB,EAAE,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAElF,gEAAgE;gBAChE,MAAM,oBAAoB,GAAG;oBAC3B;wBACE,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,oBAAoB;wBACnE,MAAM,EAAE,MAAM,EAAE,mBAAmB;qBACpC;iBACF,CAAC;gBAEF,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBAC/B,UAAU,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE;oBAC5B,QAAQ,EAAE,EAAE,UAAU,EAAE,oBAAoB,EAAE;iBAC/C,CAAC,EACJ,gEAAgE,CACjE,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wDAAwD,EAAE,KAAK;gBAChE,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,iBAAiB,EAAE,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAElF,gBAAM,CAAC,KAAK,CACV,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBAC/B,UAAU,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE;oBAC5B,QAAQ,EAAE,EAAE;iBACb,CAAC,EACF,IAAI,CACL,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;YAChC,EAAE,CAAC,4CAA4C,EAAE,KAAK;gBACpD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CAAC,MAAM,QAAQ,CAAC,gBAAgB,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,EACtE,+BAA+B,CAChC,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;gBACpD,sBAAsB;gBACtB,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE5D,sFAAsF;gBACtF,MAAM,WAAW,GAAG,EAAE,CAAC,kBAAkB,EAAE,CAAC;gBAE5C,mCAAmC;gBACnC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBACvC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;gBAC5C,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACjD,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAExC,kCAAkC;gBAClC,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;gBACpE,WAAW,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;oBAC5C,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC;oBAChE,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,CAAC;gBAChE,CAAC,CAAC,CAAC;gBAEH,uBAAuB;gBACvB,MAAM,WAAW,GAAG,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;gBACtF,WAAW,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAE9D,aAAa;gBACb,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC,CAAC;YACnE,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK;gBACxD,MAAM,eAAe,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;gBAEjD,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;gBACtC,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,WAAW,GAAG,EAAE,CAAC,kBAAkB,EAAE,CAAC;gBAE5C,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC3C,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;gBACxE,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;gBACtE,WAAW,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;YACnE,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,mDAAmD,EAAE,KAAK;gBAC3D,MAAM,kBAAkB,GAAG;oBACzB,EAAE,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE;oBACjE,EAAE,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE;oBACjE,EAAE,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE;iBAClE,CAAC;gBAEF,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,kBAAkB,CAAC,CAAC;gBACzC,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,WAAW,GAAG,EAAE,CAAC,kBAAkB,EAAE,CAAC;gBAE5C,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC3C,WAAW,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;YAClC,EAAE,CAAC,uDAAuD,EAAE,KAAK;gBAC/D,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,eAAe,GAAG,EAAE,CAAC,eAAe,CAAC;gBAE3C,eAAe,CAAC,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;gBAC7C,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,2BAA2B;YACtE,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;gBACpD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CAAC,MAAM,QAAQ,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,EAChE,+BAA+B,CAChC,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wEAAwE,EAAE,KAAK;gBAChF,0BAA0B;gBAC1B,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC9D,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC;gBAErC,qCAAqC;gBACrC,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC9D,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC;gBAErC,0DAA0D;gBAC1D,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;YAClE,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wEAAwE,EAAE,KAAK;gBAChF,0BAA0B;gBAC1B,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC9D,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC;gBAErC,oDAAoD;gBACpD,MAAM,mBAAmB,GAAG,CAAC,EAAE,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;gBAChG,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,mBAAmB,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC9D,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC;gBAErC,0DAA0D;gBAC1D,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;YACtE,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,mEAAmE,EAAE,KAAK;gBAC3E,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,0DAA0D;gBAE1D,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;gBAEpC,IAAA,gBAAM,EAAC,GAAG,EAAE,CAAC,EAAE,CAAC,eAAe,CAAC,CAAC,UAAU,CAAC,2BAA2B,CAAC,CAAC;YAC3E,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,+BAA+B,EAAE,GAAG,EAAE;YAC7C,EAAE,CAAC,+BAA+B,EAAE;gBAClC,MAAM,MAAM,GAAG,EAAS,CAAC;gBACzB,MAAM,MAAM,GAAG;oBACb,QAAQ,EAAE;wBACR,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE;wBAClD,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE;qBACnD;iBACK,CAAC;gBAET,QAAQ,CAAC,6BAA6B,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;gBAEvD,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBACxC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;YACpD,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,8BAA8B,EAAE;gBACjC,MAAM,MAAM,GAAG,EAAS,CAAC;gBACzB,MAAM,MAAM,GAAG,EAAE,QAAQ,EAAE,EAAE,EAAS,CAAC;gBAEvC,QAAQ,CAAC,6BAA6B,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;gBAEvD,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBACxC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;YACvC,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,kCAAkC,EAAE;gBACrC,MAAM,MAAM,GAAG,EAAS,CAAC;gBACzB,MAAM,MAAM,GAAG,EAAS,CAAC;gBAEzB,QAAQ,CAAC,6BAA6B,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;gBAEvD,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBACxC,CAAC,MAAM,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;YACnD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;YAC5C,EAAE,CAAC,2CAA2C,EAAE,KAAK;gBACnD,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBACpC,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;gBAElD,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE5D,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,EAAE,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBACjD,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;gBACzD,gBAAM,CAAC,QAAQ,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC;gBAC1C,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,EAAE,0BAAe,CAAC,IAAI,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,iEAAiE,EAAE,KAAK;gBACzE,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBACpC,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,iBAAiB;gBAEhE,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE5D,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,EAAE,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBACjD,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,UAAU,EAAE,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YACvD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,4BAA4B,EAAE,GAAG,EAAE;YAC1C,EAAE,CAAC,oDAAoD,EAAE,KAAK;gBAC5D,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE9D,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE9D,gBAAM,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;YAC/B,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,0DAA0D,EAAE,KAAK;gBAClE,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE9D,MAAM,mBAAmB,GAAG,CAAC,EAAE,OAAO,EAAE,QAAQ,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;gBAChG,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,mBAAmB,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE9D,gBAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;YAClC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;YAC5C,EAAE,CAAC,kCAAkC,EAAE,KAAK;gBAC1C,MAAM,UAAU,GAAG;oBACjB,SAAS,EAAE,IAAI;oBACf,QAAQ,EAAE,GAAG;oBACb,iBAAiB,EAAE,CAAC,QAAQ,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC;iBACnD,CAAC;gBAEF,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;gBAE9B,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE5D,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;gBACjC,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;gBAC/B,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,iBAAiB,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,gCAAgC,EAAE,KAAK;gBACxC,MAAM,YAAY,GAAG;oBACnB,SAAS,EAAE,WAAW,EAAE,aAAa;oBACrC,QAAQ,EAAE,MAAM;oBAChB,iBAAiB,EAAE,QAAQ,CAAC,iBAAiB;iBAC9C,CAAC;gBAEF,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;gBAEhC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE5D,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;gBACxC,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACpC,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wCAAwC,EAAE,KAAK;gBAChD,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,GAAG,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;gBAExB,gBAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAClB,gBAAM,CAAC,KAAK,CAAC,GAAG,EAAE,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC,CAAC;YACpD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;YAC5C,EAAE,CAAC,sCAAsC,EAAE,KAAK;gBAC9C,mDAAmD;gBACnD,MAAM,eAAe,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBACrD,eAAe,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAChD,eAAe,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAChD,eAAe,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAExD,MAAM,UAAU,GAAG,CAAC,MAAM,eAAe,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC1E,gBAAM,CAAC,KAAK,CAAC,UAAU,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;gBAE5C,oDAAoD;gBACpD,MAAM,WAAW,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBACjD,WAAW,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC5C,WAAW,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC5C,WAAW,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACpD,WAAW,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEtC,MAAM,MAAM,GAAG,CAAC,MAAM,WAAW,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAClE,gBAAM,CAAC,KAAK,CAAC,MAAM,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;YAC3C,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK;gBACtD,wCAAwC;gBACxC,MAAM,eAAe,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBACrD,eAAe,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAChD,eAAe,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAChD,eAAe,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAExD,MAAM,UAAU,GAAG,CAAC,MAAM,eAAe,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC1E,gBAAM,CAAC,KAAK,CAAC,UAAU,CAAC,OAAO,CAAC,EAAS,CAAC,EAAE,KAAK,CAAC,CAAC;gBAEnD,iCAAiC;gBACjC,MAAM,WAAW,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBACjD,WAAW,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC5C,WAAW,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC5C,WAAW,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACpD,WAAW,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEtC,MAAM,MAAM,GAAG,CAAC,MAAM,WAAW,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAClE,gBAAM,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,EAAS,CAAC,EAAE,IAAI,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,0CAA0C,EAAE,KAAK;gBAClD,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAE5D,gBAAM,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,EAAE,0BAAe,CAAC,IAAI,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,mCAAmC,EAAE,GAAG,EAAE;YACjD,EAAE,CAAC,iDAAiD,EAAE,KAAK;gBACzD,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC9D,MAAM,UAAU,GAAG,MAAM,GAAG,CAAC,iBAAiB,EAAE,CAAC;gBAEjD,MAAM,UAAU,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC3C,UAAU,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC3C,UAAU,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBACnD,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAErC,MAAM,GAAG,GAAG,CAAC,MAAM,UAAU,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC9D,MAAM,UAAU,GAAG,MAAM,GAAG,CAAC,iBAAiB,EAAE,CAAC;gBAEjD,gBAAM,CAAC,KAAK,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC;YACvC,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK;gBACtD,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,SAAS,GAAG,MAAM,EAAE,CAAC,iBAAiB,EAAE,CAAC;gBAE/C,6BAA6B;gBAC7B,gBAAM,CAAC,KAAK,CAAC,OAAO,SAAS,EAAE,QAAQ,CAAC,CAAC;gBACzC,gBAAM,CAAC,KAAK,CAAC,wBAAwB,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,CAAC,CAAC;gBAC7D,gBAAM,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC;gBAEzC,2BAA2B;gBAC3B,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;gBACjD,gBAAM,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC;YACzC,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK;gBACxD,MAAM,SAAS,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAC/C,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC1C,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAC1C,SAAS,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;gBAClD,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAEpC,MAAM,EAAE,GAAG,CAAC,MAAM,SAAS,CAAC,KAAK,EAAE,CAAwB,CAAC;gBAC5D,MAAM,KAAK,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;gBAC1B,MAAM,KAAK,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;gBAE1B,gBAAM,CAAC,SAAS,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YACjC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import should from 'should';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { BitGoAPI } from '@bitgo-beta/sdk-api';\nimport { Iota, TransactionBuilderFactory, TransferTransaction } from '../../src';\nimport assert from 'assert';\nimport { coins, GasTankAccountCoin } from '@bitgo-beta/statics';\nimport * as testData from '../resources/iota';\nimport { TransactionType } from '@bitgo-beta/sdk-core';\n\ndescribe('IOTA:', function () {\n  let bitgo: TestBitGoAPI;\n  let basecoin;\n\n  before(function () {\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'mock' });\n    bitgo.safeRegister('iota', Iota.createInstance);\n    bitgo.safeRegister('tiota', Iota.createInstance);\n    bitgo.initializeTestVars();\n    basecoin = bitgo.coin('tiota');\n  });\n\n  it('should return the right info', function () {\n    const iota = bitgo.coin('iota');\n    const tiota = bitgo.coin('tiota');\n    const iotaStatics = coins.get('iota') as GasTankAccountCoin;\n    const tiotaStatics = coins.get('tiota') as GasTankAccountCoin;\n\n    iota.getChain().should.equal('iota');\n    iota.getFamily().should.equal('iota');\n    iota.getFullName().should.equal('Iota');\n    iota.getBaseFactor().should.equal(1e9);\n\n    tiota.getChain().should.equal('tiota');\n    tiota.getFamily().should.equal('iota');\n    tiota.getFullName().should.equal('Testnet Iota');\n    tiota.getBaseFactor().should.equal(1e9);\n\n    iotaStatics.gasTankLowBalanceAlertFactor.should.equal(80);\n    tiotaStatics.gasTankLowBalanceAlertFactor.should.equal(80);\n    iotaStatics.gasTankMinBalanceRecommendationFactor.should.equal(200);\n    tiotaStatics.gasTankMinBalanceRecommendationFactor.should.equal(200);\n  });\n\n  it('is valid pub', function () {\n    // with 0x prefix\n    basecoin.isValidPub('0x9b4e96086d111500259f9b38680b0509a405c1904da18976455a20c691d3bb07').should.equal(false);\n    // without 0x prefix\n    basecoin.isValidPub('9b4e96086d111500259f9b38680b0509a405c1904da18976455a20c691d3bb07').should.equal(true);\n  });\n\n  describe('Address Validation', () => {\n    let keychains;\n    let commonKeychain;\n\n    before(function () {\n      commonKeychain =\n        '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781';\n      keychains = [\n        {\n          id: '6424c353eaf78d000766e95949868468',\n          source: 'user',\n          type: 'tss',\n          commonKeychain:\n            '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781',\n          encryptedPrv:\n            '{\"iv\":\"cZd5i7L4RxtwrALW2rK7UA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"5zgoH1Bd3Fw=\",\"ct\":\"9vVlnXFRtrM9FVEo+d2chbGHlM9lFZemueBuAs3BIkPo33Fo7jzwwNK/kIWkEyg+NmEBd5IaqAS157nvvvwzzsmMWlQdUz9qbmXNv3pg987cXFR08exS+4uhwP1YNOjJTRvRNcO9ZqHb46d4fmyJ/yC9/susCge7r/EsbaN5C3afv1dzybuq912FwaQElZLYYp5BICudFOMZ9k0UDMfKM/PMDkH7WexoGHr9GKq/bgCH2B39TZZyHKU6Uy47lXep2s6h0DrMwHOrnmiL3DZjOj88Ynvphlzxuo4eOlD2UHia2+nvIaISYs29Pr0DAvREutchvcBpExj1kWWPv7hQYrv8F0NAdatsbWl3w+xKyfiMKo1USlrwyJviypGtQtXOJyw0XPN0rv2+L5lW8BbjpzHfYYN13fJTedlGTFhhkzVtbbPAKE02kx7zCJcjYaiexdSTsrDLScYNT9/Jhdt27KpsooehwVohLfSKz4vbFfRu2MPZw3/+c/hfiJNgtz6esWbnxGrcE8U2IwPYCaK+Ghk4DcqWNIni59RI5B5kAsQOToII40qPN510uTgxBSPO7q7MHgkxdd4CqBq+ojr9j0P7oao8E5Y+CBDJrojDoCh1oCCDW9vo2dXlVcD8SIbw7U/9AfvEbA4xyE/5md1M7CIwLnWs2Ynv0YtaKoqhdS9x6FmHlMDhN/DKHinrwmowtrTT82fOkpO5g9saSmgU7Qy3gLt8t+VwdEyeFeQUKRSyci8qgqXQaZIg4+aXgaSOnlCFMtmB8ekYxEhTY5uzRfrNgS4s1QeqFBpNtUF+Ydi297pbVXnJoXAN+SVWd80GCx+yI2dpVC89k3rOWK9WeyqlnzuLJWp2RIOB9cdW8GFv/fN+QAJpYeVxOE4+nZDsKnsj8nKcg9t4Dlx1G6gLM1/Vq9YxNLbuzuRC0asUYvdMnoMvszmpm++TxndYisgNYscpZSoz7wvcazJNEPfhPVjEkd6tUUuN4GM35H0DmKCUQNT+a6B6hmHlTZvjxiyGAg5bY59hdjvJ+22QduazlEEC6LI3HrA7uK0TpplWzS1tCIFvTMUhj65DEZmNJ2+ZY9bQ4vsMf+DRR3OOG4t+DMlNfjOd3zNv3QoY95BjfWpryFwPzDq7bCP67JDsoj7j2TY5FRSrRkD77H0Ewlux2cWfjRTwcMHcdQxxuV0OP0aNjGDjybFN\"}',\n        },\n        {\n          id: '6424c353eaf78d000766e96137d4404b',\n          source: 'backup',\n          type: 'tss',\n          commonKeychain:\n            '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781',\n          encryptedPrv:\n            '{\"iv\":\"vi0dPef/Rx7kG/pRySQi6Q==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"9efhQsiEvVs=\",\"ct\":\"Gw6atvf6gxKzsjtl3xseipO3rAxp1mAz7Yu1ihFsi5/lf2vMZegApgZx+pyILFS9KKLHbNF3U6WgSYdrr2t4vzdLsXkH1WIxfHS+cd2C5N59yADZDnPJBT6pv/IRvaYelP0Ck3nIYQ2hSMm8op+VOWC/SzHeh7slYDqwEHTGan0Wigfvk1yRd7CCJTaEAomnc/4eFi2NY3X3gt/3opy9IAgknnwUFohn96EWpEQ0F6pbzH/Z8VF6gF+DUcrrByAxExUPnHQZiFk3YHU/vVV4FxBU/mVAE8xBsBn5ul5e5SUMPfc7TBuJWv4BByTNg9xDShF/91Yx2nbfUm5d9QmM8lpKgzzQvcK8POAPk87gRCuKnsGh5vNS0UppkHc+ocfzRQlGA6jze7QyyQO0rMj5Ly8kWjwk2vISvKYHYS1NR7VU549UIXo7NXjatunKSc3+IreoRUHIshiaLg6hl+pxCCuc0qQ43V0mdIfCjTN8gkGWLNk8R7tAGPz9jyapQPcPEGHgEz0ATIi6yMNWCsibS2eLiE1uVEJONoM4lk6FPl3Q2CHbW2MeEbqjY8hbaw18mNb2xSBH/Fwpiial+Tvi2imqgnCO4ZpO9bllKftZPcQy0stN+eGBlb5ufyflKkDSiChHYroGjEpmiFicdde48cJszF52uKNnf1q67fA9/S2FAHQab3EXojxH2Gbk+kkV2h/TYKFFZSWC3vi4e8mO+vjMUcR0AdsgPFyEIz0SCGuba3CnTLNdEuZwsauAeHkx2vUTnRgJPVgNeeuXmsVG76Sy2ggJHuals0Hj8U2Xda0qO1RuFfoCWfss9wn6HGRwPPkhSB/8oNguAqmRVGKkd8Zwt3IvrTd9fk0/rFFDJKGz7WyNHkYgUmNiGcItD12v0jx7FZ52EJzl3Av1RyJUQK18+8EYPh3SGiU9dt7VX0aF0uo6JouKhOeldUvMP+AugQz8fUclwTQsbboVg27Yxo0DyATVwThW5a56R6Qf5ZiQJluFuzs5y98rq0S5q046lE6o3vVmJpEdwjeSCJoET5CL4nTgkXyWvhm4eB8u/e66l3o0qbaSx8q9YYmT9EpRcl5TP4ThLBKETYdzVvg4exjQfektMatk5EyUpEIhZPXh5vXpJZesdfO9LJ8zTaHBsBjDPU7cdNgQMbebpataRi8A0el2/IJXl+E+olgAz5zC4i2O1Q==\"}',\n        },\n        {\n          id: '6424c353eaf78d000766e9510b125fba',\n          source: 'bitgo',\n          type: 'tss',\n          commonKeychain:\n            '19bdfe2a4b498a05511381235a8892d54267807c4a3f654e310b938b8b424ff4adedbe92f4c146de641c67508a961324c8504cdf8e0c0acbb68d6104ccccd781',\n          verifiedVssProof: true,\n          isBitGo: true,\n        },\n      ];\n    });\n\n    it('should return true when validating a well formatted address prefixed with 0x', async function () {\n      const address = '0xf941ae3cbe5645dccc15da8346b533f7f91f202089a5521653c062b2ff10b304';\n      basecoin.isValidAddress(address).should.equal(true);\n    });\n\n    it('should return false when validating an old address', async function () {\n      const address = '0x2959bfc3fdb7dc23fed8deba2fafb70f3e606a59';\n      basecoin.isValidAddress(address).should.equal(false);\n    });\n\n    it('should return false when validating an incorrectly formatted', async function () {\n      const address = 'wrongaddress';\n      basecoin.isValidAddress(address).should.equal(false);\n    });\n\n    it('should return true for isWalletAddress with valid address for index 4', async function () {\n      const newAddress = '0x3f4b2a95d9b696989814f02f899cee491b4d600b1b918e979caec307af4b8dfc';\n      const index = 4;\n\n      const params = { commonKeychain, address: newAddress, index, keychains };\n      (await basecoin.isWalletAddress(params)).should.equal(true);\n    });\n\n    it('should throw error for isWalletAddress when keychains is missing', async function () {\n      const address = '0x2959bfc3fdb7dc23fed8deba2fafb70f3e606a59';\n      const index = 0;\n\n      const params = { commonKeychain, address, index };\n      await assert.rejects(async () => basecoin.isWalletAddress(params));\n    });\n\n    it('should throw error for isWalletAddress when new address is invalid', async function () {\n      const wrongAddress = 'badAddress';\n      const index = 0;\n\n      const params = { commonKeychain, address: wrongAddress, index };\n      await assert.rejects(async () => basecoin.isWalletAddress(params), {\n        message: `invalid address: ${wrongAddress}`,\n      });\n    });\n  });\n\n  describe('Transaction Methods', () => {\n    let factory: TransactionBuilderFactory;\n\n    before(function () {\n      factory = new TransactionBuilderFactory(coins.get('tiota'));\n    });\n\n    describe('explainTransaction', () => {\n      it('should throw error for missing txHex', async function () {\n        await assert.rejects(\n          async () => await basecoin.explainTransaction({ txHex: '' }),\n          /missing required tx prebuild property txHex/\n        );\n      });\n\n      it('should throw error for invalid transaction', async function () {\n        await assert.rejects(\n          async () => await basecoin.explainTransaction({ txHex: 'invalidTxHex' }),\n          /Failed to rebuild transaction/\n        );\n      });\n\n      it('should call explainTransaction on transaction object', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const explanation = tx.explainTransaction();\n\n        explanation.should.have.property('id');\n        explanation.should.have.property('outputs');\n        explanation.should.have.property('outputAmount');\n        explanation.should.have.property('fee');\n        explanation.should.have.property('type');\n        explanation.type.should.equal(TransactionType.Send);\n        explanation.fee.fee.should.equal(testData.GAS_BUDGET.toString());\n      });\n    });\n\n    describe('verifyTransaction', () => {\n      it('should throw error for missing txHex', async function () {\n        await assert.rejects(\n          async () =>\n            await basecoin.verifyTransaction({\n              txPrebuild: {},\n              txParams: { recipients: testData.recipients },\n            }),\n          /missing required tx prebuild property txHex/\n        );\n      });\n\n      it('should verify transaction with matching recipients using JSON', async function () {\n        // Build transaction and get JSON instead of broadcast format to avoid parsing issues\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');\n        should.equal(\n          await basecoin.verifyTransaction({\n            txPrebuild: { txHex: txHex },\n            txParams: { recipients: testData.recipients },\n          }),\n          true\n        );\n      });\n\n      it('should verify transaction with recipients containing extra fields', async function () {\n        // The verification should only compare address, amount, and tokenName\n        // Extra fields should be ignored\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');\n\n        // Add extra fields to recipients that should be ignored during comparison\n        const recipientsWithExtraFields = testData.recipients.map((r) => ({\n          ...r,\n          extraField: 'should be ignored',\n          anotherField: 123,\n        }));\n\n        should.equal(\n          await basecoin.verifyTransaction({\n            txPrebuild: { txHex: txHex },\n            txParams: { recipients: recipientsWithExtraFields },\n          }),\n          true\n        );\n      });\n\n      it('should detect mismatched recipients', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');\n\n        // Create mismatched recipients with different addresses/amounts\n        const mismatchedRecipients = [\n          {\n            address: testData.addresses.validAddresses[2], // Different address\n            amount: '9999', // Different amount\n          },\n        ];\n\n        await assert.rejects(\n          async () =>\n            await basecoin.verifyTransaction({\n              txPrebuild: { txHex: txHex },\n              txParams: { recipients: mismatchedRecipients },\n            }),\n          /Tx recipients does not match with expected txParams recipients/\n        );\n      });\n\n      it('should verify transaction without recipients parameter', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const txHex = Buffer.from(await tx.toBroadcastFormat(), 'base64').toString('hex');\n\n        should.equal(\n          await basecoin.verifyTransaction({\n            txPrebuild: { txHex: txHex },\n            txParams: {},\n          }),\n          true\n        );\n      });\n    });\n\n    describe('parseTransaction', () => {\n      it('should throw error for invalid transaction', async function () {\n        await assert.rejects(\n          async () => await basecoin.parseTransaction({ txHex: 'invalidTxHex' }),\n          /Failed to rebuild transaction/\n        );\n      });\n\n      it('should parse transaction using JSON format', async function () {\n        // Build a transaction\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n\n        // Get the transaction explanation (which is what parseTransaction returns internally)\n        const explanation = tx.explainTransaction();\n\n        // Verify the parsed data structure\n        explanation.should.have.property('id');\n        explanation.should.have.property('outputs');\n        explanation.should.have.property('outputAmount');\n        explanation.should.have.property('fee');\n\n        // Verify outputs match recipients\n        explanation.outputs.length.should.equal(testData.recipients.length);\n        explanation.outputs.forEach((output, index) => {\n          output.address.should.equal(testData.recipients[index].address);\n          output.amount.should.equal(testData.recipients[index].amount);\n        });\n\n        // Verify output amount\n        const totalAmount = testData.recipients.reduce((sum, r) => sum + Number(r.amount), 0);\n        explanation.outputAmount.should.equal(totalAmount.toString());\n\n        // Verify fee\n        explanation.fee.fee.should.equal(testData.GAS_BUDGET.toString());\n      });\n\n      it('should parse transaction with single recipient', async function () {\n        const singleRecipient = [testData.recipients[0]];\n\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(singleRecipient);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const explanation = tx.explainTransaction();\n\n        explanation.outputs.length.should.equal(1);\n        explanation.outputs[0].address.should.equal(singleRecipient[0].address);\n        explanation.outputs[0].amount.should.equal(singleRecipient[0].amount);\n        explanation.outputAmount.should.equal(singleRecipient[0].amount);\n      });\n\n      it('should parse transaction with multiple recipients', async function () {\n        const multipleRecipients = [\n          { address: testData.addresses.validAddresses[0], amount: '1000' },\n          { address: testData.addresses.validAddresses[1], amount: '2000' },\n          { address: testData.addresses.validAddresses[2], amount: '3000' },\n        ];\n\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(multipleRecipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const explanation = tx.explainTransaction();\n\n        explanation.outputs.length.should.equal(3);\n        explanation.outputAmount.should.equal('6000');\n      });\n    });\n\n    describe('getSignablePayload', () => {\n      it('should get signable payload from transaction directly', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const signablePayload = tx.signablePayload;\n\n        signablePayload.should.be.instanceOf(Buffer);\n        signablePayload.length.should.equal(32); // Blake2b hash is 32 bytes\n      });\n\n      it('should throw error for invalid transaction', async function () {\n        await assert.rejects(\n          async () => await basecoin.getSignablePayload('invalidTxBase64'),\n          /Failed to rebuild transaction/\n        );\n      });\n\n      it('should generate consistent signable payload for identical transactions', async function () {\n        // Build first transaction\n        const txBuilder1 = factory.getTransferBuilder();\n        txBuilder1.sender(testData.sender.address);\n        txBuilder1.recipients(testData.recipients);\n        txBuilder1.paymentObjects(testData.paymentObjects);\n        txBuilder1.gasData(testData.gasData);\n\n        const tx1 = (await txBuilder1.build()) as TransferTransaction;\n        const payload1 = tx1.signablePayload;\n\n        // Build second identical transaction\n        const txBuilder2 = factory.getTransferBuilder();\n        txBuilder2.sender(testData.sender.address);\n        txBuilder2.recipients(testData.recipients);\n        txBuilder2.paymentObjects(testData.paymentObjects);\n        txBuilder2.gasData(testData.gasData);\n\n        const tx2 = (await txBuilder2.build()) as TransferTransaction;\n        const payload2 = tx2.signablePayload;\n\n        // Payloads should be identical for identical transactions\n        payload1.toString('hex').should.equal(payload2.toString('hex'));\n      });\n\n      it('should generate different signable payloads for different transactions', async function () {\n        // Build first transaction\n        const txBuilder1 = factory.getTransferBuilder();\n        txBuilder1.sender(testData.sender.address);\n        txBuilder1.recipients(testData.recipients);\n        txBuilder1.paymentObjects(testData.paymentObjects);\n        txBuilder1.gasData(testData.gasData);\n\n        const tx1 = (await txBuilder1.build()) as TransferTransaction;\n        const payload1 = tx1.signablePayload;\n\n        // Build second transaction with different recipient\n        const differentRecipients = [{ address: testData.addresses.validAddresses[0], amount: '5000' }];\n        const txBuilder2 = factory.getTransferBuilder();\n        txBuilder2.sender(testData.sender.address);\n        txBuilder2.recipients(differentRecipients);\n        txBuilder2.paymentObjects(testData.paymentObjects);\n        txBuilder2.gasData(testData.gasData);\n\n        const tx2 = (await txBuilder2.build()) as TransferTransaction;\n        const payload2 = tx2.signablePayload;\n\n        // Payloads should be different for different transactions\n        payload1.toString('hex').should.not.equal(payload2.toString('hex'));\n      });\n\n      it('should throw error when getting payload from simulate transaction', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        // Don't set gasData - this will be a simulate transaction\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        should.equal(tx.isSimulateTx, true);\n\n        should(() => tx.signablePayload).throwError('Cannot sign a simulate tx');\n      });\n    });\n\n    describe('setCoinSpecificFieldsInIntent', () => {\n      it('should set unspents in intent', function () {\n        const intent = {} as any;\n        const params = {\n          unspents: [\n            { objectId: '0x123', version: '1', digest: 'abc' },\n            { objectId: '0x456', version: '2', digest: 'def' },\n          ],\n        } as any;\n\n        basecoin.setCoinSpecificFieldsInIntent(intent, params);\n\n        intent.should.have.property('unspents');\n        intent.unspents.should.deepEqual(params.unspents);\n      });\n\n      it('should handle empty unspents', function () {\n        const intent = {} as any;\n        const params = { unspents: [] } as any;\n\n        basecoin.setCoinSpecificFieldsInIntent(intent, params);\n\n        intent.should.have.property('unspents');\n        intent.unspents.should.deepEqual([]);\n      });\n\n      it('should handle undefined unspents', function () {\n        const intent = {} as any;\n        const params = {} as any;\n\n        basecoin.setCoinSpecificFieldsInIntent(intent, params);\n\n        intent.should.have.property('unspents');\n        (intent.unspents === undefined).should.be.true();\n      });\n    });\n\n    describe('Transaction with Gas Sponsor', () => {\n      it('should build transaction with gas sponsor', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n        txBuilder.gasSponsor(testData.gasSponsor.address);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n\n        should.equal(tx.sender, testData.sender.address);\n        should.equal(tx.gasSponsor, testData.gasSponsor.address);\n        should.notEqual(tx.sender, tx.gasSponsor);\n        should.equal(tx.type, TransactionType.Send);\n      });\n\n      it('should handle transaction where sender and gas sponsor are same', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n        txBuilder.gasSponsor(testData.sender.address); // Same as sender\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n\n        should.equal(tx.sender, testData.sender.address);\n        should.equal(tx.gasSponsor, testData.sender.address);\n      });\n    });\n\n    describe('Transaction ID Consistency', () => {\n      it('should generate same ID for identical transactions', async function () {\n        const txBuilder1 = factory.getTransferBuilder();\n        txBuilder1.sender(testData.sender.address);\n        txBuilder1.recipients(testData.recipients);\n        txBuilder1.paymentObjects(testData.paymentObjects);\n        txBuilder1.gasData(testData.gasData);\n\n        const tx1 = (await txBuilder1.build()) as TransferTransaction;\n\n        const txBuilder2 = factory.getTransferBuilder();\n        txBuilder2.sender(testData.sender.address);\n        txBuilder2.recipients(testData.recipients);\n        txBuilder2.paymentObjects(testData.paymentObjects);\n        txBuilder2.gasData(testData.gasData);\n\n        const tx2 = (await txBuilder2.build()) as TransferTransaction;\n\n        should.equal(tx1.id, tx2.id);\n      });\n\n      it('should generate different IDs for different transactions', async function () {\n        const txBuilder1 = factory.getTransferBuilder();\n        txBuilder1.sender(testData.sender.address);\n        txBuilder1.recipients(testData.recipients);\n        txBuilder1.paymentObjects(testData.paymentObjects);\n        txBuilder1.gasData(testData.gasData);\n\n        const tx1 = (await txBuilder1.build()) as TransferTransaction;\n\n        const differentRecipients = [{ address: testData.addresses.validAddresses[0], amount: '9999' }];\n        const txBuilder2 = factory.getTransferBuilder();\n        txBuilder2.sender(testData.sender.address);\n        txBuilder2.recipients(differentRecipients);\n        txBuilder2.paymentObjects(testData.paymentObjects);\n        txBuilder2.gasData(testData.gasData);\n\n        const tx2 = (await txBuilder2.build()) as TransferTransaction;\n\n        should.notEqual(tx1.id, tx2.id);\n      });\n    });\n\n    describe('Gas Configuration Edge Cases', () => {\n      it('should handle minimum gas values', async function () {\n        const minGasData = {\n          gasBudget: 1000,\n          gasPrice: 100,\n          gasPaymentObjects: [testData.gasPaymentObjects[0]],\n        };\n\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(minGasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n\n        should.equal(tx.gasBudget, 1000);\n        should.equal(tx.gasPrice, 100);\n        should.equal(tx.gasPaymentObjects?.length, 1);\n      });\n\n      it('should handle large gas values', async function () {\n        const largeGasData = {\n          gasBudget: 50000000000, // 50 billion\n          gasPrice: 100000,\n          gasPaymentObjects: testData.gasPaymentObjects,\n        };\n\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(largeGasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n\n        should.equal(tx.gasBudget, 50000000000);\n        should.equal(tx.gasPrice, 100000);\n      });\n\n      it('should return gas fee from transaction', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const fee = tx.getFee();\n\n        should.exist(fee);\n        should.equal(fee, testData.GAS_BUDGET.toString());\n      });\n    });\n\n    describe('Transaction State Management', () => {\n      it('should track simulate mode correctly', async function () {\n        // Build without gas data - should be simulate mode\n        const simulateBuilder = factory.getTransferBuilder();\n        simulateBuilder.sender(testData.sender.address);\n        simulateBuilder.recipients(testData.recipients);\n        simulateBuilder.paymentObjects(testData.paymentObjects);\n\n        const simulateTx = (await simulateBuilder.build()) as TransferTransaction;\n        should.equal(simulateTx.isSimulateTx, true);\n\n        // Build with gas data - should not be simulate mode\n        const realBuilder = factory.getTransferBuilder();\n        realBuilder.sender(testData.sender.address);\n        realBuilder.recipients(testData.recipients);\n        realBuilder.paymentObjects(testData.paymentObjects);\n        realBuilder.gasData(testData.gasData);\n\n        const realTx = (await realBuilder.build()) as TransferTransaction;\n        should.equal(realTx.isSimulateTx, false);\n      });\n\n      it('should handle canSign based on simulate mode', async function () {\n        // Simulate transaction cannot be signed\n        const simulateBuilder = factory.getTransferBuilder();\n        simulateBuilder.sender(testData.sender.address);\n        simulateBuilder.recipients(testData.recipients);\n        simulateBuilder.paymentObjects(testData.paymentObjects);\n\n        const simulateTx = (await simulateBuilder.build()) as TransferTransaction;\n        should.equal(simulateTx.canSign({} as any), false);\n\n        // Real transaction can be signed\n        const realBuilder = factory.getTransferBuilder();\n        realBuilder.sender(testData.sender.address);\n        realBuilder.recipients(testData.recipients);\n        realBuilder.paymentObjects(testData.paymentObjects);\n        realBuilder.gasData(testData.gasData);\n\n        const realTx = (await realBuilder.build()) as TransferTransaction;\n        should.equal(realTx.canSign({} as any), true);\n      });\n\n      it('should handle transaction type correctly', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n\n        should.equal(tx.type, TransactionType.Send);\n      });\n    });\n\n    describe('Transaction Serialization Formats', () => {\n      it('should serialize to consistent broadcast format', async function () {\n        const txBuilder1 = factory.getTransferBuilder();\n        txBuilder1.sender(testData.sender.address);\n        txBuilder1.recipients(testData.recipients);\n        txBuilder1.paymentObjects(testData.paymentObjects);\n        txBuilder1.gasData(testData.gasData);\n\n        const tx1 = (await txBuilder1.build()) as TransferTransaction;\n        const broadcast1 = await tx1.toBroadcastFormat();\n\n        const txBuilder2 = factory.getTransferBuilder();\n        txBuilder2.sender(testData.sender.address);\n        txBuilder2.recipients(testData.recipients);\n        txBuilder2.paymentObjects(testData.paymentObjects);\n        txBuilder2.gasData(testData.gasData);\n\n        const tx2 = (await txBuilder2.build()) as TransferTransaction;\n        const broadcast2 = await tx2.toBroadcastFormat();\n\n        should.equal(broadcast1, broadcast2);\n      });\n\n      it('should produce valid base64 broadcast format', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const broadcast = await tx.toBroadcastFormat();\n\n        // Check if it's valid base64\n        should.equal(typeof broadcast, 'string');\n        should.equal(/^[A-Za-z0-9+/]*={0,2}$/.test(broadcast), true);\n        should.equal(broadcast.length > 0, true);\n\n        // Should be able to decode\n        const decoded = Buffer.from(broadcast, 'base64');\n        should.equal(decoded.length > 0, true);\n      });\n\n      it('should maintain JSON serialization consistency', async function () {\n        const txBuilder = factory.getTransferBuilder();\n        txBuilder.sender(testData.sender.address);\n        txBuilder.recipients(testData.recipients);\n        txBuilder.paymentObjects(testData.paymentObjects);\n        txBuilder.gasData(testData.gasData);\n\n        const tx = (await txBuilder.build()) as TransferTransaction;\n        const json1 = tx.toJson();\n        const json2 = tx.toJson();\n\n        should.deepEqual(json1, json2);\n      });\n    });\n  });\n});\n"]}