israeli-bank-scrapers 6.1.1 → 6.1.2

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 (79) hide show
  1. package/lib/assertNever.js +7 -5
  2. package/lib/constants.js +16 -13
  3. package/lib/definitions.js +113 -109
  4. package/lib/helpers/browser.js +13 -9
  5. package/lib/helpers/dates.js +19 -18
  6. package/lib/helpers/debug.js +9 -9
  7. package/lib/helpers/elements-interactions.js +82 -78
  8. package/lib/helpers/fetch.js +85 -82
  9. package/lib/helpers/navigation.js +28 -24
  10. package/lib/helpers/storage.js +11 -10
  11. package/lib/helpers/transactions.js +32 -33
  12. package/lib/helpers/waiting.js +42 -45
  13. package/lib/index.js +82 -15
  14. package/lib/scrapers/amex.js +13 -11
  15. package/lib/scrapers/amex.test.d.ts +1 -0
  16. package/lib/scrapers/amex.test.js +49 -0
  17. package/lib/scrapers/base-beinleumi-group.js +239 -233
  18. package/lib/scrapers/base-isracard-amex.js +273 -273
  19. package/lib/scrapers/base-scraper-with-browser.js +263 -241
  20. package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
  21. package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
  22. package/lib/scrapers/base-scraper.js +82 -82
  23. package/lib/scrapers/behatsdaa.js +103 -98
  24. package/lib/scrapers/behatsdaa.test.d.ts +1 -0
  25. package/lib/scrapers/behatsdaa.test.js +46 -0
  26. package/lib/scrapers/beinleumi.js +13 -11
  27. package/lib/scrapers/beinleumi.test.d.ts +1 -0
  28. package/lib/scrapers/beinleumi.test.js +47 -0
  29. package/lib/scrapers/beyahad-bishvilha.js +132 -132
  30. package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
  31. package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
  32. package/lib/scrapers/discount.js +101 -97
  33. package/lib/scrapers/discount.test.d.ts +1 -0
  34. package/lib/scrapers/discount.test.js +49 -0
  35. package/lib/scrapers/errors.js +25 -22
  36. package/lib/scrapers/factory.js +67 -66
  37. package/lib/scrapers/factory.test.d.ts +1 -0
  38. package/lib/scrapers/factory.test.js +19 -0
  39. package/lib/scrapers/hapoalim.js +175 -162
  40. package/lib/scrapers/hapoalim.test.d.ts +1 -0
  41. package/lib/scrapers/hapoalim.test.js +47 -0
  42. package/lib/scrapers/interface.js +5 -2
  43. package/lib/scrapers/isracard.js +13 -11
  44. package/lib/scrapers/isracard.test.d.ts +1 -0
  45. package/lib/scrapers/isracard.test.js +49 -0
  46. package/lib/scrapers/leumi.js +170 -167
  47. package/lib/scrapers/leumi.test.d.ts +1 -0
  48. package/lib/scrapers/leumi.test.js +47 -0
  49. package/lib/scrapers/massad.js +13 -11
  50. package/lib/scrapers/max.js +261 -261
  51. package/lib/scrapers/max.test.d.ts +1 -0
  52. package/lib/scrapers/max.test.js +65 -0
  53. package/lib/scrapers/mercantile.js +16 -14
  54. package/lib/scrapers/mercantile.test.d.ts +1 -0
  55. package/lib/scrapers/mercantile.test.js +45 -0
  56. package/lib/scrapers/mizrahi.js +154 -158
  57. package/lib/scrapers/mizrahi.test.d.ts +1 -0
  58. package/lib/scrapers/mizrahi.test.js +53 -0
  59. package/lib/scrapers/one-zero-queries.js +7 -4
  60. package/lib/scrapers/one-zero.js +221 -176
  61. package/lib/scrapers/one-zero.test.d.ts +1 -0
  62. package/lib/scrapers/one-zero.test.js +51 -0
  63. package/lib/scrapers/otsar-hahayal.js +13 -11
  64. package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
  65. package/lib/scrapers/otsar-hahayal.test.js +47 -0
  66. package/lib/scrapers/pagi.js +13 -11
  67. package/lib/scrapers/pagi.test.d.ts +1 -0
  68. package/lib/scrapers/pagi.test.js +47 -0
  69. package/lib/scrapers/union-bank.js +173 -172
  70. package/lib/scrapers/union-bank.test.d.ts +1 -0
  71. package/lib/scrapers/union-bank.test.js +47 -0
  72. package/lib/scrapers/visa-cal.js +250 -254
  73. package/lib/scrapers/visa-cal.test.d.ts +1 -0
  74. package/lib/scrapers/visa-cal.test.js +49 -0
  75. package/lib/scrapers/yahav.js +206 -190
  76. package/lib/scrapers/yahav.test.d.ts +1 -0
  77. package/lib/scrapers/yahav.test.js +49 -0
  78. package/lib/transactions.js +16 -13
  79. package/package.json +8 -3
@@ -1,281 +1,277 @@
1
1
  "use strict";
2
- var __importDefault = (this && this.__importDefault) || function (mod) {
3
- return (mod && mod.__esModule) ? mod : { "default": mod };
4
- };
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- const moment_1 = __importDefault(require("moment"));
7
- const debug_1 = require("../helpers/debug");
8
- const elements_interactions_1 = require("../helpers/elements-interactions");
9
- const fetch_1 = require("../helpers/fetch");
10
- const navigation_1 = require("../helpers/navigation");
11
- const storage_1 = require("../helpers/storage");
12
- const transactions_1 = require("../helpers/transactions");
13
- const waiting_1 = require("../helpers/waiting");
14
- const transactions_2 = require("../transactions");
15
- const base_scraper_with_browser_1 = require("./base-scraper-with-browser");
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _moment = _interopRequireDefault(require("moment"));
8
+ var _debug = require("../helpers/debug");
9
+ var _elementsInteractions = require("../helpers/elements-interactions");
10
+ var _fetch = require("../helpers/fetch");
11
+ var _navigation = require("../helpers/navigation");
12
+ var _storage = require("../helpers/storage");
13
+ var _transactions = require("../helpers/transactions");
14
+ var _waiting = require("../helpers/waiting");
15
+ var _transactions2 = require("../transactions");
16
+ var _baseScraperWithBrowser = require("./base-scraper-with-browser");
17
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
16
18
  const LOGIN_URL = 'https://www.cal-online.co.il/';
17
19
  const TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';
18
20
  const PENDING_TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/approvals/getClearanceRequests';
19
21
  const InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';
20
- const debug = (0, debug_1.getDebug)('visa-cal');
21
- var TrnTypeCode;
22
- (function (TrnTypeCode) {
23
- TrnTypeCode["regular"] = "5";
24
- TrnTypeCode["credit"] = "6";
25
- TrnTypeCode["installments"] = "8";
26
- TrnTypeCode["standingOrder"] = "9";
27
- })(TrnTypeCode || (TrnTypeCode = {}));
22
+ const debug = (0, _debug.getDebug)('visa-cal');
23
+ var TrnTypeCode = /*#__PURE__*/function (TrnTypeCode) {
24
+ TrnTypeCode["regular"] = "5";
25
+ TrnTypeCode["credit"] = "6";
26
+ TrnTypeCode["installments"] = "8";
27
+ TrnTypeCode["standingOrder"] = "9";
28
+ return TrnTypeCode;
29
+ }(TrnTypeCode || {});
28
30
  function isPending(transaction) {
29
- return transaction.debCrdDate === undefined; // an arbitrary field that only appears in a completed transaction
31
+ return transaction.debCrdDate === undefined; // an arbitrary field that only appears in a completed transaction
30
32
  }
31
33
  function isCardTransactionDetails(result) {
32
- return result.result !== undefined;
34
+ return result.result !== undefined;
33
35
  }
34
36
  function isCardPendingTransactionDetails(result) {
35
- return result.result !== undefined;
37
+ return result.result !== undefined;
36
38
  }
37
39
  async function getLoginFrame(page) {
38
- let frame = null;
39
- debug('wait until login frame found');
40
- await (0, waiting_1.waitUntil)(() => {
41
- frame = page.frames().find(f => f.url().includes('connect')) || null;
42
- return Promise.resolve(!!frame);
43
- }, 'wait for iframe with login form', 10000, 1000);
44
- if (!frame) {
45
- debug('failed to find login frame for 10 seconds');
46
- throw new Error('failed to extract login iframe');
47
- }
48
- return frame;
40
+ let frame = null;
41
+ debug('wait until login frame found');
42
+ await (0, _waiting.waitUntil)(() => {
43
+ frame = page.frames().find(f => f.url().includes('connect')) || null;
44
+ return Promise.resolve(!!frame);
45
+ }, 'wait for iframe with login form', 10000, 1000);
46
+ if (!frame) {
47
+ debug('failed to find login frame for 10 seconds');
48
+ throw new Error('failed to extract login iframe');
49
+ }
50
+ return frame;
49
51
  }
