@tomerh2001/israeli-bank-scrapers 6.3.11 → 6.4.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/lib/scrapers/hapoalim.js +314 -3
- package/lib/scrapers/visa-cal.js +2 -2
- package/lib/transactions.d.ts +12 -0
- package/lib/transactions.js +1 -1
- package/package.json +1 -1
package/lib/scrapers/hapoalim.js
CHANGED
|
@@ -6,6 +6,7 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
6
6
|
exports.default = void 0;
|
|
7
7
|
var _moment = _interopRequireDefault(require("moment"));
|
|
8
8
|
var _uuid = require("uuid");
|
|
9
|
+
var _constants = require("../constants");
|
|
9
10
|
var _debug = require("../helpers/debug");
|
|
10
11
|
var _fetch = require("../helpers/fetch");
|
|
11
12
|
var _navigation = require("../helpers/navigation");
|
|
@@ -52,7 +53,7 @@ function convertTransactions(txns) {
|
|
|
52
53
|
date: (0, _moment.default)(txn.eventDate, DATE_FORMAT).toISOString(),
|
|
53
54
|
processedDate: (0, _moment.default)(txn.valueDate, DATE_FORMAT).toISOString(),
|
|
54
55
|
originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,
|
|
55
|
-
originalCurrency:
|
|
56
|
+
originalCurrency: _constants.SHEKEL_CURRENCY,
|
|
56
57
|
chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,
|
|
57
58
|
description: txn.activityDescription || '',
|
|
58
59
|
status: txn.serialNumber === 0 ? _transactions.TransactionStatuses.Pending : _transactions.TransactionStatuses.Completed,
|
|
@@ -61,6 +62,31 @@ function convertTransactions(txns) {
|
|
|
61
62
|
return result;
|
|
62
63
|
});
|
|
63
64
|
}
|
|
65
|
+
function convertForexTransactions(txns, currency) {
|
|
66
|
+
return txns.map(txn => {
|
|
67
|
+
const isOutbound = txn.eventActivityTypeCode === 2;
|
|
68
|
+
const dateStr = txn.executingDate.toString(); // Date transaction was executed
|
|
69
|
+
const valueDateStr = txn.valueDate.toString(); // Date value was actually added to account
|
|
70
|
+
|
|
71
|
+
let memo = '';
|
|
72
|
+
if (txn.eventDetails) {
|
|
73
|
+
memo = txn.eventDetails;
|
|
74
|
+
}
|
|
75
|
+
const result = {
|
|
76
|
+
type: _transactions.TransactionTypes.Normal,
|
|
77
|
+
identifier: txn.referenceNumber || txn.recordSerialNumber,
|
|
78
|
+
date: (0, _moment.default)(valueDateStr, DATE_FORMAT).toISOString(),
|
|
79
|
+
processedDate: (0, _moment.default)(dateStr, DATE_FORMAT).toISOString(),
|
|
80
|
+
originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,
|
|
81
|
+
originalCurrency: currency,
|
|
82
|
+
chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,
|
|
83
|
+
description: txn.activityDescription || '',
|
|
84
|
+
status: _transactions.TransactionStatuses.Completed,
|
|
85
|
+
memo
|
|
86
|
+
};
|
|
87
|
+
return result;
|
|
88
|
+
});
|
|
89
|
+
}
|
|
64
90
|
async function getRestContext(page) {
|
|
65
91
|
await (0, _waiting.waitUntil)(() => {
|
|
66
92
|
return page.evaluate(() => !!window.bnhpApp);
|
|
@@ -121,6 +147,256 @@ async function getAccountBalance(apiSiteUrl, page, accountNumber) {
|
|
|
121
147
|
const balanceAndCreditLimit = await (0, _fetch.fetchGetWithinPage)(page, balanceAndCreditLimitUrl);
|
|
122
148
|
return balanceAndCreditLimit?.currentBalance;
|
|
123
149
|
}
|
|
150
|
+
async function getForexAccounts(baseUrl, page, accountNumber, startDate, endDate) {
|
|
151
|
+
debug('========== FETCHING FOREX ACCOUNTS ==========');
|
|
152
|
+
debug('Account: %s, Date range: %s to %s', accountNumber, startDate, endDate);
|
|
153
|
+
const accounts = [];
|
|
154
|
+
const currency = {
|
|
155
|
+
code: 19,
|
|
156
|
+
swift: _constants.SHEKEL_CURRENCY
|
|
157
|
+
};
|
|
158
|
+
try {
|
|
159
|
+
const detailedAccountTypeCode = 142; // For foreign currency accounts
|
|
160
|
+
const forexUrl = `${baseUrl}/ServerServices/foreign-currency/transactions?accountId=${accountNumber}&type=business&retrievalStartDate=${startDate}&retrievalEndDate=${endDate}¤cyCodeList=${currency.code}&detailedAccountTypeCodeList=${detailedAccountTypeCode}&view=details&lang=he`;
|
|
161
|
+
debug('Trying forex %s', forexUrl);
|
|
162
|
+
const forexData = await (0, _fetch.fetchGetWithinPage)(page, forexUrl, true); // ignoreErrors = true
|
|
163
|
+
|
|
164
|
+
if (forexData && forexData.balancesAndLimitsDataList && forexData.balancesAndLimitsDataList.length > 0) {
|
|
165
|
+
debug('✓ Found forex data');
|
|
166
|
+
for (const currencyData of forexData.balancesAndLimitsDataList) {
|
|
167
|
+
const currencySwiftCode = currencyData.currencySwiftCode || currency.swift;
|
|
168
|
+
const transactionCount = currencyData.transactions?.length || 0;
|
|
169
|
+
|
|
170
|
+
// Get balance from the most recent transaction's currentBalance field
|
|
171
|
+
// If no transactions, fall back to currencyData.currentBalance
|
|
172
|
+
let balance = currencyData.currentBalance;
|
|
173
|
+
if (transactionCount > 0 && currencyData.transactions) {
|
|
174
|
+
balance = currencyData.transactions[0].currentBalance;
|
|
175
|
+
debug(' - Using balance from most recent transaction: %s', balance);
|
|
176
|
+
}
|
|
177
|
+
debug(' - Currency: %s, Balance: %s, Transactions: %d', currencySwiftCode, balance, transactionCount);
|
|
178
|
+
|
|
179
|
+
// Log transaction dates for debugging
|
|
180
|
+
if (transactionCount > 0) {
|
|
181
|
+
const txnDates = currencyData.transactions?.map(t => t.executingDate).join(', ') || '';
|
|
182
|
+
debug(' Transaction dates: %s', txnDates);
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// Only add if there's actually a balance or transactions
|
|
186
|
+
if (balance !== 0 || transactionCount > 0) {
|
|
187
|
+
const txns = convertForexTransactions(currencyData.transactions || [], currencySwiftCode);
|
|
188
|
+
const forexAccountNumber = `${accountNumber}-${currencySwiftCode}`;
|
|
189
|
+
accounts.push({
|
|
190
|
+
accountNumber: forexAccountNumber,
|
|
191
|
+
balance,
|
|
192
|
+
currency: currencySwiftCode,
|
|
193
|
+
txns
|
|
194
|
+
});
|
|
195
|
+
debug(' ✓ Added forex account: %s with %d transactions', forexAccountNumber, txns.length);
|
|
196
|
+
} else {
|
|
197
|
+
debug(' - Skipping %s (zero balance and no transactions)', currencySwiftCode);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
} else {
|
|
201
|
+
debug(' - No forex data found');
|
|
202
|
+
}
|
|
203
|
+
} catch (error) {
|
|
204
|
+
debug(' - Error fetching forex: %s', error);
|
|
205
|
+
// Continue trying other currencies
|
|
206
|
+
}
|
|
207
|
+
debug('Returning %d forex accounts', accounts.length);
|
|
208
|
+
return accounts;
|
|
209
|
+
}
|
|
210
|
+
async function getSavingsAccounts(baseUrl, page, accountNumber) {
|
|
211
|
+
const savingsUrl = `${baseUrl}/ServerServices/deposits-and-savings/deposits?accountId=${accountNumber}&view=details&lang=he`;
|
|
212
|
+
const savingsData = await (0, _fetch.fetchGetWithinPage)(page, savingsUrl);
|
|
213
|
+
if (!savingsData || !savingsData.depositsWrapperData || savingsData.depositsWrapperData.length === 0) {
|
|
214
|
+
debug('No savings accounts found for account %s', accountNumber);
|
|
215
|
+
return [];
|
|
216
|
+
}
|
|
217
|
+
const accounts = [];
|
|
218
|
+
for (const wrapper of savingsData.depositsWrapperData) {
|
|
219
|
+
// Create a separate account for each individual deposit
|
|
220
|
+
for (const deposit of wrapper.data) {
|
|
221
|
+
const balance = deposit.revaluedTotalAmount || deposit.principalAmount;
|
|
222
|
+
const savingsAccountNumber = `${accountNumber}-${deposit.depositSerialId}`;
|
|
223
|
+
accounts.push({
|
|
224
|
+
accountNumber: savingsAccountNumber,
|
|
225
|
+
savingsAccount: true,
|
|
226
|
+
balance,
|
|
227
|
+
txns: [] // Savings accounts typically don't have transaction history in the same way
|
|
228
|
+
});
|
|
229
|
+
debug('Added savings account %s with balance %s', savingsAccountNumber, balance);
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
return accounts;
|
|
233
|
+
}
|
|
234
|
+
async function getInvestmentAccounts(baseUrl, page, account) {
|
|
235
|
+
debug('========== FETCHING INVESTMENT ACCOUNTS ==========');
|
|
236
|
+
const accounts = [];
|
|
237
|
+
const accountNumber = `${account.branchNumber}-${account.accountNumber}`; // Don't include bankNumber here because investment API doesn't use it
|
|
238
|
+
|
|
239
|
+
// Set up request interception to capture session headers
|
|
240
|
+
let capturedSession = null;
|
|
241
|
+
let capturedCsession = null;
|
|
242
|
+
const requestHandler = request => {
|
|
243
|
+
try {
|
|
244
|
+
const headers = request.headers();
|
|
245
|
+
if (headers.session && !capturedSession) {
|
|
246
|
+
capturedSession = headers.session;
|
|
247
|
+
debug(' - Captured session from network request');
|
|
248
|
+
}
|
|
249
|
+
if (headers.csession && !capturedCsession) {
|
|
250
|
+
capturedCsession = headers.csession;
|
|
251
|
+
debug(' - Captured csession from network request');
|
|
252
|
+
}
|
|
253
|
+
request.continue();
|
|
254
|
+
} catch (e) {
|
|
255
|
+
debug(' - Error in request handler: %s', e);
|
|
256
|
+
// Try to continue anyway
|
|
257
|
+
try {
|
|
258
|
+
request.continue();
|
|
259
|
+
} catch (continueError) {
|
|
260
|
+
// Ignore if already handled
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
};
|
|
264
|
+
try {
|
|
265
|
+
// Navigate to mytrade section to establish session
|
|
266
|
+
const mytradeUrl = `${baseUrl}/mytrade/app`;
|
|
267
|
+
debug('Navigating to mytrade section: %s', mytradeUrl);
|
|
268
|
+
await page.setRequestInterception(true);
|
|
269
|
+
page.on('request', requestHandler);
|
|
270
|
+
|
|
271
|
+
// Try to navigate, but if it fails (e.g., MyTrade not enabled), clean up and return empty
|
|
272
|
+
try {
|
|
273
|
+
await page.goto(mytradeUrl, {
|
|
274
|
+
waitUntil: 'networkidle2',
|
|
275
|
+
timeout: 60000
|
|
276
|
+
});
|
|
277
|
+
// Wait longer for the page to fully load and establish session
|
|
278
|
+
await new Promise(resolve => setTimeout(resolve, 5000));
|
|
279
|
+
} catch (navError) {
|
|
280
|
+
debug(' - Navigation to MyTrade failed (likely not enabled): %s', navError);
|
|
281
|
+
// Clean up request interception
|
|
282
|
+
page.off('request', requestHandler);
|
|
283
|
+
await page.setRequestInterception(false);
|
|
284
|
+
debug('Returning 0 investment accounts (MyTrade not accessible)');
|
|
285
|
+
return accounts;
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
// Now turn off request interception before we make API calls
|
|
289
|
+
page.off('request', requestHandler);
|
|
290
|
+
await page.setRequestInterception(false);
|
|
291
|
+
debug('Mytrade page loaded, session established');
|
|
292
|
+
const fields = 'EngName,EngSymbol,HebName,HebSymbol,Symbol,ExpirationDate,ItemType,StockType,IsEtf,IsForeign,CurrencyCode,Exchange,CreationEquityNum,EquityType,ContractType,AllowedOrderDirection,EquitySubType';
|
|
293
|
+
const investmentUrl = `${baseUrl}/ServerServices/mytrade/api/v2/json2/account/view?account=${accountNumber}&fields=${fields}`;
|
|
294
|
+
debug('Trying investment account URL: %s', investmentUrl);
|
|
295
|
+
|
|
296
|
+
// Get XSRF token and session data from cookies
|
|
297
|
+
const cookies = await page.cookies();
|
|
298
|
+
const XSRFCookie = cookies.find(cookie => cookie.name === 'XSRF-TOKEN');
|
|
299
|
+
|
|
300
|
+
// Use captured session or fallback to generated ones
|
|
301
|
+
const sessionId = capturedSession || (0, _uuid.v4)();
|
|
302
|
+
const csession = capturedCsession || Math.random().toString();
|
|
303
|
+
const headers = {
|
|
304
|
+
'Content-Type': 'application/json; charset=utf-8',
|
|
305
|
+
csession,
|
|
306
|
+
session: sessionId,
|
|
307
|
+
Referer: mytradeUrl
|
|
308
|
+
};
|
|
309
|
+
if (XSRFCookie != null) {
|
|
310
|
+
headers['X-XSRF-TOKEN'] = XSRFCookie.value;
|
|
311
|
+
debug(' - Using XSRF token: %s', XSRFCookie.value.substring(0, 10) + '...');
|
|
312
|
+
}
|
|
313
|
+
debug(' - Request headers: csession=%s, session=%s', csession, sessionId);
|
|
314
|
+
const investmentData = await (0, _fetch.fetchPostWithinPage)(page, investmentUrl, {}, headers);
|
|
315
|
+
debug(' - Response received: %s', investmentData ? 'YES' : 'NO');
|
|
316
|
+
if (investmentData) {
|
|
317
|
+
debug(' - Response has View: %s', investmentData.View ? 'YES' : 'NO');
|
|
318
|
+
debug(' - Response has View.Account: %s', investmentData.View?.Account ? 'YES' : 'NO');
|
|
319
|
+
}
|
|
320
|
+
if (investmentData?.View?.Account) {
|
|
321
|
+
debug('✓ Found investment account data');
|
|
322
|
+
const accountData = investmentData.View.Account;
|
|
323
|
+
const balance = accountData.OnlineValue || 0;
|
|
324
|
+
const currency = accountData.CurrencyCode || _constants.SHEKEL_CURRENCY;
|
|
325
|
+
|
|
326
|
+
// Get securities from the balance array
|
|
327
|
+
const securities = [];
|
|
328
|
+
const balances = accountData.AccountPosition?.Balance || [];
|
|
329
|
+
const metaSecurities = investmentData.View.Meta?.Security || [];
|
|
330
|
+
|
|
331
|
+
// Create a map of security metadata for easy lookup
|
|
332
|
+
const metaMap = new Map();
|
|
333
|
+
for (const meta of metaSecurities) {
|
|
334
|
+
metaMap.set(meta['-Key'], meta);
|
|
335
|
+
}
|
|
336
|
+
for (const securityBalance of balances) {
|
|
337
|
+
const meta = metaMap.get(securityBalance.EquityNumber);
|
|
338
|
+
securities.push({
|
|
339
|
+
name: meta?.EngName || '',
|
|
340
|
+
symbol: meta?.EngSymbol || '',
|
|
341
|
+
volume: securityBalance.OnlineNV,
|
|
342
|
+
value: securityBalance.OnlineVL,
|
|
343
|
+
currency: securityBalance.CurrencyCode || meta?.CurrencyCode || _constants.SHEKEL_CURRENCY,
|
|
344
|
+
changePercentage: securityBalance.BaseRateChangePercentage,
|
|
345
|
+
profitLoss: securityBalance.ProfitLoss
|
|
346
|
+
});
|
|
347
|
+
}
|
|
348
|
+
debug(' - Balance: %s %s, Securities: %d', balance, currency, securities.length);
|
|
349
|
+
if (balance !== 0 || securities.length > 0) {
|
|
350
|
+
const investmentAccountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}-investment`;
|
|
351
|
+
accounts.push({
|
|
352
|
+
accountNumber: investmentAccountNumber,
|
|
353
|
+
balance,
|
|
354
|
+
currency,
|
|
355
|
+
savingsAccount: true,
|
|
356
|
+
txns: [],
|
|
357
|
+
securities
|
|
358
|
+
});
|
|
359
|
+
debug(' ✓ Added investment account: %s with %d securities', investmentAccountNumber, securities.length);
|
|
360
|
+
} else {
|
|
361
|
+
debug(' - Skipping (zero balance and no securities)');
|
|
362
|
+
}
|
|
363
|
+
} else {
|
|
364
|
+
debug(' - No investment account data found');
|
|
365
|
+
}
|
|
366
|
+
} catch (error) {
|
|
367
|
+
debug(' - Error fetching investment account: %s', error);
|
|
368
|
+
// Log more details about the error
|
|
369
|
+
if (error instanceof Error) {
|
|
370
|
+
debug(' Error message: %s', error.message);
|
|
371
|
+
debug(' Error stack: %s', error.stack);
|
|
372
|
+
}
|
|
373
|
+
} finally {
|
|
374
|
+
// Clean up request interception (in case it wasn't already done)
|
|
375
|
+
try {
|
|
376
|
+
page.off('request', requestHandler);
|
|
377
|
+
await page.setRequestInterception(false);
|
|
378
|
+
} catch (e) {
|
|
379
|
+
// Ignore cleanup errors
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
// Navigate back to the main homepage to restore the session for subsequent scraping
|
|
383
|
+
try {
|
|
384
|
+
const homepageUrl = `${baseUrl}/ng-portals/rb/he/homepage`;
|
|
385
|
+
debug('Navigating back to homepage: %s', homepageUrl);
|
|
386
|
+
await page.goto(homepageUrl, {
|
|
387
|
+
waitUntil: 'networkidle2',
|
|
388
|
+
timeout: 30000
|
|
389
|
+
});
|
|
390
|
+
await new Promise(resolve => setTimeout(resolve, 2000));
|
|
391
|
+
debug('Homepage restored');
|
|
392
|
+
} catch (navError) {
|
|
393
|
+
debug(' - Failed to navigate back to homepage: %s', navError);
|
|
394
|
+
// Continue anyway, the outer try-catch in fetchAccountData will handle it
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
debug('Returning %d investment accounts', accounts.length);
|
|
398
|
+
return accounts;
|
|
399
|
+
}
|
|
124
400
|
async function fetchAccountData(page, baseUrl, options) {
|
|
125
401
|
const restContext = await getRestContext(page);
|
|
126
402
|
const apiSiteUrl = `${baseUrl}/${restContext}`;
|
|
@@ -146,12 +422,47 @@ async function fetchAccountData(page, baseUrl, options) {
|
|
|
146
422
|
} else {
|
|
147
423
|
debug('Skipping balance for a closed account, balance will be undefined');
|
|
148
424
|
}
|
|
149
|
-
|
|
425
|
+
let txns = [];
|
|
426
|
+
try {
|
|
427
|
+
txns = await getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDateStr, endDateStr, additionalTransactionInformation);
|
|
428
|
+
} catch (error) {
|
|
429
|
+
debug('Error fetching transactions for %s (possibly closed account): %s', accountNumber, error);
|
|
430
|
+
// Continue with empty transactions
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
// Add regular checking account
|
|
150
434
|
accounts.push({
|
|
151
435
|
accountNumber,
|
|
152
436
|
balance,
|
|
153
437
|
txns
|
|
154
438
|
});
|
|
439
|
+
|
|
440
|
+
// Fetch forex accounts for this account number
|
|
441
|
+
try {
|
|
442
|
+
const forexAccounts = await getForexAccounts(baseUrl, page, accountNumber, startDateStr, endDateStr);
|
|
443
|
+
accounts.push(...forexAccounts);
|
|
444
|
+
debug('Added %d forex accounts to results', forexAccounts.length);
|
|
445
|
+
} catch (error) {
|
|
446
|
+
debug('Error fetching forex accounts for %s: %s', accountNumber, error);
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
// Fetch savings accounts for this account number
|
|
450
|
+
try {
|
|
451
|
+
const savingsAccounts = await getSavingsAccounts(baseUrl, page, accountNumber);
|
|
452
|
+
accounts.push(...savingsAccounts);
|
|
453
|
+
debug('Added %d savings accounts to results', savingsAccounts.length);
|
|
454
|
+
} catch (error) {
|
|
455
|
+
debug('Error fetching savings accounts for %s: %s', accountNumber, error);
|
|
456
|
+
}
|
|
457
|
+
|
|
458
|
+
// Fetch investment accounts for this account number
|
|
459
|
+
try {
|
|
460
|
+
const investmentAccounts = await getInvestmentAccounts(baseUrl, page, account);
|
|
461
|
+
accounts.push(...investmentAccounts);
|
|
462
|
+
debug('Added %d investment accounts to results', investmentAccounts.length);
|
|
463
|
+
} catch (error) {
|
|
464
|
+
debug('Error fetching investment accounts for %s: %s', accountNumber, error);
|
|
465
|
+
}
|
|
155
466
|
}
|
|
156
467
|
const accountData = {
|
|
157
468
|
success: true,
|
|
@@ -195,4 +506,4 @@ class HapoalimScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
195
506
|
}
|
|
196
507
|
}
|
|
197
508
|
var _default = exports.default = HapoalimScraper;
|
|
198
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_uuid","_debug","_fetch","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","convertTransactions","txns","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","moment","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","waitUntil","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","uuid4","fetchPostWithinPage","getExtraScrap","txnsResult","baseUrl","accountNumber","promises","transactions","transaction","pfmDetails","extraTransactionDetails","fetchGetWithinPage","transactionNumber","res","Promise","all","getAccountTransactions","apiSiteUrl","startDate","endDate","additionalTransactionInformation","txnsUrl","finalResult","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","options","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","toDate","startMoment","max","startDateStr","format","endDateStr","accounts","account","balance","bankNumber","branchNumber","isActiveAccount","accountClosingReasonCode","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","waitForRedirect","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/hapoalim.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport { v4 as uuid4 } from 'uuid';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('hapoalim');\n\nconst DATE_FORMAT = 'YYYYMMDD';\n\n// eslint-disable-next-line @typescript-eslint/no-namespace\ndeclare namespace window {\n  const bnhpApp: any;\n}\n\ninterface ScrapedTransaction {\n  serialNumber?: number;\n  activityDescription?: string;\n  eventAmount: number;\n  valueDate?: string;\n  eventDate?: string;\n  referenceNumber?: number;\n  ScrapedTransaction?: string;\n  eventActivityTypeCode: number;\n  currentBalance: number;\n  pfmDetails: string;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ninterface ScrapedPfmTransaction {\n  transactionNumber: number;\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\n  accountClosingReasonCode: number;\n}[];\n\ntype FetchedAccountTransactionsData = {\n  transactions: ScrapedTransaction[];\n};\n\ntype BalanceAndCreditLimit = {\n  creditLimitAmount: number;\n  creditLimitDescription: string;\n  creditLimitUtilizationAmount: number;\n  creditLimitUtilizationExistanceCode: number;\n  creditLimitUtilizationPercent: number;\n  currentAccountLimitsAmount: number;\n  currentBalance: number;\n  withdrawalBalance: number;\n};\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map(txn => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const { partyHeadline, partyName, messageHeadline, messageDetail } = txn.beneficiaryDetailsData;\n      const memoLines: string[] = [];\n      if (partyHeadline) {\n        memoLines.push(partyHeadline);\n      }\n\n      if (partyName) {\n        memoLines.push(`${partyName}.`);\n      }\n\n      if (messageHeadline) {\n        memoLines.push(messageHeadline);\n      }\n\n      if (messageDetail) {\n        memoLines.push(`${messageDetail}.`);\n      }\n\n      if (memoLines.length) {\n        memo = memoLines.join(' ');\n      }\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber,\n      date: moment(txn.eventDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.valueDate, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: txn.serialNumber === 0 ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      memo,\n    };\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(() => {\n    return page.evaluate(() => !!window.bnhpApp);\n  }, 'waiting for app data load');\n\n  const result = await page.evaluate(() => {\n    return window.bnhpApp.restContext;\n  });\n\n  return result.slice(1);\n}\n\nasync function fetchPoalimXSRFWithinPage(\n  page: Page,\n  url: string,\n  pageUuid: string,\n): Promise<FetchedAccountTransactionsData | null> {\n  const cookies = await page.cookies();\n  const XSRFCookie = cookies.find(cookie => cookie.name === 'XSRF-TOKEN');\n  const headers: Record<string, any> = {};\n  if (XSRFCookie != null) {\n    headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n  }\n  headers.pageUuid = pageUuid;\n  headers.uuid = uuid4();\n  headers['Content-Type'] = 'application/json;charset=UTF-8';\n  return fetchPostWithinPage<FetchedAccountTransactionsData>(page, url, [], headers);\n}\n\nasync function getExtraScrap(\n  txnsResult: FetchedAccountTransactionsData,\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n): Promise<FetchedAccountTransactionsData> {\n  const promises = txnsResult.transactions.map(async (transaction: ScrapedTransaction): Promise<ScrapedTransaction> => {\n    const { pfmDetails, serialNumber } = transaction;\n    if (serialNumber !== 0) {\n      const url = `${baseUrl}${pfmDetails}&accountId=${accountNumber}&lang=he`;\n      const extraTransactionDetails = (await fetchGetWithinPage<ScrapedPfmTransaction[]>(page, url)) || [];\n      if (extraTransactionDetails && extraTransactionDetails.length) {\n        const { transactionNumber } = extraTransactionDetails[0];\n        if (transactionNumber) {\n          return { ...transaction, referenceNumber: transactionNumber };\n        }\n      }\n    }\n    return transaction;\n  });\n  const res = await Promise.all(promises);\n  return { transactions: res };\n}\n\nasync function getAccountTransactions(\n  baseUrl: string,\n  apiSiteUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n  additionalTransactionInformation = false,\n) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  const finalResult =\n    additionalTransactionInformation && txnsResult?.transactions.length\n      ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber)\n      : txnsResult;\n\n  return convertTransactions(finalResult?.transactions ?? []);\n}\n\nasync function getAccountBalance(apiSiteUrl: string, page: Page, accountNumber: string) {\n  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n  const balanceAndCreditLimit = await fetchGetWithinPage<BalanceAndCreditLimit>(page, balanceAndCreditLimitUrl);\n\n  return balanceAndCreditLimit?.currentBalance;\n}\n\nasync function fetchAccountData(page: Page, baseUrl: string, options: ScraperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n\n  debug('fetching accounts data');\n  const accountsInfo = (await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl)) || [];\n  debug('got %d accounts, fetching txns and balance', accountsInfo.length);\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const { additionalTransactionInformation } = options;\n\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of accountsInfo) {\n    let balance: number | undefined;\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const isActiveAccount = account.accountClosingReasonCode === 0;\n    if (isActiveAccount) {\n      balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    } else {\n      debug('Skipping balance for a closed account, balance will be undefined');\n    }\n\n    const txns = await getAccountTransactions(\n      baseUrl,\n      apiSiteUrl,\n      page,\n      accountNumber,\n      startDateStr,\n      endDateStr,\n      additionalTransactionInformation,\n    );\n\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n  debug('fetching ended');\n  return accountData;\n}\n\nfunction getPossibleLoginResults(baseUrl: string) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${baseUrl}/portalserver/HomePage`,\n    `${baseUrl}/ng-portals-bt/rb/he/homepage`,\n    `${baseUrl}/ng-portals/rb/he/homepage`,\n  ];\n  urls[LoginResults.InvalidPassword] = [\n    `${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`,\n  ];\n  urls[LoginResults.ChangePassword] = [\n    `${baseUrl}/MCP/START?flow=MCP&state=START&expiredDate=null`,\n    /\\/ABOUTTOEXPIRE\\/START/i,\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { userCode: string; password: string };\n\nclass HapoalimScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.baseUrl}/cgi-bin/poalwwwc?reqName=getLogonPage`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.login-btn',\n      postAction: async () => waitForRedirect(this.page),\n      possibleResults: getPossibleLoginResults(this.baseUrl),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.baseUrl, this.options);\n  }\n}\n\nexport default HapoalimScraper;\n"],"mappings":"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","ignoreList":[]}
|
|
509
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_uuid","_constants","_debug","_fetch","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","convertTransactions","txns","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","moment","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","convertForexTransactions","currency","dateStr","executingDate","toString","valueDateStr","eventDetails","recordSerialNumber","getRestContext","page","waitUntil","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","uuid4","fetchPostWithinPage","getExtraScrap","txnsResult","baseUrl","accountNumber","promises","transactions","transaction","pfmDetails","extraTransactionDetails","fetchGetWithinPage","transactionNumber","res","Promise","all","getAccountTransactions","apiSiteUrl","startDate","endDate","additionalTransactionInformation","txnsUrl","finalResult","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","getForexAccounts","accounts","code","swift","detailedAccountTypeCode","forexUrl","forexData","balancesAndLimitsDataList","currencyData","currencySwiftCode","transactionCount","balance","txnDates","t","forexAccountNumber","error","getSavingsAccounts","savingsUrl","savingsData","depositsWrapperData","wrapper","deposit","data","revaluedTotalAmount","principalAmount","savingsAccountNumber","depositSerialId","savingsAccount","getInvestmentAccounts","account","branchNumber","capturedSession","capturedCsession","requestHandler","request","session","csession","continue","continueError","mytradeUrl","setRequestInterception","on","goto","timeout","resolve","setTimeout","navError","off","fields","investmentUrl","sessionId","Math","random","Referer","substring","investmentData","View","Account","accountData","OnlineValue","CurrencyCode","securities","balances","AccountPosition","Balance","metaSecurities","Meta","Security","metaMap","Map","meta","set","securityBalance","get","EquityNumber","EngName","symbol","EngSymbol","volume","OnlineNV","OnlineVL","changePercentage","BaseRateChangePercentage","profitLoss","ProfitLoss","investmentAccountNumber","bankNumber","Error","message","stack","homepageUrl","fetchAccountData","options","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","toDate","startMoment","max","startDateStr","format","endDateStr","isActiveAccount","accountClosingReasonCode","forexAccounts","savingsAccounts","investmentAccounts","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","submitButtonSelector","postAction","waitForRedirect","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/hapoalim.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer';\nimport { v4 as uuid4 } from 'uuid';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport {\n  type Transaction,\n  TransactionStatuses,\n  TransactionTypes,\n  type TransactionsAccount,\n  type Security,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('hapoalim');\n\nconst DATE_FORMAT = 'YYYYMMDD';\n\n// eslint-disable-next-line @typescript-eslint/no-namespace\ndeclare namespace window {\n  const bnhpApp: any;\n}\n\ninterface ScrapedTransaction {\n  serialNumber?: number;\n  activityDescription?: string;\n  eventAmount: number;\n  valueDate?: string;\n  eventDate?: string;\n  referenceNumber?: number;\n  ScrapedTransaction?: string;\n  eventActivityTypeCode: number;\n  currentBalance: number;\n  pfmDetails: string;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ninterface ScrapedPfmTransaction {\n  transactionNumber: number;\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\n  accountClosingReasonCode: number;\n}[];\n\ntype FetchedAccountTransactionsData = {\n  transactions: ScrapedTransaction[];\n};\n\ntype BalanceAndCreditLimit = {\n  creditLimitAmount: number;\n  creditLimitDescription: string;\n  creditLimitUtilizationAmount: number;\n  creditLimitUtilizationExistanceCode: number;\n  creditLimitUtilizationPercent: number;\n  currentAccountLimitsAmount: number;\n  currentBalance: number;\n  withdrawalBalance: number;\n};\n\ninterface ForexTransaction {\n  executingDate: number;\n  valueDate: number;\n  activityDescription: string;\n  eventAmount: number;\n  currentBalance: number;\n  referenceNumber?: number;\n  eventDetails?: string;\n  eventActivityTypeCode: number;\n  currencySwiftCode: string;\n  recordSerialNumber?: number;\n}\n\ninterface ForexCurrencyData {\n  currencyCode: number;\n  currencySwiftCode: string;\n  currencySwiftDescription: string;\n  currentBalance: number;\n  transactions: ForexTransaction[];\n  detailedAccountTypeCode: number;\n}\n\ninterface ForexAccountData {\n  balancesAndLimitsDataList: ForexCurrencyData[];\n}\n\ninterface SavingsDeposit {\n  principalAmount: number;\n  revaluedTotalAmount: number;\n  depositSerialId: number;\n  productFreeText?: string;\n  shortProductName?: string;\n  formattedAgreementOpeningDate?: string;\n  formattedEndExitDate?: string;\n  nominalInterest?: number;\n  detailedAccountTypeCode: number;\n}\n\ninterface SavingsWrapper {\n  data: SavingsDeposit[];\n  amount: number;\n  revaluatedAmount: number;\n}\n\ninterface SavingsAccountData {\n  depositsWrapperData: SavingsWrapper[];\n}\n\ninterface InvestmentSecurityBalance {\n  EquityNumber: string;\n  BaseRate?: number;\n  LastRate?: number;\n  BaseRateChangePercentage?: number;\n  OnlineNV: number;\n  OnlineVL: number;\n  OnlineNisVL?: number;\n  ProfitLoss?: number;\n  CurrencyCode?: string;\n}\n\ninterface InvestmentSecurityMeta {\n  '-Key': string;\n  EngName?: string;\n  EngSymbol?: string;\n  HebName?: string;\n  HebSymbol?: string;\n  Symbol?: string;\n  ItemType?: string;\n  StockType?: string;\n  IsForeign?: boolean;\n  CurrencyCode?: string;\n  Exchange?: string;\n  EquityType?: number;\n  EquitySubType?: number;\n}\n\ninterface InvestmentAccountData {\n  View?: {\n    Account?: {\n      OnlineValue?: number;\n      OnlineCash?: number;\n      CurrencyCode?: string;\n      AccountPosition?: {\n        Balance?: InvestmentSecurityBalance[];\n      };\n    };\n    Meta?: {\n      Security?: InvestmentSecurityMeta[];\n    };\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map(txn => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const { partyHeadline, partyName, messageHeadline, messageDetail } = txn.beneficiaryDetailsData;\n      const memoLines: string[] = [];\n      if (partyHeadline) {\n        memoLines.push(partyHeadline);\n      }\n\n      if (partyName) {\n        memoLines.push(`${partyName}.`);\n      }\n\n      if (messageHeadline) {\n        memoLines.push(messageHeadline);\n      }\n\n      if (messageDetail) {\n        memoLines.push(`${messageDetail}.`);\n      }\n\n      if (memoLines.length) {\n        memo = memoLines.join(' ');\n      }\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber,\n      date: moment(txn.eventDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.valueDate, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: txn.serialNumber === 0 ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      memo,\n    };\n\n    return result;\n  });\n}\n\nfunction convertForexTransactions(txns: ForexTransaction[], currency: string): Transaction[] {\n  return txns.map(txn => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n    const dateStr = txn.executingDate.toString(); // Date transaction was executed\n    const valueDateStr = txn.valueDate.toString(); // Date value was actually added to account\n\n    let memo = '';\n    if (txn.eventDetails) {\n      memo = txn.eventDetails;\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber || txn.recordSerialNumber,\n      date: moment(valueDateStr, DATE_FORMAT).toISOString(),\n      processedDate: moment(dateStr, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: currency,\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: TransactionStatuses.Completed,\n      memo,\n    };\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(() => {\n    return page.evaluate(() => !!window.bnhpApp);\n  }, 'waiting for app data load');\n\n  const result = await page.evaluate(() => {\n    return window.bnhpApp.restContext;\n  });\n\n  return result.slice(1);\n}\n\nasync function fetchPoalimXSRFWithinPage(\n  page: Page,\n  url: string,\n  pageUuid: string,\n): Promise<FetchedAccountTransactionsData | null> {\n  const cookies = await page.cookies();\n  const XSRFCookie = cookies.find(cookie => cookie.name === 'XSRF-TOKEN');\n  const headers: Record<string, any> = {};\n  if (XSRFCookie != null) {\n    headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n  }\n  headers.pageUuid = pageUuid;\n  headers.uuid = uuid4();\n  headers['Content-Type'] = 'application/json;charset=UTF-8';\n  return fetchPostWithinPage<FetchedAccountTransactionsData>(page, url, [], headers);\n}\n\nasync function getExtraScrap(\n  txnsResult: FetchedAccountTransactionsData,\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n): Promise<FetchedAccountTransactionsData> {\n  const promises = txnsResult.transactions.map(async (transaction: ScrapedTransaction): Promise<ScrapedTransaction> => {\n    const { pfmDetails, serialNumber } = transaction;\n    if (serialNumber !== 0) {\n      const url = `${baseUrl}${pfmDetails}&accountId=${accountNumber}&lang=he`;\n      const extraTransactionDetails = (await fetchGetWithinPage<ScrapedPfmTransaction[]>(page, url)) || [];\n      if (extraTransactionDetails && extraTransactionDetails.length) {\n        const { transactionNumber } = extraTransactionDetails[0];\n        if (transactionNumber) {\n          return { ...transaction, referenceNumber: transactionNumber };\n        }\n      }\n    }\n    return transaction;\n  });\n  const res = await Promise.all(promises);\n  return { transactions: res };\n}\n\nasync function getAccountTransactions(\n  baseUrl: string,\n  apiSiteUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n  additionalTransactionInformation = false,\n) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  const finalResult =\n    additionalTransactionInformation && txnsResult?.transactions.length\n      ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber)\n      : txnsResult;\n\n  return convertTransactions(finalResult?.transactions ?? []);\n}\n\nasync function getAccountBalance(apiSiteUrl: string, page: Page, accountNumber: string) {\n  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n  const balanceAndCreditLimit = await fetchGetWithinPage<BalanceAndCreditLimit>(page, balanceAndCreditLimitUrl);\n\n  return balanceAndCreditLimit?.currentBalance;\n}\n\nasync function getForexAccounts(\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n): Promise<TransactionsAccount[]> {\n  debug('========== FETCHING FOREX ACCOUNTS ==========');\n  debug('Account: %s, Date range: %s to %s', accountNumber, startDate, endDate);\n\n  const accounts: TransactionsAccount[] = [];\n\n  const currency = { code: 19, swift: SHEKEL_CURRENCY };\n  try {\n    const detailedAccountTypeCode = 142; // For foreign currency accounts\n    const forexUrl = `${baseUrl}/ServerServices/foreign-currency/transactions?accountId=${accountNumber}&type=business&retrievalStartDate=${startDate}&retrievalEndDate=${endDate}&currencyCodeList=${currency.code}&detailedAccountTypeCodeList=${detailedAccountTypeCode}&view=details&lang=he`;\n    debug('Trying forex %s', forexUrl);\n\n    const forexData = await fetchGetWithinPage<ForexAccountData>(page, forexUrl, true); // ignoreErrors = true\n\n    if (forexData && forexData.balancesAndLimitsDataList && forexData.balancesAndLimitsDataList.length > 0) {\n      debug('✓ Found forex data');\n\n      for (const currencyData of forexData.balancesAndLimitsDataList) {\n        const currencySwiftCode = currencyData.currencySwiftCode || currency.swift;\n        const transactionCount = currencyData.transactions?.length || 0;\n\n        // Get balance from the most recent transaction's currentBalance field\n        // If no transactions, fall back to currencyData.currentBalance\n        let balance = currencyData.currentBalance;\n        if (transactionCount > 0 && currencyData.transactions) {\n          balance = currencyData.transactions[0].currentBalance;\n          debug('  - Using balance from most recent transaction: %s', balance);\n        }\n\n        debug('  - Currency: %s, Balance: %s, Transactions: %d', currencySwiftCode, balance, transactionCount);\n\n        // Log transaction dates for debugging\n        if (transactionCount > 0) {\n          const txnDates = currencyData.transactions?.map(t => t.executingDate).join(', ') || '';\n          debug('    Transaction dates: %s', txnDates);\n        }\n\n        // Only add if there's actually a balance or transactions\n        if (balance !== 0 || transactionCount > 0) {\n          const txns = convertForexTransactions(currencyData.transactions || [], currencySwiftCode);\n          const forexAccountNumber = `${accountNumber}-${currencySwiftCode}`;\n\n          accounts.push({\n            accountNumber: forexAccountNumber,\n            balance,\n            currency: currencySwiftCode,\n            txns,\n          });\n\n          debug('  ✓ Added forex account: %s with %d transactions', forexAccountNumber, txns.length);\n        } else {\n          debug('  - Skipping %s (zero balance and no transactions)', currencySwiftCode);\n        }\n      }\n    } else {\n      debug('  - No forex data found');\n    }\n  } catch (error) {\n    debug('  - Error fetching forex: %s', error);\n    // Continue trying other currencies\n  }\n\n  debug('Returning %d forex accounts', accounts.length);\n  return accounts;\n}\n\nasync function getSavingsAccounts(baseUrl: string, page: Page, accountNumber: string): Promise<TransactionsAccount[]> {\n  const savingsUrl = `${baseUrl}/ServerServices/deposits-and-savings/deposits?accountId=${accountNumber}&view=details&lang=he`;\n  const savingsData = await fetchGetWithinPage<SavingsAccountData>(page, savingsUrl);\n\n  if (!savingsData || !savingsData.depositsWrapperData || savingsData.depositsWrapperData.length === 0) {\n    debug('No savings accounts found for account %s', accountNumber);\n    return [];\n  }\n\n  const accounts: TransactionsAccount[] = [];\n\n  for (const wrapper of savingsData.depositsWrapperData) {\n    // Create a separate account for each individual deposit\n    for (const deposit of wrapper.data) {\n      const balance = deposit.revaluedTotalAmount || deposit.principalAmount;\n      const savingsAccountNumber = `${accountNumber}-${deposit.depositSerialId}`;\n\n      accounts.push({\n        accountNumber: savingsAccountNumber,\n        savingsAccount: true,\n        balance,\n        txns: [], // Savings accounts typically don't have transaction history in the same way\n      });\n\n      debug('Added savings account %s with balance %s', savingsAccountNumber, balance);\n    }\n  }\n\n  return accounts;\n}\n\nasync function getInvestmentAccounts(\n  baseUrl: string,\n  page: Page,\n  account: { bankNumber: string; branchNumber: string; accountNumber: string },\n): Promise<TransactionsAccount[]> {\n  debug('========== FETCHING INVESTMENT ACCOUNTS ==========');\n  const accounts: TransactionsAccount[] = [];\n  const accountNumber = `${account.branchNumber}-${account.accountNumber}`; // Don't include bankNumber here because investment API doesn't use it\n\n  // Set up request interception to capture session headers\n  let capturedSession: string | null = null;\n  let capturedCsession: string | null = null;\n\n  const requestHandler = (request: any) => {\n    try {\n      const headers = request.headers();\n      if (headers.session && !capturedSession) {\n        capturedSession = headers.session;\n        debug('  - Captured session from network request');\n      }\n      if (headers.csession && !capturedCsession) {\n        capturedCsession = headers.csession;\n        debug('  - Captured csession from network request');\n      }\n      request.continue();\n    } catch (e) {\n      debug('  - Error in request handler: %s', e);\n      // Try to continue anyway\n      try {\n        request.continue();\n      } catch (continueError) {\n        // Ignore if already handled\n      }\n    }\n  };\n\n  try {\n    // Navigate to mytrade section to establish session\n    const mytradeUrl = `${baseUrl}/mytrade/app`;\n    debug('Navigating to mytrade section: %s', mytradeUrl);\n\n    await page.setRequestInterception(true);\n    page.on('request', requestHandler);\n\n    // Try to navigate, but if it fails (e.g., MyTrade not enabled), clean up and return empty\n    try {\n      await page.goto(mytradeUrl, { waitUntil: 'networkidle2', timeout: 60000 });\n      // Wait longer for the page to fully load and establish session\n      await new Promise(resolve => setTimeout(resolve, 5000));\n    } catch (navError) {\n      debug('  - Navigation to MyTrade failed (likely not enabled): %s', navError);\n      // Clean up request interception\n      page.off('request', requestHandler);\n      await page.setRequestInterception(false);\n      debug('Returning 0 investment accounts (MyTrade not accessible)');\n      return accounts;\n    }\n\n    // Now turn off request interception before we make API calls\n    page.off('request', requestHandler);\n    await page.setRequestInterception(false);\n\n    debug('Mytrade page loaded, session established');\n\n    const fields =\n      'EngName,EngSymbol,HebName,HebSymbol,Symbol,ExpirationDate,ItemType,StockType,IsEtf,IsForeign,CurrencyCode,Exchange,CreationEquityNum,EquityType,ContractType,AllowedOrderDirection,EquitySubType';\n    const investmentUrl = `${baseUrl}/ServerServices/mytrade/api/v2/json2/account/view?account=${accountNumber}&fields=${fields}`;\n    debug('Trying investment account URL: %s', investmentUrl);\n\n    // Get XSRF token and session data from cookies\n    const cookies = await page.cookies();\n    const XSRFCookie = cookies.find(cookie => cookie.name === 'XSRF-TOKEN');\n\n    // Use captured session or fallback to generated ones\n    const sessionId = capturedSession || uuid4();\n    const csession = capturedCsession || Math.random().toString();\n\n    const headers: Record<string, any> = {\n      'Content-Type': 'application/json; charset=utf-8',\n      csession,\n      session: sessionId,\n      Referer: mytradeUrl,\n    };\n    if (XSRFCookie != null) {\n      headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n      debug('  - Using XSRF token: %s', XSRFCookie.value.substring(0, 10) + '...');\n    }\n\n    debug('  - Request headers: csession=%s, session=%s', csession, sessionId);\n\n    const investmentData = await fetchPostWithinPage<InvestmentAccountData>(page, investmentUrl, {}, headers);\n\n    debug('  - Response received: %s', investmentData ? 'YES' : 'NO');\n    if (investmentData) {\n      debug('  - Response has View: %s', investmentData.View ? 'YES' : 'NO');\n      debug('  - Response has View.Account: %s', investmentData.View?.Account ? 'YES' : 'NO');\n    }\n\n    if (investmentData?.View?.Account) {\n      debug('✓ Found investment account data');\n\n      const accountData = investmentData.View.Account;\n      const balance = accountData.OnlineValue || 0;\n      const currency = accountData.CurrencyCode || SHEKEL_CURRENCY;\n\n      // Get securities from the balance array\n      const securities: Security[] = [];\n      const balances = accountData.AccountPosition?.Balance || [];\n      const metaSecurities = investmentData.View.Meta?.Security || [];\n\n      // Create a map of security metadata for easy lookup\n      const metaMap = new Map<string, InvestmentSecurityMeta>();\n      for (const meta of metaSecurities) {\n        metaMap.set(meta['-Key'], meta);\n      }\n\n      for (const securityBalance of balances) {\n        const meta = metaMap.get(securityBalance.EquityNumber);\n\n        securities.push({\n          name: meta?.EngName || '',\n          symbol: meta?.EngSymbol || '',\n          volume: securityBalance.OnlineNV,\n          value: securityBalance.OnlineVL,\n          currency: securityBalance.CurrencyCode || meta?.CurrencyCode || SHEKEL_CURRENCY,\n          changePercentage: securityBalance.BaseRateChangePercentage,\n          profitLoss: securityBalance.ProfitLoss,\n        });\n      }\n\n      debug('  - Balance: %s %s, Securities: %d', balance, currency, securities.length);\n\n      if (balance !== 0 || securities.length > 0) {\n        const investmentAccountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}-investment`;\n        accounts.push({\n          accountNumber: investmentAccountNumber,\n          balance,\n          currency,\n          savingsAccount: true,\n          txns: [],\n          securities,\n        });\n\n        debug('  ✓ Added investment account: %s with %d securities', investmentAccountNumber, securities.length);\n      } else {\n        debug('  - Skipping (zero balance and no securities)');\n      }\n    } else {\n      debug('  - No investment account data found');\n    }\n  } catch (error) {\n    debug('  - Error fetching investment account: %s', error);\n    // Log more details about the error\n    if (error instanceof Error) {\n      debug('    Error message: %s', error.message);\n      debug('    Error stack: %s', error.stack);\n    }\n  } finally {\n    // Clean up request interception (in case it wasn't already done)\n    try {\n      page.off('request', requestHandler);\n      await page.setRequestInterception(false);\n    } catch (e) {\n      // Ignore cleanup errors\n    }\n\n    // Navigate back to the main homepage to restore the session for subsequent scraping\n    try {\n      const homepageUrl = `${baseUrl}/ng-portals/rb/he/homepage`;\n      debug('Navigating back to homepage: %s', homepageUrl);\n      await page.goto(homepageUrl, { waitUntil: 'networkidle2', timeout: 30000 });\n      await new Promise(resolve => setTimeout(resolve, 2000));\n      debug('Homepage restored');\n    } catch (navError) {\n      debug('  - Failed to navigate back to homepage: %s', navError);\n      // Continue anyway, the outer try-catch in fetchAccountData will handle it\n    }\n  }\n\n  debug('Returning %d investment accounts', accounts.length);\n  return accounts;\n}\n\nasync function fetchAccountData(page: Page, baseUrl: string, options: ScraperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n\n  debug('fetching accounts data');\n  const accountsInfo = (await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl)) || [];\n  debug('got %d accounts, fetching txns and balance', accountsInfo.length);\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const { additionalTransactionInformation } = options;\n\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of accountsInfo) {\n    let balance: number | undefined;\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const isActiveAccount = account.accountClosingReasonCode === 0;\n    if (isActiveAccount) {\n      balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    } else {\n      debug('Skipping balance for a closed account, balance will be undefined');\n    }\n\n    let txns: Transaction[] = [];\n    try {\n      txns = await getAccountTransactions(\n        baseUrl,\n        apiSiteUrl,\n        page,\n        accountNumber,\n        startDateStr,\n        endDateStr,\n        additionalTransactionInformation,\n      );\n    } catch (error) {\n      debug('Error fetching transactions for %s (possibly closed account): %s', accountNumber, error);\n      // Continue with empty transactions\n    }\n\n    // Add regular checking account\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n\n    // Fetch forex accounts for this account number\n    try {\n      const forexAccounts = await getForexAccounts(baseUrl, page, accountNumber, startDateStr, endDateStr);\n      accounts.push(...forexAccounts);\n      debug('Added %d forex accounts to results', forexAccounts.length);\n    } catch (error) {\n      debug('Error fetching forex accounts for %s: %s', accountNumber, error);\n    }\n\n    // Fetch savings accounts for this account number\n    try {\n      const savingsAccounts = await getSavingsAccounts(baseUrl, page, accountNumber);\n      accounts.push(...savingsAccounts);\n      debug('Added %d savings accounts to results', savingsAccounts.length);\n    } catch (error) {\n      debug('Error fetching savings accounts for %s: %s', accountNumber, error);\n    }\n\n    // Fetch investment accounts for this account number\n    try {\n      const investmentAccounts = await getInvestmentAccounts(baseUrl, page, account);\n      accounts.push(...investmentAccounts);\n      debug('Added %d investment accounts to results', investmentAccounts.length);\n    } catch (error) {\n      debug('Error fetching investment accounts for %s: %s', accountNumber, error);\n    }\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n  debug('fetching ended');\n  return accountData;\n}\n\nfunction getPossibleLoginResults(baseUrl: string) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${baseUrl}/portalserver/HomePage`,\n    `${baseUrl}/ng-portals-bt/rb/he/homepage`,\n    `${baseUrl}/ng-portals/rb/he/homepage`,\n  ];\n  urls[LoginResults.InvalidPassword] = [\n    `${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`,\n  ];\n  urls[LoginResults.ChangePassword] = [\n    `${baseUrl}/MCP/START?flow=MCP&state=START&expiredDate=null`,\n    /\\/ABOUTTOEXPIRE\\/START/i,\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { userCode: string; password: string };\n\nclass HapoalimScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.baseUrl}/cgi-bin/poalwwwc?reqName=getLogonPage`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.login-btn',\n      postAction: async () => waitForRedirect(this.page),\n      possibleResults: getPossibleLoginResults(this.baseUrl),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.baseUrl, this.options);\n  }\n}\n\nexport default HapoalimScraper;\n"],"mappings":"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","ignoreList":[]}
|
package/lib/scrapers/visa-cal.js
CHANGED
|
@@ -302,7 +302,7 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
302
302
|
const txns = this.options.outputData?.enableTransactionsFilterByDate ?? true ? (0, _transactions.filterOldTransactions)(transactions, (0, _moment.default)(startDate), this.options.combineInstallments || false) : transactions;
|
|
303
303
|
return {
|
|
304
304
|
txns,
|
|
305
|
-
balance: -frame?.nextTotalDebit,
|
|
305
|
+
balance: _lodash.default.isNumber(frame?.nextTotalDebit) ? -frame?.nextTotalDebit : undefined,
|
|
306
306
|
accountNumber: card.last4Digits
|
|
307
307
|
};
|
|
308
308
|
}));
|
|
@@ -315,4 +315,4 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
315
315
|
}
|
|
316
316
|
}
|
|
317
317
|
var _default = exports.default = VisaCalScraper;
|
|
318
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_elementsInteractions","_fetch","_navigation","_storage","_transactions","_waiting","_transactions2","_baseScraperWithBrowser","_lodash","e","__esModule","default","apiHeaders","Origin","Referer","LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","FRAMES_REQUEST_ENDPOINT","PENDING_TRANSACTIONS_REQUEST_ENDPOINT","SSO_AUTHORIZATION_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","getDebug","TrnTypeCode","isAuthModule","result","Boolean","auth","calConnectToken","String","trim","authModuleOrUndefined","undefined","isPending","transaction","debCrdDate","isCardTransactionDetails","isCardPendingTransactionDetails","getLoginFrame","page","frame","waitUntil","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","elementPresentOnPage","errorMessage","pageEval","item","innerText","hasChangePasswordForm","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","ChangePassword","createLoginFields","credentials","selector","value","username","password","convertParsedDataToTransactions","data","pendingData","pendingTransactions","cardsList","flatMap","card","authDetalisList","bankAccounts","monthData","regularDebitDays","accounts","debitDates","immediateDebitDays","immidiateDebits","debitDays","completedTransactions","debitDate","transactions","all","map","numOfPayments","numberOfPayments","installments","number","curPaymentNum","total","date","moment","trnPurchaseDate","chargedAmount","trnAmt","amtBeforeConvAndIndex","originalAmount","trnTypeCode","credit","identifier","trnIntId","type","regular","standingOrder","TransactionTypes","Normal","Installments","status","TransactionStatuses","Pending","Completed","add","toISOString","processedDate","Date","originalCurrency","trnCurrencySymbol","chargedCurrency","debCrdCurrencySymbol","description","merchantName","memo","transTypeCommentDetails","toString","category","branchCodeDesc","VisaCalScraper","BaseScraperWithBrowser","authorization","openLoginPopup","waitUntilElementFound","clickButton","getCards","initData","getFromSessionStorage","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","getXSiteId","getLoginOptions","authRequestPromise","waitForRequest","timeout","catch","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","postAction","waitForNavigation","currentUrl","getCurrentUrl","endsWith","request","headers","requiresChangePassword","userAgent","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","max","format","xSiteId","Authorization","futureMonthsToScrape","fetchPost","cardsForFrameData","finalMonthToFetchMoment","months","diff","allMonthsData","_","bankIssuedCards","cardLevelFrames","cardUniqueIDArray","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","filterOldTransactions","combineInstallments","balance","nextTotalDebit","accountNumber","JSON","stringify","success","_default","exports"],"sources":["../../src/scrapers/visa-cal.ts"],"sourcesContent":["import moment from 'moment';\nimport { type HTTPRequest, type Frame, type Page } from 'puppeteer';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, pageEval, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPost } from '../helpers/fetch';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type LoginOptions } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\nimport _ from 'lodash';\n\nconst apiHeaders = {\n  'User-Agent':\n    'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/142.0.0.0 Safari/537.36',\n  Origin: 'https://digital-web.cal-online.co.il',\n  Referer: 'https://digital-web.cal-online.co.il',\n  'Accept-Language': 'he-IL,he;q=0.9,en-US;q=0.8,en;q=0.7',\n  'Sec-Fetch-Site': 'same-site',\n  'Sec-Fetch-Mode': 'cors',\n  'Sec-Fetch-Dest': 'empty',\n};\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\nconst FRAMES_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Frames/api/Frames/GetFrameStatus';\nconst PENDING_TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/approvals/getClearanceRequests';\nconst SSO_AUTHORIZATION_REQUEST_ENDPOINT = 'https://connect.cal-online.co.il/col-rest/calconnect/authentication/SSO';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum TrnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: TrnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n  earlyPaymentInd: boolean;\n}\ninterface ScrapedPendingTransaction {\n  merchantID: string;\n  merchantName: string;\n  trnPurchaseDate: string;\n  walletTranInd: number;\n  transactionsOrigin: number;\n  trnAmt: number;\n  tpaApprovalAmount: unknown;\n  trnCurrencySymbol: CurrencySymbol;\n  trnTypeCode: TrnTypeCode;\n  trnType: string;\n  branchCodeDesc: string;\n  transCardPresentInd: boolean;\n  j5Indicator: string;\n  numberOfPayments: number;\n  firstPaymentAmount: number;\n  transTypeCommentDetails: [];\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: []; debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\ninterface CardPendingTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    cardsList: {\n      cardUniqueID: string;\n      authDetalisList: ScrapedPendingTransaction[];\n    }[];\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\ninterface AuthModule {\n  auth: {\n    calConnectToken: string | null;\n  };\n}\n\nfunction isAuthModule(result: any): result is AuthModule {\n  return Boolean(result?.auth?.calConnectToken && String(result.auth.calConnectToken).trim());\n}\n\nfunction authModuleOrUndefined(result: any): AuthModule | undefined {\n  return isAuthModule(result) ? result : undefined;\n}\n\nfunction isPending(\n  transaction: ScrapedTransaction | ScrapedPendingTransaction,\n): transaction is ScrapedPendingTransaction {\n  return (transaction as ScrapedTransaction).debCrdDate === undefined; // an arbitrary field that only appears in a completed transaction\n}\n\nfunction isCardTransactionDetails(\n  result: CardTransactionDetails | CardTransactionDetailsError,\n): result is CardTransactionDetails {\n  return (result as CardTransactionDetails).result !== undefined;\n}\n\nfunction isCardPendingTransactionDetails(\n  result: CardPendingTransactionDetails | CardTransactionDetailsError,\n): result is CardPendingTransactionDetails {\n  return (result as CardPendingTransactionDetails).result !== undefined;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(\n    () => {\n      frame = page.frames().find(f => f.url().includes('connect')) || null;\n      return Promise.resolve(!!frame);\n    },\n    'wait for iframe with login form',\n    10000,\n    1000,\n  );\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound\n    ? await pageEval(frame, 'div.general-error > div', '', item => {\n        return (item as HTMLDivElement).innerText;\n      })\n    : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nasync function hasChangePasswordForm(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, '.change-password-subtitle');\n  return errorFound;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasInvalidPasswordError(page);\n      },\n    ],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    [LoginResults.ChangePassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasChangePasswordForm(page);\n      },\n    ],\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction convertParsedDataToTransactions(\n  data: CardTransactionDetails[],\n  pendingData?: CardPendingTransactionDetails | null,\n): Transaction[] {\n  const pendingTransactions = pendingData?.result\n    ? pendingData.result.cardsList.flatMap(card => card.authDetalisList)\n    : [];\n\n  const bankAccounts = data.flatMap(monthData => monthData.result.bankAccounts);\n  const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);\n  const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);\n  const completedTransactions = [...regularDebitDays, ...immediateDebitDays].flatMap(\n    debitDate => debitDate.transactions,\n  );\n\n  const all: (ScrapedTransaction | ScrapedPendingTransaction)[] = [...pendingTransactions, ...completedTransactions];\n\n  return all.map(transaction => {\n    const numOfPayments = isPending(transaction) ? transaction.numberOfPayments : transaction.numOfPayments;\n    const installments = numOfPayments\n      ? {\n          number: isPending(transaction) ? 1 : transaction.curPaymentNum,\n          total: numOfPayments,\n        }\n      : undefined;\n\n    const date = moment(transaction.trnPurchaseDate);\n\n    const chargedAmount = (isPending(transaction) ? transaction.trnAmt : transaction.amtBeforeConvAndIndex) * -1;\n    const originalAmount = transaction.trnAmt * (transaction.trnTypeCode === TrnTypeCode.credit ? 1 : -1);\n\n    const result: Transaction = {\n      identifier: !isPending(transaction) ? transaction.trnIntId : undefined,\n      type: [TrnTypeCode.regular, TrnTypeCode.standingOrder].includes(transaction.trnTypeCode)\n        ? TransactionTypes.Normal\n        : TransactionTypes.Installments,\n      status: isPending(transaction) ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),\n      processedDate: isPending(transaction) ? date.toISOString() : new Date(transaction.debCrdDate).toISOString(),\n      originalAmount,\n      originalCurrency: transaction.trnCurrencySymbol,\n      chargedAmount,\n      chargedCurrency: !isPending(transaction) ? transaction.debCrdCurrencySymbol : undefined,\n      description: transaction.merchantName,\n      memo: transaction.transTypeCommentDetails.toString(),\n      category: transaction.branchCodeDesc,\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  private authorization: string | undefined = undefined;\n\n  private authRequestPromise: Promise<HTTPRequest | undefined> | undefined;\n\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error('could not find \"init\" data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    if (!this.authorization) {\n      debug('fetching authorization header');\n      const authModule = await waitUntil(\n        async () => authModuleOrUndefined(await getFromSessionStorage<AuthModule>(this.page, 'auth-module')),\n        'get authorization header with valid token in session storage',\n        10_000,\n        50,\n      );\n      return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n    }\n    return this.authorization;\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    this.authRequestPromise = this.page\n      .waitForRequest(SSO_AUTHORIZATION_REQUEST_ENDPOINT, { timeout: 10_000 })\n      .catch(e => {\n        debug('error while waiting for the token request', e);\n        return undefined;\n      });\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitForNavigation(this.page);\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n          const request = await this.authRequestPromise;\n          this.authorization = String(request?.headers().authorization || '').trim();\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          const requiresChangePassword = await hasChangePasswordForm(this.page);\n          if (requiresChangePassword) return;\n          throw e;\n        }\n      },\n      userAgent: apiHeaders['User-Agent'],\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const [cards, xSiteId, Authorization] = await Promise.all([\n      this.getCards(),\n      this.getXSiteId(),\n      this.getAuthorizationHeader(),\n    ]);\n\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    debug('fetch frames (misgarot) of cards');\n    const frames = await fetchPost(\n      FRAMES_REQUEST_ENDPOINT,\n      { cardsForFrameData: cards.map(({ cardUniqueId }) => ({ cardUniqueId })) },\n      {\n        Authorization,\n        'X-Site-Id': xSiteId,\n        'Content-Type': 'application/json',\n        ...apiHeaders,\n      },\n    );\n\n    const accounts = await Promise.all(\n      cards.map(async card => {\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n        const allMonthsData: CardTransactionDetails[] = [];\n        const frame = _.find(frames.result.bankIssuedCards.cardLevelFrames, { cardUniqueId: card.cardUniqueId });\n\n        debug(`fetch pending transactions for card ${card.cardUniqueId}`);\n        let pendingData = await fetchPost(\n          PENDING_TRANSACTIONS_REQUEST_ENDPOINT,\n          { cardUniqueIDArray: [card.cardUniqueId] },\n          {\n            Authorization,\n            'X-Site-Id': xSiteId,\n            'Content-Type': 'application/json',\n            ...apiHeaders,\n          },\n        );\n\n        debug(`fetch completed transactions for card ${card.cardUniqueId}`);\n        for (let i = 0; i <= months; i++) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPost(\n            TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n              ...apiHeaders,\n            },\n          );\n\n          if (monthData?.statusCode !== 1)\n            throw new Error(\n              `failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`,\n            );\n\n          if (!isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        if (pendingData?.statusCode !== 1 && pendingData?.statusCode !== 96) {\n          debug(\n            `failed to fetch pending transactions for card ${card.last4Digits}. Message: ${pendingData?.title || ''}`,\n          );\n          pendingData = null;\n        } else if (!isCardPendingTransactionDetails(pendingData)) {\n          debug('pendingData is not of type CardTransactionDetails');\n          pendingData = null;\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData, pendingData);\n\n        debug('filter out old transactions');\n        const txns =\n          (this.options.outputData?.enableTransactionsFilterByDate ?? true)\n            ? filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false)\n            : transactions;\n\n        return {\n          txns,\n          balance: -frame?.nextTotalDebit,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"],"mappings":"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","ignoreList":[]}
|
|
318
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_elementsInteractions","_fetch","_navigation","_storage","_transactions","_waiting","_transactions2","_baseScraperWithBrowser","_lodash","e","__esModule","default","apiHeaders","Origin","Referer","LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","FRAMES_REQUEST_ENDPOINT","PENDING_TRANSACTIONS_REQUEST_ENDPOINT","SSO_AUTHORIZATION_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","getDebug","TrnTypeCode","isAuthModule","result","Boolean","auth","calConnectToken","String","trim","authModuleOrUndefined","undefined","isPending","transaction","debCrdDate","isCardTransactionDetails","isCardPendingTransactionDetails","getLoginFrame","page","frame","waitUntil","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","elementPresentOnPage","errorMessage","pageEval","item","innerText","hasChangePasswordForm","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","ChangePassword","createLoginFields","credentials","selector","value","username","password","convertParsedDataToTransactions","data","pendingData","pendingTransactions","cardsList","flatMap","card","authDetalisList","bankAccounts","monthData","regularDebitDays","accounts","debitDates","immediateDebitDays","immidiateDebits","debitDays","completedTransactions","debitDate","transactions","all","map","numOfPayments","numberOfPayments","installments","number","curPaymentNum","total","date","moment","trnPurchaseDate","chargedAmount","trnAmt","amtBeforeConvAndIndex","originalAmount","trnTypeCode","credit","identifier","trnIntId","type","regular","standingOrder","TransactionTypes","Normal","Installments","status","TransactionStatuses","Pending","Completed","add","toISOString","processedDate","Date","originalCurrency","trnCurrencySymbol","chargedCurrency","debCrdCurrencySymbol","description","merchantName","memo","transTypeCommentDetails","toString","category","branchCodeDesc","VisaCalScraper","BaseScraperWithBrowser","authorization","openLoginPopup","waitUntilElementFound","clickButton","getCards","initData","getFromSessionStorage","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","getXSiteId","getLoginOptions","authRequestPromise","waitForRequest","timeout","catch","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","postAction","waitForNavigation","currentUrl","getCurrentUrl","endsWith","request","headers","requiresChangePassword","userAgent","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","max","format","xSiteId","Authorization","futureMonthsToScrape","fetchPost","cardsForFrameData","finalMonthToFetchMoment","months","diff","allMonthsData","_","bankIssuedCards","cardLevelFrames","cardUniqueIDArray","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","filterOldTransactions","combineInstallments","balance","isNumber","nextTotalDebit","accountNumber","JSON","stringify","success","_default","exports"],"sources":["../../src/scrapers/visa-cal.ts"],"sourcesContent":["import moment from 'moment';\nimport { type HTTPRequest, type Frame, type Page } from 'puppeteer';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, pageEval, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPost } from '../helpers/fetch';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type LoginOptions } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\nimport _ from 'lodash';\n\nconst apiHeaders = {\n  'User-Agent':\n    'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/142.0.0.0 Safari/537.36',\n  Origin: 'https://digital-web.cal-online.co.il',\n  Referer: 'https://digital-web.cal-online.co.il',\n  'Accept-Language': 'he-IL,he;q=0.9,en-US;q=0.8,en;q=0.7',\n  'Sec-Fetch-Site': 'same-site',\n  'Sec-Fetch-Mode': 'cors',\n  'Sec-Fetch-Dest': 'empty',\n};\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\nconst FRAMES_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Frames/api/Frames/GetFrameStatus';\nconst PENDING_TRANSACTIONS_REQUEST_ENDPOINT =\n  'https://api.cal-online.co.il/Transactions/api/approvals/getClearanceRequests';\nconst SSO_AUTHORIZATION_REQUEST_ENDPOINT = 'https://connect.cal-online.co.il/col-rest/calconnect/authentication/SSO';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum TrnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: TrnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n  earlyPaymentInd: boolean;\n}\ninterface ScrapedPendingTransaction {\n  merchantID: string;\n  merchantName: string;\n  trnPurchaseDate: string;\n  walletTranInd: number;\n  transactionsOrigin: number;\n  trnAmt: number;\n  tpaApprovalAmount: unknown;\n  trnCurrencySymbol: CurrencySymbol;\n  trnTypeCode: TrnTypeCode;\n  trnType: string;\n  branchCodeDesc: string;\n  transCardPresentInd: boolean;\n  j5Indicator: string;\n  numberOfPayments: number;\n  firstPaymentAmount: number;\n  transTypeCommentDetails: [];\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: []; debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\ninterface CardPendingTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    cardsList: {\n      cardUniqueID: string;\n      authDetalisList: ScrapedPendingTransaction[];\n    }[];\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\ninterface AuthModule {\n  auth: {\n    calConnectToken: string | null;\n  };\n}\n\nfunction isAuthModule(result: any): result is AuthModule {\n  return Boolean(result?.auth?.calConnectToken && String(result.auth.calConnectToken).trim());\n}\n\nfunction authModuleOrUndefined(result: any): AuthModule | undefined {\n  return isAuthModule(result) ? result : undefined;\n}\n\nfunction isPending(\n  transaction: ScrapedTransaction | ScrapedPendingTransaction,\n): transaction is ScrapedPendingTransaction {\n  return (transaction as ScrapedTransaction).debCrdDate === undefined; // an arbitrary field that only appears in a completed transaction\n}\n\nfunction isCardTransactionDetails(\n  result: CardTransactionDetails | CardTransactionDetailsError,\n): result is CardTransactionDetails {\n  return (result as CardTransactionDetails).result !== undefined;\n}\n\nfunction isCardPendingTransactionDetails(\n  result: CardPendingTransactionDetails | CardTransactionDetailsError,\n): result is CardPendingTransactionDetails {\n  return (result as CardPendingTransactionDetails).result !== undefined;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(\n    () => {\n      frame = page.frames().find(f => f.url().includes('connect')) || null;\n      return Promise.resolve(!!frame);\n    },\n    'wait for iframe with login form',\n    10000,\n    1000,\n  );\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound\n    ? await pageEval(frame, 'div.general-error > div', '', item => {\n        return (item as HTMLDivElement).innerText;\n      })\n    : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nasync function hasChangePasswordForm(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, '.change-password-subtitle');\n  return errorFound;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasInvalidPasswordError(page);\n      },\n    ],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    [LoginResults.ChangePassword]: [\n      async (options?: { page?: Page }) => {\n        const page = options?.page;\n        if (!page) {\n          return false;\n        }\n        return hasChangePasswordForm(page);\n      },\n    ],\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction convertParsedDataToTransactions(\n  data: CardTransactionDetails[],\n  pendingData?: CardPendingTransactionDetails | null,\n): Transaction[] {\n  const pendingTransactions = pendingData?.result\n    ? pendingData.result.cardsList.flatMap(card => card.authDetalisList)\n    : [];\n\n  const bankAccounts = data.flatMap(monthData => monthData.result.bankAccounts);\n  const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);\n  const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);\n  const completedTransactions = [...regularDebitDays, ...immediateDebitDays].flatMap(\n    debitDate => debitDate.transactions,\n  );\n\n  const all: (ScrapedTransaction | ScrapedPendingTransaction)[] = [...pendingTransactions, ...completedTransactions];\n\n  return all.map(transaction => {\n    const numOfPayments = isPending(transaction) ? transaction.numberOfPayments : transaction.numOfPayments;\n    const installments = numOfPayments\n      ? {\n          number: isPending(transaction) ? 1 : transaction.curPaymentNum,\n          total: numOfPayments,\n        }\n      : undefined;\n\n    const date = moment(transaction.trnPurchaseDate);\n\n    const chargedAmount = (isPending(transaction) ? transaction.trnAmt : transaction.amtBeforeConvAndIndex) * -1;\n    const originalAmount = transaction.trnAmt * (transaction.trnTypeCode === TrnTypeCode.credit ? 1 : -1);\n\n    const result: Transaction = {\n      identifier: !isPending(transaction) ? transaction.trnIntId : undefined,\n      type: [TrnTypeCode.regular, TrnTypeCode.standingOrder].includes(transaction.trnTypeCode)\n        ? TransactionTypes.Normal\n        : TransactionTypes.Installments,\n      status: isPending(transaction) ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),\n      processedDate: isPending(transaction) ? date.toISOString() : new Date(transaction.debCrdDate).toISOString(),\n      originalAmount,\n      originalCurrency: transaction.trnCurrencySymbol,\n      chargedAmount,\n      chargedCurrency: !isPending(transaction) ? transaction.debCrdCurrencySymbol : undefined,\n      description: transaction.merchantName,\n      memo: transaction.transTypeCommentDetails.toString(),\n      category: transaction.branchCodeDesc,\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  private authorization: string | undefined = undefined;\n\n  private authRequestPromise: Promise<HTTPRequest | undefined> | undefined;\n\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error('could not find \"init\" data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    if (!this.authorization) {\n      debug('fetching authorization header');\n      const authModule = await waitUntil(\n        async () => authModuleOrUndefined(await getFromSessionStorage<AuthModule>(this.page, 'auth-module')),\n        'get authorization header with valid token in session storage',\n        10_000,\n        50,\n      );\n      return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n    }\n    return this.authorization;\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    this.authRequestPromise = this.page\n      .waitForRequest(SSO_AUTHORIZATION_REQUEST_ENDPOINT, { timeout: 10_000 })\n      .catch(e => {\n        debug('error while waiting for the token request', e);\n        return undefined;\n      });\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitForNavigation(this.page);\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n          const request = await this.authRequestPromise;\n          this.authorization = String(request?.headers().authorization || '').trim();\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          const requiresChangePassword = await hasChangePasswordForm(this.page);\n          if (requiresChangePassword) return;\n          throw e;\n        }\n      },\n      userAgent: apiHeaders['User-Agent'],\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const [cards, xSiteId, Authorization] = await Promise.all([\n      this.getCards(),\n      this.getXSiteId(),\n      this.getAuthorizationHeader(),\n    ]);\n\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    debug('fetch frames (misgarot) of cards');\n    const frames = await fetchPost(\n      FRAMES_REQUEST_ENDPOINT,\n      { cardsForFrameData: cards.map(({ cardUniqueId }) => ({ cardUniqueId })) },\n      {\n        Authorization,\n        'X-Site-Id': xSiteId,\n        'Content-Type': 'application/json',\n        ...apiHeaders,\n      },\n    );\n\n    const accounts = await Promise.all(\n      cards.map(async card => {\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n        const allMonthsData: CardTransactionDetails[] = [];\n        const frame = _.find(frames.result.bankIssuedCards.cardLevelFrames, { cardUniqueId: card.cardUniqueId });\n\n        debug(`fetch pending transactions for card ${card.cardUniqueId}`);\n        let pendingData = await fetchPost(\n          PENDING_TRANSACTIONS_REQUEST_ENDPOINT,\n          { cardUniqueIDArray: [card.cardUniqueId] },\n          {\n            Authorization,\n            'X-Site-Id': xSiteId,\n            'Content-Type': 'application/json',\n            ...apiHeaders,\n          },\n        );\n\n        debug(`fetch completed transactions for card ${card.cardUniqueId}`);\n        for (let i = 0; i <= months; i++) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPost(\n            TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n              ...apiHeaders,\n            },\n          );\n\n          if (monthData?.statusCode !== 1)\n            throw new Error(\n              `failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`,\n            );\n\n          if (!isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        if (pendingData?.statusCode !== 1 && pendingData?.statusCode !== 96) {\n          debug(\n            `failed to fetch pending transactions for card ${card.last4Digits}. Message: ${pendingData?.title || ''}`,\n          );\n          pendingData = null;\n        } else if (!isCardPendingTransactionDetails(pendingData)) {\n          debug('pendingData is not of type CardTransactionDetails');\n          pendingData = null;\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData, pendingData);\n\n        debug('filter out old transactions');\n        const txns =\n          (this.options.outputData?.enableTransactionsFilterByDate ?? true)\n            ? filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false)\n            : transactions;\n\n        return {\n          txns,\n          balance: _.isNumber(frame?.nextTotalDebit) ? -frame?.nextTotalDebit : undefined,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"],"mappings":"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","ignoreList":[]}
|
package/lib/transactions.d.ts
CHANGED
|
@@ -1,7 +1,10 @@
|
|
|
1
1
|
export interface TransactionsAccount {
|
|
2
2
|
accountNumber: string;
|
|
3
3
|
balance?: number;
|
|
4
|
+
currency?: string;
|
|
5
|
+
savingsAccount?: boolean;
|
|
4
6
|
txns: Transaction[];
|
|
7
|
+
securities?: Security[];
|
|
5
8
|
}
|
|
6
9
|
export declare enum TransactionTypes {
|
|
7
10
|
Normal = "normal",
|
|
@@ -45,3 +48,12 @@ export interface Transaction {
|
|
|
45
48
|
installments?: TransactionInstallments;
|
|
46
49
|
category?: string;
|
|
47
50
|
}
|
|
51
|
+
export interface Security {
|
|
52
|
+
name?: string;
|
|
53
|
+
symbol: string;
|
|
54
|
+
volume: number;
|
|
55
|
+
value: number;
|
|
56
|
+
currency?: string;
|
|
57
|
+
changePercentage?: number;
|
|
58
|
+
profitLoss?: number;
|
|
59
|
+
}
|
package/lib/transactions.js
CHANGED
|
@@ -14,4 +14,4 @@ let TransactionStatuses = exports.TransactionStatuses = /*#__PURE__*/function (T
|
|
|
14
14
|
TransactionStatuses["Pending"] = "pending";
|
|
15
15
|
return TransactionStatuses;
|
|
16
16
|
}({});
|
|
17
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
17
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJUcmFuc2FjdGlvblR5cGVzIiwiZXhwb3J0cyIsIlRyYW5zYWN0aW9uU3RhdHVzZXMiXSwic291cmNlcyI6WyIuLi9zcmMvdHJhbnNhY3Rpb25zLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgVHJhbnNhY3Rpb25zQWNjb3VudCB7XG4gIGFjY291bnROdW1iZXI6IHN0cmluZztcbiAgYmFsYW5jZT86IG51bWJlcjtcbiAgY3VycmVuY3k/OiBzdHJpbmc7XG4gIHNhdmluZ3NBY2NvdW50PzogYm9vbGVhbjtcbiAgdHhuczogVHJhbnNhY3Rpb25bXTtcbiAgc2VjdXJpdGllcz86IFNlY3VyaXR5W107XG59XG5cbmV4cG9ydCBlbnVtIFRyYW5zYWN0aW9uVHlwZXMge1xuICBOb3JtYWwgPSAnbm9ybWFsJyxcbiAgSW5zdGFsbG1lbnRzID0gJ2luc3RhbGxtZW50cycsXG59XG5cbmV4cG9ydCBlbnVtIFRyYW5zYWN0aW9uU3RhdHVzZXMge1xuICBDb21wbGV0ZWQgPSAnY29tcGxldGVkJyxcbiAgUGVuZGluZyA9ICdwZW5kaW5nJyxcbn1cblxuZXhwb3J0IGludGVyZmFjZSBUcmFuc2FjdGlvbkluc3RhbGxtZW50cyB7XG4gIC8qKlxuICAgKiB0aGUgY3VycmVudCBpbnN0YWxsbWVudCBudW1iZXJcbiAgICovXG4gIG51bWJlcjogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiB0aGUgdG90YWwgbnVtYmVyIG9mIGluc3RhbGxtZW50c1xuICAgKi9cbiAgdG90YWw6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBUcmFuc2FjdGlvbiB7XG4gIHR5cGU6IFRyYW5zYWN0aW9uVHlwZXM7XG4gIC8qKlxuICAgKiBzb21ldGltZXMgY2FsbGVkIEFzbWFjaHRhXG4gICAqL1xuICBpZGVudGlmaWVyPzogc3RyaW5nIHwgbnVtYmVyO1xuICAvKipcbiAgICogSVNPIGRhdGUgc3RyaW5nXG4gICAqL1xuICBkYXRlOiBzdHJpbmc7XG4gIC8qKlxuICAgKiBJU08gZGF0ZSBzdHJpbmdcbiAgICovXG4gIHByb2Nlc3NlZERhdGU6IHN0cmluZztcbiAgb3JpZ2luYWxBbW91bnQ6IG51bWJlcjtcbiAgb3JpZ2luYWxDdXJyZW5jeTogc3RyaW5nO1xuICBjaGFyZ2VkQW1vdW50OiBudW1iZXI7XG4gIGNoYXJnZWRDdXJyZW5jeT86IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgbWVtbz86IHN0cmluZztcbiAgc3RhdHVzOiBUcmFuc2FjdGlvblN0YXR1c2VzO1xuICBpbnN0YWxsbWVudHM/OiBUcmFuc2FjdGlvbkluc3RhbGxtZW50cztcbiAgY2F0ZWdvcnk/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VjdXJpdHkge1xuICBuYW1lPzogc3RyaW5nO1xuICBzeW1ib2w6IHN0cmluZztcbiAgdm9sdW1lOiBudW1iZXI7XG4gIHZhbHVlOiBudW1iZXI7XG4gIGN1cnJlbmN5Pzogc3RyaW5nO1xuICBjaGFuZ2VQZXJjZW50YWdlPzogbnVtYmVyO1xuICBwcm9maXRMb3NzPzogbnVtYmVyO1xufVxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7SUFTWUEsZ0JBQWdCLEdBQUFDLE9BQUEsQ0FBQUQsZ0JBQUEsMEJBQWhCQSxnQkFBZ0I7RUFBaEJBLGdCQUFnQjtFQUFoQkEsZ0JBQWdCO0VBQUEsT0FBaEJBLGdCQUFnQjtBQUFBO0FBQUEsSUFLaEJFLG1CQUFtQixHQUFBRCxPQUFBLENBQUFDLG1CQUFBLDBCQUFuQkEsbUJBQW1CO0VBQW5CQSxtQkFBbUI7RUFBbkJBLG1CQUFtQjtFQUFBLE9BQW5CQSxtQkFBbUI7QUFBQSIsImlnbm9yZUxpc3QiOltdfQ==
|