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.
- package/lib/assertNever.js +7 -5
- package/lib/constants.js +16 -13
- package/lib/definitions.js +113 -109
- package/lib/helpers/browser.js +13 -9
- package/lib/helpers/dates.js +19 -18
- package/lib/helpers/debug.js +9 -9
- package/lib/helpers/elements-interactions.js +82 -78
- package/lib/helpers/fetch.js +85 -82
- package/lib/helpers/navigation.js +28 -24
- package/lib/helpers/storage.js +11 -10
- package/lib/helpers/transactions.js +32 -33
- package/lib/helpers/waiting.js +42 -45
- package/lib/index.js +82 -15
- package/lib/scrapers/amex.js +13 -11
- package/lib/scrapers/amex.test.d.ts +1 -0
- package/lib/scrapers/amex.test.js +49 -0
- package/lib/scrapers/base-beinleumi-group.js +239 -233
- package/lib/scrapers/base-isracard-amex.js +273 -273
- package/lib/scrapers/base-scraper-with-browser.js +263 -241
- package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
- package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
- package/lib/scrapers/base-scraper.js +82 -82
- package/lib/scrapers/behatsdaa.js +103 -98
- package/lib/scrapers/behatsdaa.test.d.ts +1 -0
- package/lib/scrapers/behatsdaa.test.js +46 -0
- package/lib/scrapers/beinleumi.js +13 -11
- package/lib/scrapers/beinleumi.test.d.ts +1 -0
- package/lib/scrapers/beinleumi.test.js +47 -0
- package/lib/scrapers/beyahad-bishvilha.js +132 -132
- package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
- package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
- package/lib/scrapers/discount.js +101 -97
- package/lib/scrapers/discount.test.d.ts +1 -0
- package/lib/scrapers/discount.test.js +49 -0
- package/lib/scrapers/errors.js +25 -22
- package/lib/scrapers/factory.js +67 -66
- package/lib/scrapers/factory.test.d.ts +1 -0
- package/lib/scrapers/factory.test.js +19 -0
- package/lib/scrapers/hapoalim.js +175 -162
- package/lib/scrapers/hapoalim.test.d.ts +1 -0
- package/lib/scrapers/hapoalim.test.js +47 -0
- package/lib/scrapers/interface.js +5 -2
- package/lib/scrapers/isracard.js +13 -11
- package/lib/scrapers/isracard.test.d.ts +1 -0
- package/lib/scrapers/isracard.test.js +49 -0
- package/lib/scrapers/leumi.js +170 -167
- package/lib/scrapers/leumi.test.d.ts +1 -0
- package/lib/scrapers/leumi.test.js +47 -0
- package/lib/scrapers/massad.js +13 -11
- package/lib/scrapers/max.js +261 -261
- package/lib/scrapers/max.test.d.ts +1 -0
- package/lib/scrapers/max.test.js +65 -0
- package/lib/scrapers/mercantile.js +16 -14
- package/lib/scrapers/mercantile.test.d.ts +1 -0
- package/lib/scrapers/mercantile.test.js +45 -0
- package/lib/scrapers/mizrahi.js +154 -158
- package/lib/scrapers/mizrahi.test.d.ts +1 -0
- package/lib/scrapers/mizrahi.test.js +53 -0
- package/lib/scrapers/one-zero-queries.js +7 -4
- package/lib/scrapers/one-zero.js +221 -176
- package/lib/scrapers/one-zero.test.d.ts +1 -0
- package/lib/scrapers/one-zero.test.js +51 -0
- package/lib/scrapers/otsar-hahayal.js +13 -11
- package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
- package/lib/scrapers/otsar-hahayal.test.js +47 -0
- package/lib/scrapers/pagi.js +13 -11
- package/lib/scrapers/pagi.test.d.ts +1 -0
- package/lib/scrapers/pagi.test.js +47 -0
- package/lib/scrapers/union-bank.js +173 -172
- package/lib/scrapers/union-bank.test.d.ts +1 -0
- package/lib/scrapers/union-bank.test.js +47 -0
- package/lib/scrapers/visa-cal.js +250 -254
- package/lib/scrapers/visa-cal.test.d.ts +1 -0
- package/lib/scrapers/visa-cal.test.js +49 -0
- package/lib/scrapers/yahav.js +206 -190
- package/lib/scrapers/yahav.test.d.ts +1 -0
- package/lib/scrapers/yahav.test.js +49 -0
- package/lib/transactions.js +16 -13
- package/package.json +8 -3
package/lib/scrapers/max.js
CHANGED
|
@@ -1,299 +1,299 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
exports.
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.default = void 0;
|
|
7
|
+
exports.getMemo = getMemo;
|
|
8
|
+
var _buildUrl = _interopRequireDefault(require("build-url"));
|
|
9
|
+
var _moment = _interopRequireDefault(require("moment"));
|
|
10
|
+
var _constants = require("../constants");
|
|
11
|
+
var _dates = _interopRequireDefault(require("../helpers/dates"));
|
|
12
|
+
var _debug = require("../helpers/debug");
|
|
13
|
+
var _elementsInteractions = require("../helpers/elements-interactions");
|
|
14
|
+
var _fetch = require("../helpers/fetch");
|
|
15
|
+
var _navigation = require("../helpers/navigation");
|
|
16
|
+
var _transactions = require("../helpers/transactions");
|
|
17
|
+
var _transactions2 = require("../transactions");
|
|
18
|
+
var _baseScraperWithBrowser = require("./base-scraper-with-browser");
|
|
19
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
20
|
+
const debug = (0, _debug.getDebug)('max');
|
|
19
21
|
const BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';
|
|
20
22
|
const BASE_WELCOME_URL = 'https://www.max.co.il';
|
|
21
23
|
const LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;
|
|
22
24
|
const PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;
|
|
23
25
|
const SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;
|
|
24
|
-
var MaxPlanName
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
}
|
|
26
|
+
var MaxPlanName = /*#__PURE__*/function (MaxPlanName) {
|
|
27
|
+
MaxPlanName["Normal"] = "\u05E8\u05D2\u05D9\u05DC\u05D4";
|
|
28
|
+
MaxPlanName["ImmediateCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05E2\u05E1\u05E7\u05D5\u05EA \u05DE\u05D9\u05D9\u05D3\u05D9";
|
|
29
|
+
MaxPlanName["InternetShopping"] = "\u05D0\u05D9\u05E0\u05D8\u05E8\u05E0\u05D8/\u05D7\u05D5\"\u05DC";
|
|
30
|
+
MaxPlanName["Installments"] = "\u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
|
|
31
|
+
MaxPlanName["MonthlyCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05D7\u05D5\u05D3\u05E9\u05D9";
|
|
32
|
+
MaxPlanName["OneMonthPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9";
|
|
33
|
+
MaxPlanName["MonthlyPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05DC\u05D7\u05D9\u05D5\u05D1 \u05D4\u05D7\u05D5\u05D3\u05E9\u05D9";
|
|
34
|
+
MaxPlanName["MonthlyPayment"] = "\u05EA\u05E9\u05DC\u05D5\u05DD \u05D7\u05D5\u05D3\u05E9\u05D9";
|
|
35
|
+
MaxPlanName["FuturePurchaseFinancing"] = "\u05DE\u05D9\u05DE\u05D5\u05DF \u05DC\u05E8\u05DB\u05D9\u05E9\u05D4 \u05E2\u05EA\u05D9\u05D3\u05D9\u05EA";
|
|
36
|
+
MaxPlanName["MonthlyPostponedInstallments"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9 \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
|
|
37
|
+
MaxPlanName["ThirtyDaysPlus"] = "\u05E2\u05E1\u05E7\u05EA 30 \u05E4\u05DC\u05D5\u05E1";
|
|
38
|
+
MaxPlanName["TwoMonthsPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9\u05D9\u05D9\u05DD";
|
|
39
|
+
MaxPlanName["TwoMonthsPostponed2"] = "\u05D3\u05D7\u05D5\u05D9 2 \u05D7' \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
|
|
40
|
+
MaxPlanName["MonthlyChargePlusInterest"] = "\u05D7\u05D5\u05D3\u05E9\u05D9 + \u05E8\u05D9\u05D1\u05D9\u05EA";
|
|
41
|
+
MaxPlanName["Credit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8";
|
|
42
|
+
MaxPlanName["CreditOutsideTheLimit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8-\u05DE\u05D7\u05D5\u05E5 \u05DC\u05DE\u05E1\u05D2\u05E8\u05EA";
|
|
43
|
+
MaxPlanName["AccumulatingBasket"] = "\u05E1\u05DC \u05DE\u05E6\u05D8\u05D1\u05E8";
|
|
44
|
+
MaxPlanName["PostponedTransactionInstallments"] = "\u05E4\u05E8\u05D9\u05E1\u05EA \u05D4\u05E2\u05E1\u05E7\u05D4 \u05D4\u05D3\u05D7\u05D5\u05D9\u05D4";
|
|
45
|
+
MaxPlanName["ReplacementCard"] = "\u05DB\u05E8\u05D8\u05D9\u05E1 \u05D7\u05DC\u05D9\u05E4\u05D9";
|
|
46
|
+
MaxPlanName["EarlyRepayment"] = "\u05E4\u05E8\u05E2\u05D5\u05DF \u05DE\u05D5\u05E7\u05D3\u05DD";
|
|
47
|
+
MaxPlanName["MonthlyCardFee"] = "\u05D3\u05DE\u05D9 \u05DB\u05E8\u05D8\u05D9\u05E1";
|
|
48
|
+
MaxPlanName["CurrencyPocket"] = "\u05D7\u05D9\u05D5\u05D1 \u05D0\u05E8\u05E0\u05E7 \u05DE\u05D8\u05D7";
|
|
49
|
+
return MaxPlanName;
|
|
50
|
+
}(MaxPlanName || {});
|
|
49
51
|
const INVALID_DETAILS_SELECTOR = '#popupWrongDetails';
|
|
50
52
|
const LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';
|
|
51
53
|
const categories = new Map();
|
|
52
54
|
function redirectOrDialog(page) {
|
|
53
|
-
|
|
54
|
-
(0, navigation_1.waitForRedirect)(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),
|
|
55
|
-
(0, elements_interactions_1.waitUntilElementFound)(page, INVALID_DETAILS_SELECTOR, true),
|
|
56
|
-
(0, elements_interactions_1.waitUntilElementFound)(page, LOGIN_ERROR_SELECTOR, true),
|
|
57
|
-
]);
|
|
55
|
+
return Promise.race([(0, _navigation.waitForRedirect)(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]), (0, _elementsInteractions.waitUntilElementFound)(page, INVALID_DETAILS_SELECTOR, true), (0, _elementsInteractions.waitUntilElementFound)(page, LOGIN_ERROR_SELECTOR, true)]);
|
|
58
56
|
}
|
|
59
57
|
function getTransactionsUrl(monthMoment) {
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
}
|
|
58
|
+
const month = monthMoment.month() + 1;
|
|
59
|
+
const year = monthMoment.year();
|
|
60
|
+
const date = `${year}-${month}-01`;
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* url explanation:
|
|
64
|
+
* userIndex: -1 for all account owners
|
|
65
|
+
* cardIndex: -1 for all cards under the account
|
|
66
|
+
* all other query params are static, beside the date which changes for request per month
|
|
67
|
+
*/
|
|
68
|
+
return (0, _buildUrl.default)(BASE_API_ACTIONS_URL, {
|
|
69
|
+
path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={"userIndex":-1,"cardIndex":-1,"monthView":true,"date":"${date}","dates":{"startDate":"0","endDate":"0"},"bankAccount":{"bankAccountIndex":-1,"cards":null}}&firstCallCardIndex=-1`
|
|
70
|
+
});
|
|
72
71
|
}
|
|
73
72
|
async function loadCategories(page) {
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
73
|
+
debug('Loading categories');
|
|
74
|
+
const res = await (0, _fetch.fetchGetWithinPage)(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);
|
|
75
|
+
if (res && Array.isArray(res.result)) {
|
|
76
|
+
debug(`${res.result.length} categories loaded`);
|
|
77
|
+
res.result?.forEach(({
|
|
78
|
+
id,
|
|
79
|
+
name
|
|
80
|
+
}) => categories.set(id, name));
|
|
81
|
+
}
|
|
80
82
|
}
|
|
81
83
|
function getTransactionType(planName, planTypeId) {
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
84
|
+
const cleanedUpTxnTypeStr = planName.replace('\t', ' ').trim();
|
|
85
|
+
switch (cleanedUpTxnTypeStr) {
|
|
86
|
+
case MaxPlanName.ImmediateCharge:
|
|
87
|
+
case MaxPlanName.Normal:
|
|
88
|
+
case MaxPlanName.MonthlyCharge:
|
|
89
|
+
case MaxPlanName.OneMonthPostponed:
|
|
90
|
+
case MaxPlanName.MonthlyPostponed:
|
|
91
|
+
case MaxPlanName.FuturePurchaseFinancing:
|
|
92
|
+
case MaxPlanName.MonthlyPayment:
|
|
93
|
+
case MaxPlanName.MonthlyPostponedInstallments:
|
|
94
|
+
case MaxPlanName.ThirtyDaysPlus:
|
|
95
|
+
case MaxPlanName.TwoMonthsPostponed:
|
|
96
|
+
case MaxPlanName.TwoMonthsPostponed2:
|
|
97
|
+
case MaxPlanName.AccumulatingBasket:
|
|
98
|
+
case MaxPlanName.InternetShopping:
|
|
99
|
+
case MaxPlanName.MonthlyChargePlusInterest:
|
|
100
|
+
case MaxPlanName.PostponedTransactionInstallments:
|
|
101
|
+
case MaxPlanName.ReplacementCard:
|
|
102
|
+
case MaxPlanName.EarlyRepayment:
|
|
103
|
+
case MaxPlanName.MonthlyCardFee:
|
|
104
|
+
case MaxPlanName.CurrencyPocket:
|
|
105
|
+
return _transactions2.TransactionTypes.Normal;
|
|
106
|
+
case MaxPlanName.Installments:
|
|
107
|
+
case MaxPlanName.Credit:
|
|
108
|
+
case MaxPlanName.CreditOutsideTheLimit:
|
|
109
|
+
return _transactions2.TransactionTypes.Installments;
|
|
110
|
+
default:
|
|
111
|
+
switch (planTypeId) {
|
|
112
|
+
case 2:
|
|
113
|
+
case 3:
|
|
114
|
+
return _transactions2.TransactionTypes.Installments;
|
|
115
|
+
case 5:
|
|
116
|
+
return _transactions2.TransactionTypes.Normal;
|
|
108
117
|
default:
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
return transactions_2.TransactionTypes.Installments;
|
|
113
|
-
case 5:
|
|
114
|
-
return transactions_2.TransactionTypes.Normal;
|
|
115
|
-
default:
|
|
116
|
-
throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);
|
|
117
|
-
}
|
|
118
|
-
}
|
|
118
|
+
throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
119
121
|
}
|
|
120
122
|
function getInstallmentsInfo(comments) {
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
123
|
+
if (!comments) {
|
|
124
|
+
return undefined;
|
|
125
|
+
}
|
|
126
|
+
const matches = comments.match(/\d+/g);
|
|
127
|
+
if (!matches || matches.length < 2) {
|
|
128
|
+
return undefined;
|
|
129
|
+
}
|
|
130
|
+
return {
|
|
131
|
+
number: parseInt(matches[0], 10),
|
|
132
|
+
total: parseInt(matches[1], 10)
|
|
133
|
+
};
|
|
132
134
|
}
|
|
133
135
|
function getChargedCurrency(currencyId) {
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
136
|
+
switch (currencyId) {
|
|
137
|
+
case 376:
|
|
138
|
+
return _constants.SHEKEL_CURRENCY;
|
|
139
|
+
case 840:
|
|
140
|
+
return _constants.DOLLAR_CURRENCY;
|
|
141
|
+
case 978:
|
|
142
|
+
return _constants.EURO_CURRENCY;
|
|
143
|
+
default:
|
|
144
|
+
return undefined;
|
|
145
|
+
}
|
|
144
146
|
}
|
|
145
|
-
function getMemo({
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
147
|
+
function getMemo({
|
|
148
|
+
comments,
|
|
149
|
+
fundsTransferReceiverOrTransfer,
|
|
150
|
+
fundsTransferComment
|
|
151
|
+
}) {
|
|
152
|
+
if (fundsTransferReceiverOrTransfer) {
|
|
153
|
+
const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;
|
|
154
|
+
return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;
|
|
155
|
+
}
|
|
156
|
+
return comments;
|
|
151
157
|
}
|
|
152
|
-
exports.getMemo = getMemo;
|
|
153
158
|
function mapTransaction(rawTransaction) {
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
status,
|
|
175
|
-
};
|
|
159
|
+
const isPending = rawTransaction.paymentDate === null;
|
|
160
|
+
const processedDate = (0, _moment.default)(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();
|
|
161
|
+
const status = isPending ? _transactions2.TransactionStatuses.Pending : _transactions2.TransactionStatuses.Completed;
|
|
162
|
+
const installments = getInstallmentsInfo(rawTransaction.comments);
|
|
163
|
+
const identifier = installments ? `${rawTransaction.dealData?.arn}_${installments.number}` : rawTransaction.dealData?.arn;
|
|
164
|
+
return {
|
|
165
|
+
type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),
|
|
166
|
+
date: (0, _moment.default)(rawTransaction.purchaseDate).toISOString(),
|
|
167
|
+
processedDate,
|
|
168
|
+
originalAmount: -rawTransaction.originalAmount,
|
|
169
|
+
originalCurrency: rawTransaction.originalCurrency,
|
|
170
|
+
chargedAmount: -rawTransaction.actualPaymentAmount,
|
|
171
|
+
chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),
|
|
172
|
+
description: rawTransaction.merchantName.trim(),
|
|
173
|
+
memo: getMemo(rawTransaction),
|
|
174
|
+
category: categories.get(rawTransaction?.categoryId),
|
|
175
|
+
installments,
|
|
176
|
+
identifier,
|
|
177
|
+
status
|
|
178
|
+
};
|
|
176
179
|
}
|
|
177
180
|
async function fetchTransactionsForMonth(page, monthMoment) {
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
});
|
|
193
|
-
return transactionsByAccount;
|
|
181
|
+
const url = getTransactionsUrl(monthMoment);
|
|
182
|
+
const data = await (0, _fetch.fetchGetWithinPage)(page, url);
|
|
183
|
+
const transactionsByAccount = {};
|
|
184
|
+
if (!data || !data.result) return transactionsByAccount;
|
|
185
|
+
data.result.transactions
|
|
186
|
+
// Filter out non-transactions without a plan type, e.g. summary rows
|
|
187
|
+
.filter(transaction => !!transaction.planName).forEach(transaction => {
|
|
188
|
+
if (!transactionsByAccount[transaction.shortCardNumber]) {
|
|
189
|
+
transactionsByAccount[transaction.shortCardNumber] = [];
|
|
190
|
+
}
|
|
191
|
+
const mappedTransaction = mapTransaction(transaction);
|
|
192
|
+
transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);
|
|
193
|
+
});
|
|
194
|
+
return transactionsByAccount;
|
|
194
195
|
}
|
|
195
196
|
function addResult(allResults, result) {
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
}
|
|
203
|
-
|
|
197
|
+
const clonedResults = {
|
|
198
|
+
...allResults
|
|
199
|
+
};
|
|
200
|
+
Object.keys(result).forEach(accountNumber => {
|
|
201
|
+
if (!clonedResults[accountNumber]) {
|
|
202
|
+
clonedResults[accountNumber] = [];
|
|
203
|
+
}
|
|
204
|
+
clonedResults[accountNumber].push(...result[accountNumber]);
|
|
205
|
+
});
|
|
206
|
+
return clonedResults;
|
|
204
207
|
}
|
|
205
208
|
function prepareTransactions(txns, startMoment, combineInstallments, enableTransactionsFilterByDate) {
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
: clonedTxns;
|
|
214
|
-
return clonedTxns;
|
|
209
|
+
let clonedTxns = Array.from(txns);
|
|
210
|
+
if (!combineInstallments) {
|
|
211
|
+
clonedTxns = (0, _transactions.fixInstallments)(clonedTxns);
|
|
212
|
+
}
|
|
213
|
+
clonedTxns = (0, _transactions.sortTransactionsByDate)(clonedTxns);
|
|
214
|
+
clonedTxns = enableTransactionsFilterByDate ? (0, _transactions.filterOldTransactions)(clonedTxns, startMoment, combineInstallments || false) : clonedTxns;
|
|
215
|
+
return clonedTxns;
|
|
215
216
|
}
|
|
216
217
|
async function fetchTransactions(page, options) {
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
218
|
+
const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;
|
|
219
|
+
const defaultStartMoment = (0, _moment.default)().subtract(1, 'years');
|
|
220
|
+
const startMomentLimit = (0, _moment.default)().subtract(4, 'years');
|
|
221
|
+
const startDate = options.startDate || defaultStartMoment.toDate();
|
|
222
|
+
const startMoment = _moment.default.max(startMomentLimit, (0, _moment.default)(startDate));
|
|
223
|
+
const allMonths = (0, _dates.default)(startMoment, futureMonthsToScrape);
|
|
224
|
+
await loadCategories(page);
|
|
225
|
+
let allResults = {};
|
|
226
|
+
for (let i = 0; i < allMonths.length; i += 1) {
|
|
227
|
+
const result = await fetchTransactionsForMonth(page, allMonths[i]);
|
|
228
|
+
allResults = addResult(allResults, result);
|
|
229
|
+
}
|
|
230
|
+
Object.keys(allResults).forEach(accountNumber => {
|
|
231
|
+
let txns = allResults[accountNumber];
|
|
232
|
+
txns = prepareTransactions(txns, startMoment, options.combineInstallments || false, options.outputData?.enableTransactionsFilterByDate ?? true);
|
|
233
|
+
allResults[accountNumber] = txns;
|
|
234
|
+
});
|
|
235
|
+
return allResults;
|
|
235
236
|
}
|
|
236
237
|
function getPossibleLoginResults(page) {
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
return (0, elements_interactions_1.elementPresentOnPage)(page, LOGIN_ERROR_SELECTOR);
|
|
248
|
-
},
|
|
249
|
-
];
|
|
250
|
-
return urls;
|
|
238
|
+
const urls = {};
|
|
239
|
+
urls[_baseScraperWithBrowser.LoginResults.Success] = [SUCCESS_URL];
|
|
240
|
+
urls[_baseScraperWithBrowser.LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];
|
|
241
|
+
urls[_baseScraperWithBrowser.LoginResults.InvalidPassword] = [async () => {
|
|
242
|
+
return (0, _elementsInteractions.elementPresentOnPage)(page, INVALID_DETAILS_SELECTOR);
|
|
243
|
+
}];
|
|
244
|
+
urls[_baseScraperWithBrowser.LoginResults.UnknownError] = [async () => {
|
|
245
|
+
return (0, _elementsInteractions.elementPresentOnPage)(page, LOGIN_ERROR_SELECTOR);
|
|
246
|
+
}];
|
|
247
|
+
return urls;
|
|
251
248
|
}
|
|
252
249
|
function createLoginFields(credentials) {
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
250
|
+
return [{
|
|
251
|
+
selector: '#user-name',
|
|
252
|
+
value: credentials.username
|
|
253
|
+
}, {
|
|
254
|
+
selector: '#password',
|
|
255
|
+
value: credentials.password
|
|
256
|
+
}];
|
|
257
257
|
}
|
|
258
|
-
class MaxScraper extends
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
258
|
+
class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
259
|
+
getLoginOptions(credentials) {
|
|
260
|
+
return {
|
|
261
|
+
loginUrl: LOGIN_URL,
|
|
262
|
+
fields: createLoginFields(credentials),
|
|
263
|
+
submitButtonSelector: 'app-user-login-form .general-button.send-me-code',
|
|
264
|
+
preAction: async () => {
|
|
265
|
+
if (await (0, _elementsInteractions.elementPresentOnPage)(this.page, '#closePopup')) {
|
|
266
|
+
await (0, _elementsInteractions.clickButton)(this.page, '#closePopup');
|
|
267
|
+
}
|
|
268
|
+
await (0, _elementsInteractions.clickButton)(this.page, '.personal-area > a.go-to-personal-area');
|
|
269
|
+
if (await (0, _elementsInteractions.elementPresentOnPage)(this.page, '.login-link#private')) {
|
|
270
|
+
await (0, _elementsInteractions.clickButton)(this.page, '.login-link#private');
|
|
271
|
+
}
|
|
272
|
+
await (0, _elementsInteractions.waitUntilElementFound)(this.page, '#login-password-link', true);
|
|
273
|
+
await (0, _elementsInteractions.clickButton)(this.page, '#login-password-link');
|
|
274
|
+
await (0, _elementsInteractions.waitUntilElementFound)(this.page, '#login-password.tab-pane.active app-user-login-form', true);
|
|
275
|
+
},
|
|
276
|
+
checkReadiness: async () => {
|
|
277
|
+
await (0, _elementsInteractions.waitUntilElementFound)(this.page, '.personal-area > a.go-to-personal-area', true);
|
|
278
|
+
},
|
|
279
|
+
postAction: async () => redirectOrDialog(this.page),
|
|
280
|
+
possibleResults: getPossibleLoginResults(this.page),
|
|
281
|
+
waitUntil: 'domcontentloaded'
|
|
282
|
+
};
|
|
283
|
+
}
|
|
284
|
+
async fetchData() {
|
|
285
|
+
const results = await fetchTransactions(this.page, this.options);
|
|
286
|
+
const accounts = Object.keys(results).map(accountNumber => {
|
|
287
|
+
return {
|
|
288
|
+
accountNumber,
|
|
289
|
+
txns: results[accountNumber]
|
|
290
|
+
};
|
|
291
|
+
});
|
|
292
|
+
return {
|
|
293
|
+
success: true,
|
|
294
|
+
accounts
|
|
295
|
+
};
|
|
296
|
+
}
|
|
297
297
|
}
|
|
298
|
-
exports.default = MaxScraper;
|
|
299
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"max.js","sourceRoot":"","sources":["../../src/scrapers/max.ts"],"names":[],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { DOLLAR_CURRENCY, EURO_CURRENCY, SHEKEL_CURRENCY } from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { filterOldTransactions, fixInstallments, sortTransactionsByDate } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport {\n  BaseScraperWithBrowser,\n  LoginResults,\n  type LoginOptions,\n  type PossibleLoginResults,\n} from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\nexport interface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  paymentCurrency: number | null;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  planTypeId: number;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  fundsTransferComment?: string;\n  fundsTransferReceiverOrTransfer?: string;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nenum MaxPlanName {\n  Normal = 'רגילה',\n  ImmediateCharge = 'חיוב עסקות מיידי',\n  InternetShopping = 'אינטרנט/חו\"ל',\n  Installments = 'תשלומים',\n  MonthlyCharge = 'חיוב חודשי',\n  OneMonthPostponed = 'דחוי חודש',\n  MonthlyPostponed = 'דחוי לחיוב החודשי',\n  MonthlyPayment = 'תשלום חודשי',\n  FuturePurchaseFinancing = 'מימון לרכישה עתידית',\n  MonthlyPostponedInstallments = 'דחוי חודש תשלומים',\n  ThirtyDaysPlus = 'עסקת 30 פלוס',\n  TwoMonthsPostponed = 'דחוי חודשיים',\n  TwoMonthsPostponed2 = \"דחוי 2 ח' תשלומים\",\n  MonthlyChargePlusInterest = 'חודשי + ריבית',\n  Credit = 'קרדיט',\n  CreditOutsideTheLimit = 'קרדיט-מחוץ למסגרת',\n  AccumulatingBasket = 'סל מצטבר',\n  PostponedTransactionInstallments = 'פריסת העסקה הדחויה',\n  ReplacementCard = 'כרטיס חליפי',\n  EarlyRepayment = 'פרעון מוקדם',\n  MonthlyCardFee = 'דמי כרטיס',\n  CurrencyPocket = 'חיוב ארנק מטח',\n}\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n   * url explanation:\n   * userIndex: -1 for all account owners\n   * cardIndex: -1 for all cards under the account\n   * all other query params are static, beside the date which changes for request per month\n   */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result?: Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n    res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(planName: string, planTypeId: number) {\n  const cleanedUpTxnTypeStr = planName.replace('\\t', ' ').trim() as MaxPlanName;\n  switch (cleanedUpTxnTypeStr) {\n    case MaxPlanName.ImmediateCharge:\n    case MaxPlanName.Normal:\n    case MaxPlanName.MonthlyCharge:\n    case MaxPlanName.OneMonthPostponed:\n    case MaxPlanName.MonthlyPostponed:\n    case MaxPlanName.FuturePurchaseFinancing:\n    case MaxPlanName.MonthlyPayment:\n    case MaxPlanName.MonthlyPostponedInstallments:\n    case MaxPlanName.ThirtyDaysPlus:\n    case MaxPlanName.TwoMonthsPostponed:\n    case MaxPlanName.TwoMonthsPostponed2:\n    case MaxPlanName.AccumulatingBasket:\n    case MaxPlanName.InternetShopping:\n    case MaxPlanName.MonthlyChargePlusInterest:\n    case MaxPlanName.PostponedTransactionInstallments:\n    case MaxPlanName.ReplacementCard:\n    case MaxPlanName.EarlyRepayment:\n    case MaxPlanName.MonthlyCardFee:\n    case MaxPlanName.CurrencyPocket:\n      return TransactionTypes.Normal;\n    case MaxPlanName.Installments:\n    case MaxPlanName.Credit:\n    case MaxPlanName.CreditOutsideTheLimit:\n      return TransactionTypes.Installments;\n    default:\n      switch (planTypeId) {\n        case 2:\n        case 3:\n          return TransactionTypes.Installments;\n        case 5:\n          return TransactionTypes.Normal;\n        default:\n          throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr as string}`);\n      }\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getChargedCurrency(currencyId: number | null) {\n  switch (currencyId) {\n    case 376:\n      return SHEKEL_CURRENCY;\n    case 840:\n      return DOLLAR_CURRENCY;\n    case 978:\n      return EURO_CURRENCY;\n    default:\n      return undefined;\n  }\n}\n\nexport function getMemo({\n  comments,\n  fundsTransferReceiverOrTransfer,\n  fundsTransferComment,\n}: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>) {\n  if (fundsTransferReceiverOrTransfer) {\n    const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;\n    return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;\n  }\n\n  return comments;\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments\n    ? `${rawTransaction.dealData?.arn}_${installments.number}`\n    : rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),\n    description: rawTransaction.merchantName.trim(),\n    memo: getMemo(rawTransaction),\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult {\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter(transaction => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach(accountNumber => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(\n  txns: Transaction[],\n  startMoment: moment.Moment,\n  combineInstallments: boolean,\n  enableTransactionsFilterByDate: boolean,\n) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = enableTransactionsFilterByDate\n    ? filterOldTransactions(clonedTxns, startMoment, combineInstallments || false)\n    : clonedTxns;\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startMomentLimit = moment().subtract(4, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(startMomentLimit, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach(accountNumber => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(\n      txns,\n      startMoment,\n      options.combineInstallments || false,\n      options.outputData?.enableTransactionsFilterByDate ?? true,\n    );\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [\n    async () => {\n      return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n    },\n  ];\n  urls[LoginResults.UnknownError] = [\n    async () => {\n      return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n    },\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'app-user-login-form .general-button.send-me-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        if (await elementPresentOnPage(this.page, '.login-link#private')) {\n          await clickButton(this.page, '.login-link#private');\n        }\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n      waitUntil: 'domcontentloaded',\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map(accountNumber => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"]}
|
|
298
|
+
var _default = exports.default = MaxScraper;
|
|
299
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_buildUrl","_interopRequireDefault","require","_moment","_constants","_dates","_debug","_elementsInteractions","_fetch","_navigation","_transactions","_transactions2","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","MaxPlanName","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","waitForRedirect","waitUntilElementFound","getTransactionsUrl","monthMoment","month","year","date","buildUrl","path","loadCategories","res","fetchGetWithinPage","Array","isArray","result","length","forEach","id","name","set","getTransactionType","planName","planTypeId","cleanedUpTxnTypeStr","replace","trim","ImmediateCharge","Normal","MonthlyCharge","OneMonthPostponed","MonthlyPostponed","FuturePurchaseFinancing","MonthlyPayment","MonthlyPostponedInstallments","ThirtyDaysPlus","TwoMonthsPostponed","TwoMonthsPostponed2","AccumulatingBasket","InternetShopping","MonthlyChargePlusInterest","PostponedTransactionInstallments","ReplacementCard","EarlyRepayment","MonthlyCardFee","CurrencyPocket","TransactionTypes","Installments","Credit","CreditOutsideTheLimit","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","getChargedCurrency","currencyId","SHEKEL_CURRENCY","DOLLAR_CURRENCY","EURO_CURRENCY","getMemo","fundsTransferReceiverOrTransfer","fundsTransferComment","memo","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","moment","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","chargedCurrency","paymentCurrency","description","merchantName","category","get","categoryId","fetchTransactionsForMonth","url","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","enableTransactionsFilterByDate","clonedTxns","from","fixInstallments","sortTransactionsByDate","filterOldTransactions","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startMomentLimit","startDate","toDate","max","allMonths","getAllMonthMoments","i","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","elementPresentOnPage","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","clickButton","checkReadiness","postAction","possibleResults","waitUntil","fetchData","results","accounts","map","success","_default","exports"],"sources":["../../src/scrapers/max.ts"],"sourcesContent":["import buildUrl from 'build-url';\nimport moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { DOLLAR_CURRENCY, EURO_CURRENCY, SHEKEL_CURRENCY } from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { filterOldTransactions, fixInstallments, sortTransactionsByDate } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport {\n  BaseScraperWithBrowser,\n  LoginResults,\n  type LoginOptions,\n  type PossibleLoginResults,\n} from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\nexport interface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  paymentCurrency: number | null;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  planTypeId: number;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  fundsTransferComment?: string;\n  fundsTransferReceiverOrTransfer?: string;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nenum MaxPlanName {\n  Normal = 'רגילה',\n  ImmediateCharge = 'חיוב עסקות מיידי',\n  InternetShopping = 'אינטרנט/חו\"ל',\n  Installments = 'תשלומים',\n  MonthlyCharge = 'חיוב חודשי',\n  OneMonthPostponed = 'דחוי חודש',\n  MonthlyPostponed = 'דחוי לחיוב החודשי',\n  MonthlyPayment = 'תשלום חודשי',\n  FuturePurchaseFinancing = 'מימון לרכישה עתידית',\n  MonthlyPostponedInstallments = 'דחוי חודש תשלומים',\n  ThirtyDaysPlus = 'עסקת 30 פלוס',\n  TwoMonthsPostponed = 'דחוי חודשיים',\n  TwoMonthsPostponed2 = \"דחוי 2 ח' תשלומים\",\n  MonthlyChargePlusInterest = 'חודשי + ריבית',\n  Credit = 'קרדיט',\n  CreditOutsideTheLimit = 'קרדיט-מחוץ למסגרת',\n  AccumulatingBasket = 'סל מצטבר',\n  PostponedTransactionInstallments = 'פריסת העסקה הדחויה',\n  ReplacementCard = 'כרטיס חליפי',\n  EarlyRepayment = 'פרעון מוקדם',\n  MonthlyCardFee = 'דמי כרטיס',\n  CurrencyPocket = 'חיוב ארנק מטח',\n}\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n   * url explanation:\n   * userIndex: -1 for all account owners\n   * cardIndex: -1 for all cards under the account\n   * all other query params are static, beside the date which changes for request per month\n   */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result?: Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n    res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(planName: string, planTypeId: number) {\n  const cleanedUpTxnTypeStr = planName.replace('\\t', ' ').trim() as MaxPlanName;\n  switch (cleanedUpTxnTypeStr) {\n    case MaxPlanName.ImmediateCharge:\n    case MaxPlanName.Normal:\n    case MaxPlanName.MonthlyCharge:\n    case MaxPlanName.OneMonthPostponed:\n    case MaxPlanName.MonthlyPostponed:\n    case MaxPlanName.FuturePurchaseFinancing:\n    case MaxPlanName.MonthlyPayment:\n    case MaxPlanName.MonthlyPostponedInstallments:\n    case MaxPlanName.ThirtyDaysPlus:\n    case MaxPlanName.TwoMonthsPostponed:\n    case MaxPlanName.TwoMonthsPostponed2:\n    case MaxPlanName.AccumulatingBasket:\n    case MaxPlanName.InternetShopping:\n    case MaxPlanName.MonthlyChargePlusInterest:\n    case MaxPlanName.PostponedTransactionInstallments:\n    case MaxPlanName.ReplacementCard:\n    case MaxPlanName.EarlyRepayment:\n    case MaxPlanName.MonthlyCardFee:\n    case MaxPlanName.CurrencyPocket:\n      return TransactionTypes.Normal;\n    case MaxPlanName.Installments:\n    case MaxPlanName.Credit:\n    case MaxPlanName.CreditOutsideTheLimit:\n      return TransactionTypes.Installments;\n    default:\n      switch (planTypeId) {\n        case 2:\n        case 3:\n          return TransactionTypes.Installments;\n        case 5:\n          return TransactionTypes.Normal;\n        default:\n          throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr as string}`);\n      }\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getChargedCurrency(currencyId: number | null) {\n  switch (currencyId) {\n    case 376:\n      return SHEKEL_CURRENCY;\n    case 840:\n      return DOLLAR_CURRENCY;\n    case 978:\n      return EURO_CURRENCY;\n    default:\n      return undefined;\n  }\n}\n\nexport function getMemo({\n  comments,\n  fundsTransferReceiverOrTransfer,\n  fundsTransferComment,\n}: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>) {\n  if (fundsTransferReceiverOrTransfer) {\n    const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;\n    return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;\n  }\n\n  return comments;\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments\n    ? `${rawTransaction.dealData?.arn}_${installments.number}`\n    : rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),\n    description: rawTransaction.merchantName.trim(),\n    memo: getMemo(rawTransaction),\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult {\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter(transaction => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach(accountNumber => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(\n  txns: Transaction[],\n  startMoment: moment.Moment,\n  combineInstallments: boolean,\n  enableTransactionsFilterByDate: boolean,\n) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = enableTransactionsFilterByDate\n    ? filterOldTransactions(clonedTxns, startMoment, combineInstallments || false)\n    : clonedTxns;\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startMomentLimit = moment().subtract(4, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(startMomentLimit, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach(accountNumber => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(\n      txns,\n      startMoment,\n      options.combineInstallments || false,\n      options.outputData?.enableTransactionsFilterByDate ?? true,\n    );\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [\n    async () => {\n      return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n    },\n  ];\n  urls[LoginResults.UnknownError] = [\n    async () => {\n      return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n    },\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'app-user-login-form .general-button.send-me-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        if (await elementPresentOnPage(this.page, '.login-link#private')) {\n          await clickButton(this.page, '.login-link#private');\n        }\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n      waitUntil: 'domcontentloaded',\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map(accountNumber => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"],"mappings":";;;;;;;AAAA,IAAAA,SAAA,GAAAC,sBAAA,CAAAC,OAAA;AACA,IAAAC,OAAA,GAAAF,sBAAA,CAAAC,OAAA;AAEA,IAAAE,UAAA,GAAAF,OAAA;AACA,IAAAG,MAAA,GAAAJ,sBAAA,CAAAC,OAAA;AACA,IAAAI,MAAA,GAAAJ,OAAA;AACA,IAAAK,qBAAA,GAAAL,OAAA;AACA,IAAAM,MAAA,GAAAN,OAAA;AACA,IAAAO,WAAA,GAAAP,OAAA;AACA,IAAAQ,aAAA,GAAAR,OAAA;AACA,IAAAS,cAAA,GAAAT,OAAA;AACA,IAAAU,uBAAA,GAAAV,OAAA;AAKqC,SAAAD,uBAAAY,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAGrC,MAAMG,KAAK,GAAG,IAAAC,eAAQ,EAAC,KAAK,CAAC;AAsB7B,MAAMC,oBAAoB,GAAG,+BAA+B;AAC5D,MAAMC,gBAAgB,GAAG,uBAAuB;AAEhD,MAAMC,SAAS,GAAG,GAAGD,gBAAgB,mBAAmB;AACxD,MAAME,oBAAoB,GAAG,GAAGF,gBAAgB,iBAAiB;AACjE,MAAMG,WAAW,GAAG,GAAGH,gBAAgB,oBAAoB;AAAC,IAEvDI,WAAW,0BAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAXA,WAAW;EAAA,OAAXA,WAAW;AAAA,EAAXA,WAAW;AAyBhB,MAAMC,wBAAwB,GAAG,oBAAoB;AACrD,MAAMC,oBAAoB,GAAG,6BAA6B;AAE1D,MAAMC,UAAU,GAAG,IAAIC,GAAG,CAAiB,CAAC;AAE5C,SAASC,gBAAgBA,CAACC,IAAU,EAAE;EACpC,OAAOC,OAAO,CAACC,IAAI,CAAC,CAClB,IAAAC,2BAAe,EAACH,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAACV,gBAAgB,EAAE,GAAGA,gBAAgB,GAAG,CAAC,CAAC,EAC/E,IAAAc,2CAAqB,EAACJ,IAAI,EAAEL,wBAAwB,EAAE,IAAI,CAAC,EAC3D,IAAAS,2CAAqB,EAACJ,IAAI,EAAEJ,oBAAoB,EAAE,IAAI,CAAC,CACxD,CAAC;AACJ;AAEA,SAASS,kBAAkBA,CAACC,WAAmB,EAAE;EAC/C,MAAMC,KAAK,GAAGD,WAAW,CAACC,KAAK,CAAC,CAAC,GAAG,CAAC;EACrC,MAAMC,IAAI,GAAGF,WAAW,CAACE,IAAI,CAAC,CAAC;EAC/B,MAAMC,IAAI,GAAG,GAAGD,IAAI,IAAID,KAAK,KAAK;;EAElC;AACF;AACA;AACA;AACA;AACA;EACE,OAAO,IAAAG,iBAAQ,EAACrB,oBAAoB,EAAE;IACpCsB,IAAI,EAAE,kIAAkIF,IAAI;EAC9I,CAAC,CAAC;AACJ;AASA,eAAeG,cAAcA,CAACZ,IAAU,EAAE;EACxCb,KAAK,CAAC,oBAAoB,CAAC;EAC3B,MAAM0B,GAAG,GAAG,MAAM,IAAAC,yBAAkB,EAAsBd,IAAI,EAAE,GAAGX,oBAAoB,6BAA6B,CAAC;EACrH,IAAIwB,GAAG,IAAIE,KAAK,CAACC,OAAO,CAACH,GAAG,CAACI,MAAM,CAAC,EAAE;IACpC9B,KAAK,CAAC,GAAG0B,GAAG,CAACI,MAAM,CAACC,MAAM,oBAAoB,CAAC;IAC/CL,GAAG,CAACI,MAAM,EAAEE,OAAO,CAAC,CAAC;MAAEC,EAAE;MAAEC;IAAK,CAAC,KAAKxB,UAAU,CAACyB,GAAG,CAACF,EAAE,EAAEC,IAAI,CAAC,CAAC;EACjE;AACF;AAEA,SAASE,kBAAkBA,CAACC,QAAgB,EAAEC,UAAkB,EAAE;EAChE,MAAMC,mBAAmB,GAAGF,QAAQ,CAACG,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAACC,IAAI,CAAC,CAAgB;EAC7E,QAAQF,mBAAmB;IACzB,KAAKhC,WAAW,CAACmC,eAAe;IAChC,KAAKnC,WAAW,CAACoC,MAAM;IACvB,KAAKpC,WAAW,CAACqC,aAAa;IAC9B,KAAKrC,WAAW,CAACsC,iBAAiB;IAClC,KAAKtC,WAAW,CAACuC,gBAAgB;IACjC,KAAKvC,WAAW,CAACwC,uBAAuB;IACxC,KAAKxC,WAAW,CAACyC,cAAc;IAC/B,KAAKzC,WAAW,CAAC0C,4BAA4B;IAC7C,KAAK1C,WAAW,CAAC2C,cAAc;IAC/B,KAAK3C,WAAW,CAAC4C,kBAAkB;IACnC,KAAK5C,WAAW,CAAC6C,mBAAmB;IACpC,KAAK7C,WAAW,CAAC8C,kBAAkB;IACnC,KAAK9C,WAAW,CAAC+C,gBAAgB;IACjC,KAAK/C,WAAW,CAACgD,yBAAyB;IAC1C,KAAKhD,WAAW,CAACiD,gCAAgC;IACjD,KAAKjD,WAAW,CAACkD,eAAe;IAChC,KAAKlD,WAAW,CAACmD,cAAc;IAC/B,KAAKnD,WAAW,CAACoD,cAAc;IAC/B,KAAKpD,WAAW,CAACqD,cAAc;MAC7B,OAAOC,+BAAgB,CAAClB,MAAM;IAChC,KAAKpC,WAAW,CAACuD,YAAY;IAC7B,KAAKvD,WAAW,CAACwD,MAAM;IACvB,KAAKxD,WAAW,CAACyD,qBAAqB;MACpC,OAAOH,+BAAgB,CAACC,YAAY;IACtC;MACE,QAAQxB,UAAU;QAChB,KAAK,CAAC;QACN,KAAK,CAAC;UACJ,OAAOuB,+BAAgB,CAACC,YAAY;QACtC,KAAK,CAAC;UACJ,OAAOD,+BAAgB,CAAClB,MAAM;QAChC;UACE,MAAM,IAAIsB,KAAK,CAAC,4BAA4B1B,mBAAmB,EAAY,CAAC;MAChF;EACJ;AACF;AAEA,SAAS2B,mBAAmBA,CAACC,QAAgB,EAAE;EAC7C,IAAI,CAACA,QAAQ,EAAE;IACb,OAAOC,SAAS;EAClB;EACA,MAAMC,OAAO,GAAGF,QAAQ,CAACG,KAAK,CAAC,MAAM,CAAC;EACtC,IAAI,CAACD,OAAO,IAAIA,OAAO,CAACtC,MAAM,GAAG,CAAC,EAAE;IAClC,OAAOqC,SAAS;EAClB;EAEA,OAAO;IACLG,MAAM,EAAEC,QAAQ,CAACH,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;IAChCI,KAAK,EAAED,QAAQ,CAACH,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;EAChC,CAAC;AACH;AAEA,SAASK,kBAAkBA,CAACC,UAAyB,EAAE;EACrD,QAAQA,UAAU;IAChB,KAAK,GAAG;MACN,OAAOC,0BAAe;IACxB,KAAK,GAAG;MACN,OAAOC,0BAAe;IACxB,KAAK,GAAG;MACN,OAAOC,wBAAa;IACtB;MACE,OAAOV,SAAS;EACpB;AACF;AAEO,SAASW,OAAOA,CAAC;EACtBZ,QAAQ;EACRa,+BAA+B;EAC/BC;AACiG,CAAC,EAAE;EACpG,IAAID,+BAA+B,EAAE;IACnC,MAAME,IAAI,GAAGf,QAAQ,GAAG,GAAGA,QAAQ,IAAIa,+BAA+B,EAAE,GAAGA,+BAA+B;IAC1G,OAAOC,oBAAoB,GAAG,GAAGC,IAAI,KAAKD,oBAAoB,EAAE,GAAGC,IAAI;EACzE;EAEA,OAAOf,QAAQ;AACjB;AAEA,SAASgB,cAAcA,CAACC,cAAkC,EAAe;EACvE,MAAMC,SAAS,GAAGD,cAAc,CAACE,WAAW,KAAK,IAAI;EACrD,MAAMC,aAAa,GAAG,IAAAC,eAAM,EAACH,SAAS,GAAGD,cAAc,CAACK,YAAY,GAAGL,cAAc,CAACE,WAAW,CAAC,CAACI,WAAW,CAAC,CAAC;EAChH,MAAMC,MAAM,GAAGN,SAAS,GAAGO,kCAAmB,CAACC,OAAO,GAAGD,kCAAmB,CAACE,SAAS;EAEtF,MAAMC,YAAY,GAAG7B,mBAAmB,CAACkB,cAAc,CAACjB,QAAQ,CAAC;EACjE,MAAM6B,UAAU,GAAGD,YAAY,GAC3B,GAAGX,cAAc,CAACa,QAAQ,EAAEC,GAAG,IAAIH,YAAY,CAACxB,MAAM,EAAE,GACxDa,cAAc,CAACa,QAAQ,EAAEC,GAAG;EAEhC,OAAO;IACLC,IAAI,EAAE/D,kBAAkB,CAACgD,cAAc,CAAC/C,QAAQ,EAAE+C,cAAc,CAAC9C,UAAU,CAAC;IAC5EhB,IAAI,EAAE,IAAAkE,eAAM,EAACJ,cAAc,CAACK,YAAY,CAAC,CAACC,WAAW,CAAC,CAAC;IACvDH,aAAa;IACba,cAAc,EAAE,CAAChB,cAAc,CAACgB,cAAc;IAC9CC,gBAAgB,EAAEjB,cAAc,CAACiB,gBAAgB;IACjDC,aAAa,EAAE,CAAClB,cAAc,CAACmB,mBAAmB;IAClDC,eAAe,EAAE9B,kBAAkB,CAACU,cAAc,CAACqB,eAAe,CAAC;IACnEC,WAAW,EAAEtB,cAAc,CAACuB,YAAY,CAAClE,IAAI,CAAC,CAAC;IAC/CyC,IAAI,EAAEH,OAAO,CAACK,cAAc,CAAC;IAC7BwB,QAAQ,EAAElG,UAAU,CAACmG,GAAG,CAACzB,cAAc,EAAE0B,UAAU,CAAC;IACpDf,YAAY;IACZC,UAAU;IACVL;EACF,CAAC;AACH;AAOA,eAAeoB,yBAAyBA,CAAClG,IAAU,EAAEM,WAAmB,EAAE;EACxE,MAAM6F,GAAG,GAAG9F,kBAAkB,CAACC,WAAW,CAAC;EAE3C,MAAM8F,IAAI,GAAG,MAAM,IAAAtF,yBAAkB,EAA4Bd,IAAI,EAAEmG,GAAG,CAAC;EAC3E,MAAME,qBAAoD,GAAG,CAAC,CAAC;EAE/D,IAAI,CAACD,IAAI,IAAI,CAACA,IAAI,CAACnF,MAAM,EAAE,OAAOoF,qBAAqB;EAEvDD,IAAI,CAACnF,MAAM,CAACqF;EACV;EAAA,CACCC,MAAM,CAACC,WAAW,IAAI,CAAC,CAACA,WAAW,CAAChF,QAAQ,CAAC,CAC7CL,OAAO,CAAEqF,WAA+B,IAAK;IAC5C,IAAI,CAACH,qBAAqB,CAACG,WAAW,CAACC,eAAe,CAAC,EAAE;MACvDJ,qBAAqB,CAACG,WAAW,CAACC,eAAe,CAAC,GAAG,EAAE;IACzD;IAEA,MAAMC,iBAAiB,GAAGpC,cAAc,CAACkC,WAAW,CAAC;IACrDH,qBAAqB,CAACG,WAAW,CAACC,eAAe,CAAC,CAACE,IAAI,CAACD,iBAAiB,CAAC;EAC5E,CAAC,CAAC;EAEJ,OAAOL,qBAAqB;AAC9B;AAEA,SAASO,SAASA,CAACC,UAAyC,EAAE5F,MAAqC,EAAE;EACnG,MAAM6F,aAA4C,GAAG;IAAE,GAAGD;EAAW,CAAC;EACtEE,MAAM,CAACC,IAAI,CAAC/F,MAAM,CAAC,CAACE,OAAO,CAAC8F,aAAa,IAAI;IAC3C,IAAI,CAACH,aAAa,CAACG,aAAa,CAAC,EAAE;MACjCH,aAAa,CAACG,aAAa,CAAC,GAAG,EAAE;IACnC;IACAH,aAAa,CAACG,aAAa,CAAC,CAACN,IAAI,CAAC,GAAG1F,MAAM,CAACgG,aAAa,CAAC,CAAC;EAC7D,CAAC,CAAC;EACF,OAAOH,aAAa;AACtB;AAEA,SAASI,mBAAmBA,CAC1BC,IAAmB,EACnBC,WAA0B,EAC1BC,mBAA4B,EAC5BC,8BAAuC,EACvC;EACA,IAAIC,UAAU,GAAGxG,KAAK,CAACyG,IAAI,CAACL,IAAI,CAAC;EACjC,IAAI,CAACE,mBAAmB,EAAE;IACxBE,UAAU,GAAG,IAAAE,6BAAe,EAACF,UAAU,CAAC;EAC1C;EACAA,UAAU,GAAG,IAAAG,oCAAsB,EAACH,UAAU,CAAC;EAC/CA,UAAU,GAAGD,8BAA8B,GACvC,IAAAK,mCAAqB,EAACJ,UAAU,EAAEH,WAAW,EAAEC,mBAAmB,IAAI,KAAK,CAAC,GAC5EE,UAAU;EACd,OAAOA,UAAU;AACnB;AAEA,eAAeK,iBAAiBA,CAAC5H,IAAU,EAAE6H,OAAuB,EAAE;EACpE,MAAMC,oBAAoB,GAAGD,OAAO,CAACC,oBAAoB,IAAI,CAAC;EAC9D,MAAMC,kBAAkB,GAAG,IAAApD,eAAM,EAAC,CAAC,CAACqD,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC;EACxD,MAAMC,gBAAgB,GAAG,IAAAtD,eAAM,EAAC,CAAC,CAACqD,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC;EACtD,MAAME,SAAS,GAAGL,OAAO,CAACK,SAAS,IAAIH,kBAAkB,CAACI,MAAM,CAAC,CAAC;EAClE,MAAMf,WAAW,GAAGzC,eAAM,CAACyD,GAAG,CAACH,gBAAgB,EAAE,IAAAtD,eAAM,EAACuD,SAAS,CAAC,CAAC;EACnE,MAAMG,SAAS,GAAG,IAAAC,cAAkB,EAAClB,WAAW,EAAEU,oBAAoB,CAAC;EAEvE,MAAMlH,cAAc,CAACZ,IAAI,CAAC;EAE1B,IAAI6G,UAAyC,GAAG,CAAC,CAAC;EAClD,KAAK,IAAI0B,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGF,SAAS,CAACnH,MAAM,EAAEqH,CAAC,IAAI,CAAC,EAAE;IAC5C,MAAMtH,MAAM,GAAG,MAAMiF,yBAAyB,CAAClG,IAAI,EAAEqI,SAAS,CAACE,CAAC,CAAC,CAAC;IAClE1B,UAAU,GAAGD,SAAS,CAACC,UAAU,EAAE5F,MAAM,CAAC;EAC5C;EAEA8F,MAAM,CAACC,IAAI,CAACH,UAAU,CAAC,CAAC1F,OAAO,CAAC8F,aAAa,IAAI;IAC/C,IAAIE,IAAI,GAAGN,UAAU,CAACI,aAAa,CAAC;IACpCE,IAAI,GAAGD,mBAAmB,CACxBC,IAAI,EACJC,WAAW,EACXS,OAAO,CAACR,mBAAmB,IAAI,KAAK,EACpCQ,OAAO,CAACW,UAAU,EAAElB,8BAA8B,IAAI,IACxD,CAAC;IACDT,UAAU,CAACI,aAAa,CAAC,GAAGE,IAAI;EAClC,CAAC,CAAC;EAEF,OAAON,UAAU;AACnB;AAEA,SAAS4B,uBAAuBA,CAACzI,IAAU,EAAwB;EACjE,MAAM0I,IAA0B,GAAG,CAAC,CAAC;EACrCA,IAAI,CAACC,oCAAY,CAACC,OAAO,CAAC,GAAG,CAACnJ,WAAW,CAAC;EAC1CiJ,IAAI,CAACC,oCAAY,CAACE,cAAc,CAAC,GAAG,CAACrJ,oBAAoB,CAAC;EAC1DkJ,IAAI,CAACC,oCAAY,CAACG,eAAe,CAAC,GAAG,CACnC,YAAY;IACV,OAAO,IAAAC,0CAAoB,EAAC/I,IAAI,EAAEL,wBAAwB,CAAC;EAC7D,CAAC,CACF;EACD+I,IAAI,CAACC,oCAAY,CAACK,YAAY,CAAC,GAAG,CAChC,YAAY;IACV,OAAO,IAAAD,0CAAoB,EAAC/I,IAAI,EAAEJ,oBAAoB,CAAC;EACzD,CAAC,CACF;EACD,OAAO8I,IAAI;AACb;AAEA,SAASO,iBAAiBA,CAACC,WAAuC,EAAE;EAClE,OAAO,CACL;IAAEC,QAAQ,EAAE,YAAY;IAAEC,KAAK,EAAEF,WAAW,CAACG;EAAS,CAAC,EACvD;IAAEF,QAAQ,EAAE,WAAW;IAAEC,KAAK,EAAEF,WAAW,CAACI;EAAS,CAAC,CACvD;AACH;AAIA,MAAMC,UAAU,SAASC,8CAAsB,CAA6B;EAC1EC,eAAeA,CAACP,WAAuC,EAAgB;IACrE,OAAO;MACLQ,QAAQ,EAAEnK,SAAS;MACnBoK,MAAM,EAAEV,iBAAiB,CAACC,WAAW,CAAC;MACtCU,oBAAoB,EAAE,kDAAkD;MACxEC,SAAS,EAAE,MAAAA,CAAA,KAAY;QACrB,IAAI,MAAM,IAAAd,0CAAoB,EAAC,IAAI,CAAC/I,IAAI,EAAE,aAAa,CAAC,EAAE;UACxD,MAAM,IAAA8J,iCAAW,EAAC,IAAI,CAAC9J,IAAI,EAAE,aAAa,CAAC;QAC7C;QACA,MAAM,IAAA8J,iCAAW,EAAC,IAAI,CAAC9J,IAAI,EAAE,wCAAwC,CAAC;QACtE,IAAI,MAAM,IAAA+I,0CAAoB,EAAC,IAAI,CAAC/I,IAAI,EAAE,qBAAqB,CAAC,EAAE;UAChE,MAAM,IAAA8J,iCAAW,EAAC,IAAI,CAAC9J,IAAI,EAAE,qBAAqB,CAAC;QACrD;QACA,MAAM,IAAAI,2CAAqB,EAAC,IAAI,CAACJ,IAAI,EAAE,sBAAsB,EAAE,IAAI,CAAC;QACpE,MAAM,IAAA8J,iCAAW,EAAC,IAAI,CAAC9J,IAAI,EAAE,sBAAsB,CAAC;QACpD,MAAM,IAAAI,2CAAqB,EAAC,IAAI,CAACJ,IAAI,EAAE,qDAAqD,EAAE,IAAI,CAAC;MACrG,CAAC;MACD+J,cAAc,EAAE,MAAAA,CAAA,KAAY;QAC1B,MAAM,IAAA3J,2CAAqB,EAAC,IAAI,CAACJ,IAAI,EAAE,wCAAwC,EAAE,IAAI,CAAC;MACxF,CAAC;MACDgK,UAAU,EAAE,MAAAA,CAAA,KAAYjK,gBAAgB,CAAC,IAAI,CAACC,IAAI,CAAC;MACnDiK,eAAe,EAAExB,uBAAuB,CAAC,IAAI,CAACzI,IAAI,CAAC;MACnDkK,SAAS,EAAE;IACb,CAAC;EACH;EAEA,MAAMC,SAASA,CAAA,EAAG;IAChB,MAAMC,OAAO,GAAG,MAAMxC,iBAAiB,CAAC,IAAI,CAAC5H,IAAI,EAAE,IAAI,CAAC6H,OAAO,CAAC;IAChE,MAAMwC,QAAQ,GAAGtD,MAAM,CAACC,IAAI,CAACoD,OAAO,CAAC,CAACE,GAAG,CAACrD,aAAa,IAAI;MACzD,OAAO;QACLA,aAAa;QACbE,IAAI,EAAEiD,OAAO,CAACnD,aAAa;MAC7B,CAAC;IACH,CAAC,CAAC;IAEF,OAAO;MACLsD,OAAO,EAAE,IAAI;MACbF;IACF,CAAC;EACH;AACF;AAAC,IAAAG,QAAA,GAAAC,OAAA,CAAAvL,OAAA,GAEcqK,UAAU","ignoreList":[]}
|