50
52
  async function hasInvalidPasswordError(page) {
51
- const frame = await getLoginFrame(page);
52
- const errorFound = await (0, elements_interactions_1.elementPresentOnPage)(frame, 'div.general-error > div');
53
- const errorMessage = errorFound
54
- ? await (0, elements_interactions_1.pageEval)(frame, 'div.general-error > div', '', item => {
55
- return item.innerText;
56
- })
57
- : '';
58
- return errorMessage === InvalidPasswordMessage;
53
+ const frame = await getLoginFrame(page);
54
+ const errorFound = await (0, _elementsInteractions.elementPresentOnPage)(frame, 'div.general-error > div');
55
+ const errorMessage = errorFound ? await (0, _elementsInteractions.pageEval)(frame, 'div.general-error > div', '', item => {
56
+ return item.innerText;
57
+ }) : '';
58
+ return errorMessage === InvalidPasswordMessage;
59
59
  }
60
60
  async function hasChangePasswordForm(page) {
61
- const frame = await getLoginFrame(page);
62
- const errorFound = await (0, elements_interactions_1.elementPresentOnPage)(frame, '.change-password-subtitle');
63
- return errorFound;
61
+ const frame = await getLoginFrame(page);
62
+ const errorFound = await (0, _elementsInteractions.elementPresentOnPage)(frame, '.change-password-subtitle');
63
+ return errorFound;
64
64
  }
65
65
  function getPossibleLoginResults() {
66
- debug('return possible login results');
67
- const urls = {
68
- [base_scraper_with_browser_1.LoginResults.Success]: [/dashboard/i],
69
- [base_scraper_with_browser_1.LoginResults.InvalidPassword]: [
70
- async (options) => {
71
- const page = options?.page;
72
- if (!page) {
73
- return false;
74
- }
75
- return hasInvalidPasswordError(page);
76
- },
77
- ],
78
- // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario
79
- [base_scraper_with_browser_1.LoginResults.ChangePassword]: [
80
- async (options) => {
81
- const page = options?.page;
82
- if (!page) {
83
- return false;
84
- }
85
- return hasChangePasswordForm(page);
86
- },
87
- ],
88
- };
89
- return urls;
66
+ debug('return possible login results');
67
+ const urls = {
68
+ [_baseScraperWithBrowser.LoginResults.Success]: [/dashboard/i],
69
+ [_baseScraperWithBrowser.LoginResults.InvalidPassword]: [async options => {
70
+ const page = options?.page;
71
+ if (!page) {
72
+ return false;
73
+ }
74
+ return hasInvalidPasswordError(page);
75
+ }],
76
+ // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario
77
+ [_baseScraperWithBrowser.LoginResults.ChangePassword]: [async options => {
78
+ const page = options?.page;
79
+ if (!page) {
80
+ return false;
81
+ }
82
+ return hasChangePasswordForm(page);
83
+ }]
84
+ };
85
+ return urls;
90
86
  }
91
87
  function createLoginFields(credentials) {
92
- debug('create login fields for username and password');
93
- return [
94
- { selector: '[formcontrolname="userName"]', value: credentials.username },
95
- { selector: '[formcontrolname="password"]', value: credentials.password },
96
- ];
88
+ debug('create login fields for username and password');
89
+ return [{
90
+ selector: '[formcontrolname="userName"]',
91
+ value: credentials.username
92
+ }, {
93
+ selector: '[formcontrolname="password"]',
94
+ value: credentials.password
95
+ }];
97
96
  }
