israeli-bank-scrapers 3.4.1 → 3.7.0
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/README.md +42 -0
- package/lib/definitions.d.ts +6 -1
- package/lib/definitions.js +6 -1
- package/lib/helpers/fetch.d.ts +2 -1
- package/lib/helpers/fetch.js +26 -9
- package/lib/helpers/storage.d.ts +2 -0
- package/lib/helpers/storage.js +17 -0
- package/lib/index.d.ts +2 -1
- package/lib/index.js +47 -3
- package/lib/scrapers/amex.d.ts +1 -1
- package/lib/scrapers/amex.js +1 -1
- package/lib/scrapers/amex.test.js +3 -2
- package/lib/scrapers/base-beinleumi-group.d.ts +7 -4
- package/lib/scrapers/base-beinleumi-group.js +1 -1
- package/lib/scrapers/base-isracard-amex.d.ts +8 -3
- package/lib/scrapers/base-isracard-amex.js +13 -11
- package/lib/scrapers/base-scraper-with-browser.d.ts +6 -3
- package/lib/scrapers/base-scraper-with-browser.js +14 -11
- package/lib/scrapers/base-scraper.d.ts +9 -119
- package/lib/scrapers/base-scraper.js +34 -50
- package/lib/scrapers/beyahad-bishvilha.d.ts +6 -3
- package/lib/scrapers/beyahad-bishvilha.js +1 -1
- package/lib/scrapers/discount.d.ts +9 -4
- package/lib/scrapers/discount.js +4 -4
- package/lib/scrapers/discount.test.js +4 -3
- package/lib/scrapers/errors.d.ts +16 -0
- package/lib/scrapers/errors.js +37 -0
- package/lib/scrapers/factory.d.ts +2 -16
- package/lib/scrapers/factory.js +6 -1
- package/lib/scrapers/hapoalim.d.ts +6 -3
- package/lib/scrapers/hapoalim.js +1 -1
- package/lib/scrapers/hapoalim.test.js +2 -2
- package/lib/scrapers/interface.d.ts +146 -0
- package/lib/scrapers/interface.js +2 -0
- package/lib/scrapers/isracard.d.ts +1 -1
- package/lib/scrapers/isracard.js +1 -1
- package/lib/scrapers/isracard.test.js +4 -3
- package/lib/scrapers/leumi.d.ts +8 -4
- package/lib/scrapers/leumi.js +1 -1
- package/lib/scrapers/max.d.ts +6 -3
- package/lib/scrapers/max.js +1 -1
- package/lib/scrapers/mizrahi.d.ts +7 -3
- package/lib/scrapers/mizrahi.js +11 -9
- package/lib/scrapers/one-zero-queries.d.ts +2 -0
- package/lib/scrapers/one-zero-queries.js +562 -0
- package/lib/scrapers/one-zero.d.ts +36 -0
- package/lib/scrapers/one-zero.js +304 -0
- package/lib/scrapers/one-zero.test.d.ts +1 -0
- package/lib/scrapers/one-zero.test.js +67 -0
- package/lib/scrapers/otsar-hahayal.d.ts +6 -3
- package/lib/scrapers/otsar-hahayal.js +1 -1
- package/lib/scrapers/union-bank.d.ts +6 -3
- package/lib/scrapers/union-bank.js +1 -1
- package/lib/scrapers/visa-cal.d.ts +103 -15
- package/lib/scrapers/visa-cal.js +154 -265
- package/lib/scrapers/yahav.d.ts +7 -3
- package/lib/scrapers/yahav.js +1 -1
- package/package.json +2 -2
package/lib/scrapers/visa-cal.js
CHANGED
|
@@ -1,14 +1,12 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
|
-
require("core-js/modules/es.
|
|
3
|
+
require("core-js/modules/es.array.flat-map");
|
|
4
4
|
|
|
5
5
|
require("core-js/modules/es.array.iterator");
|
|
6
6
|
|
|
7
|
-
require("core-js/modules/es.
|
|
8
|
-
|
|
9
|
-
require("core-js/modules/es.string.replace");
|
|
7
|
+
require("core-js/modules/es.array.unscopables.flat-map");
|
|
10
8
|
|
|
11
|
-
require("core-js/modules/es.
|
|
9
|
+
require("core-js/modules/es.promise");
|
|
12
10
|
|
|
13
11
|
Object.defineProperty(exports, "__esModule", {
|
|
14
12
|
value: true
|
|
@@ -17,30 +15,42 @@ exports.default = void 0;
|
|
|
17
15
|
|
|
18
16
|
var _moment = _interopRequireDefault(require("moment"));
|
|
19
17
|
|
|
20
|
-
var
|
|
18
|
+
var _constants = require("../constants");
|
|
19
|
+
|
|
20
|
+
var _debug = require("../helpers/debug");
|
|
21
21
|
|
|
22
22
|
var _elementsInteractions = require("../helpers/elements-interactions");
|
|
23
23
|
|
|
24
|
-
var
|
|
24
|
+
var _fetch = require("../helpers/fetch");
|
|
25
25
|
|
|
26
|
-
var
|
|
26
|
+
var _navigation = require("../helpers/navigation");
|
|
27
|
+
|
|
28
|
+
var _storage = require("../helpers/storage");
|
|
29
|
+
|
|
30
|
+
var _transactions = require("../helpers/transactions");
|
|
27
31
|
|
|
28
32
|
var _waiting = require("../helpers/waiting");
|
|
29
33
|
|
|
30
|
-
var _transactions2 = require("../
|
|
34
|
+
var _transactions2 = require("../transactions");
|
|
31
35
|
|
|
32
|
-
var
|
|
36
|
+
var _baseScraperWithBrowser = require("./base-scraper-with-browser");
|
|
33
37
|
|
|
34
38
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
35
39
|
|
|
36
40
|
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
37
41
|
|
|
38
42
|
const LOGIN_URL = 'https://www.cal-online.co.il/';
|
|
39
|
-
const
|
|
40
|
-
const LONG_DATE_FORMAT = 'DD/MM/YYYY';
|
|
41
|
-
const DATE_FORMAT = 'DD/MM/YY';
|
|
43
|
+
const TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';
|
|
42
44
|
const InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';
|
|
43
45
|
const debug = (0, _debug.getDebug)('visa-cal');
|
|
46
|
+
var trnTypeCode;
|
|
47
|
+
|
|
48
|
+
(function (trnTypeCode) {
|
|
49
|
+
trnTypeCode["regular"] = "5";
|
|
50
|
+
trnTypeCode["credit"] = "6";
|
|
51
|
+
trnTypeCode["installments"] = "8";
|
|
52
|
+
trnTypeCode["standingOrder"] = "9";
|
|
53
|
+
})(trnTypeCode || (trnTypeCode = {}));
|
|
44
54
|
|
|
45
55
|
async function getLoginFrame(page) {
|
|
46
56
|
let frame = null;
|
|
@@ -70,7 +80,7 @@ async function hasInvalidPasswordError(page) {
|
|
|
70
80
|
function getPossibleLoginResults() {
|
|
71
81
|
debug('return possible login results');
|
|
72
82
|
const urls = {
|
|
73
|
-
[_baseScraperWithBrowser.LoginResults.Success]: [/
|
|
83
|
+
[_baseScraperWithBrowser.LoginResults.Success]: [/dashboard/i],
|
|
74
84
|
[_baseScraperWithBrowser.LoginResults.InvalidPassword]: [async options => {
|
|
75
85
|
const page = options === null || options === void 0 ? void 0 : options.page;
|
|
76
86
|
|
|
@@ -97,70 +107,39 @@ function createLoginFields(credentials) {
|
|
|
97
107
|
}];
|
|
98
108
|
}
|
|
99
109
|
|
|
100
|
-
function
|
|
101
|
-
|
|
102
|
-
let currency = null;
|
|
103
|
-
let amount = null;
|
|
104
|
-
|
|
105
|
-
if (amountStrCln.includes(_constants.SHEKEL_CURRENCY_SYMBOL)) {
|
|
106
|
-
amount = -parseFloat(amountStrCln.replace(_constants.SHEKEL_CURRENCY_SYMBOL, ''));
|
|
107
|
-
currency = _constants.SHEKEL_CURRENCY;
|
|
108
|
-
} else if (amountStrCln.includes(_constants.DOLLAR_CURRENCY_SYMBOL)) {
|
|
109
|
-
amount = -parseFloat(amountStrCln.replace(_constants.DOLLAR_CURRENCY_SYMBOL, ''));
|
|
110
|
-
currency = _constants.DOLLAR_CURRENCY;
|
|
111
|
-
} else if (amountStrCln.includes(_constants.EURO_CURRENCY_SYMBOL)) {
|
|
112
|
-
amount = -parseFloat(amountStrCln.replace(_constants.EURO_CURRENCY_SYMBOL, ''));
|
|
113
|
-
currency = _constants.EURO_CURRENCY;
|
|
114
|
-
} else {
|
|
115
|
-
const parts = amountStrCln.split(' ');
|
|
116
|
-
[currency] = parts;
|
|
117
|
-
amount = -parseFloat(parts[1]);
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
return {
|
|
121
|
-
amount,
|
|
122
|
-
currency
|
|
123
|
-
};
|
|
110
|
+
function cardAndTransactionCurrencySymbolIsShekel(transaction) {
|
|
111
|
+
return transaction.debCrdCurrencySymbol === _constants.SHEKEL_CURRENCY_SYMBOL && transaction.trnCurrencySymbol === _constants.SHEKEL_CURRENCY_SYMBOL;
|
|
124
112
|
}
|
|
125
113
|
|
|
126
|
-
function
|
|
127
|
-
|
|
114
|
+
function convertParsedDataToTransactions(parsedData) {
|
|
115
|
+
return parsedData.flatMap(monthData => monthData.result.bankAccounts).flatMap(accounts => accounts.debitDates).flatMap(debitDate => debitDate.transactions).map(transaction => {
|
|
116
|
+
const installments = transaction.curPaymentNum && transaction.numOfPayments && {
|
|
117
|
+
number: transaction.curPaymentNum,
|
|
118
|
+
total: transaction.numOfPayments
|
|
119
|
+
} || undefined;
|
|
120
|
+
const date = (0, _moment.default)(transaction.trnPurchaseDate); // I didn't test `amtBeforeConvAndIndex` with a foreign currency as I don't have such transactions
|
|
128
121
|
|
|
129
|
-
|
|
130
|
-
return null;
|
|
131
|
-
}
|
|
122
|
+
let chargedAmount;
|
|
132
123
|
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
}
|
|
124
|
+
if (cardAndTransactionCurrencySymbolIsShekel(transaction)) {
|
|
125
|
+
chargedAmount = transaction.amtBeforeConvAndIndex * -1;
|
|
126
|
+
} else {
|
|
127
|
+
chargedAmount = transaction.trnAmt * -1;
|
|
138
128
|
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
const originalAmountTuple = getAmountData(txn.originalAmount || '');
|
|
143
|
-
const chargedAmountTuple = getAmountData(txn.chargedAmount || '');
|
|
144
|
-
const installments = getTransactionInstallments(txn.memo);
|
|
145
|
-
const txnDate = (0, _moment.default)(txn.date, DATE_FORMAT);
|
|
146
|
-
const processedDateFormat = txn.processedDate.length === 8 ? DATE_FORMAT : txn.processedDate.length === 9 || txn.processedDate.length === 10 ? LONG_DATE_FORMAT : null;
|
|
147
|
-
|
|
148
|
-
if (!processedDateFormat) {
|
|
149
|
-
throw new Error('invalid processed date');
|
|
129
|
+
if (transaction.trnTypeCode === trnTypeCode.credit) {
|
|
130
|
+
chargedAmount = transaction.trnAmt;
|
|
131
|
+
}
|
|
150
132
|
}
|
|
151
133
|
|
|
152
|
-
const txnProcessedDate = (0, _moment.default)(txn.processedDate, processedDateFormat);
|
|
153
134
|
const result = {
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
description: txn.description || '',
|
|
163
|
-
memo: txn.memo || ''
|
|
135
|
+
chargedAmount,
|
|
136
|
+
description: transaction.merchantName,
|
|
137
|
+
originalAmount: transaction.amtBeforeConvAndIndex,
|
|
138
|
+
originalCurrency: transaction.trnCurrencySymbol,
|
|
139
|
+
processedDate: transaction.debCrdDate,
|
|
140
|
+
status: _transactions2.TransactionStatuses.Completed,
|
|
141
|
+
date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),
|
|
142
|
+
type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ? _transactions2.TransactionTypes.Normal : _transactions2.TransactionTypes.Installments
|
|
164
143
|
};
|
|
165
144
|
|
|
166
145
|
if (installments) {
|
|
@@ -171,191 +150,6 @@ function convertTransactions(txns) {
|
|
|
171
150
|
});
|
|
172
151
|
}
|
|
173
152
|
|
|
174
|
-
async function fetchTransactionsForAccount(page, startDate, accountNumber, scraperOptions) {
|
|
175
|
-
var _scraperOptions$outpu, _scraperOptions$outpu2;
|
|
176
|
-
|
|
177
|
-
const startDateValue = startDate.format('MM/YYYY');
|
|
178
|
-
const dateSelector = '[id$="FormAreaNoBorder_FormArea_clndrDebitDateScope_TextBox"]';
|
|
179
|
-
const dateHiddenFieldSelector = '[id$="FormAreaNoBorder_FormArea_clndrDebitDateScope_HiddenField"]';
|
|
180
|
-
const buttonSelector = '[id$="FormAreaNoBorder_FormArea_ctlSubmitRequest"]';
|
|
181
|
-
const nextPageSelector = '[id$="FormAreaNoBorder_FormArea_ctlGridPager_btnNext"]';
|
|
182
|
-
const billingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlMainToolBar_lblCaption]';
|
|
183
|
-
const secondaryBillingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlSecondaryToolBar_lblCaption]';
|
|
184
|
-
const noDataSelector = '[id$=FormAreaNoBorder_FormArea_msgboxErrorMessages]';
|
|
185
|
-
debug('find the start date index in the dropbox');
|
|
186
|
-
const options = await (0, _elementsInteractions.pageEvalAll)(page, '[id$="FormAreaNoBorder_FormArea_clndrDebitDateScope_OptionList"] li', [], items => {
|
|
187
|
-
return items.map(el => el.innerText);
|
|
188
|
-
});
|
|
189
|
-
const startDateIndex = options.findIndex(option => option === startDateValue);
|
|
190
|
-
debug(`scrape ${options.length - startDateIndex} billing cycles`);
|
|
191
|
-
const accountTransactions = [];
|
|
192
|
-
|
|
193
|
-
for (let currentDateIndex = startDateIndex; currentDateIndex < options.length; currentDateIndex += 1) {
|
|
194
|
-
debug('wait for date selector to be found');
|
|
195
|
-
await (0, _elementsInteractions.waitUntilElementFound)(page, dateSelector, true);
|
|
196
|
-
debug(`set hidden value of the date selector to be the index ${currentDateIndex}`);
|
|
197
|
-
await (0, _elementsInteractions.setValue)(page, dateHiddenFieldSelector, `${currentDateIndex}`);
|
|
198
|
-
debug('wait a second to workaround navigation issue in headless browser mode');
|
|
199
|
-
await page.waitForTimeout(1000);
|
|
200
|
-
debug('click on the filter submit button and wait for navigation');
|
|
201
|
-
await Promise.all([page.waitForNavigation({
|
|
202
|
-
waitUntil: 'domcontentloaded'
|
|
203
|
-
}), (0, _elementsInteractions.clickButton)(page, buttonSelector)]);
|
|
204
|
-
debug('check if month has no transactions');
|
|
205
|
-
const pageHasNoTransactions = await (0, _elementsInteractions.pageEval)(page, noDataSelector, false, element => {
|
|
206
|
-
const siteValue = (element.innerText || '').replace(/[^ א-ת]/g, '');
|
|
207
|
-
return siteValue === 'לא נמצאו נתונים';
|
|
208
|
-
});
|
|
209
|
-
|
|
210
|
-
if (pageHasNoTransactions) {
|
|
211
|
-
debug('page has no transactions');
|
|
212
|
-
} else {
|
|
213
|
-
var _settlementDateRegex$;
|
|
214
|
-
|
|
215
|
-
debug('find the billing date');
|
|
216
|
-
let billingDateLabel = await (0, _elementsInteractions.pageEval)(page, billingLabelSelector, '', element => {
|
|
217
|
-
return element.innerText;
|
|
218
|
-
});
|
|
219
|
-
let settlementDateRegex = /\d{1,2}[/]\d{2}[/]\d{2,4}/;
|
|
220
|
-
|
|
221
|
-
if (billingDateLabel === '') {
|
|
222
|
-
billingDateLabel = await (0, _elementsInteractions.pageEval)(page, secondaryBillingLabelSelector, '', element => {
|
|
223
|
-
return element.innerText;
|
|
224
|
-
});
|
|
225
|
-
settlementDateRegex = /\d{1,2}[/]\d{2,4}/;
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
const billingDate = (_settlementDateRegex$ = settlementDateRegex.exec(billingDateLabel)) === null || _settlementDateRegex$ === void 0 ? void 0 : _settlementDateRegex$[0];
|
|
229
|
-
|
|
230
|
-
if (!billingDate) {
|
|
231
|
-
throw new Error('failed to fetch process date');
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
debug(`found the billing date for that month ${billingDate}`);
|
|
235
|
-
let hasNextPage = false;
|
|
236
|
-
|
|
237
|
-
do {
|
|
238
|
-
debug('fetch raw transactions from page');
|
|
239
|
-
const rawTransactions = await (0, _elementsInteractions.pageEvalAll)(page, '#ctlMainGrid > tbody tr, #ctlSecondaryGrid > tbody tr', [], (items, billingDate) => {
|
|
240
|
-
return items.map(el => {
|
|
241
|
-
const columns = el.getElementsByTagName('td');
|
|
242
|
-
|
|
243
|
-
if (columns.length === 6) {
|
|
244
|
-
return {
|
|
245
|
-
processedDate: columns[0].innerText,
|
|
246
|
-
date: columns[1].innerText,
|
|
247
|
-
description: columns[2].innerText,
|
|
248
|
-
originalAmount: columns[3].innerText,
|
|
249
|
-
chargedAmount: columns[4].innerText,
|
|
250
|
-
memo: columns[5].innerText
|
|
251
|
-
};
|
|
252
|
-
}
|
|
253
|
-
|
|
254
|
-
if (columns.length === 5) {
|
|
255
|
-
return {
|
|
256
|
-
processedDate: billingDate,
|
|
257
|
-
date: columns[0].innerText,
|
|
258
|
-
description: columns[1].innerText,
|
|
259
|
-
originalAmount: columns[2].innerText,
|
|
260
|
-
chargedAmount: columns[3].innerText,
|
|
261
|
-
memo: columns[4].innerText
|
|
262
|
-
};
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
return null;
|
|
266
|
-
});
|
|
267
|
-
}, billingDate);
|
|
268
|
-
debug(`fetched ${rawTransactions.length} raw transactions from page`);
|
|
269
|
-
accountTransactions.push(...convertTransactions(rawTransactions.filter(item => !!item)));
|
|
270
|
-
debug('check for existance of another page');
|
|
271
|
-
hasNextPage = await (0, _elementsInteractions.elementPresentOnPage)(page, nextPageSelector);
|
|
272
|
-
|
|
273
|
-
if (hasNextPage) {
|
|
274
|
-
debug('has another page, click on button next and wait for page navigation');
|
|
275
|
-
await Promise.all([page.waitForNavigation({
|
|
276
|
-
waitUntil: 'domcontentloaded'
|
|
277
|
-
}), await (0, _elementsInteractions.clickButton)(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]')]);
|
|
278
|
-
}
|
|
279
|
-
} while (hasNextPage);
|
|
280
|
-
}
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
debug('filer out old transactions');
|
|
284
|
-
const txns = ((_scraperOptions$outpu = (_scraperOptions$outpu2 = scraperOptions.outputData) === null || _scraperOptions$outpu2 === void 0 ? void 0 : _scraperOptions$outpu2.enableTransactionsFilterByDate) !== null && _scraperOptions$outpu !== void 0 ? _scraperOptions$outpu : true) ? (0, _transactions2.filterOldTransactions)(accountTransactions, startDate, scraperOptions.combineInstallments || false) : accountTransactions;
|
|
285
|
-
debug(`found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`);
|
|
286
|
-
return {
|
|
287
|
-
accountNumber,
|
|
288
|
-
txns
|
|
289
|
-
};
|
|
290
|
-
}
|
|
291
|
-
|
|
292
|
-
async function getAccountNumbers(page) {
|
|
293
|
-
return (0, _elementsInteractions.pageEvalAll)(page, '[id$=lnkItem]', [], elements => elements.map(e => e.text)).then(res => res.map(text => {
|
|
294
|
-
var _$exec$, _$exec;
|
|
295
|
-
|
|
296
|
-
return (_$exec$ = (_$exec = /\d+$/.exec(text.trim())) === null || _$exec === void 0 ? void 0 : _$exec[0]) !== null && _$exec$ !== void 0 ? _$exec$ : '';
|
|
297
|
-
}));
|
|
298
|
-
}
|
|
299
|
-
|
|
300
|
-
async function setAccount(page, account) {
|
|
301
|
-
await (0, _elementsInteractions.pageEvalAll)(page, '[id$=lnkItem]', null, (elements, account) => {
|
|
302
|
-
for (const elem of elements) {
|
|
303
|
-
const a = elem;
|
|
304
|
-
|
|
305
|
-
if (a.text.includes(account)) {
|
|
306
|
-
a.click();
|
|
307
|
-
}
|
|
308
|
-
}
|
|
309
|
-
}, account);
|
|
310
|
-
}
|
|
311
|
-
|
|
312
|
-
async function fetchTransactions(page, startDate, scraperOptions) {
|
|
313
|
-
const accountNumbers = await getAccountNumbers(page);
|
|
314
|
-
const accounts = [];
|
|
315
|
-
|
|
316
|
-
for (const account of accountNumbers) {
|
|
317
|
-
debug(`setting account: ${account}`);
|
|
318
|
-
await setAccount(page, account);
|
|
319
|
-
await page.waitForTimeout(1000);
|
|
320
|
-
accounts.push((await fetchTransactionsForAccount(page, startDate, account, scraperOptions)));
|
|
321
|
-
}
|
|
322
|
-
|
|
323
|
-
return accounts;
|
|
324
|
-
}
|
|
325
|
-
|
|
326
|
-
async function fetchFutureDebits(page) {
|
|
327
|
-
const futureDebitsSelector = '.homepage-banks-top';
|
|
328
|
-
const result = await (0, _elementsInteractions.pageEvalAll)(page, futureDebitsSelector, [], items => {
|
|
329
|
-
const debitMountClass = 'amount';
|
|
330
|
-
const debitWhenChargeClass = 'when-charge';
|
|
331
|
-
const debitBankNumberClass = 'bankDesc';
|
|
332
|
-
return items.map(currBankEl => {
|
|
333
|
-
const amount = currBankEl.getElementsByClassName(debitMountClass)[0].innerText;
|
|
334
|
-
const whenCharge = currBankEl.getElementsByClassName(debitWhenChargeClass)[0].innerText;
|
|
335
|
-
const bankNumber = currBankEl.getElementsByClassName(debitBankNumberClass)[0].innerText;
|
|
336
|
-
return {
|
|
337
|
-
amount,
|
|
338
|
-
whenCharge,
|
|
339
|
-
bankNumber
|
|
340
|
-
};
|
|
341
|
-
});
|
|
342
|
-
});
|
|
343
|
-
const futureDebits = result.map(item => {
|
|
344
|
-
var _$exec2, _$exec3;
|
|
345
|
-
|
|
346
|
-
const amountData = getAmountData(item.amount);
|
|
347
|
-
const chargeDate = (_$exec2 = /\d{1,2}[/]\d{2}[/]\d{2,4}/.exec(item.whenCharge)) === null || _$exec2 === void 0 ? void 0 : _$exec2[0];
|
|
348
|
-
const bankAccountNumber = (_$exec3 = /\d+-\d+/.exec(item.bankNumber)) === null || _$exec3 === void 0 ? void 0 : _$exec3[0];
|
|
349
|
-
return {
|
|
350
|
-
amount: amountData.amount,
|
|
351
|
-
amountCurrency: amountData.currency,
|
|
352
|
-
chargeDate,
|
|
353
|
-
bankAccountNumber
|
|
354
|
-
};
|
|
355
|
-
});
|
|
356
|
-
return futureDebits;
|
|
357
|
-
}
|
|
358
|
-
|
|
359
153
|
class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
360
154
|
constructor(...args) {
|
|
361
155
|
super(...args);
|
|
@@ -377,6 +171,48 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
377
171
|
});
|
|
378
172
|
}
|
|
379
173
|
|
|
174
|
+
async getCards() {
|
|
175
|
+
const initData = await (0, _storage.getFromSessionStorage)(this.page, 'init');
|
|
176
|
+
|
|
177
|
+
if (!initData) {
|
|
178
|
+
throw new Error('could not find \'init\' data in session storage');
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
return initData === null || initData === void 0 ? void 0 : initData.result.cards.map(({
|
|
182
|
+
cardUniqueId,
|
|
183
|
+
last4Digits
|
|
184
|
+
}) => ({
|
|
185
|
+
cardUniqueId,
|
|
186
|
+
last4Digits
|
|
187
|
+
}));
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
async getAuthorizationHeader() {
|
|
191
|
+
const authModule = await (0, _storage.getFromSessionStorage)(this.page, 'auth-module');
|
|
192
|
+
|
|
193
|
+
if (!authModule) {
|
|
194
|
+
throw new Error('could not find \'auth-module\' in session storage');
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
return `CALAuthScheme ${authModule.auth.calConnectToken}`;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
async getXSiteId() {
|
|
201
|
+
/*
|
|
202
|
+
I don't know if the constant below will change in the feature.
|
|
203
|
+
If so, use the next code:
|
|
204
|
+
return this.page.evaluate(() => new Ut().xSiteId);
|
|
205
|
+
To get the classname search for 'xSiteId' in the page source
|
|
206
|
+
class Ut {
|
|
207
|
+
constructor(_e, on, yn) {
|
|
208
|
+
this.store = _e,
|
|
209
|
+
this.config = on,
|
|
210
|
+
this.eventBusService = yn,
|
|
211
|
+
this.xSiteId = "09031987-273E-2311-906C-8AF85B17C8D9",
|
|
212
|
+
*/
|
|
213
|
+
return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');
|
|
214
|
+
}
|
|
215
|
+
|
|
380
216
|
getLoginOptions(credentials) {
|
|
381
217
|
return {
|
|
382
218
|
loginUrl: `${LOGIN_URL}`,
|
|
@@ -385,28 +221,81 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
385
221
|
possibleResults: getPossibleLoginResults(),
|
|
386
222
|
checkReadiness: async () => (0, _elementsInteractions.waitUntilElementFound)(this.page, '#ccLoginDesktopBtn'),
|
|
387
223
|
preAction: this.openLoginPopup,
|
|
224
|
+
postAction: async () => {
|
|
225
|
+
try {
|
|
226
|
+
await (0, _elementsInteractions.waitUntilElementFound)(this.page, 'button.btn-close');
|
|
227
|
+
const currentUrl = await (0, _navigation.getCurrentUrl)(this.page);
|
|
228
|
+
|
|
229
|
+
if (currentUrl.endsWith('site-tutorial')) {
|
|
230
|
+
await (0, _elementsInteractions.clickButton)(this.page, 'button.btn-close');
|
|
231
|
+
}
|
|
232
|
+
} catch (e) {
|
|
233
|
+
const currentUrl = await (0, _navigation.getCurrentUrl)(this.page);
|
|
234
|
+
if (currentUrl.endsWith('dashboard')) return;
|
|
235
|
+
throw e;
|
|
236
|
+
}
|
|
237
|
+
},
|
|
388
238
|
userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36'
|
|
389
239
|
};
|
|
390
240
|
}
|
|
391
241
|
|
|
242
|
+
isCardTransactionDetails(result) {
|
|
243
|
+
return result.result !== undefined;
|
|
244
|
+
}
|
|
245
|
+
|
|
392
246
|
async fetchData() {
|
|
393
|
-
const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
|
|
247
|
+
const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').subtract(6, 'months').add(1, 'day');
|
|
394
248
|
const startDate = this.options.startDate || defaultStartMoment.toDate();
|
|
395
249
|
|
|
396
250
|
const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
|
|
397
251
|
|
|
398
252
|
debug(`fetch transactions starting ${startMoment.format()}`);
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
await this.
|
|
403
|
-
|
|
404
|
-
const accounts = await
|
|
253
|
+
const Authorization = await this.getAuthorizationHeader(); // Wait a little before `this.getCards` so that it would exist
|
|
254
|
+
|
|
255
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
256
|
+
const cards = await this.getCards();
|
|
257
|
+
const xSiteId = await this.getXSiteId();
|
|
258
|
+
const accounts = await Promise.all(cards.map(async card => {
|
|
259
|
+
var _this$options$outputD, _this$options$outputD2;
|
|
260
|
+
|
|
261
|
+
debug(`fetch transactions for card ${card.cardUniqueId}`);
|
|
262
|
+
const nextBillingMonth = (0, _moment.default)().add(1, 'month');
|
|
263
|
+
const months = nextBillingMonth.diff(startMoment, 'months');
|
|
264
|
+
const allMonthsData = [];
|
|
265
|
+
|
|
266
|
+
for (let i = 0; i <= months; i += 1) {
|
|
267
|
+
const month = nextBillingMonth.clone().subtract(i, 'months');
|
|
268
|
+
const monthData = await (0, _fetch.fetchPostWithinPage)(this.page, TRANSACTIONS_REQUEST_ENDPOINT, {
|
|
269
|
+
cardUniqueId: card.cardUniqueId,
|
|
270
|
+
month: month.format('M'),
|
|
271
|
+
year: month.format('YYYY')
|
|
272
|
+
}, {
|
|
273
|
+
Authorization,
|
|
274
|
+
'X-Site-Id': xSiteId,
|
|
275
|
+
'Content-Type': 'application/json'
|
|
276
|
+
});
|
|
277
|
+
if ((monthData === null || monthData === void 0 ? void 0 : monthData.statusCode) !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${(monthData === null || monthData === void 0 ? void 0 : monthData.title) || ''}`);
|
|
278
|
+
|
|
279
|
+
if (!this.isCardTransactionDetails(monthData)) {
|
|
280
|
+
throw new Error('monthData is not of type CardTransactionDetails');
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
allMonthsData.push(monthData);
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
const transactions = convertParsedDataToTransactions(allMonthsData);
|
|
287
|
+
debug('filer out old transactions');
|
|
288
|
+
const txns = ((_this$options$outputD = (_this$options$outputD2 = this.options.outputData) === null || _this$options$outputD2 === void 0 ? void 0 : _this$options$outputD2.enableTransactionsFilterByDate) !== null && _this$options$outputD !== void 0 ? _this$options$outputD : true) ? (0, _transactions.filterOldTransactions)(transactions, (0, _moment.default)(startDate), this.options.combineInstallments || false) : transactions;
|
|
289
|
+
return {
|
|
290
|
+
txns,
|
|
291
|
+
accountNumber: card.last4Digits
|
|
292
|
+
};
|
|
293
|
+
}));
|
|
405
294
|
debug('return the scraped accounts');
|
|
295
|
+
debug(JSON.stringify(accounts, null, 2));
|
|
406
296
|
return {
|
|
407
297
|
success: true,
|
|
408
|
-
accounts
|
|
409
|
-
futureDebits
|
|
298
|
+
accounts
|
|
410
299
|
};
|
|
411
300
|
}
|
|
412
301
|
|
|
@@ -414,4 +303,4 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
414
303
|
|
|
415
304
|
var _default = VisaCalScraper;
|
|
416
305
|
exports.default = _default;
|
|
417
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_URL","LONG_DATE_FORMAT","DATE_FORMAT","InvalidPasswordMessage","debug","getLoginFrame","page","frame","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","errorMessage","item","innerText","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","amountStrCln","replace","currency","amount","SHEKEL_CURRENCY_SYMBOL","parseFloat","SHEKEL_CURRENCY","DOLLAR_CURRENCY_SYMBOL","DOLLAR_CURRENCY","EURO_CURRENCY_SYMBOL","EURO_CURRENCY","parts","split","getTransactionInstallments","memo","parsedMemo","exec","length","number","parseInt","total","convertTransactions","txns","map","txn","originalAmountTuple","originalAmount","chargedAmountTuple","chargedAmount","installments","txnDate","date","processedDateFormat","processedDate","txnProcessedDate","result","type","TransactionTypes","Installments","Normal","status","TransactionStatuses","Completed","add","toISOString","originalCurrency","chargedCurrency","description","fetchTransactionsForAccount","startDate","accountNumber","scraperOptions","startDateValue","format","dateSelector","dateHiddenFieldSelector","buttonSelector","nextPageSelector","billingLabelSelector","secondaryBillingLabelSelector","noDataSelector","items","el","startDateIndex","findIndex","option","accountTransactions","currentDateIndex","waitForTimeout","all","waitForNavigation","waitUntil","pageHasNoTransactions","element","siteValue","billingDateLabel","settlementDateRegex","billingDate","hasNextPage","rawTransactions","columns","getElementsByTagName","push","filter","outputData","enableTransactionsFilterByDate","combineInstallments","substring","getAccountNumbers","elements","e","text","then","res","trim","setAccount","account","elem","a","click","fetchTransactions","accountNumbers","accounts","fetchFutureDebits","futureDebitsSelector","debitMountClass","debitWhenChargeClass","debitBankNumberClass","currBankEl","getElementsByClassName","whenCharge","bankNumber","futureDebits","amountData","chargeDate","bankAccountNumber","amountCurrency","VisaCalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","userAgent","fetchData","defaultStartMoment","subtract","toDate","startMoment","moment","max","navigateTo","undefined","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Frame, Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport {\n  clickButton, elementPresentOnPage, pageEval, pageEvalAll, setValue, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport {\n  Transaction,\n  TransactionInstallments,\n  TransactionsAccount,\n  TransactionStatuses,\n  TransactionTypes,\n} from '../transactions';\nimport { ScraperOptions, ScaperScrapingResult, ScraperCredentials } from './base-scraper';\nimport {\n  DOLLAR_CURRENCY, DOLLAR_CURRENCY_SYMBOL, EURO_CURRENCY, EURO_CURRENCY_SYMBOL, SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL,\n} from '../constants';\nimport { waitUntil } from '../helpers/waiting';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { getDebug } from '../helpers/debug';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_URL = 'https://services.cal-online.co.il/Card-Holders/Screens/Transactions/Transactions.aspx';\nconst LONG_DATE_FORMAT = 'DD/MM/YYYY';\nconst DATE_FORMAT = 'DD/MM/YY';\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\ninterface ScrapedTransaction {\n  date: string;\n  processedDate: string;\n  description: string;\n  originalAmount: string;\n  chargedAmount: string;\n  memo: string;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/AccountManagement/i],\n    [LoginResults.InvalidPassword]: [async (options?: { page?: Page}) => {\n      const page = options?.page;\n      if (!page) {\n        return false;\n      }\n      return hasInvalidPasswordError(page);\n    }],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    // [LoginResults.ChangePassword]: [], // TODO add when reaching this scenario\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(DOLLAR_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(DOLLAR_CURRENCY_SYMBOL, ''));\n    currency = DOLLAR_CURRENCY;\n  } else if (amountStrCln.includes(EURO_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(EURO_CURRENCY_SYMBOL, ''));\n    currency = EURO_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    [currency] = parts;\n    amount = -parseFloat(parts[1]);\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction getTransactionInstallments(memo: string): TransactionInstallments | null {\n  const parsedMemo = (/תשלום (\\d+) מתוך (\\d+)/).exec(memo || '');\n\n  if (!parsedMemo || parsedMemo.length === 0) {\n    return null;\n  }\n\n  return {\n    number: parseInt(parsedMemo[1], 10),\n    total: parseInt(parsedMemo[2], 10),\n  };\n}\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  debug(`convert ${txns.length} raw transactions to official Transaction structure`);\n  return txns.map((txn) => {\n    const originalAmountTuple = getAmountData(txn.originalAmount || '');\n    const chargedAmountTuple = getAmountData(txn.chargedAmount || '');\n\n    const installments = getTransactionInstallments(txn.memo);\n    const txnDate = moment(txn.date, DATE_FORMAT);\n    const processedDateFormat =\n      txn.processedDate.length === 8 ?\n        DATE_FORMAT :\n        txn.processedDate.length === 9 || txn.processedDate.length === 10 ?\n          LONG_DATE_FORMAT :\n          null;\n    if (!processedDateFormat) {\n      throw new Error('invalid processed date');\n    }\n    const txnProcessedDate = moment(txn.processedDate, processedDateFormat);\n\n    const result: Transaction = {\n      type: installments ? TransactionTypes.Installments : TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      date: installments ? txnDate.add(installments.number - 1, 'month').toISOString() : txnDate.toISOString(),\n      processedDate: txnProcessedDate.toISOString(),\n      originalAmount: originalAmountTuple.amount,\n      originalCurrency: originalAmountTuple.currency,\n      chargedAmount: chargedAmountTuple.amount,\n      chargedCurrency: chargedAmountTuple.currency,\n      description: txn.description || '',\n      memo: txn.memo || '',\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment, accountNumber: string, scraperOptions: ScraperOptions): Promise<TransactionsAccount> {\n  const startDateValue = startDate.format('MM/YYYY');\n  const dateSelector = '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_TextBox\"]';\n  const dateHiddenFieldSelector = '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_HiddenField\"]';\n  const buttonSelector = '[id$=\"FormAreaNoBorder_FormArea_ctlSubmitRequest\"]';\n  const nextPageSelector = '[id$=\"FormAreaNoBorder_FormArea_ctlGridPager_btnNext\"]';\n  const billingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlMainToolBar_lblCaption]';\n  const secondaryBillingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlSecondaryToolBar_lblCaption]';\n  const noDataSelector = '[id$=FormAreaNoBorder_FormArea_msgboxErrorMessages]';\n\n  debug('find the start date index in the dropbox');\n  const options = await pageEvalAll(page, '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_OptionList\"] li', [], (items) => {\n    return items.map((el: any) => el.innerText);\n  });\n  const startDateIndex = options.findIndex((option) => option === startDateValue);\n\n  debug(`scrape ${options.length - startDateIndex} billing cycles`);\n  const accountTransactions: Transaction[] = [];\n  for (let currentDateIndex = startDateIndex; currentDateIndex < options.length; currentDateIndex += 1) {\n    debug('wait for date selector to be found');\n    await waitUntilElementFound(page, dateSelector, true);\n    debug(`set hidden value of the date selector to be the index ${currentDateIndex}`);\n    await setValue(page, dateHiddenFieldSelector, `${currentDateIndex}`);\n    debug('wait a second to workaround navigation issue in headless browser mode');\n    await page.waitForTimeout(1000);\n    debug('click on the filter submit button and wait for navigation');\n    await Promise.all([\n      page.waitForNavigation({ waitUntil: 'domcontentloaded' }),\n      clickButton(page, buttonSelector),\n    ]);\n    debug('check if month has no transactions');\n    const pageHasNoTransactions = await pageEval(page, noDataSelector, false, ((element) => {\n      const siteValue = ((element as HTMLSpanElement).innerText || '').replace(/[^ א-ת]/g, '');\n      return siteValue === 'לא נמצאו נתונים';\n    }));\n\n    if (pageHasNoTransactions) {\n      debug('page has no transactions');\n    } else {\n      debug('find the billing date');\n      let billingDateLabel = await pageEval(page, billingLabelSelector, '', ((element) => {\n        return (element as HTMLSpanElement).innerText;\n      }));\n      let settlementDateRegex = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/;\n\n      if (billingDateLabel === '') {\n        billingDateLabel = await pageEval(page, secondaryBillingLabelSelector, '', ((element) => {\n          return (element as HTMLSpanElement).innerText;\n        }));\n        settlementDateRegex = /\\d{1,2}[/]\\d{2,4}/;\n      }\n\n      const billingDate = settlementDateRegex.exec(billingDateLabel)?.[0];\n\n      if (!billingDate) {\n        throw new Error('failed to fetch process date');\n      }\n\n      debug(`found the billing date for that month ${billingDate}`);\n      let hasNextPage = false;\n      do {\n        debug('fetch raw transactions from page');\n        const rawTransactions = await pageEvalAll<(ScrapedTransaction | null)[]>(page, '#ctlMainGrid > tbody tr, #ctlSecondaryGrid > tbody tr', [], (items, billingDate) => {\n          return (items).map((el) => {\n            const columns = el.getElementsByTagName('td');\n            if (columns.length === 6) {\n              return {\n                processedDate: columns[0].innerText,\n                date: columns[1].innerText,\n                description: columns[2].innerText,\n                originalAmount: columns[3].innerText,\n                chargedAmount: columns[4].innerText,\n                memo: columns[5].innerText,\n              };\n            }\n            if (columns.length === 5) {\n              return {\n                processedDate: billingDate,\n                date: columns[0].innerText,\n                description: columns[1].innerText,\n                originalAmount: columns[2].innerText,\n                chargedAmount: columns[3].innerText,\n                memo: columns[4].innerText,\n              };\n            }\n            return null;\n          });\n        }, billingDate);\n        debug(`fetched ${rawTransactions.length} raw transactions from page`);\n        accountTransactions.push(...convertTransactions((rawTransactions as ScrapedTransaction[])\n          .filter((item) => !!item)));\n\n        debug('check for existance of another page');\n        hasNextPage = await elementPresentOnPage(page, nextPageSelector);\n        if (hasNextPage) {\n          debug('has another page, click on button next and wait for page navigation');\n          await Promise.all([\n            page.waitForNavigation({ waitUntil: 'domcontentloaded' }),\n            await clickButton(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]'),\n          ]);\n        }\n      } while (hasNextPage);\n    }\n  }\n\n  debug('filer out old transactions');\n  const txns = (scraperOptions.outputData?.enableTransactionsFilterByDate ?? true) ?\n    filterOldTransactions(accountTransactions, startDate, scraperOptions.combineInstallments || false) :\n    accountTransactions;\n  debug(`found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`);\n  return {\n    accountNumber,\n    txns,\n  };\n}\n\nasync function getAccountNumbers(page: Page): Promise<string[]> {\n  return pageEvalAll(page, '[id$=lnkItem]', [], (elements) => elements.map((e) => (e as HTMLAnchorElement).text)).then((res) => res.map((text) => /\\d+$/.exec(text.trim())?.[0] ?? ''));\n}\n\nasync function setAccount(page: Page, account: string) {\n  await pageEvalAll(\n    page,\n    '[id$=lnkItem]',\n    null,\n    (elements, account) => {\n      for (const elem of elements) {\n        const a = elem as HTMLAnchorElement;\n        if (a.text.includes(account)) {\n          a.click();\n        }\n      }\n    },\n    account,\n  );\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment, scraperOptions: ScraperOptions): Promise<TransactionsAccount[]> {\n  const accountNumbers: string[] = await getAccountNumbers(page);\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of accountNumbers) {\n    debug(`setting account: ${account}`);\n    await setAccount(page, account);\n    await page.waitForTimeout(1000);\n    accounts.push(\n      await fetchTransactionsForAccount(\n        page,\n        startDate,\n        account,\n        scraperOptions,\n      ),\n    );\n  }\n\n  return accounts;\n}\n\nasync function fetchFutureDebits(page: Page) {\n  const futureDebitsSelector = '.homepage-banks-top';\n\n  const result = await pageEvalAll(page, futureDebitsSelector, [], (items) => {\n    const debitMountClass = 'amount';\n    const debitWhenChargeClass = 'when-charge';\n    const debitBankNumberClass = 'bankDesc';\n\n    return items.map((currBankEl: any) => {\n      const amount = currBankEl.getElementsByClassName(debitMountClass)[0].innerText;\n      const whenCharge = currBankEl.getElementsByClassName(debitWhenChargeClass)[0].innerText;\n      const bankNumber = currBankEl.getElementsByClassName(debitBankNumberClass)[0].innerText;\n      return {\n        amount,\n        whenCharge,\n        bankNumber,\n      };\n    });\n  });\n  const futureDebits = result.map((item) => {\n    const amountData = getAmountData(item.amount);\n    const chargeDate = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/.exec(item.whenCharge)?.[0];\n    const bankAccountNumber = /\\d+-\\d+/.exec(item.bankNumber)?.[0];\n    return {\n      amount: amountData.amount,\n      amountCurrency: amountData.currency,\n      chargeDate,\n      bankAccountNumber,\n    };\n  });\n  return futureDebits;\n}\n\nclass VisaCalScraper extends BaseScraperWithBrowser {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  getLoginOptions(credentials: Record<string, string>) {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\n    };\n  }\n\n  async fetchData(): Promise<ScaperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    debug('fetch future debits');\n    const futureDebits = await fetchFutureDebits(this.page);\n\n    debug('navigate to transactions page');\n    await this.navigateTo(TRANSACTIONS_URL, undefined, 60000);\n\n    debug('fetch accounts transactions');\n    const accounts = await fetchTransactions(this.page, startMoment, this.options);\n\n    debug('return the scraped accounts');\n    return {\n      success: true,\n      accounts,\n      futureDebits,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
|
|
306
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","trnTypeCode","getLoginFrame","page","frame","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","errorMessage","item","innerText","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","createLoginFields","credentials","selector","value","username","password","cardAndTransactionCurrencySymbolIsShekel","transaction","debCrdCurrencySymbol","SHEKEL_CURRENCY_SYMBOL","trnCurrencySymbol","convertParsedDataToTransactions","parsedData","flatMap","monthData","result","bankAccounts","accounts","debitDates","debitDate","transactions","map","installments","curPaymentNum","numOfPayments","number","total","undefined","date","trnPurchaseDate","chargedAmount","amtBeforeConvAndIndex","trnAmt","credit","description","merchantName","originalAmount","originalCurrency","processedDate","debCrdDate","status","TransactionStatuses","Completed","add","toISOString","type","regular","standingOrder","TransactionTypes","Normal","Installments","VisaCalScraper","BaseScraperWithBrowser","getCards","initData","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","auth","calConnectToken","getXSiteId","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","postAction","currentUrl","endsWith","e","userAgent","isCardTransactionDetails","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","moment","max","format","Authorization","setTimeout","xSiteId","all","card","nextBillingMonth","months","diff","allMonthsData","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","combineInstallments","accountNumber","JSON","stringify","success"],"mappings":";;;;;;;;;;;;;;;AAAA;;AAGA;;AACA;;AACA;;AAGA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAMA;;;;;;AAGA,MAAMA,SAAS,GAAG,+BAAlB;AACA,MAAMC,6BAA6B,GAAG,8FAAtC;AAEA,MAAMC,sBAAsB,GAAG,mCAA/B;AAEA,MAAMC,KAAK,GAAG,qBAAS,UAAT,CAAd;IAEKC,W;;WAAAA,W;AAAAA,EAAAA,W;AAAAA,EAAAA,W;AAAAA,EAAAA,W;AAAAA,EAAAA,W;GAAAA,W,KAAAA,W;;AAiGL,eAAeC,aAAf,CAA6BC,IAA7B,EAAyC;AACvC,MAAIC,KAAmB,GAAG,IAA1B;AACAJ,EAAAA,KAAK,CAAC,8BAAD,CAAL;AACA,QAAM,wBAAU,MAAM;AACpBI,IAAAA,KAAK,GAAGD,IAAI,CACTE,MADK,GAELC,IAFK,CAECC,CAAD,IAAOA,CAAC,CAACC,GAAF,GAAQC,QAAR,CAAiB,YAAjB,CAFP,KAE0C,IAFlD;AAGA,WAAOC,OAAO,CAACC,OAAR,CAAgB,CAAC,CAACP,KAAlB,CAAP;AACD,GALK,EAKH,iCALG,EAKgC,KALhC,EAKuC,IALvC,CAAN;;AAOA,MAAI,CAACA,KAAL,EAAY;AACVJ,IAAAA,KAAK,CAAC,2CAAD,CAAL;AACA,UAAM,IAAIY,KAAJ,CAAU,gCAAV,CAAN;AACD;;AAED,SAAOR,KAAP;AACD;;AAED,eAAeS,uBAAf,CAAuCV,IAAvC,EAAmD;AACjD,QAAMC,KAAK,GAAG,MAAMF,aAAa,CAACC,IAAD,CAAjC;AACA,QAAMW,UAAU,GAAG,MAAM,gDAAqBV,KAArB,EAA4B,yBAA5B,CAAzB;AACA,QAAMW,YAAY,GAAGD,UAAU,GAAG,MAAM,oCAASV,KAAT,EAAgB,yBAAhB,EAA2C,EAA3C,EAAgDY,IAAD,IAAU;AAC/F,WAAQA,IAAD,CAAyBC,SAAhC;AACD,GAFuC,CAAT,GAE1B,EAFL;AAGA,SAAOF,YAAY,KAAKhB,sBAAxB;AACD;;AAED,SAASmB,uBAAT,GAAmC;AACjClB,EAAAA,KAAK,CAAC,+BAAD,CAAL;AACA,QAAMmB,IAAqC,GAAG;AAC5C,KAACC,qCAAaC,OAAd,GAAwB,CAAC,YAAD,CADoB;AAE5C,KAACD,qCAAaE,eAAd,GAAgC,CAAC,MAAOC,OAAP,IAAqC;AACpE,YAAMpB,IAAI,GAAGoB,OAAH,aAAGA,OAAH,uBAAGA,OAAO,CAAEpB,IAAtB;;AACA,UAAI,CAACA,IAAL,EAAW;AACT,eAAO,KAAP;AACD;;AACD,aAAOU,uBAAuB,CAACV,IAAD,CAA9B;AACD,KAN+B,CAFY,CAS5C;AACA;;AAV4C,GAA9C;AAYA,SAAOgB,IAAP;AACD;;AAED,SAASK,iBAAT,CAA2BC,WAA3B,EAAoE;AAClEzB,EAAAA,KAAK,CAAC,+CAAD,CAAL;AACA,SAAO,CACL;AAAE0B,IAAAA,QAAQ,EAAE,8BAAZ;AAA4CC,IAAAA,KAAK,EAAEF,WAAW,CAACG;AAA/D,GADK,EAEL;AAAEF,IAAAA,QAAQ,EAAE,8BAAZ;AAA4CC,IAAAA,KAAK,EAAEF,WAAW,CAACI;AAA/D,GAFK,CAAP;AAID;;AAED,SAASC,wCAAT,CAAkDC,WAAlD,EAAmF;AACjF,SAAOA,WAAW,CAACC,oBAAZ,KAAqCC,iCAArC,IACLF,WAAW,CAACG,iBAAZ,KAAkCD,iCADpC;AAED;;AACD,SAASE,+BAAT,CAAyCC,UAAzC,EAA8F;AAC5F,SAAOA,UAAU,CACdC,OADI,CACKC,SAAD,IAAeA,SAAS,CAACC,MAAV,CAAiBC,YADpC,EAEJH,OAFI,CAEKI,QAAD,IAAcA,QAAQ,CAACC,UAF3B,EAGJL,OAHI,CAGKM,SAAD,IAAeA,SAAS,CAACC,YAH7B,EAIJC,GAJI,CAICd,WAAD,IAAiB;AACpB,UAAMe,YAAY,GAAIf,WAAW,CAACgB,aAAZ,IAA6BhB,WAAW,CAACiB,aAAzC,IACtB;AACEC,MAAAA,MAAM,EAAElB,WAAW,CAACgB,aADtB;AAEEG,MAAAA,KAAK,EAAEnB,WAAW,CAACiB;AAFrB,KADqB,IAKnBG,SALF;AAOA,UAAMC,IAAI,GAAG,qBAAOrB,WAAW,CAACsB,eAAnB,CAAb,CARoB,CAUpB;;AACA,QAAIC,aAAJ;;AACA,QAAIxB,wCAAwC,CAACC,WAAD,CAA5C,EAA2D;AACzDuB,MAAAA,aAAa,GAAGvB,WAAW,CAACwB,qBAAZ,GAAqC,CAAC,CAAtD;AACD,KAFD,MAEO;AACLD,MAAAA,aAAa,GAAGvB,WAAW,CAACyB,MAAZ,GAAsB,CAAC,CAAvC;;AAEA,UAAIzB,WAAW,CAAC9B,WAAZ,KAA4BA,WAAW,CAACwD,MAA5C,EAAoD;AAClDH,QAAAA,aAAa,GAAGvB,WAAW,CAACyB,MAA5B;AACD;AACF;;AAED,UAAMjB,MAAmB,GAAG;AAC1Be,MAAAA,aAD0B;AAE1BI,MAAAA,WAAW,EAAE3B,WAAW,CAAC4B,YAFC;AAG1BC,MAAAA,cAAc,EAAE7B,WAAW,CAACwB,qBAHF;AAI1BM,MAAAA,gBAAgB,EAAE9B,WAAW,CAACG,iBAJJ;AAK1B4B,MAAAA,aAAa,EAAE/B,WAAW,CAACgC,UALD;AAM1BC,MAAAA,MAAM,EAAEC,mCAAoBC,SANF;AAO1Bd,MAAAA,IAAI,EAAEN,YAAY,GAChBM,IAAI,CAACe,GAAL,CAASrB,YAAY,CAACG,MAAb,GAAsB,CAA/B,EAAkC,OAAlC,EAA2CmB,WAA3C,EADgB,GAEhBhB,IAAI,CAACgB,WAAL,EATwB;AAU1BC,MAAAA,IAAI,EAAE,CAACpE,WAAW,CAACqE,OAAb,EAAsBrE,WAAW,CAACsE,aAAlC,EAAiD9D,QAAjD,CAA0DsB,WAAW,CAAC9B,WAAtE,IACJuE,gCAAiBC,MADb,GAEJD,gCAAiBE;AAZO,KAA5B;;AAeA,QAAI5B,YAAJ,EAAkB;AAChBP,MAAAA,MAAM,CAACO,YAAP,GAAsBA,YAAtB;AACD;;AAED,WAAOP,MAAP;AACD,GA9CI,CAAP;AA+CD;;AAID,MAAMoC,cAAN,SAA6BC,8CAA7B,CAAgF;AAAA;AAAA;;AAAA,4CAC7D,YAAY;AAC3B5E,MAAAA,KAAK,CAAC,qDAAD,CAAL;AACA,YAAM,iDAAsB,KAAKG,IAA3B,EAAiC,oBAAjC,EAAuD,IAAvD,CAAN;AACAH,MAAAA,KAAK,CAAC,2BAAD,CAAL;AACA,YAAM,uCAAY,KAAKG,IAAjB,EAAuB,oBAAvB,CAAN;AACAH,MAAAA,KAAK,CAAC,oCAAD,CAAL;AACA,YAAMI,KAAK,GAAG,MAAMF,aAAa,CAAC,KAAKC,IAAN,CAAjC;AACAH,MAAAA,KAAK,CAAC,uDAAD,CAAL;AACA,YAAM,iDAAsBI,KAAtB,EAA6B,gBAA7B,CAAN;AACAJ,MAAAA,KAAK,CAAC,oCAAD,CAAL;AACA,YAAM,uCAAYI,KAAZ,EAAmB,gBAAnB,CAAN;AACAJ,MAAAA,KAAK,CAAC,6CAAD,CAAL;AACA,YAAM,iDAAsBI,KAAtB,EAA6B,eAA7B,CAAN;AAEA,aAAOA,KAAP;AACD,KAhB6E;AAAA;;AAkB9E,QAAMyE,QAAN,GAAiB;AACf,UAAMC,QAAQ,GAAG,MAAM,oCAAoC,KAAK3E,IAAzC,EAA+C,MAA/C,CAAvB;;AACA,QAAI,CAAC2E,QAAL,EAAe;AACb,YAAM,IAAIlE,KAAJ,CAAU,iDAAV,CAAN;AACD;;AACD,WAAOkE,QAAP,aAAOA,QAAP,uBAAOA,QAAQ,CAAEvC,MAAV,CAAiBwC,KAAjB,CAAuBlC,GAAvB,CAA2B,CAAC;AAAEmC,MAAAA,YAAF;AAAgBC,MAAAA;AAAhB,KAAD,MAAoC;AAAED,MAAAA,YAAF;AAAgBC,MAAAA;AAAhB,KAApC,CAA3B,CAAP;AACD;;AAED,QAAMC,sBAAN,GAA+B;AAC7B,UAAMC,UAAU,GAAG,MAAM,oCAA6D,KAAKhF,IAAlE,EAAwE,aAAxE,CAAzB;;AACA,QAAI,CAACgF,UAAL,EAAiB;AACf,YAAM,IAAIvE,KAAJ,CAAU,mDAAV,CAAN;AACD;;AACD,WAAQ,iBAAgBuE,UAAU,CAACC,IAAX,CAAgBC,eAAgB,EAAxD;AACD;;AAED,QAAMC,UAAN,GAAmB;AACjB;;;;;;;;;;;;AAcA,WAAO5E,OAAO,CAACC,OAAR,CAAgB,sCAAhB,CAAP;AACD;;AAED4E,EAAAA,eAAe,CAAC9D,WAAD,EAAwD;AACrE,WAAO;AACL+D,MAAAA,QAAQ,EAAG,GAAE3F,SAAU,EADlB;AAEL4F,MAAAA,MAAM,EAAEjE,iBAAiB,CAACC,WAAD,CAFpB;AAGLiE,MAAAA,oBAAoB,EAAE,uBAHjB;AAILC,MAAAA,eAAe,EAAEzE,uBAAuB,EAJnC;AAKL0E,MAAAA,cAAc,EAAE,YAAY,iDAAsB,KAAKzF,IAA3B,EAAiC,oBAAjC,CALvB;AAML0F,MAAAA,SAAS,EAAE,KAAKC,cANX;AAOLC,MAAAA,UAAU,EAAE,YAAY;AACtB,YAAI;AACF,gBAAM,iDAAsB,KAAK5F,IAA3B,EAAiC,kBAAjC,CAAN;AACA,gBAAM6F,UAAU,GAAG,MAAM,+BAAc,KAAK7F,IAAnB,CAAzB;;AACA,cAAI6F,UAAU,CAACC,QAAX,CAAoB,eAApB,CAAJ,EAA0C;AACxC,kBAAM,uCAAY,KAAK9F,IAAjB,EAAuB,kBAAvB,CAAN;AACD;AACF,SAND,CAME,OAAO+F,CAAP,EAAU;AACV,gBAAMF,UAAU,GAAG,MAAM,+BAAc,KAAK7F,IAAnB,CAAzB;AACA,cAAI6F,UAAU,CAACC,QAAX,CAAoB,WAApB,CAAJ,EAAsC;AACtC,gBAAMC,CAAN;AACD;AACF,OAnBI;AAoBLC,MAAAA,SAAS,EAAE;AApBN,KAAP;AAsBD;;AAEDC,EAAAA,wBAAwB,CAAC7D,MAAD,EACW;AACjC,WAAQA,MAAD,CAAmCA,MAAnC,KAA8CY,SAArD;AACD;;AAED,QAAMkD,SAAN,GAAkD;AAChD,UAAMC,kBAAkB,GAAG,uBAASC,QAAT,CAAkB,CAAlB,EAAqB,OAArB,EAA8BA,QAA9B,CAAuC,CAAvC,EAA0C,QAA1C,EAAoDpC,GAApD,CAAwD,CAAxD,EAA2D,KAA3D,CAA3B;AACA,UAAMqC,SAAS,GAAG,KAAKjF,OAAL,CAAaiF,SAAb,IAA0BF,kBAAkB,CAACG,MAAnB,EAA5C;;AACA,UAAMC,WAAW,GAAGC,gBAAOC,GAAP,CAAWN,kBAAX,EAA+B,qBAAOE,SAAP,CAA/B,CAApB;;AACAxG,IAAAA,KAAK,CAAE,+BAA8B0G,WAAW,CAACG,MAAZ,EAAqB,EAArD,CAAL;AAEA,UAAMC,aAAa,GAAG,MAAM,KAAK5B,sBAAL,EAA5B,CANgD,CAOhD;;AACA,UAAM,IAAIxE,OAAJ,CAAaC,OAAD,IAAaoG,UAAU,CAACpG,OAAD,EAAU,IAAV,CAAnC,CAAN;AACA,UAAMoE,KAAK,GAAG,MAAM,KAAKF,QAAL,EAApB;AACA,UAAMmC,OAAO,GAAG,MAAM,KAAK1B,UAAL,EAAtB;AAGA,UAAM7C,QAAQ,GAAG,MAAM/B,OAAO,CAACuG,GAAR,CACrBlC,KAAK,CAAClC,GAAN,CAAU,MAAOqE,IAAP,IAAgB;AAAA;;AACxBlH,MAAAA,KAAK,CAAE,+BAA8BkH,IAAI,CAAClC,YAAa,EAAlD,CAAL;AAEA,YAAMmC,gBAAgB,GAAG,uBAAShD,GAAT,CAAa,CAAb,EAAgB,OAAhB,CAAzB;AACA,YAAMiD,MAAM,GAAGD,gBAAgB,CAACE,IAAjB,CAAsBX,WAAtB,EAAmC,QAAnC,CAAf;AAEA,YAAMY,aAAyC,GAAG,EAAlD;;AACA,WAAK,IAAIC,CAAC,GAAG,CAAb,EAAgBA,CAAC,IAAIH,MAArB,EAA6BG,CAAC,IAAI,CAAlC,EAAqC;AACnC,cAAMC,KAAK,GAAGL,gBAAgB,CAACM,KAAjB,GAAyBlB,QAAzB,CAAkCgB,CAAlC,EAAqC,QAArC,CAAd;AACA,cAAMjF,SAAS,GAAG,MAAM,gCACtB,KAAKnC,IADiB,EACXL,6BADW,EAEtB;AAAEkF,UAAAA,YAAY,EAAEkC,IAAI,CAAClC,YAArB;AAAmCwC,UAAAA,KAAK,EAAEA,KAAK,CAACX,MAAN,CAAa,GAAb,CAA1C;AAA6Da,UAAAA,IAAI,EAAEF,KAAK,CAACX,MAAN,CAAa,MAAb;AAAnE,SAFsB,EAGtB;AACEC,UAAAA,aADF;AAEE,uBAAaE,OAFf;AAGE,0BAAgB;AAHlB,SAHsB,CAAxB;AAUA,YAAI,CAAA1E,SAAS,SAAT,IAAAA,SAAS,WAAT,YAAAA,SAAS,CAAEqF,UAAX,MAA0B,CAA9B,EAAiC,MAAM,IAAI/G,KAAJ,CAAW,yCAAwCsG,IAAI,CAACjC,WAAY,cAAa,CAAA3C,SAAS,SAAT,IAAAA,SAAS,WAAT,YAAAA,SAAS,CAAEsF,KAAX,KAAoB,EAAG,EAAxG,CAAN;;AAEjC,YAAI,CAAC,KAAKxB,wBAAL,CAA8B9D,SAA9B,CAAL,EAA+C;AAC7C,gBAAM,IAAI1B,KAAJ,CAAU,iDAAV,CAAN;AACD;;AAED0G,QAAAA,aAAa,CAACO,IAAd,CAAmBvF,SAAnB;AACD;;AAED,YAAMM,YAAY,GAAGT,+BAA+B,CAACmF,aAAD,CAApD;AAEAtH,MAAAA,KAAK,CAAC,4BAAD,CAAL;AACA,YAAM8H,IAAI,GAAG,oDAAC,KAAKvG,OAAL,CAAawG,UAAd,2DAAC,uBAAyBC,8BAA1B,yEAA4D,IAA5D,IACX,yCAAsBpF,YAAtB,EAAoC,qBAAO4D,SAAP,CAApC,EAAuD,KAAKjF,OAAL,CAAa0G,mBAAb,IAAoC,KAA3F,CADW,GAEXrF,YAFF;AAIA,aAAO;AACLkF,QAAAA,IADK;AAELI,QAAAA,aAAa,EAAEhB,IAAI,CAACjC;AAFf,OAAP;AAID,KAvCD,CADqB,CAAvB;AA2CAjF,IAAAA,KAAK,CAAC,6BAAD,CAAL;AAEAA,IAAAA,KAAK,CAACmI,IAAI,CAACC,SAAL,CAAe3F,QAAf,EAAyB,IAAzB,EAA+B,CAA/B,CAAD,CAAL;AACA,WAAO;AACL4F,MAAAA,OAAO,EAAE,IADJ;AAEL5F,MAAAA;AAFK,KAAP;AAID;;AAjJ6E;;eAoJjEkC,c","sourcesContent":["import moment from 'moment';\nimport { Frame, Page } from 'puppeteer';\n\nimport { SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport { getDebug } from '../helpers/debug';\nimport {\n  clickButton, elementPresentOnPage, pageEval, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport {\n  Transaction,\n  TransactionStatuses,\n  TransactionTypes,\n  TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum trnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: trnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = '₪' | string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: [], debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [async (options?: { page?: Page }) => {\n      const page = options?.page;\n      if (!page) {\n        return false;\n      }\n      return hasInvalidPasswordError(page);\n    }],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    // [LoginResults.ChangePassword]: [], // TODO add when reaching this scenario\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction cardAndTransactionCurrencySymbolIsShekel(transaction: ScrapedTransaction) {\n  return transaction.debCrdCurrencySymbol === SHEKEL_CURRENCY_SYMBOL &&\n    transaction.trnCurrencySymbol === SHEKEL_CURRENCY_SYMBOL;\n}\nfunction convertParsedDataToTransactions(parsedData: CardTransactionDetails[]): Transaction[] {\n  return parsedData\n    .flatMap((monthData) => monthData.result.bankAccounts)\n    .flatMap((accounts) => accounts.debitDates)\n    .flatMap((debitDate) => debitDate.transactions)\n    .map((transaction) => {\n      const installments = (transaction.curPaymentNum && transaction.numOfPayments &&\n      {\n        number: transaction.curPaymentNum,\n        total: transaction.numOfPayments,\n      }) ||\n        undefined;\n\n      const date = moment(transaction.trnPurchaseDate);\n\n      // I didn't test `amtBeforeConvAndIndex` with a foreign currency as I don't have such transactions\n      let chargedAmount: number;\n      if (cardAndTransactionCurrencySymbolIsShekel(transaction)) {\n        chargedAmount = transaction.amtBeforeConvAndIndex * (-1);\n      } else {\n        chargedAmount = transaction.trnAmt * (-1);\n\n        if (transaction.trnTypeCode === trnTypeCode.credit) {\n          chargedAmount = transaction.trnAmt;\n        }\n      }\n\n      const result: Transaction = {\n        chargedAmount,\n        description: transaction.merchantName,\n        originalAmount: transaction.amtBeforeConvAndIndex,\n        originalCurrency: transaction.trnCurrencySymbol,\n        processedDate: transaction.debCrdDate,\n        status: TransactionStatuses.Completed,\n        date: installments ?\n          date.add(installments.number - 1, 'month').toISOString() :\n          date.toISOString(),\n        type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ?\n          TransactionTypes.Normal :\n          TransactionTypes.Installments,\n      };\n\n      if (installments) {\n        result.installments = installments;\n      }\n\n      return result;\n    });\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await getFromSessionStorage<InitResponse>(this.page, 'init');\n    if (!initData) {\n      throw new Error('could not find \\'init\\' data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error('could not find \\'auth-module\\' in session storage');\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitUntilElementFound(this.page, 'button.btn-close');\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          throw e;\n        }\n      },\n      userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\n    };\n  }\n\n  isCardTransactionDetails(result: CardTransactionDetails | CardTransactionDetailsError):\n    result is CardTransactionDetails {\n    return (result as CardTransactionDetails).result !== undefined;\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const Authorization = await this.getAuthorizationHeader();\n    // Wait a little before `this.getCards` so that it would exist\n    await new Promise((resolve) => setTimeout(resolve, 1000));\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n\n\n    const accounts = await Promise.all(\n      cards.map(async (card) => {\n        debug(`fetch transactions for card ${card.cardUniqueId}`);\n\n        const nextBillingMonth = moment().add(1, 'month');\n        const months = nextBillingMonth.diff(startMoment, 'months');\n\n        const allMonthsData: (CardTransactionDetails)[] = [];\n        for (let i = 0; i <= months; i += 1) {\n          const month = nextBillingMonth.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page, TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);\n\n          if (!this.isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData);\n\n        debug('filer out old transactions');\n        const txns = (this.options.outputData?.enableTransactionsFilterByDate ?? true) ?\n          filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false) :\n          transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
|