israeli-bank-scrapers 6.1.0 → 6.1.1
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 +5 -7
- package/lib/constants.js +13 -16
- package/lib/definitions.js +109 -113
- package/lib/helpers/browser.js +9 -16
- package/lib/helpers/dates.js +18 -19
- package/lib/helpers/debug.js +9 -9
- package/lib/helpers/elements-interactions.js +78 -84
- package/lib/helpers/fetch.js +82 -89
- package/lib/helpers/navigation.js +24 -31
- package/lib/helpers/storage.js +10 -12
- package/lib/helpers/transactions.js +33 -35
- package/lib/helpers/waiting.js +45 -44
- package/lib/index.js +15 -82
- package/lib/scrapers/amex.js +11 -13
- package/lib/scrapers/base-beinleumi-group.js +233 -252
- package/lib/scrapers/base-isracard-amex.js +273 -286
- package/lib/scrapers/base-scraper-with-browser.js +240 -274
- package/lib/scrapers/base-scraper.js +82 -86
- package/lib/scrapers/behatsdaa.js +98 -107
- package/lib/scrapers/beinleumi.js +11 -20
- package/lib/scrapers/beyahad-bishvilha.js +132 -138
- package/lib/scrapers/discount.js +97 -103
- package/lib/scrapers/errors.js +22 -25
- package/lib/scrapers/factory.js +66 -67
- package/lib/scrapers/hapoalim.js +162 -182
- package/lib/scrapers/interface.js +2 -5
- package/lib/scrapers/isracard.js +11 -13
- package/lib/scrapers/leumi.js +167 -176
- package/lib/scrapers/massad.js +11 -20
- package/lib/scrapers/max.js +256 -268
- package/lib/scrapers/mercantile.js +14 -20
- package/lib/scrapers/mizrahi.js +158 -159
- package/lib/scrapers/one-zero-queries.js +4 -7
- package/lib/scrapers/one-zero.js +176 -240
- package/lib/scrapers/otsar-hahayal.js +11 -20
- package/lib/scrapers/pagi.js +11 -20
- package/lib/scrapers/union-bank.js +172 -179
- package/lib/scrapers/visa-cal.js +254 -263
- package/lib/scrapers/yahav.js +190 -211
- package/lib/transactions.js +13 -16
- package/package.json +12 -14
- package/lib/scrapers/amex.test.d.ts +0 -1
- package/lib/scrapers/amex.test.js +0 -54
- package/lib/scrapers/base-scraper-with-browser.test.d.ts +0 -1
- package/lib/scrapers/base-scraper-with-browser.test.js +0 -58
- package/lib/scrapers/behatsdaa.test.d.ts +0 -1
- package/lib/scrapers/behatsdaa.test.js +0 -50
- package/lib/scrapers/beinleumi.test.d.ts +0 -1
- package/lib/scrapers/beinleumi.test.js +0 -52
- package/lib/scrapers/beyahad-bishvilha.test.d.ts +0 -1
- package/lib/scrapers/beyahad-bishvilha.test.js +0 -52
- package/lib/scrapers/discount.test.d.ts +0 -1
- package/lib/scrapers/discount.test.js +0 -54
- package/lib/scrapers/factory.test.d.ts +0 -1
- package/lib/scrapers/factory.test.js +0 -19
- package/lib/scrapers/hapoalim.test.d.ts +0 -1
- package/lib/scrapers/hapoalim.test.js +0 -52
- package/lib/scrapers/isracard.test.d.ts +0 -1
- package/lib/scrapers/isracard.test.js +0 -54
- package/lib/scrapers/leumi.test.d.ts +0 -1
- package/lib/scrapers/leumi.test.js +0 -52
- package/lib/scrapers/max.test.d.ts +0 -1
- package/lib/scrapers/max.test.js +0 -71
- package/lib/scrapers/mercantile.test.d.ts +0 -1
- package/lib/scrapers/mercantile.test.js +0 -50
- package/lib/scrapers/mizrahi.test.d.ts +0 -1
- package/lib/scrapers/mizrahi.test.js +0 -58
- package/lib/scrapers/one-zero.test.d.ts +0 -1
- package/lib/scrapers/one-zero.test.js +0 -56
- package/lib/scrapers/otsar-hahayal.test.d.ts +0 -1
- package/lib/scrapers/otsar-hahayal.test.js +0 -52
- package/lib/scrapers/pagi.test.d.ts +0 -1
- package/lib/scrapers/pagi.test.js +0 -52
- package/lib/scrapers/union-bank.test.d.ts +0 -1
- package/lib/scrapers/union-bank.test.js +0 -52
- package/lib/scrapers/visa-cal.test.d.ts +0 -1
- package/lib/scrapers/visa-cal.test.js +0 -54
- package/lib/scrapers/yahav.test.d.ts +0 -1
- package/lib/scrapers/yahav.test.js +0 -54
package/lib/scrapers/max.js
CHANGED
|
@@ -1,311 +1,299 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
});
|
|
6
|
-
exports.
|
|
7
|
-
|
|
8
|
-
require("
|
|
9
|
-
require("
|
|
10
|
-
require("
|
|
11
|
-
require("
|
|
12
|
-
require("
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
var _fetch = require("../helpers/fetch");
|
|
20
|
-
var _navigation = require("../helpers/navigation");
|
|
21
|
-
var _transactions = require("../helpers/transactions");
|
|
22
|
-
var _transactions2 = require("../transactions");
|
|
23
|
-
var _baseScraperWithBrowser = require("./base-scraper-with-browser");
|
|
24
|
-
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
25
|
-
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
26
|
-
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
|
|
27
|
-
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
|
|
28
|
-
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
|
|
29
|
-
function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
|
|
30
|
-
const debug = (0, _debug.getDebug)('max');
|
|
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
|
+
exports.getMemo = void 0;
|
|
7
|
+
const build_url_1 = __importDefault(require("build-url"));
|
|
8
|
+
const moment_1 = __importDefault(require("moment"));
|
|
9
|
+
const constants_1 = require("../constants");
|
|
10
|
+
const dates_1 = __importDefault(require("../helpers/dates"));
|
|
11
|
+
const debug_1 = require("../helpers/debug");
|
|
12
|
+
const elements_interactions_1 = require("../helpers/elements-interactions");
|
|
13
|
+
const fetch_1 = require("../helpers/fetch");
|
|
14
|
+
const navigation_1 = require("../helpers/navigation");
|
|
15
|
+
const transactions_1 = require("../helpers/transactions");
|
|
16
|
+
const transactions_2 = require("../transactions");
|
|
17
|
+
const base_scraper_with_browser_1 = require("./base-scraper-with-browser");
|
|
18
|
+
const debug = (0, debug_1.getDebug)('max');
|
|
31
19
|
const BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';
|
|
32
20
|
const BASE_WELCOME_URL = 'https://www.max.co.il';
|
|
33
21
|
const LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;
|
|
34
22
|
const PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;
|
|
35
23
|
const SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;
|
|
36
|
-
var MaxPlanName
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
}(MaxPlanName || {});
|
|
24
|
+
var MaxPlanName;
|
|
25
|
+
(function (MaxPlanName) {
|
|
26
|
+
MaxPlanName["Normal"] = "\u05E8\u05D2\u05D9\u05DC\u05D4";
|
|
27
|
+
MaxPlanName["ImmediateCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05E2\u05E1\u05E7\u05D5\u05EA \u05DE\u05D9\u05D9\u05D3\u05D9";
|
|
28
|
+
MaxPlanName["InternetShopping"] = "\u05D0\u05D9\u05E0\u05D8\u05E8\u05E0\u05D8/\u05D7\u05D5\"\u05DC";
|
|
29
|
+
MaxPlanName["Installments"] = "\u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
|
|
30
|
+
MaxPlanName["MonthlyCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05D7\u05D5\u05D3\u05E9\u05D9";
|
|
31
|
+
MaxPlanName["OneMonthPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9";
|
|
32
|
+
MaxPlanName["MonthlyPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05DC\u05D7\u05D9\u05D5\u05D1 \u05D4\u05D7\u05D5\u05D3\u05E9\u05D9";
|
|
33
|
+
MaxPlanName["MonthlyPayment"] = "\u05EA\u05E9\u05DC\u05D5\u05DD \u05D7\u05D5\u05D3\u05E9\u05D9";
|
|
34
|
+
MaxPlanName["FuturePurchaseFinancing"] = "\u05DE\u05D9\u05DE\u05D5\u05DF \u05DC\u05E8\u05DB\u05D9\u05E9\u05D4 \u05E2\u05EA\u05D9\u05D3\u05D9\u05EA";
|
|
35
|
+
MaxPlanName["MonthlyPostponedInstallments"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9 \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
|
|
36
|
+
MaxPlanName["ThirtyDaysPlus"] = "\u05E2\u05E1\u05E7\u05EA 30 \u05E4\u05DC\u05D5\u05E1";
|
|
37
|
+
MaxPlanName["TwoMonthsPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9\u05D9\u05D9\u05DD";
|
|
38
|
+
MaxPlanName["TwoMonthsPostponed2"] = "\u05D3\u05D7\u05D5\u05D9 2 \u05D7' \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
|
|
39
|
+
MaxPlanName["MonthlyChargePlusInterest"] = "\u05D7\u05D5\u05D3\u05E9\u05D9 + \u05E8\u05D9\u05D1\u05D9\u05EA";
|
|
40
|
+
MaxPlanName["Credit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8";
|
|
41
|
+
MaxPlanName["CreditOutsideTheLimit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8-\u05DE\u05D7\u05D5\u05E5 \u05DC\u05DE\u05E1\u05D2\u05E8\u05EA";
|
|
42
|
+
MaxPlanName["AccumulatingBasket"] = "\u05E1\u05DC \u05DE\u05E6\u05D8\u05D1\u05E8";
|
|
43
|
+
MaxPlanName["PostponedTransactionInstallments"] = "\u05E4\u05E8\u05D9\u05E1\u05EA \u05D4\u05E2\u05E1\u05E7\u05D4 \u05D4\u05D3\u05D7\u05D5\u05D9\u05D4";
|
|
44
|
+
MaxPlanName["ReplacementCard"] = "\u05DB\u05E8\u05D8\u05D9\u05E1 \u05D7\u05DC\u05D9\u05E4\u05D9";
|
|
45
|
+
MaxPlanName["EarlyRepayment"] = "\u05E4\u05E8\u05E2\u05D5\u05DF \u05DE\u05D5\u05E7\u05D3\u05DD";
|
|
46
|
+
MaxPlanName["MonthlyCardFee"] = "\u05D3\u05DE\u05D9 \u05DB\u05E8\u05D8\u05D9\u05E1";
|
|
47
|
+
MaxPlanName["CurrencyPocket"] = "\u05D7\u05D9\u05D5\u05D1 \u05D0\u05E8\u05E0\u05E7 \u05DE\u05D8\u05D7";
|
|
48
|
+
})(MaxPlanName || (MaxPlanName = {}));
|
|
61
49
|
const INVALID_DETAILS_SELECTOR = '#popupWrongDetails';
|
|
62
50
|
const LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';
|
|
63
51
|
const categories = new Map();
|
|
64
52
|
function redirectOrDialog(page) {
|
|
65
|
-
|
|
53
|
+
return Promise.race([
|
|
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
|
+
]);
|
|
66
58
|
}
|
|
67
59
|
function getTransactionsUrl(monthMoment) {
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
/**
|
|
60
|
+
const month = monthMoment.month() + 1;
|
|
61
|
+
const year = monthMoment.year();
|
|
62
|
+
const date = `${year}-${month}-01`;
|
|
63
|
+
/**
|
|
73
64
|
* url explanation:
|
|
74
65
|
* userIndex: -1 for all account owners
|
|
75
66
|
* cardIndex: -1 for all cards under the account
|
|
76
67
|
* all other query params are static, beside the date which changes for request per month
|
|
77
68
|
*/
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
69
|
+
return (0, build_url_1.default)(BASE_API_ACTIONS_URL, {
|
|
70
|
+
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`,
|
|
71
|
+
});
|
|
81
72
|
}
|
|
82
73
|
async function loadCategories(page) {
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
id,
|
|
90
|
-
name
|
|
91
|
-
}) => categories.set(id, name));
|
|
92
|
-
}
|
|
74
|
+
debug('Loading categories');
|
|
75
|
+
const res = await (0, fetch_1.fetchGetWithinPage)(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);
|
|
76
|
+
if (res && Array.isArray(res.result)) {
|
|
77
|
+
debug(`${res.result.length} categories loaded`);
|
|
78
|
+
res.result?.forEach(({ id, name }) => categories.set(id, name));
|
|
79
|
+
}
|
|
93
80
|
}
|
|
94
81
|
function getTransactionType(planName, planTypeId) {
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
default:
|
|
122
|
-
switch (planTypeId) {
|
|
123
|
-
case 2:
|
|
124
|
-
case 3:
|
|
125
|
-
return _transactions2.TransactionTypes.Installments;
|
|
126
|
-
case 5:
|
|
127
|
-
return _transactions2.TransactionTypes.Normal;
|
|
82
|
+
const cleanedUpTxnTypeStr = planName.replace('\t', ' ').trim();
|
|
83
|
+
switch (cleanedUpTxnTypeStr) {
|
|
84
|
+
case MaxPlanName.ImmediateCharge:
|
|
85
|
+
case MaxPlanName.Normal:
|
|
86
|
+
case MaxPlanName.MonthlyCharge:
|
|
87
|
+
case MaxPlanName.OneMonthPostponed:
|
|
88
|
+
case MaxPlanName.MonthlyPostponed:
|
|
89
|
+
case MaxPlanName.FuturePurchaseFinancing:
|
|
90
|
+
case MaxPlanName.MonthlyPayment:
|
|
91
|
+
case MaxPlanName.MonthlyPostponedInstallments:
|
|
92
|
+
case MaxPlanName.ThirtyDaysPlus:
|
|
93
|
+
case MaxPlanName.TwoMonthsPostponed:
|
|
94
|
+
case MaxPlanName.TwoMonthsPostponed2:
|
|
95
|
+
case MaxPlanName.AccumulatingBasket:
|
|
96
|
+
case MaxPlanName.InternetShopping:
|
|
97
|
+
case MaxPlanName.MonthlyChargePlusInterest:
|
|
98
|
+
case MaxPlanName.PostponedTransactionInstallments:
|
|
99
|
+
case MaxPlanName.ReplacementCard:
|
|
100
|
+
case MaxPlanName.EarlyRepayment:
|
|
101
|
+
case MaxPlanName.MonthlyCardFee:
|
|
102
|
+
case MaxPlanName.CurrencyPocket:
|
|
103
|
+
return transactions_2.TransactionTypes.Normal;
|
|
104
|
+
case MaxPlanName.Installments:
|
|
105
|
+
case MaxPlanName.Credit:
|
|
106
|
+
case MaxPlanName.CreditOutsideTheLimit:
|
|
107
|
+
return transactions_2.TransactionTypes.Installments;
|
|
128
108
|
default:
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
109
|
+
switch (planTypeId) {
|
|
110
|
+
case 2:
|
|
111
|
+
case 3:
|
|
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
|
+
}
|
|
132
119
|
}
|
|
133
120
|
function getInstallmentsInfo(comments) {
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
121
|
+
if (!comments) {
|
|
122
|
+
return undefined;
|
|
123
|
+
}
|
|
124
|
+
const matches = comments.match(/\d+/g);
|
|
125
|
+
if (!matches || matches.length < 2) {
|
|
126
|
+
return undefined;
|
|
127
|
+
}
|
|
128
|
+
return {
|
|
129
|
+
number: parseInt(matches[0], 10),
|
|
130
|
+
total: parseInt(matches[1], 10),
|
|
131
|
+
};
|
|
145
132
|
}
|
|
146
133
|
function getChargedCurrency(currencyId) {
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
134
|
+
switch (currencyId) {
|
|
135
|
+
case 376:
|
|
136
|
+
return constants_1.SHEKEL_CURRENCY;
|
|
137
|
+
case 840:
|
|
138
|
+
return constants_1.DOLLAR_CURRENCY;
|
|
139
|
+
case 978:
|
|
140
|
+
return constants_1.EURO_CURRENCY;
|
|
141
|
+
default:
|
|
142
|
+
return undefined;
|
|
143
|
+
}
|
|
157
144
|
}
|
|
158
|
-
function getMemo({
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;
|
|
165
|
-
return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;
|
|
166
|
-
}
|
|
167
|
-
return comments;
|
|
145
|
+
function getMemo({ comments, fundsTransferReceiverOrTransfer, fundsTransferComment, }) {
|
|
146
|
+
if (fundsTransferReceiverOrTransfer) {
|
|
147
|
+
const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;
|
|
148
|
+
return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;
|
|
149
|
+
}
|
|
150
|
+
return comments;
|
|
168
151
|
}
|
|
152
|
+
exports.getMemo = getMemo;
|
|
169
153
|
function mapTransaction(rawTransaction) {
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
154
|
+
const isPending = rawTransaction.paymentDate === null;
|
|
155
|
+
const processedDate = (0, moment_1.default)(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();
|
|
156
|
+
const status = isPending ? transactions_2.TransactionStatuses.Pending : transactions_2.TransactionStatuses.Completed;
|
|
157
|
+
const installments = getInstallmentsInfo(rawTransaction.comments);
|
|
158
|
+
const identifier = installments
|
|
159
|
+
? `${rawTransaction.dealData?.arn}_${installments.number}`
|
|
160
|
+
: rawTransaction.dealData?.arn;
|
|
161
|
+
return {
|
|
162
|
+
type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),
|
|
163
|
+
date: (0, moment_1.default)(rawTransaction.purchaseDate).toISOString(),
|
|
164
|
+
processedDate,
|
|
165
|
+
originalAmount: -rawTransaction.originalAmount,
|
|
166
|
+
originalCurrency: rawTransaction.originalCurrency,
|
|
167
|
+
chargedAmount: -rawTransaction.actualPaymentAmount,
|
|
168
|
+
chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),
|
|
169
|
+
description: rawTransaction.merchantName.trim(),
|
|
170
|
+
memo: getMemo(rawTransaction),
|
|
171
|
+
category: categories.get(rawTransaction?.categoryId),
|
|
172
|
+
installments,
|
|
173
|
+
identifier,
|
|
174
|
+
status,
|
|
175
|
+
};
|
|
191
176
|
}
|
|
192
177
|
async function fetchTransactionsForMonth(page, monthMoment) {
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
178
|
+
const url = getTransactionsUrl(monthMoment);
|
|
179
|
+
const data = await (0, fetch_1.fetchGetWithinPage)(page, url);
|
|
180
|
+
const transactionsByAccount = {};
|
|
181
|
+
if (!data || !data.result)
|
|
182
|
+
return transactionsByAccount;
|
|
183
|
+
data.result.transactions
|
|
184
|
+
// Filter out non-transactions without a plan type, e.g. summary rows
|
|
185
|
+
.filter(transaction => !!transaction.planName)
|
|
186
|
+
.forEach((transaction) => {
|
|
187
|
+
if (!transactionsByAccount[transaction.shortCardNumber]) {
|
|
188
|
+
transactionsByAccount[transaction.shortCardNumber] = [];
|
|
189
|
+
}
|
|
190
|
+
const mappedTransaction = mapTransaction(transaction);
|
|
191
|
+
transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);
|
|
192
|
+
});
|
|
193
|
+
return transactionsByAccount;
|
|
207
194
|
}
|
|
208
195
|
function addResult(allResults, result) {
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
196
|
+
const clonedResults = { ...allResults };
|
|
197
|
+
Object.keys(result).forEach(accountNumber => {
|
|
198
|
+
if (!clonedResults[accountNumber]) {
|
|
199
|
+
clonedResults[accountNumber] = [];
|
|
200
|
+
}
|
|
201
|
+
clonedResults[accountNumber].push(...result[accountNumber]);
|
|
202
|
+
});
|
|
203
|
+
return clonedResults;
|
|
217
204
|
}
|
|
218
205
|
function prepareTransactions(txns, startMoment, combineInstallments, enableTransactionsFilterByDate) {
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
206
|
+
let clonedTxns = Array.from(txns);
|
|
207
|
+
if (!combineInstallments) {
|
|
208
|
+
clonedTxns = (0, transactions_1.fixInstallments)(clonedTxns);
|
|
209
|
+
}
|
|
210
|
+
clonedTxns = (0, transactions_1.sortTransactionsByDate)(clonedTxns);
|
|
211
|
+
clonedTxns = enableTransactionsFilterByDate
|
|
212
|
+
? (0, transactions_1.filterOldTransactions)(clonedTxns, startMoment, combineInstallments || false)
|
|
213
|
+
: clonedTxns;
|
|
214
|
+
return clonedTxns;
|
|
226
215
|
}
|
|
227
216
|
async function fetchTransactions(page, options) {
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
allResults
|
|
246
|
-
});
|
|
247
|
-
return allResults;
|
|
217
|
+
const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;
|
|
218
|
+
const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years');
|
|
219
|
+
const startMomentLimit = (0, moment_1.default)().subtract(4, 'years');
|
|
220
|
+
const startDate = options.startDate || defaultStartMoment.toDate();
|
|
221
|
+
const startMoment = moment_1.default.max(startMomentLimit, (0, moment_1.default)(startDate));
|
|
222
|
+
const allMonths = (0, dates_1.default)(startMoment, futureMonthsToScrape);
|
|
223
|
+
await loadCategories(page);
|
|
224
|
+
let allResults = {};
|
|
225
|
+
for (let i = 0; i < allMonths.length; i += 1) {
|
|
226
|
+
const result = await fetchTransactionsForMonth(page, allMonths[i]);
|
|
227
|
+
allResults = addResult(allResults, result);
|
|
228
|
+
}
|
|
229
|
+
Object.keys(allResults).forEach(accountNumber => {
|
|
230
|
+
let txns = allResults[accountNumber];
|
|
231
|
+
txns = prepareTransactions(txns, startMoment, options.combineInstallments || false, options.outputData?.enableTransactionsFilterByDate ?? true);
|
|
232
|
+
allResults[accountNumber] = txns;
|
|
233
|
+
});
|
|
234
|
+
return allResults;
|
|
248
235
|
}
|
|
249
236
|
function getPossibleLoginResults(page) {
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
237
|
+
const urls = {};
|
|
238
|
+
urls[base_scraper_with_browser_1.LoginResults.Success] = [SUCCESS_URL];
|
|
239
|
+
urls[base_scraper_with_browser_1.LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];
|
|
240
|
+
urls[base_scraper_with_browser_1.LoginResults.InvalidPassword] = [
|
|
241
|
+
async () => {
|
|
242
|
+
return (0, elements_interactions_1.elementPresentOnPage)(page, INVALID_DETAILS_SELECTOR);
|
|
243
|
+
},
|
|
244
|
+
];
|
|
245
|
+
urls[base_scraper_with_browser_1.LoginResults.UnknownError] = [
|
|
246
|
+
async () => {
|
|
247
|
+
return (0, elements_interactions_1.elementPresentOnPage)(page, LOGIN_ERROR_SELECTOR);
|
|
248
|
+
},
|
|
249
|
+
];
|
|
250
|
+
return urls;
|
|
260
251
|
}
|
|
261
252
|
function createLoginFields(credentials) {
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
selector: '#password',
|
|
267
|
-
value: credentials.password
|
|
268
|
-
}];
|
|
253
|
+
return [
|
|
254
|
+
{ selector: '#user-name', value: credentials.username },
|
|
255
|
+
{ selector: '#password', value: credentials.password },
|
|
256
|
+
];
|
|
269
257
|
}
|
|
270
|
-
class MaxScraper extends
|
|
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
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
258
|
+
class MaxScraper extends base_scraper_with_browser_1.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, elements_interactions_1.elementPresentOnPage)(this.page, '#closePopup')) {
|
|
266
|
+
await (0, elements_interactions_1.clickButton)(this.page, '#closePopup');
|
|
267
|
+
}
|
|
268
|
+
await (0, elements_interactions_1.clickButton)(this.page, '.personal-area > a.go-to-personal-area');
|
|
269
|
+
if (await (0, elements_interactions_1.elementPresentOnPage)(this.page, '.login-link#private')) {
|
|
270
|
+
await (0, elements_interactions_1.clickButton)(this.page, '.login-link#private');
|
|
271
|
+
}
|
|
272
|
+
await (0, elements_interactions_1.waitUntilElementFound)(this.page, '#login-password-link', true);
|
|
273
|
+
await (0, elements_interactions_1.clickButton)(this.page, '#login-password-link');
|
|
274
|
+
await (0, elements_interactions_1.waitUntilElementFound)(this.page, '#login-password.tab-pane.active app-user-login-form', true);
|
|
275
|
+
},
|
|
276
|
+
checkReadiness: async () => {
|
|
277
|
+
await (0, elements_interactions_1.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
|
+
}
|
|
309
297
|
}
|
|
310
|
-
|
|
311
|
-
//# 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","ownKeys","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty","getOwnPropertyDescriptors","defineProperties","defineProperty","_toPropertyKey","value","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","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","_res$result","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","_rawTransaction$dealD","_rawTransaction$dealD2","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","transaction","shortCardNumber","mappedTransaction","addResult","allResults","clonedResults","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","enableTransactionsFilterByDate","clonedTxns","from","fixInstallments","sortTransactionsByDate","filterOldTransactions","fetchTransactions","options","_options$futureMonths","futureMonthsToScrape","defaultStartMoment","subtract","startMomentLimit","startDate","toDate","max","allMonths","getAllMonthMoments","_options$outputData$e","_options$outputData","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","elementPresentOnPage","UnknownError","createLoginFields","credentials","selector","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, fundsTransferReceiverOrTransfer, 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 ?\n    rawTransaction.purchaseDate :\n    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(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean, enableTransactionsFilterByDate: boolean) {\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(txns, startMoment, options.combineInstallments || false,\n      (options.outputData?.enableTransactionsFilterByDate ?? true));\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] = [async () => {\n    return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n  }];\n  urls[LoginResults.UnknownError] = [async () => {\n    return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\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":"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","ignoreList":[]}
|
|
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"]}
|