israeli-bank-scrapers 6.1.1 → 6.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (79) hide show
  1. package/lib/assertNever.js +7 -5
  2. package/lib/constants.js +16 -13
  3. package/lib/definitions.js +113 -109
  4. package/lib/helpers/browser.js +13 -9
  5. package/lib/helpers/dates.js +19 -18
  6. package/lib/helpers/debug.js +9 -9
  7. package/lib/helpers/elements-interactions.js +82 -78
  8. package/lib/helpers/fetch.js +85 -82
  9. package/lib/helpers/navigation.js +28 -24
  10. package/lib/helpers/storage.js +11 -10
  11. package/lib/helpers/transactions.js +32 -33
  12. package/lib/helpers/waiting.js +42 -45
  13. package/lib/index.js +82 -15
  14. package/lib/scrapers/amex.js +13 -11
  15. package/lib/scrapers/amex.test.d.ts +1 -0
  16. package/lib/scrapers/amex.test.js +49 -0
  17. package/lib/scrapers/base-beinleumi-group.js +239 -233
  18. package/lib/scrapers/base-isracard-amex.js +273 -273
  19. package/lib/scrapers/base-scraper-with-browser.js +263 -241
  20. package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
  21. package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
  22. package/lib/scrapers/base-scraper.js +82 -82
  23. package/lib/scrapers/behatsdaa.js +103 -98
  24. package/lib/scrapers/behatsdaa.test.d.ts +1 -0
  25. package/lib/scrapers/behatsdaa.test.js +46 -0
  26. package/lib/scrapers/beinleumi.js +13 -11
  27. package/lib/scrapers/beinleumi.test.d.ts +1 -0
  28. package/lib/scrapers/beinleumi.test.js +47 -0
  29. package/lib/scrapers/beyahad-bishvilha.js +132 -132
  30. package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
  31. package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
  32. package/lib/scrapers/discount.js +101 -97
  33. package/lib/scrapers/discount.test.d.ts +1 -0
  34. package/lib/scrapers/discount.test.js +49 -0
  35. package/lib/scrapers/errors.js +25 -22
  36. package/lib/scrapers/factory.js +67 -66
  37. package/lib/scrapers/factory.test.d.ts +1 -0
  38. package/lib/scrapers/factory.test.js +19 -0
  39. package/lib/scrapers/hapoalim.js +175 -162
  40. package/lib/scrapers/hapoalim.test.d.ts +1 -0
  41. package/lib/scrapers/hapoalim.test.js +47 -0
  42. package/lib/scrapers/interface.js +5 -2
  43. package/lib/scrapers/isracard.js +13 -11
  44. package/lib/scrapers/isracard.test.d.ts +1 -0
  45. package/lib/scrapers/isracard.test.js +49 -0
  46. package/lib/scrapers/leumi.js +170 -167
  47. package/lib/scrapers/leumi.test.d.ts +1 -0
  48. package/lib/scrapers/leumi.test.js +47 -0
  49. package/lib/scrapers/massad.js +13 -11
  50. package/lib/scrapers/max.js +261 -261
  51. package/lib/scrapers/max.test.d.ts +1 -0
  52. package/lib/scrapers/max.test.js +65 -0
  53. package/lib/scrapers/mercantile.js +16 -14
  54. package/lib/scrapers/mercantile.test.d.ts +1 -0
  55. package/lib/scrapers/mercantile.test.js +45 -0
  56. package/lib/scrapers/mizrahi.js +154 -158
  57. package/lib/scrapers/mizrahi.test.d.ts +1 -0
  58. package/lib/scrapers/mizrahi.test.js +53 -0
  59. package/lib/scrapers/one-zero-queries.js +7 -4
  60. package/lib/scrapers/one-zero.js +221 -176
  61. package/lib/scrapers/one-zero.test.d.ts +1 -0
  62. package/lib/scrapers/one-zero.test.js +51 -0
  63. package/lib/scrapers/otsar-hahayal.js +13 -11
  64. package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
  65. package/lib/scrapers/otsar-hahayal.test.js +47 -0
  66. package/lib/scrapers/pagi.js +13 -11
  67. package/lib/scrapers/pagi.test.d.ts +1 -0
  68. package/lib/scrapers/pagi.test.js +47 -0
  69. package/lib/scrapers/union-bank.js +173 -172
  70. package/lib/scrapers/union-bank.test.d.ts +1 -0
  71. package/lib/scrapers/union-bank.test.js +47 -0
  72. package/lib/scrapers/visa-cal.js +250 -254
  73. package/lib/scrapers/visa-cal.test.d.ts +1 -0
  74. package/lib/scrapers/visa-cal.test.js +49 -0
  75. package/lib/scrapers/yahav.js +206 -190
  76. package/lib/scrapers/yahav.test.d.ts +1 -0
  77. package/lib/scrapers/yahav.test.js +49 -0
  78. package/lib/transactions.js +16 -13
  79. package/package.json +8 -3
@@ -1,16 +1,20 @@
1
1
  "use strict";
2
- var __importDefault = (this && this.__importDefault) || function (mod) {
3
- return (mod && mod.__esModule) ? mod : { "default": mod };
4
- };
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.waitForPostLogin = exports.createLoginFields = exports.getPossibleLoginResults = void 0;
7
- const moment_1 = __importDefault(require("moment"));
8
- const constants_1 = require("../constants");
9
- const elements_interactions_1 = require("../helpers/elements-interactions");
10
- const navigation_1 = require("../helpers/navigation");
11
- const waiting_1 = require("../helpers/waiting");
12
- const transactions_1 = require("../transactions");
13
- const base_scraper_with_browser_1 = require("./base-scraper-with-browser");
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.createLoginFields = createLoginFields;
7
+ exports.default = void 0;
8
+ exports.getPossibleLoginResults = getPossibleLoginResults;
9
+ exports.waitForPostLogin = waitForPostLogin;
10
+ var _moment = _interopRequireDefault(require("moment"));
11
+ var _constants = require("../constants");
12
+ var _elementsInteractions = require("../helpers/elements-interactions");
13
+ var _navigation = require("../helpers/navigation");
14
+ var _waiting = require("../helpers/waiting");
15
+ var _transactions = require("../transactions");
16
+ var _baseScraperWithBrowser = require("./base-scraper-with-browser");
17
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
14
18
  const DATE_FORMAT = 'DD/MM/YYYY';
15
19
  const NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא נמצאו נתונים בנושא המבוקש';
16
20
  const DATE_COLUMN_CLASS_COMPLETED = 'date first';
@@ -30,286 +34,288 @@ const NEXT_PAGE_LINK = 'a#Npage.paging';
30
34
  const CURRENT_BALANCE = '.main_balance';
31
35
  const IFRAME_NAME = 'iframe-old-pages';
32
36
  function getPossibleLoginResults() {
33
- const urls = {};
34
- urls[base_scraper_with_browser_1.LoginResults.Success] = [
35
- /fibi.*accountSummary/,
36
- /FibiMenu\/Online/, // Old UI pattern
37
- ];
38
- urls[base_scraper_with_browser_1.LoginResults.InvalidPassword] = [/FibiMenu\/Marketing\/Private\/Home/];
39
- return urls;
37
+ const urls = {};
38
+ urls[_baseScraperWithBrowser.LoginResults.Success] = [/fibi.*accountSummary/,
39
+ // New UI pattern
40
+ /FibiMenu\/Online/ // Old UI pattern
41
+ ];
42
+ urls[_baseScraperWithBrowser.LoginResults.InvalidPassword] = [/FibiMenu\/Marketing\/Private\/Home/];
43
+ return urls;
40
44
  }