98
97
  function convertParsedDataToTransactions(data, pendingData) {
99
- const pendingTransactions = pendingData?.result
100
- ? pendingData.result.cardsList.flatMap(card => card.authDetalisList)
101
- : [];
102
- const bankAccounts = data.flatMap(monthData => monthData.result.bankAccounts);
103
- const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);
104
- const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);
105
- const completedTransactions = [...regularDebitDays, ...immediateDebitDays].flatMap(debitDate => debitDate.transactions);
106
- const all = [...pendingTransactions, ...completedTransactions];
107
- return all.map(transaction => {
108
- const numOfPayments = isPending(transaction) ? transaction.numberOfPayments : transaction.numOfPayments;
109
- const installments = numOfPayments
110
- ? {
111
- number: isPending(transaction) ? 1 : transaction.curPaymentNum,
112
- total: numOfPayments,
113
- }
114
- : undefined;
115
- const date = (0, moment_1.default)(transaction.trnPurchaseDate);
116
- let chargedAmount = isPending(transaction) ? transaction.trnAmt * -1 : transaction.amtBeforeConvAndIndex * -1;
117
- let originalAmount = transaction.trnAmt * -1;
118
- if (transaction.trnTypeCode === TrnTypeCode.credit) {
119
- chargedAmount = isPending(transaction) ? transaction.trnAmt : transaction.amtBeforeConvAndIndex;
120
- originalAmount = transaction.trnAmt;
121
- }
122
- const result = {
123
- identifier: !isPending(transaction) ? transaction.trnIntId : undefined,
124
- type: [TrnTypeCode.regular, TrnTypeCode.standingOrder].includes(transaction.trnTypeCode)
125
- ? transactions_2.TransactionTypes.Normal
126
- : transactions_2.TransactionTypes.Installments,
127
- status: isPending(transaction) ? transactions_2.TransactionStatuses.Pending : transactions_2.TransactionStatuses.Completed,
128
- date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),
129
- processedDate: isPending(transaction) ? date.toISOString() : new Date(transaction.debCrdDate).toISOString(),
130
- originalAmount,
131
- originalCurrency: transaction.trnCurrencySymbol,
132
- chargedAmount,
133
- chargedCurrency: !isPending(transaction) ? transaction.debCrdCurrencySymbol : undefined,
134
- description: transaction.merchantName,
135
- memo: transaction.transTypeCommentDetails.toString(),
136
- category: transaction.branchCodeDesc,
137
- };
138
- if (installments) {
139
- result.installments = installments;
140
- }
141
- return result;
142
- });
143
- }
144
- class VisaCalScraper extends base_scraper_with_browser_1.BaseScraperWithBrowser {
145
- openLoginPopup = async () => {
146
- debug('open login popup, wait until login button available');
147
- await (0, elements_interactions_1.waitUntilElementFound)(this.page, '#ccLoginDesktopBtn', true);
148
- debug('click on the login button');
149
- await (0, elements_interactions_1.clickButton)(this.page, '#ccLoginDesktopBtn');
150
- debug('get the frame that holds the login');
151
- const frame = await getLoginFrame(this.page);
152
- debug('wait until the password login tab header is available');
153
- await (0, elements_interactions_1.waitUntilElementFound)(frame, '#regular-login');
154
- debug('navigate to the password login tab');
155
- await (0, elements_interactions_1.clickButton)(frame, '#regular-login');
156
- debug('wait until the password login tab is active');
157
- await (0, elements_interactions_1.waitUntilElementFound)(frame, 'regular-login');
158
- return frame;
159
- };
160
- async getCards() {
161
- const initData = await (0, waiting_1.waitUntil)(() => (0, storage_1.getFromSessionStorage)(this.page, 'init'), 'get init data in session storage', 10000, 1000);
162
- if (!initData) {
163
- throw new Error("could not find 'init' data in session storage");
164
- }
165
- return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));
98
+ const pendingTransactions = pendingData?.result ? pendingData.result.cardsList.flatMap(card => card.authDetalisList) : [];
99
+ const bankAccounts = data.flatMap(monthData => monthData.result.bankAccounts);
100
+ const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);
101
+ const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);
102
+ const completedTransactions = [...regularDebitDays, ...immediateDebitDays].flatMap(debitDate => debitDate.transactions);
103
+ const all = [...pendingTransactions, ...completedTransactions];
104
+ return all.map(transaction => {
105
+ const numOfPayments = isPending(transaction) ? transaction.numberOfPayments : transaction.numOfPayments;
106
+ const installments = numOfPayments ? {
107
+ number: isPending(transaction) ? 1 : transaction.curPaymentNum,
108
+ total: numOfPayments
109
+ } : undefined;
110
+ const date = (0, _moment.default)(transaction.trnPurchaseDate);
111
+ let chargedAmount = isPending(transaction) ? transaction.trnAmt * -1 : transaction.amtBeforeConvAndIndex * -1;
112
+ let originalAmount = transaction.trnAmt * -1;
113
+ if (transaction.trnTypeCode === TrnTypeCode.credit) {
114
+ chargedAmount = isPending(transaction) ? transaction.trnAmt : transaction.amtBeforeConvAndIndex;
115
+ originalAmount = transaction.trnAmt;
166
116
  }
167
- async getAuthorizationHeader() {
168
- const authModule = await (0, storage_1.getFromSessionStorage)(this.page, 'auth-module');
169
- if (!authModule) {
170
- throw new Error("could not find 'auth-module' in session storage");
171
- }
172
- return `CALAuthScheme ${authModule.auth.calConnectToken}`;
173
- }
174
- async getXSiteId() {
175
- /*
176
- I don't know if the constant below will change in the feature.
177
- If so, use the next code:
178
-
179
- return this.page.evaluate(() => new Ut().xSiteId);
180
-
181
- To get the classname search for 'xSiteId' in the page source
182
- class Ut {
183
- constructor(_e, on, yn) {
184
- this.store = _e,
185
- this.config = on,
186
- this.eventBusService = yn,
187
- this.xSiteId = "09031987-273E-2311-906C-8AF85B17C8D9",
188
- */
189
- return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');
117
+ const result = {
118
+ identifier: !isPending(transaction) ? transaction.trnIntId : undefined,
119
+ type: [TrnTypeCode.regular, TrnTypeCode.standingOrder].includes(transaction.trnTypeCode) ? _transactions2.TransactionTypes.Normal : _transactions2.TransactionTypes.Installments,
120
+ status: isPending(transaction) ? _transactions2.TransactionStatuses.Pending : _transactions2.TransactionStatuses.Completed,
121
+ date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),
122
+ processedDate: isPending(transaction) ? date.toISOString() : new Date(transaction.debCrdDate).toISOString(),
123
+ originalAmount,
124
+ originalCurrency: transaction.trnCurrencySymbol,
125
+ chargedAmount,
126
+ chargedCurrency: !isPending(transaction) ? transaction.debCrdCurrencySymbol : undefined,
127
+ description: transaction.merchantName,
128
+ memo: transaction.transTypeCommentDetails.toString(),
129
+ category: transaction.branchCodeDesc
130
+ };
131
+ if (installments) {
132
+ result.installments = installments;
190
133
  }
191
- getLoginOptions(credentials) {
192
- return {
193
- loginUrl: `${LOGIN_URL}`,
194
- fields: createLoginFields(credentials),
195
- submitButtonSelector: 'button[type="submit"]',
196
- possibleResults: getPossibleLoginResults(),
197
- checkReadiness: async () => (0, elements_interactions_1.waitUntilElementFound)(this.page, '#ccLoginDesktopBtn'),
198
- preAction: this.openLoginPopup,
199
- postAction: async () => {
200
- try {
201
- await (0, navigation_1.waitForNavigation)(this.page);
202
- const currentUrl = await (0, navigation_1.getCurrentUrl)(this.page);
203
- if (currentUrl.endsWith('site-tutorial')) {
204
- await (0, elements_interactions_1.clickButton)(this.page, 'button.btn-close');
205
- }
206
- }
207
- catch (e) {
208
- const currentUrl = await (0, navigation_1.getCurrentUrl)(this.page);
209
- if (currentUrl.endsWith('dashboard'))
210
- return;
211
- const requiresChangePassword = await hasChangePasswordForm(this.page);
212
- if (requiresChangePassword)
213
- return;
214
- throw e;
215
- }
216
- },
217
- userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',
218
- };
134
+ return result;
135
+ });
136
+ }
137
+ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
138
+ openLoginPopup = async () => {
139
+ debug('open login popup, wait until login button available');
140
+ await (0, _elementsInteractions.waitUntilElementFound)(this.page, '#ccLoginDesktopBtn', true);
141
+ debug('click on the login button');
142
+ await (0, _elementsInteractions.clickButton)(this.page, '#ccLoginDesktopBtn');
143
+ debug('get the frame that holds the login');
144
+ const frame = await getLoginFrame(this.page);
145
+ debug('wait until the password login tab header is available');
146
+ await (0, _elementsInteractions.waitUntilElementFound)(frame, '#regular-login');
147
+ debug('navigate to the password login tab');
148
+ await (0, _elementsInteractions.clickButton)(frame, '#regular-login');
149
+ debug('wait until the password login tab is active');
150
+ await (0, _elementsInteractions.waitUntilElementFound)(frame, 'regular-login');
151
+ return frame;
152
+ };
153
+ async getCards() {
154
+ const initData = await (0, _waiting.waitUntil)(() => (0, _storage.getFromSessionStorage)(this.page, 'init'), 'get init data in session storage', 10000, 1000);
155
+ if (!initData) {
156
+ throw new Error("could not find 'init' data in session storage");
219
157
  }
220
- async fetchData() {
221
- const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years').subtract(6, 'months').add(1, 'day');
222
- const startDate = this.options.startDate || defaultStartMoment.toDate();
223
- const startMoment = moment_1.default.max(defaultStartMoment, (0, moment_1.default)(startDate));
224
- debug(`fetch transactions starting ${startMoment.format()}`);
225
- const Authorization = await this.getAuthorizationHeader();
226
- const cards = await this.getCards();
227
- const xSiteId = await this.getXSiteId();
228
- const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;
229
- const accounts = await Promise.all(cards.map(async (card) => {
230
- const finalMonthToFetchMoment = (0, moment_1.default)().add(futureMonthsToScrape, 'month');
231
- const months = finalMonthToFetchMoment.diff(startMoment, 'months');
232
- const allMonthsData = [];
233
- debug(`fetch pending transactions for card ${card.cardUniqueId}`);
234
- let pendingData = await (0, fetch_1.fetchPostWithinPage)(this.page, PENDING_TRANSACTIONS_REQUEST_ENDPOINT, { cardUniqueIDArray: [card.cardUniqueId] }, {
235
- Authorization,
236
- 'X-Site-Id': xSiteId,
237
- 'Content-Type': 'application/json',
238
- });
239
- debug(`fetch completed transactions for card ${card.cardUniqueId}`);
240
- for (let i = 0; i <= months; i += 1) {
241
- const month = finalMonthToFetchMoment.clone().subtract(i, 'months');
242
- const monthData = await (0, fetch_1.fetchPostWithinPage)(this.page, TRANSACTIONS_REQUEST_ENDPOINT, { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') }, {
243
- Authorization,
244
- 'X-Site-Id': xSiteId,
245
- 'Content-Type': 'application/json',
246
- });
247
- if (monthData?.statusCode !== 1)
248
- throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);
249
- if (!isCardTransactionDetails(monthData)) {
250
- throw new Error('monthData is not of type CardTransactionDetails');
251
- }
252
- allMonthsData.push(monthData);
253
- }
254
- if (pendingData?.statusCode !== 1 && pendingData?.statusCode !== 96) {
255
- debug(`failed to fetch pending transactions for card ${card.last4Digits}. Message: ${pendingData?.title || ''}`);
256
- pendingData = null;
257
- }
258
- else if (!isCardPendingTransactionDetails(pendingData)) {
259
- debug('pendingData is not of type CardTransactionDetails');
260
- pendingData = null;
261
- }
262
- const transactions = convertParsedDataToTransactions(allMonthsData, pendingData);
263
- debug('filer out old transactions');
264
- const txns = (this.options.outputData?.enableTransactionsFilterByDate ?? true)
265
- ? (0, transactions_1.filterOldTransactions)(transactions, (0, moment_1.default)(startDate), this.options.combineInstallments || false)
266
- : transactions;
267
- return {
268
- txns,
269
- accountNumber: card.last4Digits,
270
- };
271
- }));
272
- debug('return the scraped accounts');
273
- debug(JSON.stringify(accounts, null, 2));
274
- return {
275
- success: true,
276
- accounts,
277
- };
158
+ return initData?.result.cards.map(({
159
+ cardUniqueId,
160
+ last4Digits
161
+ }) => ({
162
+ cardUniqueId,
163
+ last4Digits
164
+ }));
165
+ }
166
+ async getAuthorizationHeader() {
167
+ const authModule = await (0, _storage.getFromSessionStorage)(this.page, 'auth-module');
168
+ if (!authModule) {
169
+ throw new Error("could not find 'auth-module' in session storage");
278
170
  }
171
+ return `CALAuthScheme ${authModule.auth.calConnectToken}`;
172
+ }
173
+ async getXSiteId() {
174
+ /*
175
+ I don't know if the constant below will change in the feature.
176
+ If so, use the next code:
177
+ return this.page.evaluate(() => new Ut().xSiteId);
178
+ To get the classname search for 'xSiteId' in the page source
179
+ class Ut {
180
+ constructor(_e, on, yn) {
181
+ this.store = _e,
182
+ this.config = on,
183
+ this.eventBusService = yn,
184
+ this.xSiteId = "09031987-273E-2311-906C-8AF85B17C8D9",
185
+ */
186
+ return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');
187
+ }
188
+ getLoginOptions(credentials) {
189
+ return {
190
+ loginUrl: `${LOGIN_URL}`,
191
+ fields: createLoginFields(credentials),
192
+ submitButtonSelector: 'button[type="submit"]',
193
+ possibleResults: getPossibleLoginResults(),
194
+ checkReadiness: async () => (0, _elementsInteractions.waitUntilElementFound)(this.page, '#ccLoginDesktopBtn'),
195
+ preAction: this.openLoginPopup,
196
+ postAction: async () => {
197
+ try {
198
+ await (0, _navigation.waitForNavigation)(this.page);
199
+ const currentUrl = await (0, _navigation.getCurrentUrl)(this.page);
200
+ if (currentUrl.endsWith('site-tutorial')) {
201
+ await (0, _elementsInteractions.clickButton)(this.page, 'button.btn-close');
202
+ }
203
+ } catch (e) {
204
+ const currentUrl = await (0, _navigation.getCurrentUrl)(this.page);
205
+ if (currentUrl.endsWith('dashboard')) return;
206
+ const requiresChangePassword = await hasChangePasswordForm(this.page);
207
+ if (requiresChangePassword) return;
208
+ throw e;
209
+ }
210
+ },
211
+ userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36'
212
+ };
213
+ }
214
+ async fetchData() {
215
+ const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').subtract(6, 'months').add(1, 'day');
216
+ const startDate = this.options.startDate || defaultStartMoment.toDate();
217
+ const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
218
+ debug(`fetch transactions starting ${startMoment.format()}`);
219
+ const Authorization = await this.getAuthorizationHeader();
220
+ const cards = await this.getCards();
221
+ const xSiteId = await this.getXSiteId();
222
+ const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;
223
+ const accounts = await Promise.all(cards.map(async card => {
224
+ const finalMonthToFetchMoment = (0, _moment.default)().add(futureMonthsToScrape, 'month');
225
+ const months = finalMonthToFetchMoment.diff(startMoment, 'months');
226
+ const allMonthsData = [];
227
+ debug(`fetch pending transactions for card ${card.cardUniqueId}`);
228
+ let pendingData = await (0, _fetch.fetchPostWithinPage)(this.page, PENDING_TRANSACTIONS_REQUEST_ENDPOINT, {
229
+ cardUniqueIDArray: [card.cardUniqueId]
230
+ }, {
231
+ Authorization,
232
+ 'X-Site-Id': xSiteId,
233
+ 'Content-Type': 'application/json'
234
+ });
235
+ debug(`fetch completed transactions for card ${card.cardUniqueId}`);
236
+ for (let i = 0; i <= months; i += 1) {
237
+ const month = finalMonthToFetchMoment.clone().subtract(i, 'months');
238
+ const monthData = await (0, _fetch.fetchPostWithinPage)(this.page, TRANSACTIONS_REQUEST_ENDPOINT, {
239
+ cardUniqueId: card.cardUniqueId,
240
+ month: month.format('M'),
241
+ year: month.format('YYYY')
242
+ }, {
243
+ Authorization,
244
+ 'X-Site-Id': xSiteId,
245
+ 'Content-Type': 'application/json'
246
+ });
247
+ if (monthData?.statusCode !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);
248
+ if (!isCardTransactionDetails(monthData)) {
249
+ throw new Error('monthData is not of type CardTransactionDetails');
250
+ }
251
+ allMonthsData.push(monthData);
252
+ }
253
+ if (pendingData?.statusCode !== 1 && pendingData?.statusCode !== 96) {
254
+ debug(`failed to fetch pending transactions for card ${card.last4Digits}. Message: ${pendingData?.title || ''}`);
255
+ pendingData = null;
256
+ } else if (!isCardPendingTransactionDetails(pendingData)) {
257
+ debug('pendingData is not of type CardTransactionDetails');
258
+ pendingData = null;
259
+ }
260
+ const transactions = convertParsedDataToTransactions(allMonthsData, pendingData);
261
+ debug('filer out old transactions');
262
+ const txns = this.options.outputData?.enableTransactionsFilterByDate ?? true ? (0, _transactions.filterOldTransactions)(transactions, (0, _moment.default)(startDate), this.options.combineInstallments || false) : transactions;
263
+ return {
264
+ txns,
265
+ accountNumber: card.last4Digits
266
+ };
267
+ }));
268
+ debug('return the scraped accounts');
269
+ debug(JSON.stringify(accounts, null, 2));
270
+ return {
271
+ success: true,
272
+ accounts
273
+ };
274
+ }
279
275
  }