41
- exports.getPossibleLoginResults = getPossibleLoginResults;
42
45
  function createLoginFields(credentials) {
43
- return [
44
- { selector: '#username', value: credentials.username },
45
- { selector: '#password', value: credentials.password },
46
- ];
46
+ return [{
47
+ selector: '#username',
48
+ value: credentials.username
49
+ }, {
50
+ selector: '#password',
51
+ value: credentials.password
52
+ }];
47
53
  }
48
- exports.createLoginFields = createLoginFields;
49
54
  function getAmountData(amountStr) {
50
- let amountStrCopy = amountStr.replace(constants_1.SHEKEL_CURRENCY_SYMBOL, '');
51
- amountStrCopy = amountStrCopy.replaceAll(',', '');
52
- return parseFloat(amountStrCopy);
55
+ let amountStrCopy = amountStr.replace(_constants.SHEKEL_CURRENCY_SYMBOL, '');
56
+ amountStrCopy = amountStrCopy.replaceAll(',', '');
57
+ return parseFloat(amountStrCopy);
53
58
  }
54
59
  function getTxnAmount(txn) {
55
- const credit = getAmountData(txn.credit);
56
- const debit = getAmountData(txn.debit);
57
- return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);
60
+ const credit = getAmountData(txn.credit);
61
+ const debit = getAmountData(txn.debit);
62
+ return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);
58
63
  }
59
64
  function convertTransactions(txns) {
60
- return txns.map((txn) => {
61
- const convertedDate = (0, moment_1.default)(txn.date, DATE_FORMAT).toISOString();
62
- const convertedAmount = getTxnAmount(txn);
63
- return {
64
- type: transactions_1.TransactionTypes.Normal,
65
- identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,
66
- date: convertedDate,
67
- processedDate: convertedDate,
68
- originalAmount: convertedAmount,
69
- originalCurrency: constants_1.SHEKEL_CURRENCY,
70
- chargedAmount: convertedAmount,
71
- status: txn.status,
72
- description: txn.description,
73
- memo: txn.memo,
74
- };
75
- });
65
+ return txns.map(txn => {
66
+ const convertedDate = (0, _moment.default)(txn.date, DATE_FORMAT).toISOString();
67
+ const convertedAmount = getTxnAmount(txn);
68
+ return {
69
+ type: _transactions.TransactionTypes.Normal,
70
+ identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,
71
+ date: convertedDate,
72
+ processedDate: convertedDate,
73
+ originalAmount: convertedAmount,
74
+ originalCurrency: _constants.SHEKEL_CURRENCY,
75
+ chargedAmount: convertedAmount,
76
+ status: txn.status,
77
+ description: txn.description,
78
+ memo: txn.memo
79
+ };
80
+ });
76
81
  }
77
82
  function getTransactionDate(tds, transactionType, transactionsColsTypes) {
78
- if (transactionType === 'completed') {
79
- return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_COMPLETED]] || '').trim();
80
- }
81
- return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_PENDING]] || '').trim();
83
+ if (transactionType === 'completed') {
84
+ return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_COMPLETED]] || '').trim();
85
+ }
86
+ return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_PENDING]] || '').trim();
82
87
  }
83
88
  function getTransactionDescription(tds, transactionType, transactionsColsTypes) {
84
- if (transactionType === 'completed') {
85
- return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_COMPLETED]] || '').trim();
86
- }
87
- return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_PENDING]] || '').trim();
89
+ if (transactionType === 'completed') {
90
+ return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_COMPLETED]] || '').trim();
91
+ }
92
+ return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_PENDING]] || '').trim();
88
93
  }
89
94
  function getTransactionReference(tds, transactionsColsTypes) {
90
- return (tds[transactionsColsTypes[REFERENCE_COLUMN_CLASS]] || '').trim();
95
+ return (tds[transactionsColsTypes[REFERENCE_COLUMN_CLASS]] || '').trim();
91
96
  }
92
97
  function getTransactionDebit(tds, transactionsColsTypes) {
93
- return (tds[transactionsColsTypes[DEBIT_COLUMN_CLASS]] || '').trim();
98
+ return (tds[transactionsColsTypes[DEBIT_COLUMN_CLASS]] || '').trim();
94
99
  }
95
100
  function getTransactionCredit(tds, transactionsColsTypes) {
96
- return (tds[transactionsColsTypes[CREDIT_COLUMN_CLASS]] || '').trim();
101
+ return (tds[transactionsColsTypes[CREDIT_COLUMN_CLASS]] || '').trim();
97
102
  }
98
103
  function extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes) {
99
- const tds = txnRow.innerTds;
100
- const item = {
101
- status: transactionStatus,
102
- date: getTransactionDate(tds, transactionStatus, transactionsColsTypes),
103
- description: getTransactionDescription(tds, transactionStatus, transactionsColsTypes),
104
- reference: getTransactionReference(tds, transactionsColsTypes),
105
- debit: getTransactionDebit(tds, transactionsColsTypes),
106
- credit: getTransactionCredit(tds, transactionsColsTypes),
107
- };
108
- return item;
104
+ const tds = txnRow.innerTds;
105
+ const item = {
106
+ status: transactionStatus,
107
+ date: getTransactionDate(tds, transactionStatus, transactionsColsTypes),
108
+ description: getTransactionDescription(tds, transactionStatus, transactionsColsTypes),
109
+ reference: getTransactionReference(tds, transactionsColsTypes),
110
+ debit: getTransactionDebit(tds, transactionsColsTypes),
111
+ credit: getTransactionCredit(tds, transactionsColsTypes)
112
+ };
113
+ return item;
109
114
  }
110
115
  async function getTransactionsColsTypeClasses(page, tableLocator) {
111
- const result = {};
112
- const typeClassesObjs = await (0, elements_interactions_1.pageEvalAll)(page, `${tableLocator} tbody tr:first-of-type td`, null, tds => {
113
- return tds.map((td, index) => ({
114
- colClass: td.getAttribute('class'),
115
- index,
116
- }));
117
- });
118
- for (const typeClassObj of typeClassesObjs) {
119
- if (typeClassObj.colClass) {
120
- result[typeClassObj.colClass] = typeClassObj.index;
121
- }
116
+ const result = {};
117
+ const typeClassesObjs = await (0, _elementsInteractions.pageEvalAll)(page, `${tableLocator} tbody tr:first-of-type td`, null, tds => {
118
+ return tds.map((td, index) => ({
119
+ colClass: td.getAttribute('class'),
120
+ index
121
+ }));
122
+ });
123
+ for (const typeClassObj of typeClassesObjs) {
124
+ if (typeClassObj.colClass) {
125
+ result[typeClassObj.colClass] = typeClassObj.index;
122
126
  }
123
- return result;
127
+ }
128
+ return result;
124
129
  }
125
130
  function extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes) {
126
- const txn = extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes);
127
- if (txn.date !== '') {
128
- txns.push(txn);
129
- }
131
+ const txn = extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes);
132
+ if (txn.date !== '') {
133
+ txns.push(txn);
134
+ }
130
135
  }