280
- exports.default = VisaCalScraper;
281
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"visa-cal.js","sourceRoot":"","sources":["../../src/scrapers/visa-cal.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { type Frame, type Page } from 'puppeteer';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, pageEval, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type LoginOptions } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\nconst PENDING_TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/approvals/getClearanceRequests';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum TrnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: TrnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n  earlyPaymentInd: boolean;\n}\ninterface ScrapedPendingTransaction {\n  merchantID: string;\n  merchantName: string;\n  trnPurchaseDate: string;\n  walletTranInd: number;\n  transactionsOrigin: number;\n  trnAmt: number;\n  tpaApprovalAmount: unknown;\n  trnCurrencySymbol: CurrencySymbol;\n  trnTypeCode: TrnTypeCode;\n  trnType: string;\n  branchCodeDesc: string;\n  transCardPresentInd: boolean;\n  j5Indicator: string;\n  numberOfPayments: number;\n  firstPaymentAmount: number;\n  transTypeCommentDetails: [];\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: []; debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\ninterface CardPendingTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    cardsList: {\n      cardUniqueID: string;\n      authDetalisList: ScrapedPendingTransaction[];\n    }[];\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\nfunction isPending(\n  transaction: ScrapedTransaction | ScrapedPendingTransaction,\n): transaction is ScrapedPendingTransaction {\n  return (transaction as ScrapedTransaction).debCrdDate === undefined; // an arbitrary field that only appears in a completed transaction\n}\n\nfunction isCardTransactionDetails(\n  result: CardTransactionDetails | CardTransactionDetailsError,\n): result is CardTransactionDetails {\n  return (result as CardTransactionDetails).result !== undefined;\n}\n\nfunction isCardPendingTransactionDetails(\n  result: CardPendingTransactionDetails | CardTransactionDetailsError,\n): result is CardPendingTransactionDetails {\n  return (result as CardPendingTransactionDetails).result !== undefined;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(\n    () => {\n      frame = page.frames().find(f => f.url().includes('connect')) || null;\n      return Promise.resolve(!!frame);\n    },\n    'wait for iframe with login form',\n    10000,\n    1000,\n  );\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound\n    ? await pageEval(frame, 'div.general-error > div', '', item => {\n        return (item as HTMLDivElement).innerText;\n      })\n    : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nasync function hasChangePasswordForm(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, '.change-password-subtitle');\n  return errorFound;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasInvalidPasswordError(page);\n      },\n    ],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    [LoginResults.ChangePassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasChangePasswordForm(page);\n      },\n    ],\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction convertParsedDataToTransactions(\n  data: CardTransactionDetails[],\n  pendingData?: CardPendingTransactionDetails | null,\n): Transaction[] {\n  const pendingTransactions = pendingData?.result\n    ? pendingData.result.cardsList.flatMap(card => card.authDetalisList)\n    : [];\n\n  const bankAccounts = data.flatMap(monthData => monthData.result.bankAccounts);\n  const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);\n  const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);\n  const completedTransactions = [...regularDebitDays, ...immediateDebitDays].flatMap(\n    debitDate => debitDate.transactions,\n  );\n\n  const all: (ScrapedTransaction | ScrapedPendingTransaction)[] = [...pendingTransactions, ...completedTransactions];\n\n  return all.map(transaction => {\n    const numOfPayments = isPending(transaction) ? transaction.numberOfPayments : transaction.numOfPayments;\n    const installments = numOfPayments\n      ? {\n          number: isPending(transaction) ? 1 : transaction.curPaymentNum,\n          total: numOfPayments,\n        }\n      : undefined;\n\n    const date = moment(transaction.trnPurchaseDate);\n\n    let chargedAmount = isPending(transaction) ? transaction.trnAmt * -1 : transaction.amtBeforeConvAndIndex * -1;\n    let originalAmount = transaction.trnAmt * -1;\n\n    if (transaction.trnTypeCode === TrnTypeCode.credit) {\n      chargedAmount = isPending(transaction) ? transaction.trnAmt : transaction.amtBeforeConvAndIndex;\n      originalAmount = transaction.trnAmt;\n    }\n\n    const result: Transaction = {\n      identifier: !isPending(transaction) ? transaction.trnIntId : undefined,\n      type: [TrnTypeCode.regular, TrnTypeCode.standingOrder].includes(transaction.trnTypeCode)\n        ? TransactionTypes.Normal\n        : TransactionTypes.Installments,\n      status: isPending(transaction) ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),\n      processedDate: isPending(transaction) ? date.toISOString() : new Date(transaction.debCrdDate).toISOString(),\n      originalAmount,\n      originalCurrency: transaction.trnCurrencySymbol,\n      chargedAmount,\n      chargedCurrency: !isPending(transaction) ? transaction.debCrdCurrencySymbol : undefined,\n      description: transaction.merchantName,\n      memo: transaction.transTypeCommentDetails.toString(),\n      category: transaction.branchCodeDesc,\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error(\"could not find 'init' data in session storage\");\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error(\"could not find 'auth-module' in session storage\");\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitForNavigation(this.page);\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          const requiresChangePassword = await hasChangePasswordForm(this.page);\n          if (requiresChangePassword) return;\n          throw e;\n        }\n      },\n      userAgent:\n        'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const Authorization = await this.getAuthorizationHeader();\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    const accounts = await Promise.all(\n      cards.map(async card => {\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n\n        const allMonthsData: CardTransactionDetails[] = [];\n\n        debug(`fetch pending transactions for card ${card.cardUniqueId}`);\n        let pendingData = await fetchPostWithinPage<CardPendingTransactionDetails | CardTransactionDetailsError>(\n          this.page,\n          PENDING_TRANSACTIONS_REQUEST_ENDPOINT,\n          { cardUniqueIDArray: [card.cardUniqueId] },\n          {\n            Authorization,\n            'X-Site-Id': xSiteId,\n            'Content-Type': 'application/json',\n          },\n        );\n\n        debug(`fetch completed transactions for card ${card.cardUniqueId}`);\n        for (let i = 0; i <= months; i += 1) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page,\n            TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1)\n            throw new Error(\n              `failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`,\n            );\n\n          if (!isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        if (pendingData?.statusCode !== 1 && pendingData?.statusCode !== 96) {\n          debug(\n            `failed to fetch pending transactions for card ${card.last4Digits}. Message: ${pendingData?.title || ''}`,\n          );\n          pendingData = null;\n        } else if (!isCardPendingTransactionDetails(pendingData)) {\n          debug('pendingData is not of type CardTransactionDetails');\n          pendingData = null;\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData, pendingData);\n\n        debug('filer out old transactions');\n        const txns =\n          (this.options.outputData?.enableTransactionsFilterByDate ?? true)\n            ? filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false)\n            : transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
276
+ var _default = exports.default = VisaCalScraper;
277
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_elementsInteractions","_fetch","_navigation","_storage","_transactions","_waiting","_transactions2","_baseScraperWithBrowser","e","__esModule","default","LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","PENDING_TRANSACTIONS_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","getDebug","TrnTypeCode","isPending","transaction","debCrdDate","undefined","isCardTransactionDetails","result","isCardPendingTransactionDetails","getLoginFrame","page","frame","waitUntil","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","elementPresentOnPage","errorMessage","pageEval","item","innerText","hasChangePasswordForm","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","ChangePassword","createLoginFields","credentials","selector","value","username","password","convertParsedDataToTransactions","data","pendingData","pendingTransactions","cardsList","flatMap","card","authDetalisList","bankAccounts","monthData","regularDebitDays","accounts","debitDates","immediateDebitDays","immidiateDebits","debitDays","completedTransactions","debitDate","transactions","all","map","numOfPayments","numberOfPayments","installments","number","curPaymentNum","total","date","moment","trnPurchaseDate","chargedAmount","trnAmt","amtBeforeConvAndIndex","originalAmount","trnTypeCode","credit","identifier","trnIntId","type","regular","standingOrder","TransactionTypes","Normal","Installments","status","TransactionStatuses","Pending","Completed","add","toISOString","processedDate","Date","originalCurrency","trnCurrencySymbol","chargedCurrency","debCrdCurrencySymbol","description","merchantName","memo","transTypeCommentDetails","toString","category","branchCodeDesc","VisaCalScraper","BaseScraperWithBrowser","openLoginPopup","waitUntilElementFound","clickButton","getCards","initData","getFromSessionStorage","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","auth","calConnectToken","getXSiteId","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","postAction","waitForNavigation","currentUrl","getCurrentUrl","endsWith","requiresChangePassword","userAgent","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","max","format","Authorization","xSiteId","futureMonthsToScrape","finalMonthToFetchMoment","months","diff","allMonthsData","fetchPostWithinPage","cardUniqueIDArray","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","filterOldTransactions","combineInstallments","accountNumber","JSON","stringify","success","_default","exports"],"sources":["../../src/scrapers/visa-cal.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Frame, type Page } from 'puppeteer';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, pageEval, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type LoginOptions } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\nconst PENDING_TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/approvals/getClearanceRequests';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum TrnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: TrnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n  earlyPaymentInd: boolean;\n}\ninterface ScrapedPendingTransaction {\n  merchantID: string;\n  merchantName: string;\n  trnPurchaseDate: string;\n  walletTranInd: number;\n  transactionsOrigin: number;\n  trnAmt: number;\n  tpaApprovalAmount: unknown;\n  trnCurrencySymbol: CurrencySymbol;\n  trnTypeCode: TrnTypeCode;\n  trnType: string;\n  branchCodeDesc: string;\n  transCardPresentInd: boolean;\n  j5Indicator: string;\n  numberOfPayments: number;\n  firstPaymentAmount: number;\n  transTypeCommentDetails: [];\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: []; debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\ninterface CardPendingTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    cardsList: {\n      cardUniqueID: string;\n      authDetalisList: ScrapedPendingTransaction[];\n    }[];\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\nfunction isPending(\n  transaction: ScrapedTransaction | ScrapedPendingTransaction,\n): transaction is ScrapedPendingTransaction {\n  return (transaction as ScrapedTransaction).debCrdDate === undefined; // an arbitrary field that only appears in a completed transaction\n}\n\nfunction isCardTransactionDetails(\n  result: CardTransactionDetails | CardTransactionDetailsError,\n): result is CardTransactionDetails {\n  return (result as CardTransactionDetails).result !== undefined;\n}\n\nfunction isCardPendingTransactionDetails(\n  result: CardPendingTransactionDetails | CardTransactionDetailsError,\n): result is CardPendingTransactionDetails {\n  return (result as CardPendingTransactionDetails).result !== undefined;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(\n    () => {\n      frame = page.frames().find(f => f.url().includes('connect')) || null;\n      return Promise.resolve(!!frame);\n    },\n    'wait for iframe with login form',\n    10000,\n    1000,\n  );\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound\n    ? await pageEval(frame, 'div.general-error > div', '', item => {\n        return (item as HTMLDivElement).innerText;\n      })\n    : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nasync function hasChangePasswordForm(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, '.change-password-subtitle');\n  return errorFound;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasInvalidPasswordError(page);\n      },\n    ],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    [LoginResults.ChangePassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasChangePasswordForm(page);\n      },\n    ],\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction convertParsedDataToTransactions(\n  data: CardTransactionDetails[],\n  pendingData?: CardPendingTransactionDetails | null,\n): Transaction[] {\n  const pendingTransactions = pendingData?.result\n    ? pendingData.result.cardsList.flatMap(card => card.authDetalisList)\n    : [];\n\n  const bankAccounts = data.flatMap(monthData => monthData.result.bankAccounts);\n  const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);\n  const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);\n  const completedTransactions = [...regularDebitDays, ...immediateDebitDays].flatMap(\n    debitDate => debitDate.transactions,\n  );\n\n  const all: (ScrapedTransaction | ScrapedPendingTransaction)[] = [...pendingTransactions, ...completedTransactions];\n\n  return all.map(transaction => {\n    const numOfPayments = isPending(transaction) ? transaction.numberOfPayments : transaction.numOfPayments;\n    const installments = numOfPayments\n      ? {\n          number: isPending(transaction) ? 1 : transaction.curPaymentNum,\n          total: numOfPayments,\n        }\n      : undefined;\n\n    const date = moment(transaction.trnPurchaseDate);\n\n    let chargedAmount = isPending(transaction) ? transaction.trnAmt * -1 : transaction.amtBeforeConvAndIndex * -1;\n    let originalAmount = transaction.trnAmt * -1;\n\n    if (transaction.trnTypeCode === TrnTypeCode.credit) {\n      chargedAmount = isPending(transaction) ? transaction.trnAmt : transaction.amtBeforeConvAndIndex;\n      originalAmount = transaction.trnAmt;\n    }\n\n    const result: Transaction = {\n      identifier: !isPending(transaction) ? transaction.trnIntId : undefined,\n      type: [TrnTypeCode.regular, TrnTypeCode.standingOrder].includes(transaction.trnTypeCode)\n        ? TransactionTypes.Normal\n        : TransactionTypes.Installments,\n      status: isPending(transaction) ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),\n      processedDate: isPending(transaction) ? date.toISOString() : new Date(transaction.debCrdDate).toISOString(),\n      originalAmount,\n      originalCurrency: transaction.trnCurrencySymbol,\n      chargedAmount,\n      chargedCurrency: !isPending(transaction) ? transaction.debCrdCurrencySymbol : undefined,\n      description: transaction.merchantName,\n      memo: transaction.transTypeCommentDetails.toString(),\n      category: transaction.branchCodeDesc,\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error(\"could not find 'init' data in session storage\");\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error(\"could not find 'auth-module' in session storage\");\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitForNavigation(this.page);\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          const requiresChangePassword = await hasChangePasswordForm(this.page);\n          if (requiresChangePassword) return;\n          throw e;\n        }\n      },\n      userAgent:\n        'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const Authorization = await this.getAuthorizationHeader();\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    const accounts = await Promise.all(\n      cards.map(async card => {\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n\n        const allMonthsData: CardTransactionDetails[] = [];\n\n        debug(`fetch pending transactions for card ${card.cardUniqueId}`);\n        let pendingData = await fetchPostWithinPage<CardPendingTransactionDetails | CardTransactionDetailsError>(\n          this.page,\n          PENDING_TRANSACTIONS_REQUEST_ENDPOINT,\n          { cardUniqueIDArray: [card.cardUniqueId] },\n          {\n            Authorization,\n            'X-Site-Id': xSiteId,\n            'Content-Type': 'application/json',\n          },\n        );\n\n        debug(`fetch completed transactions for card ${card.cardUniqueId}`);\n        for (let i = 0; i <= months; i += 1) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page,\n            TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1)\n            throw new Error(\n              `failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`,\n            );\n\n          if (!isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        if (pendingData?.statusCode !== 1 && pendingData?.statusCode !== 96) {\n          debug(\n            `failed to fetch pending transactions for card ${card.last4Digits}. Message: ${pendingData?.title || ''}`,\n          );\n          pendingData = null;\n        } else if (!isCardPendingTransactionDetails(pendingData)) {\n          debug('pendingData is not of type CardTransactionDetails');\n          pendingData = null;\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData, pendingData);\n\n        debug('filer out old transactions');\n        const txns =\n          (this.options.outputData?.enableTransactionsFilterByDate ?? true)\n            ? filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false)\n            : transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"],"mappings":";;;;;;AAAA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AAEA,IAAAC,MAAA,GAAAD,OAAA;AACA,IAAAE,qBAAA,GAAAF,OAAA;AACA,IAAAG,MAAA,GAAAH,OAAA;AACA,IAAAI,WAAA,GAAAJ,OAAA;AACA,IAAAK,QAAA,GAAAL,OAAA;AACA,IAAAM,aAAA,GAAAN,OAAA;AACA,IAAAO,QAAA,GAAAP,OAAA;AACA,IAAAQ,cAAA,GAAAR,OAAA;AACA,IAAAS,uBAAA,GAAAT,OAAA;AAAsG,SAAAD,uBAAAW,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAGtG,MAAMG,SAAS,GAAG,+BAA+B;AACjD,MAAMC,6BAA6B,GACjC,8FAA8F;AAChG,MAAMC,qCAAqC,GACzC,8EAA8E;AAEhF,MAAMC,sBAAsB,GAAG,mCAAmC;AAElE,MAAMC,KAAK,GAAG,IAAAC,eAAQ,EAAC,UAAU,CAAC;AAAC,IAE9BC,WAAW,0BAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAA,OAAXA,WAAW;AAAA,EAAXA,WAAW;AA8HhB,SAASC,SAASA,CAChBC,WAA2D,EACjB;EAC1C,OAAQA,WAAW,CAAwBC,UAAU,KAAKC,SAAS,CAAC,CAAC;AACvE;AAEA,SAASC,wBAAwBA,CAC/BC,MAA4D,EAC1B;EAClC,OAAQA,MAAM,CAA4BA,MAAM,KAAKF,SAAS;AAChE;AAEA,SAASG,+BAA+BA,CACtCD,MAAmE,EAC1B;EACzC,OAAQA,MAAM,CAAmCA,MAAM,KAAKF,SAAS;AACvE;AAEA,eAAeI,aAAaA,CAACC,IAAU,EAAE;EACvC,IAAIC,KAAmB,GAAG,IAAI;EAC9BZ,KAAK,CAAC,8BAA8B,CAAC;EACrC,MAAM,IAAAa,kBAAS,EACb,MAAM;IACJD,KAAK,GAAGD,IAAI,CAACG,MAAM,CAAC,CAAC,CAACC,IAAI,CAACC,CAAC,IAAIA,CAAC,CAACC,GAAG,CAAC,CAAC,CAACC,QAAQ,CAAC,SAAS,CAAC,CAAC,IAAI,IAAI;IACpE,OAAOC,OAAO,CAACC,OAAO,CAAC,CAAC,CAACR,KAAK,CAAC;EACjC,CAAC,EACD,iCAAiC,EACjC,KAAK,EACL,IACF,CAAC;EAED,IAAI,CAACA,KAAK,EAAE;IACVZ,KAAK,CAAC,2CAA2C,CAAC;IAClD,MAAM,IAAIqB,KAAK,CAAC,gCAAgC,CAAC;EACnD;EAEA,OAAOT,KAAK;AACd;AAEA,eAAeU,uBAAuBA,CAACX,IAAU,EAAE;EACjD,MAAMC,KAAK,GAAG,MAAMF,aAAa,CAACC,IAAI,CAAC;EACvC,MAAMY,UAAU,GAAG,MAAM,IAAAC,0CAAoB,EAACZ,KAAK,EAAE,yBAAyB,CAAC;EAC/E,MAAMa,YAAY,GAAGF,UAAU,GAC3B,MAAM,IAAAG,8BAAQ,EAACd,KAAK,EAAE,yBAAyB,EAAE,EAAE,EAAEe,IAAI,IAAI;IAC3D,OAAQA,IAAI,CAAoBC,SAAS;EAC3C,CAAC,CAAC,GACF,EAAE;EACN,OAAOH,YAAY,KAAK1B,sBAAsB;AAChD;AAEA,eAAe8B,qBAAqBA,CAAClB,IAAU,EAAE;EAC/C,MAAMC,KAAK,GAAG,MAAMF,aAAa,CAACC,IAAI,CAAC;EACvC,MAAMY,UAAU,GAAG,MAAM,IAAAC,0CAAoB,EAACZ,KAAK,EAAE,2BAA2B,CAAC;EACjF,OAAOW,UAAU;AACnB;AAEA,SAASO,uBAAuBA,CAAA,EAAG;EACjC9B,KAAK,CAAC,+BAA+B,CAAC;EACtC,MAAM+B,IAAqC,GAAG;IAC5C,CAACC,oCAAY,CAACC,OAAO,GAAG,CAAC,YAAY,CAAC;IACtC,CAACD,oCAAY,CAACE,eAAe,GAAG,CAC9B,MAAOC,OAAyB,IAAK;MACnC,MAAMxB,IAAI,GAAGwB,OAAO,EAAExB,IAAI;MAC1B,IAAI,CAACA,IAAI,EAAE;QACT,OAAO,KAAK;MACd;MACA,OAAOW,uBAAuB,CAACX,IAAI,CAAC;IACtC,CAAC,CACF;IACD;IACA,CAACqB,oCAAY,CAACI,cAAc,GAAG,CAC7B,MAAOD,OAAyB,IAAK;MACnC,MAAMxB,IAAI,GAAGwB,OAAO,EAAExB,IAAI;MAC1B,IAAI,CAACA,IAAI,EAAE;QACT,OAAO,KAAK;MACd;MACA,OAAOkB,qBAAqB,CAAClB,IAAI,CAAC;IACpC,CAAC;EAEL,CAAC;EACD,OAAOoB,IAAI;AACb;AAEA,SAASM,iBAAiBA,CAACC,WAAuC,EAAE;EAClEtC,KAAK,CAAC,+CAA+C,CAAC;EACtD,OAAO,CACL;IAAEuC,QAAQ,EAAE,8BAA8B;IAAEC,KAAK,EAAEF,WAAW,CAACG;EAAS,CAAC,EACzE;IAAEF,QAAQ,EAAE,8BAA8B;IAAEC,KAAK,EAAEF,WAAW,CAACI;EAAS,CAAC,CAC1E;AACH;AAEA,SAASC,+BAA+BA,CACtCC,IAA8B,EAC9BC,WAAkD,EACnC;EACf,MAAMC,mBAAmB,GAAGD,WAAW,EAAErC,MAAM,GAC3CqC,WAAW,CAACrC,MAAM,CAACuC,SAAS,CAACC,OAAO,CAACC,IAAI,IAAIA,IAAI,CAACC,eAAe,CAAC,GAClE,EAAE;EAEN,MAAMC,YAAY,GAAGP,IAAI,CAACI,OAAO,CAACI,SAAS,IAAIA,SAAS,CAAC5C,MAAM,CAAC2C,YAAY,CAAC;EAC7E,MAAME,gBAAgB,GAAGF,YAAY,CAACH,OAAO,CAACM,QAAQ,IAAIA,QAAQ,CAACC,UAAU,CAAC;EAC9E,MAAMC,kBAAkB,GAAGL,YAAY,CAACH,OAAO,CAACM,QAAQ,IAAIA,QAAQ,CAACG,eAAe,CAACC,SAAS,CAAC;EAC/F,MAAMC,qBAAqB,GAAG,CAAC,GAAGN,gBAAgB,EAAE,GAAGG,kBAAkB,CAAC,CAACR,OAAO,CAChFY,SAAS,IAAIA,SAAS,CAACC,YACzB,CAAC;EAED,MAAMC,GAAuD,GAAG,CAAC,GAAGhB,mBAAmB,EAAE,GAAGa,qBAAqB,CAAC;EAElH,OAAOG,GAAG,CAACC,GAAG,CAAC3D,WAAW,IAAI;IAC5B,MAAM4D,aAAa,GAAG7D,SAAS,CAACC,WAAW,CAAC,GAAGA,WAAW,CAAC6D,gBAAgB,GAAG7D,WAAW,CAAC4D,aAAa;IACvG,MAAME,YAAY,GAAGF,aAAa,GAC9B;MACEG,MAAM,EAAEhE,SAAS,CAACC,WAAW,CAAC,GAAG,CAAC,GAAGA,WAAW,CAACgE,aAAa;MAC9DC,KAAK,EAAEL;IACT,CAAC,GACD1D,SAAS;IAEb,MAAMgE,IAAI,GAAG,IAAAC,eAAM,EAACnE,WAAW,CAACoE,eAAe,CAAC;IAEhD,IAAIC,aAAa,GAAGtE,SAAS,CAACC,WAAW,CAAC,GAAGA,WAAW,CAACsE,MAAM,GAAG,CAAC,CAAC,GAAGtE,WAAW,CAACuE,qBAAqB,GAAG,CAAC,CAAC;IAC7G,IAAIC,cAAc,GAAGxE,WAAW,CAACsE,MAAM,GAAG,CAAC,CAAC;IAE5C,IAAItE,WAAW,CAACyE,WAAW,KAAK3E,WAAW,CAAC4E,MAAM,EAAE;MAClDL,aAAa,GAAGtE,SAAS,CAACC,WAAW,CAAC,GAAGA,WAAW,CAACsE,MAAM,GAAGtE,WAAW,CAACuE,qBAAqB;MAC/FC,cAAc,GAAGxE,WAAW,CAACsE,MAAM;IACrC;IAEA,MAAMlE,MAAmB,GAAG;MAC1BuE,UAAU,EAAE,CAAC5E,SAAS,CAACC,WAAW,CAAC,GAAGA,WAAW,CAAC4E,QAAQ,GAAG1E,SAAS;MACtE2E,IAAI,EAAE,CAAC/E,WAAW,CAACgF,OAAO,EAAEhF,WAAW,CAACiF,aAAa,CAAC,CAACjE,QAAQ,CAACd,WAAW,CAACyE,WAAW,CAAC,GACpFO,+BAAgB,CAACC,MAAM,GACvBD,+BAAgB,CAACE,YAAY;MACjCC,MAAM,EAAEpF,SAAS,CAACC,WAAW,CAAC,GAAGoF,kCAAmB,CAACC,OAAO,GAAGD,kCAAmB,CAACE,SAAS;MAC5FpB,IAAI,EAAEJ,YAAY,GAAGI,IAAI,CAACqB,GAAG,CAACzB,YAAY,CAACC,MAAM,GAAG,CAAC,EAAE,OAAO,CAAC,CAACyB,WAAW,CAAC,CAAC,GAAGtB,IAAI,CAACsB,WAAW,CAAC,CAAC;MAClGC,aAAa,EAAE1F,SAAS,CAACC,WAAW,CAAC,GAAGkE,IAAI,CAACsB,WAAW,CAAC,CAAC,GAAG,IAAIE,IAAI,CAAC1F,WAAW,CAACC,UAAU,CAAC,CAACuF,WAAW,CAAC,CAAC;MAC3GhB,cAAc;MACdmB,gBAAgB,EAAE3F,WAAW,CAAC4F,iBAAiB;MAC/CvB,aAAa;MACbwB,eAAe,EAAE,CAAC9F,SAAS,CAACC,WAAW,CAAC,GAAGA,WAAW,CAAC8F,oBAAoB,GAAG5F,SAAS;MACvF6F,WAAW,EAAE/F,WAAW,CAACgG,YAAY;MACrCC,IAAI,EAAEjG,WAAW,CAACkG,uBAAuB,CAACC,QAAQ,CAAC,CAAC;MACpDC,QAAQ,EAAEpG,WAAW,CAACqG;IACxB,CAAC;IAED,IAAIvC,YAAY,EAAE;MAChB1D,MAAM,CAAC0D,YAAY,GAAGA,YAAY;IACpC;IAEA,OAAO1D,MAAM;EACf,CAAC,CAAC;AACJ;AAIA,MAAMkG,cAAc,SAASC,8CAAsB,CAA6B;EAC9EC,cAAc,GAAG,MAAAA,CAAA,KAAY;IAC3B5G,KAAK,CAAC,qDAAqD,CAAC;IAC5D,MAAM,IAAA6G,2CAAqB,EAAC,IAAI,CAAClG,IAAI,EAAE,oBAAoB,EAAE,IAAI,CAAC;IAClEX,KAAK,CAAC,2BAA2B,CAAC;IAClC,MAAM,IAAA8G,iCAAW,EAAC,IAAI,CAACnG,IAAI,EAAE,oBAAoB,CAAC;IAClDX,KAAK,CAAC,oCAAoC,CAAC;IAC3C,MAAMY,KAAK,GAAG,MAAMF,aAAa,CAAC,IAAI,CAACC,IAAI,CAAC;IAC5CX,KAAK,CAAC,uDAAuD,CAAC;IAC9D,MAAM,IAAA6G,2CAAqB,EAACjG,KAAK,EAAE,gBAAgB,CAAC;IACpDZ,KAAK,CAAC,oCAAoC,CAAC;IAC3C,MAAM,IAAA8G,iCAAW,EAAClG,KAAK,EAAE,gBAAgB,CAAC;IAC1CZ,KAAK,CAAC,6CAA6C,CAAC;IACpD,MAAM,IAAA6G,2CAAqB,EAACjG,KAAK,EAAE,eAAe,CAAC;IAEnD,OAAOA,KAAK;EACd,CAAC;EAED,MAAMmG,QAAQA,CAAA,EAAG;IACf,MAAMC,QAAQ,GAAG,MAAM,IAAAnG,kBAAS,EAC9B,MAAM,IAAAoG,8BAAqB,EAAe,IAAI,CAACtG,IAAI,EAAE,MAAM,CAAC,EAC5D,kCAAkC,EAClC,KAAK,EACL,IACF,CAAC;IACD,IAAI,CAACqG,QAAQ,EAAE;MACb,MAAM,IAAI3F,KAAK,CAAC,+CAA+C,CAAC;IAClE;IACA,OAAO2F,QAAQ,EAAExG,MAAM,CAAC0G,KAAK,CAACnD,GAAG,CAAC,CAAC;MAAEoD,YAAY;MAAEC;IAAY,CAAC,MAAM;MAAED,YAAY;MAAEC;IAAY,CAAC,CAAC,CAAC;EACvG;EAEA,MAAMC,sBAAsBA,CAAA,EAAG;IAC7B,MAAMC,UAAU,GAAG,MAAM,IAAAL,8BAAqB,EAAwC,IAAI,CAACtG,IAAI,EAAE,aAAa,CAAC;IAC/G,IAAI,CAAC2G,UAAU,EAAE;MACf,MAAM,IAAIjG,KAAK,CAAC,iDAAiD,CAAC;IACpE;IACA,OAAO,iBAAiBiG,UAAU,CAACC,IAAI,CAACC,eAAe,EAAE;EAC3D;EAEA,MAAMC,UAAUA,CAAA,EAAG;IACjB;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;IAGI,OAAOtG,OAAO,CAACC,OAAO,CAAC,sCAAsC,CAAC;EAChE;EAEAsG,eAAeA,CAACpF,WAAuC,EAAgB;IACrE,OAAO;MACLqF,QAAQ,EAAE,GAAG/H,SAAS,EAAE;MACxBgI,MAAM,EAAEvF,iBAAiB,CAACC,WAAW,CAAC;MACtCuF,oBAAoB,EAAE,uBAAuB;MAC7CC,eAAe,EAAEhG,uBAAuB,CAAC,CAAC;MAC1CiG,cAAc,EAAE,MAAAA,CAAA,KAAY,IAAAlB,2CAAqB,EAAC,IAAI,CAAClG,IAAI,EAAE,oBAAoB,CAAC;MAClFqH,SAAS,EAAE,IAAI,CAACpB,cAAc;MAC9BqB,UAAU,EAAE,MAAAA,CAAA,KAAY;QACtB,IAAI;UACF,MAAM,IAAAC,6BAAiB,EAAC,IAAI,CAACvH,IAAI,CAAC;UAClC,MAAMwH,UAAU,GAAG,MAAM,IAAAC,yBAAa,EAAC,IAAI,CAACzH,IAAI,CAAC;UACjD,IAAIwH,UAAU,CAACE,QAAQ,CAAC,eAAe,CAAC,EAAE;YACxC,MAAM,IAAAvB,iCAAW,EAAC,IAAI,CAACnG,IAAI,EAAE,kBAAkB,CAAC;UAClD;QACF,CAAC,CAAC,OAAOlB,CAAC,EAAE;UACV,MAAM0I,UAAU,GAAG,MAAM,IAAAC,yBAAa,EAAC,IAAI,CAACzH,IAAI,CAAC;UACjD,IAAIwH,UAAU,CAACE,QAAQ,CAAC,WAAW,CAAC,EAAE;UACtC,MAAMC,sBAAsB,GAAG,MAAMzG,qBAAqB,CAAC,IAAI,CAAClB,IAAI,CAAC;UACrE,IAAI2H,sBAAsB,EAAE;UAC5B,MAAM7I,CAAC;QACT;MACF,CAAC;MACD8I,SAAS,EACP;IACJ,CAAC;EACH;EAEA,MAAMC,SAASA,CAAA,EAAmC;IAChD,MAAMC,kBAAkB,GAAG,IAAAlE,eAAM,EAAC,CAAC,CAACmE,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC,CAACA,QAAQ,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC/C,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC;IAC5F,MAAMgD,SAAS,GAAG,IAAI,CAACxG,OAAO,CAACwG,SAAS,IAAIF,kBAAkB,CAACG,MAAM,CAAC,CAAC;IACvE,MAAMC,WAAW,GAAGtE,eAAM,CAACuE,GAAG,CAACL,kBAAkB,EAAE,IAAAlE,eAAM,EAACoE,SAAS,CAAC,CAAC;IACrE3I,KAAK,CAAC,+BAA+B6I,WAAW,CAACE,MAAM,CAAC,CAAC,EAAE,CAAC;IAE5D,MAAMC,aAAa,GAAG,MAAM,IAAI,CAAC3B,sBAAsB,CAAC,CAAC;IACzD,MAAMH,KAAK,GAAG,MAAM,IAAI,CAACH,QAAQ,CAAC,CAAC;IACnC,MAAMkC,OAAO,GAAG,MAAM,IAAI,CAACxB,UAAU,CAAC,CAAC;IACvC,MAAMyB,oBAAoB,GAAG,IAAI,CAAC/G,OAAO,CAAC+G,oBAAoB,IAAI,CAAC;IAEnE,MAAM5F,QAAQ,GAAG,MAAMnC,OAAO,CAAC2C,GAAG,CAChCoD,KAAK,CAACnD,GAAG,CAAC,MAAMd,IAAI,IAAI;MACtB,MAAMkG,uBAAuB,GAAG,IAAA5E,eAAM,EAAC,CAAC,CAACoB,GAAG,CAACuD,oBAAoB,EAAE,OAAO,CAAC;MAC3E,MAAME,MAAM,GAAGD,uBAAuB,CAACE,IAAI,CAACR,WAAW,EAAE,QAAQ,CAAC;MAElE,MAAMS,aAAuC,GAAG,EAAE;MAElDtJ,KAAK,CAAC,uCAAuCiD,IAAI,CAACkE,YAAY,EAAE,CAAC;MACjE,IAAItE,WAAW,GAAG,MAAM,IAAA0G,0BAAmB,EACzC,IAAI,CAAC5I,IAAI,EACTb,qCAAqC,EACrC;QAAE0J,iBAAiB,EAAE,CAACvG,IAAI,CAACkE,YAAY;MAAE,CAAC,EAC1C;QACE6B,aAAa;QACb,WAAW,EAAEC,OAAO;QACpB,cAAc,EAAE;MAClB,CACF,CAAC;MAEDjJ,KAAK,CAAC,yCAAyCiD,IAAI,CAACkE,YAAY,EAAE,CAAC;MACnE,KAAK,IAAIsC,CAAC,GAAG,CAAC,EAAEA,CAAC,IAAIL,MAAM,EAAEK,CAAC,IAAI,CAAC,EAAE;QACnC,MAAMC,KAAK,GAAGP,uBAAuB,CAACQ,KAAK,CAAC,CAAC,CAACjB,QAAQ,CAACe,CAAC,EAAE,QAAQ,CAAC;QACnE,MAAMrG,SAAS,GAAG,MAAM,IAAAmG,0BAAmB,EACzC,IAAI,CAAC5I,IAAI,EACTd,6BAA6B,EAC7B;UAAEsH,YAAY,EAAElE,IAAI,CAACkE,YAAY;UAAEuC,KAAK,EAAEA,KAAK,CAACX,MAAM,CAAC,GAAG,CAAC;UAAEa,IAAI,EAAEF,KAAK,CAACX,MAAM,CAAC,MAAM;QAAE,CAAC,EACzF;UACEC,aAAa;UACb,WAAW,EAAEC,OAAO;UACpB,cAAc,EAAE;QAClB,CACF,CAAC;QAED,IAAI7F,SAAS,EAAEyG,UAAU,KAAK,CAAC,EAC7B,MAAM,IAAIxI,KAAK,CACb,yCAAyC4B,IAAI,CAACmE,WAAW,cAAchE,SAAS,EAAE0G,KAAK,IAAI,EAAE,EAC/F,CAAC;QAEH,IAAI,CAACvJ,wBAAwB,CAAC6C,SAAS,CAAC,EAAE;UACxC,MAAM,IAAI/B,KAAK,CAAC,iDAAiD,CAAC;QACpE;QAEAiI,aAAa,CAACS,IAAI,CAAC3G,SAAS,CAAC;MAC/B;MAEA,IAAIP,WAAW,EAAEgH,UAAU,KAAK,CAAC,IAAIhH,WAAW,EAAEgH,UAAU,KAAK,EAAE,EAAE;QACnE7J,KAAK,CACH,iDAAiDiD,IAAI,CAACmE,WAAW,cAAcvE,WAAW,EAAEiH,KAAK,IAAI,EAAE,EACzG,CAAC;QACDjH,WAAW,GAAG,IAAI;MACpB,CAAC,MAAM,IAAI,CAACpC,+BAA+B,CAACoC,WAAW,CAAC,EAAE;QACxD7C,KAAK,CAAC,mDAAmD,CAAC;QAC1D6C,WAAW,GAAG,IAAI;MACpB;MAEA,MAAMgB,YAAY,GAAGlB,+BAA+B,CAAC2G,aAAa,EAAEzG,WAAW,CAAC;MAEhF7C,KAAK,CAAC,4BAA4B,CAAC;MACnC,MAAMgK,IAAI,GACP,IAAI,CAAC7H,OAAO,CAAC8H,UAAU,EAAEC,8BAA8B,IAAI,IAAI,GAC5D,IAAAC,mCAAqB,EAACtG,YAAY,EAAE,IAAAU,eAAM,EAACoE,SAAS,CAAC,EAAE,IAAI,CAACxG,OAAO,CAACiI,mBAAmB,IAAI,KAAK,CAAC,GACjGvG,YAAY;MAElB,OAAO;QACLmG,IAAI;QACJK,aAAa,EAAEpH,IAAI,CAACmE;MACtB,CAAC;IACH,CAAC,CACH,CAAC;IAEDpH,KAAK,CAAC,6BAA6B,CAAC;IAEpCA,KAAK,CAACsK,IAAI,CAACC,SAAS,CAACjH,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;IACxC,OAAO;MACLkH,OAAO,EAAE,IAAI;MACblH;IACF,CAAC;EACH;AACF;AAAC,IAAAmH,QAAA,GAAAC,OAAA,CAAA/K,OAAA,GAEc+G,cAAc","ignoreList":[]}