131
136
  async function extractTransactions(page, tableLocator, transactionStatus) {
132
- const txns = [];
133
- const transactionsColsTypes = await getTransactionsColsTypeClasses(page, tableLocator);
134
- const transactionsRows = await (0, elements_interactions_1.pageEvalAll)(page, `${tableLocator} tbody tr`, [], trs => {
135
- return trs.map(tr => ({
136
- innerTds: Array.from(tr.getElementsByTagName('td')).map(td => td.innerText),
137
- }));
138
- });
139
- for (const txnRow of transactionsRows) {
140
- extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes);
141
- }
142
- return txns;
137
+ const txns = [];
138
+ const transactionsColsTypes = await getTransactionsColsTypeClasses(page, tableLocator);
139
+ const transactionsRows = await (0, _elementsInteractions.pageEvalAll)(page, `${tableLocator} tbody tr`, [], trs => {
140
+ return trs.map(tr => ({
141
+ innerTds: Array.from(tr.getElementsByTagName('td')).map(td => td.innerText)
142
+ }));
143
+ });
144
+ for (const txnRow of transactionsRows) {
145
+ extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes);
146
+ }
147
+ return txns;
143
148
  }
144
149
  async function isNoTransactionInDateRangeError(page) {
145
- const hasErrorInfoElement = await (0, elements_interactions_1.elementPresentOnPage)(page, `.${ERROR_MESSAGE_CLASS}`);
146
- if (hasErrorInfoElement) {
147
- const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, errorElement => {
148
- return errorElement.innerText;
149
- });
150
- return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;
151
- }
152
- return false;
150
+ const hasErrorInfoElement = await (0, _elementsInteractions.elementPresentOnPage)(page, `.${ERROR_MESSAGE_CLASS}`);
151
+ if (hasErrorInfoElement) {
152
+ const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, errorElement => {
153
+ return errorElement.innerText;
154
+ });
155
+ return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;
156
+ }
157
+ return false;
153
158
  }
154
159
  async function searchByDates(page, startDate) {
155
- await (0, elements_interactions_1.clickButton)(page, 'a#tabHeader4');
156
- await (0, elements_interactions_1.waitUntilElementFound)(page, 'div#fibi_dates');
157
- await (0, elements_interactions_1.fillInput)(page, 'input#fromDate', startDate.format(DATE_FORMAT));
158
- await (0, elements_interactions_1.clickButton)(page, `button[class*=${CLOSE_SEARCH_BY_DATES_BUTTON_CLASS}]`);
159
- await (0, elements_interactions_1.clickButton)(page, `input[value=${SHOW_SEARCH_BY_DATES_BUTTON_VALUE}]`);
160
- await (0, navigation_1.waitForNavigation)(page);
160
+ await (0, _elementsInteractions.clickButton)(page, 'a#tabHeader4');
161
+ await (0, _elementsInteractions.waitUntilElementFound)(page, 'div#fibi_dates');
162
+ await (0, _elementsInteractions.fillInput)(page, 'input#fromDate', startDate.format(DATE_FORMAT));
163
+ await (0, _elementsInteractions.clickButton)(page, `button[class*=${CLOSE_SEARCH_BY_DATES_BUTTON_CLASS}]`);
164
+ await (0, _elementsInteractions.clickButton)(page, `input[value=${SHOW_SEARCH_BY_DATES_BUTTON_VALUE}]`);
165
+ await (0, _navigation.waitForNavigation)(page);
161
166
  }
162
167
  async function getAccountNumber(page) {
163
- const selectedSnifAccount = await page.$eval(ACCOUNTS_NUMBER, option => {
164
- return option.innerText;
165
- });
166
- return selectedSnifAccount.replace('/', '_').trim();
168
+ const selectedSnifAccount = await page.$eval(ACCOUNTS_NUMBER, option => {
169
+ return option.innerText;
170
+ });
171
+ return selectedSnifAccount.replace('/', '_').trim();
167
172
  }
168
173
  async function checkIfHasNextPage(page) {
169
- return (0, elements_interactions_1.elementPresentOnPage)(page, NEXT_PAGE_LINK);
174
+ return (0, _elementsInteractions.elementPresentOnPage)(page, NEXT_PAGE_LINK);
170
175
  }
171
176
  async function navigateToNextPage(page) {
172
- await (0, elements_interactions_1.clickButton)(page, NEXT_PAGE_LINK);
173
- await (0, navigation_1.waitForNavigation)(page);
177
+ await (0, _elementsInteractions.clickButton)(page, NEXT_PAGE_LINK);
178
+ await (0, _navigation.waitForNavigation)(page);
174
179
  }
180
+
175
181
  /* Couldn't reproduce scenario with multiple pages of pending transactions - Should support if exists such case.
176
182
  needToPaginate is false if scraping pending transactions */
177
183
  async function scrapeTransactions(page, tableLocator, transactionStatus, needToPaginate) {
178
- const txns = [];
179
- let hasNextPage = false;
180
- do {
181
- const currentPageTxns = await extractTransactions(page, tableLocator, transactionStatus);
182
- txns.push(...currentPageTxns);
183
- if (needToPaginate) {
184
- hasNextPage = await checkIfHasNextPage(page);
185
- if (hasNextPage) {
186
- await navigateToNextPage(page);
187
- }
188
- }
189
- } while (hasNextPage);
190
- return convertTransactions(txns);
184
+ const txns = [];
185
+ let hasNextPage = false;
186
+ do {
187
+ const currentPageTxns = await extractTransactions(page, tableLocator, transactionStatus);
188
+ txns.push(...currentPageTxns);
189
+ if (needToPaginate) {
190
+ hasNextPage = await checkIfHasNextPage(page);
191
+ if (hasNextPage) {
192
+ await navigateToNextPage(page);
193
+ }
194
+ }
195
+ } while (hasNextPage);
196
+ return convertTransactions(txns);
191
197
  }
192
198
  async function getAccountTransactions(page) {
193
- await Promise.race([
194
- (0, elements_interactions_1.waitUntilElementFound)(page, "div[id*='divTable']", false),
195
- (0, elements_interactions_1.waitUntilElementFound)(page, `.${ERROR_MESSAGE_CLASS}`, false),
196
- ]);
197
- const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);
198
- if (noTransactionInRangeError) {
199
- return [];
200
- }
201
- const pendingTxns = await scrapeTransactions(page, PENDING_TRANSACTIONS_TABLE, transactions_1.TransactionStatuses.Pending, false);
202
- const completedTxns = await scrapeTransactions(page, COMPLETED_TRANSACTIONS_TABLE, transactions_1.TransactionStatuses.Completed, true);
203
- const txns = [...pendingTxns, ...completedTxns];
204
- return txns;
199
+ await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, "div[id*='divTable']", false), (0, _elementsInteractions.waitUntilElementFound)(page, `.${ERROR_MESSAGE_CLASS}`, false)]);
200
+ const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);
201
+ if (noTransactionInRangeError) {
202
+ return [];
203
+ }
204
+ const pendingTxns = await scrapeTransactions(page, PENDING_TRANSACTIONS_TABLE, _transactions.TransactionStatuses.Pending, false);
205
+ const completedTxns = await scrapeTransactions(page, COMPLETED_TRANSACTIONS_TABLE, _transactions.TransactionStatuses.Completed, true);
206
+ const txns = [...pendingTxns, ...completedTxns];
207
+ return txns;
205
208
  }
206
209
  async function getCurrentBalance(page) {
207
- const balanceElement = await page.$(CURRENT_BALANCE);
208
- if (!balanceElement) {
209
- return undefined;
210
- }
211
- const balanceStr = await balanceElement.evaluate(option => {
212
- return option.innerText;
213
- });
214
- return getAmountData(balanceStr);
210
+ const balanceElement = await page.$(CURRENT_BALANCE);
211
+ if (!balanceElement) {
212
+ return undefined;
213
+ }
214
+ const balanceStr = await balanceElement.evaluate(option => {
215
+ return option.innerText;
216
+ });
217
+ return getAmountData(balanceStr);
215
218
  }
216
219
  async function waitForPostLogin(page) {
217
- return Promise.race([
218
- (0, elements_interactions_1.waitUntilElementFound)(page, '#card-header', true),
219
- (0, elements_interactions_1.waitUntilElementFound)(page, '#account_num', true),
220
- (0, elements_interactions_1.waitUntilElementFound)(page, '#matafLogoutLink', true),
221
- (0, elements_interactions_1.waitUntilElementFound)(page, '#validationMsg', true), // Old UI
222
- ]);
220
+ return Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, '#card-header', true),
221
+ // New UI
222
+ (0, _elementsInteractions.waitUntilElementFound)(page, '#account_num', true),
223
+ // New UI
224
+ (0, _elementsInteractions.waitUntilElementFound)(page, '#matafLogoutLink', true),
225
+ // Old UI
226
+ (0, _elementsInteractions.waitUntilElementFound)(page, '#validationMsg', true) // Old UI
227
+ ]);
223
228
  }
224
- exports.waitForPostLogin = waitForPostLogin;
225
229
  async function fetchAccountData(page, startDate) {
226
- await searchByDates(page, startDate);
227
- const accountNumber = await getAccountNumber(page);
228
- const balance = await getCurrentBalance(page);
229
- const txns = await getAccountTransactions(page);
230
- return {
231
- accountNumber,
232
- txns,
233
- balance,
234
- };
230
+ await searchByDates(page, startDate);
231
+ const accountNumber = await getAccountNumber(page);
232
+ const balance = await getCurrentBalance(page);
233
+ const txns = await getAccountTransactions(page);
234
+ return {
235
+ accountNumber,
236
+ txns,
237
+ balance
238
+ };
235
239
  }
236
240
  async function getAccountIdsBySelector(page) {
237
- const accountsIds = await page.evaluate(() => {
238
- const selectElement = document.getElementById('account_num_select');
239
- const options = selectElement ? selectElement.querySelectorAll('option') : [];
240
- if (!options)
241
- return [];
242
- return Array.from(options, option => option.value);
243
- });
244
- return accountsIds;
241
+ const accountsIds = await page.evaluate(() => {
242
+ const selectElement = document.getElementById('account_num_select');
243
+ const options = selectElement ? selectElement.querySelectorAll('option') : [];
244
+ if (!options) return [];
245
+ return Array.from(options, option => option.value);
246
+ });
247
+ return accountsIds;
245
248
  }
246
249
  async function getTransactionsFrame(page) {
247
- // Try a few times to find the iframe, as it might not be immediately available
248
- for (let attempt = 0; attempt < 3; attempt++) {
249
- await (0, waiting_1.sleep)(2000);
250
- const frames = page.frames();
251
- const targetFrame = frames.find(f => f.name() === IFRAME_NAME);
252
- if (targetFrame) {
253
- return targetFrame;
254
- }
250
+ // Try a few times to find the iframe, as it might not be immediately available
251
+ for (let attempt = 0; attempt < 3; attempt++) {
252
+ await (0, _waiting.sleep)(2000);
253
+ const frames = page.frames();
254
+ const targetFrame = frames.find(f => f.name() === IFRAME_NAME);
255
+ if (targetFrame) {
256
+ return targetFrame;
255
257
  }
256
- return null;
258
+ }
259
+ return null;
257
260
  }
258
261
  async function selectAccount(page, accountId) {
259
- await page.select('#account_num_select', accountId);
260
- await (0, elements_interactions_1.waitUntilElementFound)(page, '#account_num_select', true);
262
+ await page.select('#account_num_select', accountId);
263
+ await (0, _elementsInteractions.waitUntilElementFound)(page, '#account_num_select', true);
261
264
  }
262
265
  async function fetchAccountDataBothUIs(page, startDate) {
263
- // Try to get the iframe for the new UI
264
- const frame = await getTransactionsFrame(page);
265
- // Use the frame if available (new UI), otherwise use the page directly (old UI)
266
- const targetPage = frame || page;
267
- return fetchAccountData(targetPage, startDate);
266
+ // Try to get the iframe for the new UI
267
+ const frame = await getTransactionsFrame(page);
268
+
269
+ // Use the frame if available (new UI), otherwise use the page directly (old UI)
270
+ const targetPage = frame || page;
271
+ return fetchAccountData(targetPage, startDate);
268
272
  }
269
273
  async function fetchAccounts(page, startDate) {
270
- const accountsIds = await getAccountIdsBySelector(page);
271
- if (accountsIds.length <= 1) {
272
- const accountData = await fetchAccountDataBothUIs(page, startDate);
273
- return [accountData];
274
- }
275
- const accounts = [];
276
- for (const accountId of accountsIds) {
277
- await selectAccount(page, accountId);
278
- const accountData = await fetchAccountDataBothUIs(page, startDate);
279
- accounts.push(accountData);
280
- }
281
- return accounts;
274
+ const accountsIds = await getAccountIdsBySelector(page);
275
+ if (accountsIds.length <= 1) {
276
+ const accountData = await fetchAccountDataBothUIs(page, startDate);
277
+ return [accountData];
278
+ }
279
+ const accounts = [];
280
+ for (const accountId of accountsIds) {
281
+ await selectAccount(page, accountId);
282
+ const accountData = await fetchAccountDataBothUIs(page, startDate);
283
+ accounts.push(accountData);
284
+ }
285
+ return accounts;
282
286
  }
283
- class BeinleumiGroupBaseScraper extends base_scraper_with_browser_1.BaseScraperWithBrowser {
284
- BASE_URL = '';
285
- LOGIN_URL = '';
286
- TRANSACTIONS_URL = '';
287
- getLoginOptions(credentials) {
288
- return {
289
- loginUrl: `${this.LOGIN_URL}`,
290
- fields: createLoginFields(credentials),
291
- submitButtonSelector: '#continueBtn',
292
- postAction: async () => waitForPostLogin(this.page),
293
- possibleResults: getPossibleLoginResults(),
294
- // HACK: For some reason, though the login button (#continueBtn) is present and visible, the click action does not perform.
295
- // Adding this delay fixes the issue.
296
- preAction: async () => {
297
- await (0, waiting_1.sleep)(1000);
298
- },
299
- };
300
- }
301
- async fetchData() {
302
- const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years').add(1, 'day');
303
- const startMomentLimit = (0, moment_1.default)({ year: 1600 });
304
- const startDate = this.options.startDate || defaultStartMoment.toDate();
305
- const startMoment = moment_1.default.max(startMomentLimit, (0, moment_1.default)(startDate));
306
- await this.navigateTo(this.TRANSACTIONS_URL);
307
- const accounts = await fetchAccounts(this.page, startMoment);
308
- return {
309
- success: true,
310
- accounts,
311
- };
312
- }
287
+ class BeinleumiGroupBaseScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
288
+ BASE_URL = '';
289
+ LOGIN_URL = '';
290
+ TRANSACTIONS_URL = '';
291
+ getLoginOptions(credentials) {
292
+ return {
293
+ loginUrl: `${this.LOGIN_URL}`,
294
+ fields: createLoginFields(credentials),
295
+ submitButtonSelector: '#continueBtn',
296
+ postAction: async () => waitForPostLogin(this.page),
297
+ possibleResults: getPossibleLoginResults(),
298
+ // HACK: For some reason, though the login button (#continueBtn) is present and visible, the click action does not perform.
299
+ // Adding this delay fixes the issue.
300
+ preAction: async () => {
301
+ await (0, _waiting.sleep)(1000);
302
+ }
303
+ };
304
+ }
305
+ async fetchData() {
306
+ const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
307
+ const startMomentLimit = (0, _moment.default)({
308
+ year: 1600
309
+ });
310
+ const startDate = this.options.startDate || defaultStartMoment.toDate();
311
+ const startMoment = _moment.default.max(startMomentLimit, (0, _moment.default)(startDate));
312
+ await this.navigateTo(this.TRANSACTIONS_URL);
313
+ const accounts = await fetchAccounts(this.page, startMoment);
314
+ return {
315
+ success: true,
316
+ accounts
317
+ };
318
+ }
313
319
  }
314
- exports.default = BeinleumiGroupBaseScraper;
315
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-beinleumi-group.js","sourceRoot":"","sources":["../../src/scrapers/base-beinleumi-group.ts"],"names":[],"mappings":"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","sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Frame, type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport {\n  clickButton,\n  elementPresentOnPage,\n  fillInput,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { sleep } from '../helpers/waiting';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\n\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא נמצאו נתונים בנושא המבוקש';\nconst DATE_COLUMN_CLASS_COMPLETED = 'date first';\nconst DATE_COLUMN_CLASS_PENDING = 'first date';\nconst DESCRIPTION_COLUMN_CLASS_COMPLETED = 'reference wrap_normal';\nconst DESCRIPTION_COLUMN_CLASS_PENDING = 'details wrap_normal';\nconst REFERENCE_COLUMN_CLASS = 'details';\nconst DEBIT_COLUMN_CLASS = 'debit';\nconst CREDIT_COLUMN_CLASS = 'credit';\nconst ERROR_MESSAGE_CLASS = 'NO_DATA';\nconst ACCOUNTS_NUMBER = 'div.fibi_account span.acc_num';\nconst CLOSE_SEARCH_BY_DATES_BUTTON_CLASS = 'ui-datepicker-close';\nconst SHOW_SEARCH_BY_DATES_BUTTON_VALUE = 'הצג';\nconst COMPLETED_TRANSACTIONS_TABLE = 'table#dataTable077';\nconst PENDING_TRANSACTIONS_TABLE = 'table#dataTable023';\nconst NEXT_PAGE_LINK = 'a#Npage.paging';\nconst CURRENT_BALANCE = '.main_balance';\nconst IFRAME_NAME = 'iframe-old-pages';\n\ntype TransactionsColsTypes = Record<string, number>;\ntype TransactionsTrTds = string[];\ntype TransactionsTr = { innerTds: TransactionsTrTds };\n\ninterface ScrapedTransaction {\n  reference: string;\n  date: string;\n  credit: string;\n  debit: string;\n  memo?: string;\n  description: string;\n  status: TransactionStatuses;\n}\n\nexport function getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    /fibi.*accountSummary/, // New UI pattern\n    /FibiMenu\\/Online/, // Old UI pattern\n  ];\n  urls[LoginResults.InvalidPassword] = [/FibiMenu\\/Marketing\\/Private\\/Home/];\n  return urls;\n}\n\nexport function createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string) {\n  let amountStrCopy = amountStr.replace(SHEKEL_CURRENCY_SYMBOL, '');\n  amountStrCopy = amountStrCopy.replaceAll(',', '');\n  return parseFloat(amountStrCopy);\n}\n\nfunction getTxnAmount(txn: ScrapedTransaction) {\n  const credit = getAmountData(txn.credit);\n  const debit = getAmountData(txn.debit);\n  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn): Transaction => {\n    const convertedDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const convertedAmount = getTxnAmount(txn);\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: convertedDate,\n      processedDate: convertedDate,\n      originalAmount: convertedAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: convertedAmount,\n      status: txn.status,\n      description: txn.description,\n      memo: txn.memo,\n    };\n  });\n}\n\nfunction getTransactionDate(\n  tds: TransactionsTrTds,\n  transactionType: string,\n  transactionsColsTypes: TransactionsColsTypes,\n) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionDescription(\n  tds: TransactionsTrTds,\n  transactionType: string,\n  transactionsColsTypes: TransactionsColsTypes,\n) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionReference(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[REFERENCE_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionDebit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[DEBIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionCredit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[CREDIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction extractTransactionDetails(\n  txnRow: TransactionsTr,\n  transactionStatus: TransactionStatuses,\n  transactionsColsTypes: TransactionsColsTypes,\n): ScrapedTransaction {\n  const tds = txnRow.innerTds;\n  const item = {\n    status: transactionStatus,\n    date: getTransactionDate(tds, transactionStatus, transactionsColsTypes),\n    description: getTransactionDescription(tds, transactionStatus, transactionsColsTypes),\n    reference: getTransactionReference(tds, transactionsColsTypes),\n    debit: getTransactionDebit(tds, transactionsColsTypes),\n    credit: getTransactionCredit(tds, transactionsColsTypes),\n  };\n\n  return item;\n}\n\nasync function getTransactionsColsTypeClasses(\n  page: Page | Frame,\n  tableLocator: string,\n): Promise<TransactionsColsTypes> {\n  const result: TransactionsColsTypes = {};\n  const typeClassesObjs = await pageEvalAll(page, `${tableLocator} tbody tr:first-of-type td`, null, tds => {\n    return tds.map((td, index) => ({\n      colClass: td.getAttribute('class'),\n      index,\n    }));\n  });\n\n  for (const typeClassObj of typeClassesObjs) {\n    if (typeClassObj.colClass) {\n      result[typeClassObj.colClass] = typeClassObj.index;\n    }\n  }\n  return result;\n}\n\nfunction extractTransaction(\n  txns: ScrapedTransaction[],\n  transactionStatus: TransactionStatuses,\n  txnRow: TransactionsTr,\n  transactionsColsTypes: TransactionsColsTypes,\n) {\n  const txn = extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes);\n  if (txn.date !== '') {\n    txns.push(txn);\n  }\n}\n\nasync function extractTransactions(page: Page | Frame, tableLocator: string, transactionStatus: TransactionStatuses) {\n  const txns: ScrapedTransaction[] = [];\n  const transactionsColsTypes = await getTransactionsColsTypeClasses(page, tableLocator);\n\n  const transactionsRows = await pageEvalAll<TransactionsTr[]>(page, `${tableLocator} tbody tr`, [], trs => {\n    return trs.map(tr => ({\n      innerTds: Array.from(tr.getElementsByTagName('td')).map(td => td.innerText),\n    }));\n  });\n\n  for (const txnRow of transactionsRows) {\n    extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes);\n  }\n  return txns;\n}\n\nasync function isNoTransactionInDateRangeError(page: Page | Frame) {\n  const hasErrorInfoElement = await elementPresentOnPage(page, `.${ERROR_MESSAGE_CLASS}`);\n  if (hasErrorInfoElement) {\n    const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, errorElement => {\n      return (errorElement as HTMLElement).innerText;\n    });\n    return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;\n  }\n  return false;\n}\n\nasync function searchByDates(page: Page | Frame, startDate: Moment) {\n  await clickButton(page, 'a#tabHeader4');\n  await waitUntilElementFound(page, 'div#fibi_dates');\n  await fillInput(page, 'input#fromDate', startDate.format(DATE_FORMAT));\n  await clickButton(page, `button[class*=${CLOSE_SEARCH_BY_DATES_BUTTON_CLASS}]`);\n  await clickButton(page, `input[value=${SHOW_SEARCH_BY_DATES_BUTTON_VALUE}]`);\n  await waitForNavigation(page);\n}\n\nasync function getAccountNumber(page: Page | Frame) {\n  const selectedSnifAccount = await page.$eval(ACCOUNTS_NUMBER, option => {\n    return (option as HTMLElement).innerText;\n  });\n\n  return selectedSnifAccount.replace('/', '_').trim();\n}\n\nasync function checkIfHasNextPage(page: Page | Frame) {\n  return elementPresentOnPage(page, NEXT_PAGE_LINK);\n}\n\nasync function navigateToNextPage(page: Page | Frame) {\n  await clickButton(page, NEXT_PAGE_LINK);\n  await waitForNavigation(page);\n}\n\n/* Couldn't reproduce scenario with multiple pages of pending transactions - Should support if exists such case.\n   needToPaginate is false if scraping pending transactions */\nasync function scrapeTransactions(\n  page: Page | Frame,\n  tableLocator: string,\n  transactionStatus: TransactionStatuses,\n  needToPaginate: boolean,\n) {\n  const txns = [];\n  let hasNextPage = false;\n\n  do {\n    const currentPageTxns = await extractTransactions(page, tableLocator, transactionStatus);\n    txns.push(...currentPageTxns);\n    if (needToPaginate) {\n      hasNextPage = await checkIfHasNextPage(page);\n      if (hasNextPage) {\n        await navigateToNextPage(page);\n      }\n    }\n  } while (hasNextPage);\n\n  return convertTransactions(txns);\n}\n\nasync function getAccountTransactions(page: Page | Frame) {\n  await Promise.race([\n    waitUntilElementFound(page, \"div[id*='divTable']\", false),\n    waitUntilElementFound(page, `.${ERROR_MESSAGE_CLASS}`, false),\n  ]);\n\n  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);\n  if (noTransactionInRangeError) {\n    return [];\n  }\n\n  const pendingTxns = await scrapeTransactions(page, PENDING_TRANSACTIONS_TABLE, TransactionStatuses.Pending, false);\n  const completedTxns = await scrapeTransactions(\n    page,\n    COMPLETED_TRANSACTIONS_TABLE,\n    TransactionStatuses.Completed,\n    true,\n  );\n  const txns = [...pendingTxns, ...completedTxns];\n  return txns;\n}\n\nasync function getCurrentBalance(page: Page | Frame) {\n  const balanceElement = await page.$(CURRENT_BALANCE);\n  if (!balanceElement) {\n    return undefined;\n  }\n  const balanceStr = await balanceElement.evaluate(option => {\n    return (option as HTMLElement).innerText;\n  });\n  return getAmountData(balanceStr);\n}\n\nexport async function waitForPostLogin(page: Page) {\n  return Promise.race([\n    waitUntilElementFound(page, '#card-header', true), // New UI\n    waitUntilElementFound(page, '#account_num', true), // New UI\n    waitUntilElementFound(page, '#matafLogoutLink', true), // Old UI\n    waitUntilElementFound(page, '#validationMsg', true), // Old UI\n  ]);\n}\n\nasync function fetchAccountData(page: Page | Frame, startDate: Moment) {\n  await searchByDates(page, startDate);\n  const accountNumber = await getAccountNumber(page);\n  const balance = await getCurrentBalance(page);\n  const txns = await getAccountTransactions(page);\n\n  return {\n    accountNumber,\n    txns,\n    balance,\n  };\n}\n\nasync function getAccountIdsBySelector(page: Page): Promise<string[]> {\n  const accountsIds = await page.evaluate(() => {\n    const selectElement = document.getElementById('account_num_select');\n    const options = selectElement ? selectElement.querySelectorAll('option') : [];\n    if (!options) return [];\n    return Array.from(options, option => option.value);\n  });\n  return accountsIds;\n}\n\nasync function getTransactionsFrame(page: Page): Promise<Frame | null> {\n  // Try a few times to find the iframe, as it might not be immediately available\n  for (let attempt = 0; attempt < 3; attempt++) {\n    await sleep(2000);\n    const frames = page.frames();\n    const targetFrame = frames.find(f => f.name() === IFRAME_NAME);\n\n    if (targetFrame) {\n      return targetFrame;\n    }\n  }\n\n  return null;\n}\n\nasync function selectAccount(page: Page, accountId: string) {\n  await page.select('#account_num_select', accountId);\n  await waitUntilElementFound(page, '#account_num_select', true);\n}\n\nasync function fetchAccountDataBothUIs(page: Page, startDate: Moment) {\n  // Try to get the iframe for the new UI\n  const frame = await getTransactionsFrame(page);\n\n  // Use the frame if available (new UI), otherwise use the page directly (old UI)\n  const targetPage = frame || page;\n  return fetchAccountData(targetPage, startDate);\n}\n\nasync function fetchAccounts(page: Page, startDate: Moment): Promise<TransactionsAccount[]> {\n  const accountsIds = await getAccountIdsBySelector(page);\n\n  if (accountsIds.length <= 1) {\n    const accountData = await fetchAccountDataBothUIs(page, startDate);\n    return [accountData];\n  }\n\n  const accounts: TransactionsAccount[] = [];\n  for (const accountId of accountsIds) {\n    await selectAccount(page, accountId);\n\n    const accountData = await fetchAccountDataBothUIs(page, startDate);\n    accounts.push(accountData);\n  }\n\n  return accounts;\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass BeinleumiGroupBaseScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  BASE_URL = '';\n\n  LOGIN_URL = '';\n\n  TRANSACTIONS_URL = '';\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n      // HACK: For some reason, though the login button (#continueBtn) is present and visible, the click action does not perform.\n      // Adding this delay fixes the issue.\n      preAction: async () => {\n        await sleep(1000);\n      },\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startMomentLimit = moment({ year: 1600 });\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(startMomentLimit, moment(startDate));\n\n    await this.navigateTo(this.TRANSACTIONS_URL);\n\n    const accounts = await fetchAccounts(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default BeinleumiGroupBaseScraper;\n"]}
320
+ var _default = exports.default = BeinleumiGroupBaseScraper;
321
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","DATE_FORMAT","NO_TRANSACTION_IN_DATE_RANGE_TEXT","DATE_COLUMN_CLASS_COMPLETED","DATE_COLUMN_CLASS_PENDING","DESCRIPTION_COLUMN_CLASS_COMPLETED","DESCRIPTION_COLUMN_CLASS_PENDING","REFERENCE_COLUMN_CLASS","DEBIT_COLUMN_CLASS","CREDIT_COLUMN_CLASS","ERROR_MESSAGE_CLASS","ACCOUNTS_NUMBER","CLOSE_SEARCH_BY_DATES_BUTTON_CLASS","SHOW_SEARCH_BY_DATES_BUTTON_VALUE","COMPLETED_TRANSACTIONS_TABLE","PENDING_TRANSACTIONS_TABLE","NEXT_PAGE_LINK","CURRENT_BALANCE","IFRAME_NAME","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","amountStrCopy","replace","SHEKEL_CURRENCY_SYMBOL","replaceAll","parseFloat","getTxnAmount","txn","credit","debit","Number","isNaN","convertTransactions","txns","map","convertedDate","moment","date","toISOString","convertedAmount","type","TransactionTypes","Normal","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","status","description","memo","getTransactionDate","tds","transactionType","transactionsColsTypes","trim","getTransactionDescription","getTransactionReference","getTransactionDebit","getTransactionCredit","extractTransactionDetails","txnRow","transactionStatus","innerTds","item","getTransactionsColsTypeClasses","page","tableLocator","result","typeClassesObjs","pageEvalAll","td","index","colClass","getAttribute","typeClassObj","extractTransaction","push","extractTransactions","transactionsRows","trs","tr","Array","from","getElementsByTagName","innerText","isNoTransactionInDateRangeError","hasErrorInfoElement","elementPresentOnPage","errorText","$eval","errorElement","searchByDates","startDate","clickButton","waitUntilElementFound","fillInput","format","waitForNavigation","getAccountNumber","selectedSnifAccount","option","checkIfHasNextPage","navigateToNextPage","scrapeTransactions","needToPaginate","hasNextPage","currentPageTxns","getAccountTransactions","Promise","race","noTransactionInRangeError","pendingTxns","TransactionStatuses","Pending","completedTxns","Completed","getCurrentBalance","balanceElement","$","balanceStr","evaluate","waitForPostLogin","fetchAccountData","accountNumber","balance","getAccountIdsBySelector","accountsIds","selectElement","document","getElementById","options","querySelectorAll","getTransactionsFrame","attempt","sleep","frames","targetFrame","find","f","name","selectAccount","accountId","select","fetchAccountDataBothUIs","frame","targetPage","fetchAccounts","length","accountData","accounts","BeinleumiGroupBaseScraper","BaseScraperWithBrowser","BASE_URL","LOGIN_URL","TRANSACTIONS_URL","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","preAction","fetchData","defaultStartMoment","subtract","add","startMomentLimit","year","toDate","startMoment","max","navigateTo","success","_default","exports"],"sources":["../../src/scrapers/base-beinleumi-group.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Frame, type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport {\n  clickButton,\n  elementPresentOnPage,\n  fillInput,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { sleep } from '../helpers/waiting';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\n\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא נמצאו נתונים בנושא המבוקש';\nconst DATE_COLUMN_CLASS_COMPLETED = 'date first';\nconst DATE_COLUMN_CLASS_PENDING = 'first date';\nconst DESCRIPTION_COLUMN_CLASS_COMPLETED = 'reference wrap_normal';\nconst DESCRIPTION_COLUMN_CLASS_PENDING = 'details wrap_normal';\nconst REFERENCE_COLUMN_CLASS = 'details';\nconst DEBIT_COLUMN_CLASS = 'debit';\nconst CREDIT_COLUMN_CLASS = 'credit';\nconst ERROR_MESSAGE_CLASS = 'NO_DATA';\nconst ACCOUNTS_NUMBER = 'div.fibi_account span.acc_num';\nconst CLOSE_SEARCH_BY_DATES_BUTTON_CLASS = 'ui-datepicker-close';\nconst SHOW_SEARCH_BY_DATES_BUTTON_VALUE = 'הצג';\nconst COMPLETED_TRANSACTIONS_TABLE = 'table#dataTable077';\nconst PENDING_TRANSACTIONS_TABLE = 'table#dataTable023';\nconst NEXT_PAGE_LINK = 'a#Npage.paging';\nconst CURRENT_BALANCE = '.main_balance';\nconst IFRAME_NAME = 'iframe-old-pages';\n\ntype TransactionsColsTypes = Record<string, number>;\ntype TransactionsTrTds = string[];\ntype TransactionsTr = { innerTds: TransactionsTrTds };\n\ninterface ScrapedTransaction {\n  reference: string;\n  date: string;\n  credit: string;\n  debit: string;\n  memo?: string;\n  description: string;\n  status: TransactionStatuses;\n}\n\nexport function getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    /fibi.*accountSummary/, // New UI pattern\n    /FibiMenu\\/Online/, // Old UI pattern\n  ];\n  urls[LoginResults.InvalidPassword] = [/FibiMenu\\/Marketing\\/Private\\/Home/];\n  return urls;\n}\n\nexport function createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string) {\n  let amountStrCopy = amountStr.replace(SHEKEL_CURRENCY_SYMBOL, '');\n  amountStrCopy = amountStrCopy.replaceAll(',', '');\n  return parseFloat(amountStrCopy);\n}\n\nfunction getTxnAmount(txn: ScrapedTransaction) {\n  const credit = getAmountData(txn.credit);\n  const debit = getAmountData(txn.debit);\n  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn): Transaction => {\n    const convertedDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const convertedAmount = getTxnAmount(txn);\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: convertedDate,\n      processedDate: convertedDate,\n      originalAmount: convertedAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: convertedAmount,\n      status: txn.status,\n      description: txn.description,\n      memo: txn.memo,\n    };\n  });\n}\n\nfunction getTransactionDate(\n  tds: TransactionsTrTds,\n  transactionType: string,\n  transactionsColsTypes: TransactionsColsTypes,\n) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionDescription(\n  tds: TransactionsTrTds,\n  transactionType: string,\n  transactionsColsTypes: TransactionsColsTypes,\n) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionReference(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[REFERENCE_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionDebit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[DEBIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionCredit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[CREDIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction extractTransactionDetails(\n  txnRow: TransactionsTr,\n  transactionStatus: TransactionStatuses,\n  transactionsColsTypes: TransactionsColsTypes,\n): ScrapedTransaction {\n  const tds = txnRow.innerTds;\n  const item = {\n    status: transactionStatus,\n    date: getTransactionDate(tds, transactionStatus, transactionsColsTypes),\n    description: getTransactionDescription(tds, transactionStatus, transactionsColsTypes),\n    reference: getTransactionReference(tds, transactionsColsTypes),\n    debit: getTransactionDebit(tds, transactionsColsTypes),\n    credit: getTransactionCredit(tds, transactionsColsTypes),\n  };\n\n  return item;\n}\n\nasync function getTransactionsColsTypeClasses(\n  page: Page | Frame,\n  tableLocator: string,\n): Promise<TransactionsColsTypes> {\n  const result: TransactionsColsTypes = {};\n  const typeClassesObjs = await pageEvalAll(page, `${tableLocator} tbody tr:first-of-type td`, null, tds => {\n    return tds.map((td, index) => ({\n      colClass: td.getAttribute('class'),\n      index,\n    }));\n  });\n\n  for (const typeClassObj of typeClassesObjs) {\n    if (typeClassObj.colClass) {\n      result[typeClassObj.colClass] = typeClassObj.index;\n    }\n  }\n  return result;\n}\n\nfunction extractTransaction(\n  txns: ScrapedTransaction[],\n  transactionStatus: TransactionStatuses,\n  txnRow: TransactionsTr,\n  transactionsColsTypes: TransactionsColsTypes,\n) {\n  const txn = extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes);\n  if (txn.date !== '') {\n    txns.push(txn);\n  }\n}\n\nasync function extractTransactions(page: Page | Frame, tableLocator: string, transactionStatus: TransactionStatuses) {\n  const txns: ScrapedTransaction[] = [];\n  const transactionsColsTypes = await getTransactionsColsTypeClasses(page, tableLocator);\n\n  const transactionsRows = await pageEvalAll<TransactionsTr[]>(page, `${tableLocator} tbody tr`, [], trs => {\n    return trs.map(tr => ({\n      innerTds: Array.from(tr.getElementsByTagName('td')).map(td => td.innerText),\n    }));\n  });\n\n  for (const txnRow of transactionsRows) {\n    extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes);\n  }\n  return txns;\n}\n\nasync function isNoTransactionInDateRangeError(page: Page | Frame) {\n  const hasErrorInfoElement = await elementPresentOnPage(page, `.${ERROR_MESSAGE_CLASS}`);\n  if (hasErrorInfoElement) {\n    const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, errorElement => {\n      return (errorElement as HTMLElement).innerText;\n    });\n    return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;\n  }\n  return false;\n}\n\nasync function searchByDates(page: Page | Frame, startDate: Moment) {\n  await clickButton(page, 'a#tabHeader4');\n  await waitUntilElementFound(page, 'div#fibi_dates');\n  await fillInput(page, 'input#fromDate', startDate.format(DATE_FORMAT));\n  await clickButton(page, `button[class*=${CLOSE_SEARCH_BY_DATES_BUTTON_CLASS}]`);\n  await clickButton(page, `input[value=${SHOW_SEARCH_BY_DATES_BUTTON_VALUE}]`);\n  await waitForNavigation(page);\n}\n\nasync function getAccountNumber(page: Page | Frame) {\n  const selectedSnifAccount = await page.$eval(ACCOUNTS_NUMBER, option => {\n    return (option as HTMLElement).innerText;\n  });\n\n  return selectedSnifAccount.replace('/', '_').trim();\n}\n\nasync function checkIfHasNextPage(page: Page | Frame) {\n  return elementPresentOnPage(page, NEXT_PAGE_LINK);\n}\n\nasync function navigateToNextPage(page: Page | Frame) {\n  await clickButton(page, NEXT_PAGE_LINK);\n  await waitForNavigation(page);\n}\n\n/* Couldn't reproduce scenario with multiple pages of pending transactions - Should support if exists such case.\n   needToPaginate is false if scraping pending transactions */\nasync function scrapeTransactions(\n  page: Page | Frame,\n  tableLocator: string,\n  transactionStatus: TransactionStatuses,\n  needToPaginate: boolean,\n) {\n  const txns = [];\n  let hasNextPage = false;\n\n  do {\n    const currentPageTxns = await extractTransactions(page, tableLocator, transactionStatus);\n    txns.push(...currentPageTxns);\n    if (needToPaginate) {\n      hasNextPage = await checkIfHasNextPage(page);\n      if (hasNextPage) {\n        await navigateToNextPage(page);\n      }\n    }\n  } while (hasNextPage);\n\n  return convertTransactions(txns);\n}\n\nasync function getAccountTransactions(page: Page | Frame) {\n  await Promise.race([\n    waitUntilElementFound(page, \"div[id*='divTable']\", false),\n    waitUntilElementFound(page, `.${ERROR_MESSAGE_CLASS}`, false),\n  ]);\n\n  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);\n  if (noTransactionInRangeError) {\n    return [];\n  }\n\n  const pendingTxns = await scrapeTransactions(page, PENDING_TRANSACTIONS_TABLE, TransactionStatuses.Pending, false);\n  const completedTxns = await scrapeTransactions(\n    page,\n    COMPLETED_TRANSACTIONS_TABLE,\n    TransactionStatuses.Completed,\n    true,\n  );\n  const txns = [...pendingTxns, ...completedTxns];\n  return txns;\n}\n\nasync function getCurrentBalance(page: Page | Frame) {\n  const balanceElement = await page.$(CURRENT_BALANCE);\n  if (!balanceElement) {\n    return undefined;\n  }\n  const balanceStr = await balanceElement.evaluate(option => {\n    return (option as HTMLElement).innerText;\n  });\n  return getAmountData(balanceStr);\n}\n\nexport async function waitForPostLogin(page: Page) {\n  return Promise.race([\n    waitUntilElementFound(page, '#card-header', true), // New UI\n    waitUntilElementFound(page, '#account_num', true), // New UI\n    waitUntilElementFound(page, '#matafLogoutLink', true), // Old UI\n    waitUntilElementFound(page, '#validationMsg', true), // Old UI\n  ]);\n}\n\nasync function fetchAccountData(page: Page | Frame, startDate: Moment) {\n  await searchByDates(page, startDate);\n  const accountNumber = await getAccountNumber(page);\n  const balance = await getCurrentBalance(page);\n  const txns = await getAccountTransactions(page);\n\n  return {\n    accountNumber,\n    txns,\n    balance,\n  };\n}\n\nasync function getAccountIdsBySelector(page: Page): Promise<string[]> {\n  const accountsIds = await page.evaluate(() => {\n    const selectElement = document.getElementById('account_num_select');\n    const options = selectElement ? selectElement.querySelectorAll('option') : [];\n    if (!options) return [];\n    return Array.from(options, option => option.value);\n  });\n  return accountsIds;\n}\n\nasync function getTransactionsFrame(page: Page): Promise<Frame | null> {\n  // Try a few times to find the iframe, as it might not be immediately available\n  for (let attempt = 0; attempt < 3; attempt++) {\n    await sleep(2000);\n    const frames = page.frames();\n    const targetFrame = frames.find(f => f.name() === IFRAME_NAME);\n\n    if (targetFrame) {\n      return targetFrame;\n    }\n  }\n\n  return null;\n}\n\nasync function selectAccount(page: Page, accountId: string) {\n  await page.select('#account_num_select', accountId);\n  await waitUntilElementFound(page, '#account_num_select', true);\n}\n\nasync function fetchAccountDataBothUIs(page: Page, startDate: Moment) {\n  // Try to get the iframe for the new UI\n  const frame = await getTransactionsFrame(page);\n\n  // Use the frame if available (new UI), otherwise use the page directly (old UI)\n  const targetPage = frame || page;\n  return fetchAccountData(targetPage, startDate);\n}\n\nasync function fetchAccounts(page: Page, startDate: Moment): Promise<TransactionsAccount[]> {\n  const accountsIds = await getAccountIdsBySelector(page);\n\n  if (accountsIds.length <= 1) {\n    const accountData = await fetchAccountDataBothUIs(page, startDate);\n    return [accountData];\n  }\n\n  const accounts: TransactionsAccount[] = [];\n  for (const accountId of accountsIds) {\n    await selectAccount(page, accountId);\n\n    const accountData = await fetchAccountDataBothUIs(page, startDate);\n    accounts.push(accountData);\n  }\n\n  return accounts;\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass BeinleumiGroupBaseScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  BASE_URL = '';\n\n  LOGIN_URL = '';\n\n  TRANSACTIONS_URL = '';\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n      // HACK: For some reason, though the login button (#continueBtn) is present and visible, the click action does not perform.\n      // Adding this delay fixes the issue.\n      preAction: async () => {\n        await sleep(1000);\n      },\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startMomentLimit = moment({ year: 1600 });\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(startMomentLimit, moment(startDate));\n\n    await this.navigateTo(this.TRANSACTIONS_URL);\n\n    const accounts = await fetchAccounts(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default BeinleumiGroupBaseScraper;\n"],"mappings":"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","ignoreList":[]}