israeli-bank-scrapers 6.1.0 → 6.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (79) hide show
  1. package/lib/assertNever.js +5 -7
  2. package/lib/constants.js +13 -16
  3. package/lib/definitions.js +109 -113
  4. package/lib/helpers/browser.js +9 -16
  5. package/lib/helpers/dates.js +18 -19
  6. package/lib/helpers/debug.js +9 -9
  7. package/lib/helpers/elements-interactions.js +78 -84
  8. package/lib/helpers/fetch.js +82 -89
  9. package/lib/helpers/navigation.js +24 -31
  10. package/lib/helpers/storage.js +10 -12
  11. package/lib/helpers/transactions.js +33 -35
  12. package/lib/helpers/waiting.js +45 -44
  13. package/lib/index.js +15 -82
  14. package/lib/scrapers/amex.js +11 -13
  15. package/lib/scrapers/base-beinleumi-group.js +233 -252
  16. package/lib/scrapers/base-isracard-amex.js +273 -286
  17. package/lib/scrapers/base-scraper-with-browser.js +240 -274
  18. package/lib/scrapers/base-scraper.js +82 -86
  19. package/lib/scrapers/behatsdaa.js +98 -107
  20. package/lib/scrapers/beinleumi.js +11 -20
  21. package/lib/scrapers/beyahad-bishvilha.js +132 -138
  22. package/lib/scrapers/discount.js +97 -103
  23. package/lib/scrapers/errors.js +22 -25
  24. package/lib/scrapers/factory.js +66 -67
  25. package/lib/scrapers/hapoalim.js +162 -182
  26. package/lib/scrapers/interface.js +2 -5
  27. package/lib/scrapers/isracard.js +11 -13
  28. package/lib/scrapers/leumi.js +167 -176
  29. package/lib/scrapers/massad.js +11 -20
  30. package/lib/scrapers/max.js +256 -268
  31. package/lib/scrapers/mercantile.js +14 -20
  32. package/lib/scrapers/mizrahi.js +158 -159
  33. package/lib/scrapers/one-zero-queries.js +4 -7
  34. package/lib/scrapers/one-zero.js +176 -240
  35. package/lib/scrapers/otsar-hahayal.js +11 -20
  36. package/lib/scrapers/pagi.js +11 -20
  37. package/lib/scrapers/union-bank.js +172 -179
  38. package/lib/scrapers/visa-cal.js +254 -263
  39. package/lib/scrapers/yahav.js +190 -211
  40. package/lib/transactions.js +13 -16
  41. package/package.json +12 -14
  42. package/lib/scrapers/amex.test.d.ts +0 -1
  43. package/lib/scrapers/amex.test.js +0 -54
  44. package/lib/scrapers/base-scraper-with-browser.test.d.ts +0 -1
  45. package/lib/scrapers/base-scraper-with-browser.test.js +0 -58
  46. package/lib/scrapers/behatsdaa.test.d.ts +0 -1
  47. package/lib/scrapers/behatsdaa.test.js +0 -50
  48. package/lib/scrapers/beinleumi.test.d.ts +0 -1
  49. package/lib/scrapers/beinleumi.test.js +0 -52
  50. package/lib/scrapers/beyahad-bishvilha.test.d.ts +0 -1
  51. package/lib/scrapers/beyahad-bishvilha.test.js +0 -52
  52. package/lib/scrapers/discount.test.d.ts +0 -1
  53. package/lib/scrapers/discount.test.js +0 -54
  54. package/lib/scrapers/factory.test.d.ts +0 -1
  55. package/lib/scrapers/factory.test.js +0 -19
  56. package/lib/scrapers/hapoalim.test.d.ts +0 -1
  57. package/lib/scrapers/hapoalim.test.js +0 -52
  58. package/lib/scrapers/isracard.test.d.ts +0 -1
  59. package/lib/scrapers/isracard.test.js +0 -54
  60. package/lib/scrapers/leumi.test.d.ts +0 -1
  61. package/lib/scrapers/leumi.test.js +0 -52
  62. package/lib/scrapers/max.test.d.ts +0 -1
  63. package/lib/scrapers/max.test.js +0 -71
  64. package/lib/scrapers/mercantile.test.d.ts +0 -1
  65. package/lib/scrapers/mercantile.test.js +0 -50
  66. package/lib/scrapers/mizrahi.test.d.ts +0 -1
  67. package/lib/scrapers/mizrahi.test.js +0 -58
  68. package/lib/scrapers/one-zero.test.d.ts +0 -1
  69. package/lib/scrapers/one-zero.test.js +0 -56
  70. package/lib/scrapers/otsar-hahayal.test.d.ts +0 -1
  71. package/lib/scrapers/otsar-hahayal.test.js +0 -52
  72. package/lib/scrapers/pagi.test.d.ts +0 -1
  73. package/lib/scrapers/pagi.test.js +0 -52
  74. package/lib/scrapers/union-bank.test.d.ts +0 -1
  75. package/lib/scrapers/union-bank.test.js +0 -52
  76. package/lib/scrapers/visa-cal.test.d.ts +0 -1
  77. package/lib/scrapers/visa-cal.test.js +0 -54
  78. package/lib/scrapers/yahav.test.d.ts +0 -1
  79. package/lib/scrapers/yahav.test.js +0 -54
@@ -1,22 +1,16 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.default = void 0;
7
- require("core-js/modules/es.array.iterator.js");
8
- require("core-js/modules/es.promise.js");
9
- require("core-js/modules/es.regexp.exec.js");
10
- require("core-js/modules/es.string.replace.js");
11
- var _moment = _interopRequireDefault(require("moment"));
12
- var _constants = require("../constants");
13
- var _debug = require("../helpers/debug");
14
- var _elementsInteractions = require("../helpers/elements-interactions");
15
- var _navigation = require("../helpers/navigation");
16
- var _transactions = require("../transactions");
17
- var _baseScraperWithBrowser = require("./base-scraper-with-browser");
18
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
19
- const debug = (0, _debug.getDebug)('leumi');
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
+ const moment_1 = __importDefault(require("moment"));
7
+ const constants_1 = require("../constants");
8
+ const debug_1 = require("../helpers/debug");
9
+ const elements_interactions_1 = require("../helpers/elements-interactions");
10
+ const navigation_1 = require("../helpers/navigation");
11
+ const transactions_1 = require("../transactions");
12
+ const base_scraper_with_browser_1 = require("./base-scraper-with-browser");
13
+ const debug = (0, debug_1.getDebug)('leumi');
20
14
  const BASE_URL = 'https://hb2.bankleumi.co.il';
21
15
  const LOGIN_URL = 'https://www.leumi.co.il/';
22
16
  const TRANSACTIONS_URL = `${BASE_URL}/eBanking/SO/SPA.aspx#/ts/BusinessAccountTrx?WidgetPar=1`;
@@ -25,182 +19,179 @@ const DATE_FORMAT = 'DD.MM.YY';
25
19
  const ACCOUNT_BLOCKED_MSG = 'המנוי חסום';
26
20
  const INVALID_PASSWORD_MSG = 'אחד או יותר מפרטי ההזדהות שמסרת שגויים. ניתן לנסות שוב';
27
21
  function getPossibleLoginResults() {
28
- const urls = {
29
- [_baseScraperWithBrowser.LoginResults.Success]: [/ebanking\/SO\/SPA.aspx/i],
30
- [_baseScraperWithBrowser.LoginResults.InvalidPassword]: [async options => {
31
- if (!options || !options.page) {
32
- throw new Error('missing page options argument');
33
- }
34
- const errorMessage = await (0, _elementsInteractions.pageEvalAll)(options.page, 'svg#Capa_1', '', element => {
35
- var _element$;
36
- return (_element$ = element[0]) === null || _element$ === void 0 || (_element$ = _element$.parentElement) === null || _element$ === void 0 || (_element$ = _element$.children[1]) === null || _element$ === void 0 ? void 0 : _element$.innerText;
37
- });
38
- return errorMessage === null || errorMessage === void 0 ? void 0 : errorMessage.startsWith(INVALID_PASSWORD_MSG);
39
- }],
40
- [_baseScraperWithBrowser.LoginResults.AccountBlocked]: [
41
- // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep
42
- async options => {
43
- if (!options || !options.page) {
44
- throw new Error('missing page options argument');
45
- }
46
- const errorMessage = await (0, _elementsInteractions.pageEvalAll)(options.page, '.errHeader', '', label => {
47
- var _label$;
48
- return (_label$ = label[0]) === null || _label$ === void 0 ? void 0 : _label$.innerText;
49
- });
50
- return errorMessage === null || errorMessage === void 0 ? void 0 : errorMessage.startsWith(ACCOUNT_BLOCKED_MSG);
51
- }],
52
- [_baseScraperWithBrowser.LoginResults.ChangePassword]: ['https://hb2.bankleumi.co.il/authenticate'] // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep
53
- };
54
- return urls;
22
+ const urls = {
23
+ [base_scraper_with_browser_1.LoginResults.Success]: [/ebanking\/SO\/SPA.aspx/i],
24
+ [base_scraper_with_browser_1.LoginResults.InvalidPassword]: [
25
+ async (options) => {
26
+ if (!options || !options.page) {
27
+ throw new Error('missing page options argument');
28
+ }
29
+ const errorMessage = await (0, elements_interactions_1.pageEvalAll)(options.page, 'svg#Capa_1', '', element => {
30
+ return element[0]?.parentElement?.children[1]?.innerText;
31
+ });
32
+ return errorMessage?.startsWith(INVALID_PASSWORD_MSG);
33
+ },
34
+ ],
35
+ [base_scraper_with_browser_1.LoginResults.AccountBlocked]: [
36
+ // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep
37
+ async (options) => {
38
+ if (!options || !options.page) {
39
+ throw new Error('missing page options argument');
40
+ }
41
+ const errorMessage = await (0, elements_interactions_1.pageEvalAll)(options.page, '.errHeader', '', label => {
42
+ return label[0]?.innerText;
43
+ });
44
+ return errorMessage?.startsWith(ACCOUNT_BLOCKED_MSG);
45
+ },
46
+ ],
47
+ [base_scraper_with_browser_1.LoginResults.ChangePassword]: ['https://hb2.bankleumi.co.il/authenticate'], // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep
48
+ };
49
+ return urls;
55
50
  }
56
51
  function createLoginFields(credentials) {
57
- return [{
58
- selector: 'input[placeholder="שם משתמש"]',
59
- value: credentials.username
60
- }, {
61
- selector: 'input[placeholder="סיסמה"]',
62
- value: credentials.password
63
- }];
52
+ return [
53
+ { selector: 'input[placeholder="שם משתמש"]', value: credentials.username },
54
+ { selector: 'input[placeholder="סיסמה"]', value: credentials.password },
55
+ ];
64
56
  }
65
57
  function extractTransactionsFromPage(transactions, status) {
66
- if (transactions === null || transactions.length === 0) {
67
- return [];
68
- }
69
- const result = transactions.map(rawTransaction => {
70
- const date = (0, _moment.default)(rawTransaction.DateUTC).milliseconds(0).toISOString();
71
- const newTransaction = {
72
- status,
73
- type: _transactions.TransactionTypes.Normal,
74
- date,
75
- processedDate: date,
76
- description: rawTransaction.Description || '',
77
- identifier: rawTransaction.ReferenceNumberLong,
78
- memo: rawTransaction.AdditionalData || '',
79
- originalCurrency: _constants.SHEKEL_CURRENCY,
80
- chargedAmount: rawTransaction.Amount,
81
- originalAmount: rawTransaction.Amount
82
- };
83
- return newTransaction;
84
- });
85
- return result;
58
+ if (transactions === null || transactions.length === 0) {
59
+ return [];
60
+ }
61
+ const result = transactions.map(rawTransaction => {
62
+ const date = (0, moment_1.default)(rawTransaction.DateUTC).milliseconds(0).toISOString();
63
+ const newTransaction = {
64
+ status,
65
+ type: transactions_1.TransactionTypes.Normal,
66
+ date,
67
+ processedDate: date,
68
+ description: rawTransaction.Description || '',
69
+ identifier: rawTransaction.ReferenceNumberLong,
70
+ memo: rawTransaction.AdditionalData || '',
71
+ originalCurrency: constants_1.SHEKEL_CURRENCY,
72
+ chargedAmount: rawTransaction.Amount,
73
+ originalAmount: rawTransaction.Amount,
74
+ };
75
+ return newTransaction;
76
+ });
77
+ return result;
86
78
  }
87
79
  function hangProcess(timeout) {
88
- return new Promise(resolve => {
89
- setTimeout(() => {
90
- resolve();
91
- }, timeout);
92
- });
80
+ return new Promise(resolve => {
81
+ setTimeout(() => {
82
+ resolve();
83
+ }, timeout);
84
+ });
93
85
  }
94
86
  async function clickByXPath(page, xpath) {
95
- await page.waitForSelector(xpath, {
96
- timeout: 30000,
97
- visible: true
98
- });
99
- const elm = await page.$$(xpath);
100
- await elm[0].click();
87
+ await page.waitForSelector(xpath, { timeout: 30000, visible: true });
88
+ const elm = await page.$$(xpath);
89
+ await elm[0].click();
101
90
  }
102
91
  function removeSpecialCharacters(str) {
103
- return str.replace(/[^0-9/-]/g, '');
92
+ return str.replace(/[^0-9/-]/g, '');
104
93
  }
105
94
  async function fetchTransactionsForAccount(page, startDate, accountId) {
106
- // DEVELOPER NOTICE the account number received from the server is being altered at
107
- // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
108
- await hangProcess(4000);
109
- await (0, _elementsInteractions.waitUntilElementFound)(page, 'button[title="חיפוש מתקדם"]', true);
110
- await (0, _elementsInteractions.clickButton)(page, 'button[title="חיפוש מתקדם"]');
111
- await (0, _elementsInteractions.waitUntilElementFound)(page, 'bll-radio-button', true);
112
- await (0, _elementsInteractions.clickButton)(page, 'bll-radio-button:not([checked])');
113
- await (0, _elementsInteractions.waitUntilElementFound)(page, 'input[formcontrolname="txtInputFrom"]', true);
114
- await (0, _elementsInteractions.fillInput)(page, 'input[formcontrolname="txtInputFrom"]', startDate.format(DATE_FORMAT));
115
-
116
- // we must blur the from control otherwise the search will use the previous value
117
- await page.focus("button[aria-label='סנן']");
118
- await (0, _elementsInteractions.clickButton)(page, "button[aria-label='סנן']");
119
- const finalResponse = await page.waitForResponse(response => {
120
- return response.url() === FILTERED_TRANSACTIONS_URL && response.request().method() === 'POST';
121
- });
122
- const responseJson = await finalResponse.json();
123
- const accountNumber = accountId.replace('/', '_').replace(/[^\d-_]/g, '');
124
- const response = JSON.parse(responseJson.jsonResp);
125
- const pendingTransactions = response.TodayTransactionsItems;
126
- const transactions = response.HistoryTransactionsItems;
127
- const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;
128
- const pendingTxns = extractTransactionsFromPage(pendingTransactions, _transactions.TransactionStatuses.Pending);
129
- const completedTxns = extractTransactionsFromPage(transactions, _transactions.TransactionStatuses.Completed);
130
- const txns = [...pendingTxns, ...completedTxns];
131
- return {
132
- accountNumber,
133
- balance,
134
- txns
135
- };
95
+ // DEVELOPER NOTICE the account number received from the server is being altered at
96
+ // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
97
+ await hangProcess(4000);
98
+ await (0, elements_interactions_1.waitUntilElementFound)(page, 'button[title="חיפוש מתקדם"]', true);
99
+ await (0, elements_interactions_1.clickButton)(page, 'button[title="חיפוש מתקדם"]');
100
+ await (0, elements_interactions_1.waitUntilElementFound)(page, 'bll-radio-button', true);
101
+ await (0, elements_interactions_1.clickButton)(page, 'bll-radio-button:not([checked])');
102
+ await (0, elements_interactions_1.waitUntilElementFound)(page, 'input[formcontrolname="txtInputFrom"]', true);
103
+ await (0, elements_interactions_1.fillInput)(page, 'input[formcontrolname="txtInputFrom"]', startDate.format(DATE_FORMAT));
104
+ // we must blur the from control otherwise the search will use the previous value
105
+ await page.focus("button[aria-label='סנן']");
106
+ await (0, elements_interactions_1.clickButton)(page, "button[aria-label='סנן']");
107
+ const finalResponse = await page.waitForResponse(response => {
108
+ return response.url() === FILTERED_TRANSACTIONS_URL && response.request().method() === 'POST';
109
+ });
110
+ const responseJson = await finalResponse.json();
111
+ const accountNumber = accountId.replace('/', '_').replace(/[^\d-_]/g, '');
112
+ const response = JSON.parse(responseJson.jsonResp);
113
+ const pendingTransactions = response.TodayTransactionsItems;
114
+ const transactions = response.HistoryTransactionsItems;
115
+ const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;
116
+ const pendingTxns = extractTransactionsFromPage(pendingTransactions, transactions_1.TransactionStatuses.Pending);
117
+ const completedTxns = extractTransactionsFromPage(transactions, transactions_1.TransactionStatuses.Completed);
118
+ const txns = [...pendingTxns, ...completedTxns];
119
+ return {
120
+ accountNumber,
121
+ balance,
122
+ txns,
123
+ };
136
124
  }
137
125
  async function fetchTransactions(page, startDate) {
138
- const accounts = [];
139
-
140
- // DEVELOPER NOTICE the account number received from the server is being altered at
141
- // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
142
- await hangProcess(4000);
143
- const accountsIds = await page.evaluate(() => Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), e => e.textContent));
144
-
145
- // due to a bug, the altered value might include undesired signs like & that should be removed
146
-
147
- if (!accountsIds.length) {
148
- throw new Error('Failed to extract or parse the account number');
149
- }
150
- for (const accountId of accountsIds) {
151
- if (accountsIds.length > 1) {
152
- // get list of accounts and check accountId
153
- await clickByXPath(page, 'xpath///*[contains(@class, "number") and contains(@class, "combo-inner")]');
154
- await clickByXPath(page, `xpath///span[contains(text(), '${accountId}')]`);
126
+ const accounts = [];
127
+ // DEVELOPER NOTICE the account number received from the server is being altered at
128
+ // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
129
+ await hangProcess(4000);
130
+ const accountsIds = (await page.evaluate(() => Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), e => e.textContent)));
131
+ // due to a bug, the altered value might include undesired signs like & that should be removed
132
+ if (!accountsIds.length) {
133
+ throw new Error('Failed to extract or parse the account number');
134
+ }
135
+ for (const accountId of accountsIds) {
136
+ if (accountsIds.length > 1) {
137
+ // get list of accounts and check accountId
138
+ await clickByXPath(page, 'xpath///*[contains(@class, "number") and contains(@class, "combo-inner")]');
139
+ await clickByXPath(page, `xpath///span[contains(text(), '${accountId}')]`);
140
+ }
141
+ accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));
155
142
  }
156
- accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));
157
- }
158
- return accounts;
143
+ return accounts;
159
144
  }
160
145
  async function navigateToLogin(page) {
161
- const loginButtonSelector = '.enter-account a[originaltitle="כניסה לחשבונך"]';
162
- debug('wait for homepage to click on login button');
163
- await (0, _elementsInteractions.waitUntilElementFound)(page, loginButtonSelector);
164
- debug('navigate to login page');
165
- const loginUrl = await (0, _elementsInteractions.pageEval)(page, loginButtonSelector, null, element => {
166
- return element.href;
167
- });
168
- debug(`navigating to page (${loginUrl})`);
169
- await page.goto(loginUrl);
170
- debug('waiting for page to be loaded (networkidle2)');
171
- await (0, _navigation.waitForNavigation)(page, {
172
- waitUntil: 'networkidle2'
173
- });
174
- debug('waiting for components of login to enter credentials');
175
- await Promise.all([(0, _elementsInteractions.waitUntilElementFound)(page, 'input[placeholder="שם משתמש"]', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'input[placeholder="סיסמה"]', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'button[type="submit"]', true)]);
146
+ const loginButtonSelector = '.enter-account a[originaltitle="כניסה לחשבונך"]';
147
+ debug('wait for homepage to click on login button');
148
+ await (0, elements_interactions_1.waitUntilElementFound)(page, loginButtonSelector);
149
+ debug('navigate to login page');
150
+ const loginUrl = await (0, elements_interactions_1.pageEval)(page, loginButtonSelector, null, element => {
151
+ return element.href;
152
+ });
153
+ debug(`navigating to page (${loginUrl})`);
154
+ await page.goto(loginUrl);
155
+ debug('waiting for page to be loaded (networkidle2)');
156
+ await (0, navigation_1.waitForNavigation)(page, { waitUntil: 'networkidle2' });
157
+ debug('waiting for components of login to enter credentials');
158
+ await Promise.all([
159
+ (0, elements_interactions_1.waitUntilElementFound)(page, 'input[placeholder="שם משתמש"]', true),
160
+ (0, elements_interactions_1.waitUntilElementFound)(page, 'input[placeholder="סיסמה"]', true),
161
+ (0, elements_interactions_1.waitUntilElementFound)(page, 'button[type="submit"]', true),
162
+ ]);
176
163
  }
177
164
  async function waitForPostLogin(page) {
178
- await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, 'a[title="דלג לחשבון"]', true, 60000), (0, _elementsInteractions.waitUntilElementFound)(page, 'div.main-content', false, 60000), page.waitForSelector(`xpath//div[contains(string(),"${INVALID_PASSWORD_MSG}")]`), (0, _elementsInteractions.waitUntilElementFound)(page, 'form[action="/changepassword"]', true, 60000) // not sure if they kept this one
179
- ]);
165
+ await Promise.race([
166
+ (0, elements_interactions_1.waitUntilElementFound)(page, 'a[title="דלג לחשבון"]', true, 60000),
167
+ (0, elements_interactions_1.waitUntilElementFound)(page, 'div.main-content', false, 60000),
168
+ page.waitForSelector(`xpath//div[contains(string(),"${INVALID_PASSWORD_MSG}")]`),
169
+ (0, elements_interactions_1.waitUntilElementFound)(page, 'form[action="/changepassword"]', true, 60000), // not sure if they kept this one
170
+ ]);
180
171
  }
181
- class LeumiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
182
- getLoginOptions(credentials) {
183
- return {
184
- loginUrl: LOGIN_URL,
185
- fields: createLoginFields(credentials),
186
- submitButtonSelector: "button[type='submit']",
187
- checkReadiness: async () => navigateToLogin(this.page),
188
- postAction: async () => waitForPostLogin(this.page),
189
- possibleResults: getPossibleLoginResults()
190
- };
191
- }
192
- async fetchData() {
193
- const minimumStartMoment = (0, _moment.default)().subtract(3, 'years').add(1, 'day');
194
- const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
195
- const startDate = this.options.startDate || defaultStartMoment.toDate();
196
- const startMoment = _moment.default.max(minimumStartMoment, (0, _moment.default)(startDate));
197
- await this.navigateTo(TRANSACTIONS_URL);
198
- const accounts = await fetchTransactions(this.page, startMoment);
199
- return {
200
- success: true,
201
- accounts
202
- };
203
- }
172
+ class LeumiScraper extends base_scraper_with_browser_1.BaseScraperWithBrowser {
173
+ getLoginOptions(credentials) {
174
+ return {
175
+ loginUrl: LOGIN_URL,
176
+ fields: createLoginFields(credentials),
177
+ submitButtonSelector: "button[type='submit']",
178
+ checkReadiness: async () => navigateToLogin(this.page),
179
+ postAction: async () => waitForPostLogin(this.page),
180
+ possibleResults: getPossibleLoginResults(),
181
+ };
182
+ }
183
+ async fetchData() {
184
+ const minimumStartMoment = (0, moment_1.default)().subtract(3, 'years').add(1, 'day');
185
+ const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years').add(1, 'day');
186
+ const startDate = this.options.startDate || defaultStartMoment.toDate();
187
+ const startMoment = moment_1.default.max(minimumStartMoment, (0, moment_1.default)(startDate));
188
+ await this.navigateTo(TRANSACTIONS_URL);
189
+ const accounts = await fetchTransactions(this.page, startMoment);
190
+ return {
191
+ success: true,
192
+ accounts,
193
+ };
194
+ }
204
195
  }
205
- var _default = exports.default = LeumiScraper;
206
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_debug","_elementsInteractions","_navigation","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","BASE_URL","LOGIN_URL","TRANSACTIONS_URL","FILTERED_TRANSACTIONS_URL","DATE_FORMAT","ACCOUNT_BLOCKED_MSG","INVALID_PASSWORD_MSG","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","page","Error","errorMessage","pageEvalAll","element","_element$","parentElement","children","innerText","startsWith","AccountBlocked","label","_label$","ChangePassword","createLoginFields","credentials","selector","value","username","password","extractTransactionsFromPage","transactions","status","length","result","map","rawTransaction","date","moment","DateUTC","milliseconds","toISOString","newTransaction","type","TransactionTypes","Normal","processedDate","description","Description","identifier","ReferenceNumberLong","memo","AdditionalData","originalCurrency","SHEKEL_CURRENCY","chargedAmount","Amount","originalAmount","hangProcess","timeout","Promise","resolve","setTimeout","clickByXPath","xpath","waitForSelector","visible","elm","$$","click","removeSpecialCharacters","str","replace","fetchTransactionsForAccount","startDate","accountId","waitUntilElementFound","clickButton","fillInput","format","focus","finalResponse","waitForResponse","response","url","request","method","responseJson","json","accountNumber","JSON","parse","jsonResp","pendingTransactions","TodayTransactionsItems","HistoryTransactionsItems","balance","BalanceDisplay","parseFloat","undefined","pendingTxns","TransactionStatuses","Pending","completedTxns","Completed","txns","fetchTransactions","accounts","accountsIds","evaluate","Array","from","document","querySelectorAll","textContent","push","navigateToLogin","loginButtonSelector","loginUrl","pageEval","href","goto","waitForNavigation","waitUntil","all","waitForPostLogin","race","LeumiScraper","BaseScraperWithBrowser","getLoginOptions","fields","submitButtonSelector","checkReadiness","postAction","possibleResults","fetchData","minimumStartMoment","subtract","add","defaultStartMoment","toDate","startMoment","max","navigateTo","success","_default","exports"],"sources":["../../src/scrapers/leumi.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport { getDebug } from '../helpers/debug';\nimport {\n  clickButton,\n  fillInput,\n  pageEval,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport {\n  TransactionStatuses, TransactionTypes,\n  type Transaction,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type LoginOptions } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\n\nconst debug = getDebug('leumi');\nconst BASE_URL = 'https://hb2.bankleumi.co.il';\nconst LOGIN_URL = 'https://www.leumi.co.il/';\nconst TRANSACTIONS_URL = `${BASE_URL}/eBanking/SO/SPA.aspx#/ts/BusinessAccountTrx?WidgetPar=1`;\nconst FILTERED_TRANSACTIONS_URL = `${BASE_URL}/ChannelWCF/Broker.svc/ProcessRequest?moduleName=UC_SO_27_GetBusinessAccountTrx`;\n\nconst DATE_FORMAT = 'DD.MM.YY';\nconst ACCOUNT_BLOCKED_MSG = 'המנוי חסום';\nconst INVALID_PASSWORD_MSG = 'אחד או יותר מפרטי ההזדהות שמסרת שגויים. ניתן לנסות שוב';\n\nfunction getPossibleLoginResults() {\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/ebanking\\/SO\\/SPA.aspx/i],\n    [LoginResults.InvalidPassword]: [\n      async (options) => {\n        if (!options || !options.page) {\n          throw new Error('missing page options argument');\n        }\n        const errorMessage = await pageEvalAll(options.page, 'svg#Capa_1', '', (element) => {\n          return (element[0]?.parentElement?.children[1] as HTMLDivElement)?.innerText;\n        });\n\n        return errorMessage?.startsWith(INVALID_PASSWORD_MSG);\n      },\n    ],\n    [LoginResults.AccountBlocked]: [ // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep\n      async (options) => {\n        if (!options || !options.page) {\n          throw new Error('missing page options argument');\n        }\n        const errorMessage = await pageEvalAll(options.page, '.errHeader', '', (label) => {\n          return (label[0] as HTMLElement)?.innerText;\n        });\n\n        return errorMessage?.startsWith(ACCOUNT_BLOCKED_MSG);\n      },\n    ],\n    [LoginResults.ChangePassword]: ['https://hb2.bankleumi.co.il/authenticate'], // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: 'input[placeholder=\"שם משתמש\"]', value: credentials.username },\n    { selector: 'input[placeholder=\"סיסמה\"]', value: credentials.password },\n  ];\n}\n\nfunction extractTransactionsFromPage(transactions: any[], status: TransactionStatuses): Transaction[] {\n  if (transactions === null || transactions.length === 0) {\n    return [];\n  }\n\n  const result: Transaction[] = transactions.map((rawTransaction) => {\n    const date = moment(rawTransaction.DateUTC).milliseconds(0).toISOString();\n    const newTransaction: Transaction = {\n      status,\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      description: rawTransaction.Description || '',\n      identifier: rawTransaction.ReferenceNumberLong,\n      memo: rawTransaction.AdditionalData || '',\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: rawTransaction.Amount,\n      originalAmount: rawTransaction.Amount,\n    };\n\n    return newTransaction;\n  });\n\n  return result;\n}\n\nfunction hangProcess(timeout: number) {\n  return new Promise<void>((resolve) => {\n    setTimeout(() => {\n      resolve();\n    }, timeout);\n  });\n}\n\nasync function clickByXPath(page: Page, xpath: string): Promise<void> {\n  await page.waitForSelector(xpath, { timeout: 30000, visible: true });\n  const elm = await page.$$(xpath);\n  await elm[0].click();\n}\n\nfunction removeSpecialCharacters(str: string): string {\n  return str.replace(/[^0-9/-]/g, '');\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment, accountId: string): Promise<TransactionsAccount> {\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  await waitUntilElementFound(page, 'button[title=\"חיפוש מתקדם\"]', true);\n  await clickButton(page, 'button[title=\"חיפוש מתקדם\"]');\n  await waitUntilElementFound(page, 'bll-radio-button', true);\n  await clickButton(page, 'bll-radio-button:not([checked])');\n\n  await waitUntilElementFound(page, 'input[formcontrolname=\"txtInputFrom\"]', true);\n\n  await fillInput(\n    page,\n    'input[formcontrolname=\"txtInputFrom\"]',\n    startDate.format(DATE_FORMAT),\n  );\n\n  // we must blur the from control otherwise the search will use the previous value\n  await page.focus(\"button[aria-label='סנן']\");\n\n  await clickButton(page, \"button[aria-label='סנן']\");\n  const finalResponse = await page.waitForResponse((response) => {\n    return response.url() === FILTERED_TRANSACTIONS_URL &&\n      response.request().method() === 'POST';\n  });\n\n  const responseJson: any = await finalResponse.json();\n\n  const accountNumber = accountId.replace('/', '_').replace(/[^\\d-_]/g, '');\n\n  const response = JSON.parse(responseJson.jsonResp);\n\n  const pendingTransactions = response.TodayTransactionsItems;\n  const transactions = response.HistoryTransactionsItems;\n  const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;\n\n  const pendingTxns = extractTransactionsFromPage(pendingTransactions, TransactionStatuses.Pending);\n  const completedTxns = extractTransactionsFromPage(transactions, TransactionStatuses.Completed);\n  const txns = [\n    ...pendingTxns,\n    ...completedTxns,\n  ];\n\n  return {\n    accountNumber,\n    balance,\n    txns,\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment): Promise<TransactionsAccount[]> {\n  const accounts: TransactionsAccount[] = [];\n\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  const accountsIds = await page.evaluate(() => Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), (e) => e.textContent)) as string[];\n\n  // due to a bug, the altered value might include undesired signs like & that should be removed\n\n  if (!accountsIds.length) {\n    throw new Error('Failed to extract or parse the account number');\n  }\n\n  for (const accountId of accountsIds) {\n    if (accountsIds.length > 1) {\n      // get list of accounts and check accountId\n      await clickByXPath(page, 'xpath///*[contains(@class, \"number\") and contains(@class, \"combo-inner\")]');\n      await clickByXPath(page, `xpath///span[contains(text(), '${accountId}')]`);\n    }\n\n    accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));\n  }\n\n  return accounts;\n}\n\nasync function navigateToLogin(page: Page): Promise<void> {\n  const loginButtonSelector = '.enter-account a[originaltitle=\"כניסה לחשבונך\"]';\n  debug('wait for homepage to click on login button');\n  await waitUntilElementFound(page, loginButtonSelector);\n  debug('navigate to login page');\n  const loginUrl = await pageEval(page, loginButtonSelector, null, (element) => {\n    return (element as any).href;\n  });\n  debug(`navigating to page (${loginUrl})`);\n  await page.goto(loginUrl);\n  debug('waiting for page to be loaded (networkidle2)');\n  await waitForNavigation(page, { waitUntil: 'networkidle2' });\n  debug('waiting for components of login to enter credentials');\n  await Promise.all([\n    waitUntilElementFound(page, 'input[placeholder=\"שם משתמש\"]', true),\n    waitUntilElementFound(page, 'input[placeholder=\"סיסמה\"]', true),\n    waitUntilElementFound(page, 'button[type=\"submit\"]', true),\n  ]);\n}\n\nasync function waitForPostLogin(page: Page): Promise<void> {\n  await Promise.race([\n    waitUntilElementFound(page, 'a[title=\"דלג לחשבון\"]', true, 60000),\n    waitUntilElementFound(page, 'div.main-content', false, 60000),\n    page.waitForSelector(`xpath//div[contains(string(),\"${INVALID_PASSWORD_MSG}\")]`),\n    waitUntilElementFound(page, 'form[action=\"/changepassword\"]', true, 60000), // not sure if they kept this one\n  ]);\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass LeumiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: \"button[type='submit']\",\n      checkReadiness: async () => navigateToLogin(this.page),\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const minimumStartMoment = moment().subtract(3, 'years').add(1, 'day');\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(minimumStartMoment, moment(startDate));\n\n    await this.navigateTo(TRANSACTIONS_URL);\n\n    const accounts = await fetchTransactions(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default LeumiScraper;\n"],"mappings":"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","ignoreList":[]}
196
+ exports.default = LeumiScraper;
197
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"leumi.js","sourceRoot":"","sources":["../../src/scrapers/leumi.ts"],"names":[],"mappings":"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","sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, fillInput, pageEval, pageEvalAll, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { TransactionStatuses, TransactionTypes, type Transaction, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type LoginOptions } from './base-scraper-with-browser';\nimport { type ScraperScrapingResult } from './interface';\n\nconst debug = getDebug('leumi');\nconst BASE_URL = 'https://hb2.bankleumi.co.il';\nconst LOGIN_URL = 'https://www.leumi.co.il/';\nconst TRANSACTIONS_URL = `${BASE_URL}/eBanking/SO/SPA.aspx#/ts/BusinessAccountTrx?WidgetPar=1`;\nconst FILTERED_TRANSACTIONS_URL = `${BASE_URL}/ChannelWCF/Broker.svc/ProcessRequest?moduleName=UC_SO_27_GetBusinessAccountTrx`;\n\nconst DATE_FORMAT = 'DD.MM.YY';\nconst ACCOUNT_BLOCKED_MSG = 'המנוי חסום';\nconst INVALID_PASSWORD_MSG = 'אחד או יותר מפרטי ההזדהות שמסרת שגויים. ניתן לנסות שוב';\n\nfunction getPossibleLoginResults() {\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/ebanking\\/SO\\/SPA.aspx/i],\n    [LoginResults.InvalidPassword]: [\n      async options => {\n        if (!options || !options.page) {\n          throw new Error('missing page options argument');\n        }\n        const errorMessage = await pageEvalAll(options.page, 'svg#Capa_1', '', element => {\n          return (element[0]?.parentElement?.children[1] as HTMLDivElement)?.innerText;\n        });\n\n        return errorMessage?.startsWith(INVALID_PASSWORD_MSG);\n      },\n    ],\n    [LoginResults.AccountBlocked]: [\n      // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep\n      async options => {\n        if (!options || !options.page) {\n          throw new Error('missing page options argument');\n        }\n        const errorMessage = await pageEvalAll(options.page, '.errHeader', '', label => {\n          return (label[0] as HTMLElement)?.innerText;\n        });\n\n        return errorMessage?.startsWith(ACCOUNT_BLOCKED_MSG);\n      },\n    ],\n    [LoginResults.ChangePassword]: ['https://hb2.bankleumi.co.il/authenticate'], // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: 'input[placeholder=\"שם משתמש\"]', value: credentials.username },\n    { selector: 'input[placeholder=\"סיסמה\"]', value: credentials.password },\n  ];\n}\n\nfunction extractTransactionsFromPage(transactions: any[], status: TransactionStatuses): Transaction[] {\n  if (transactions === null || transactions.length === 0) {\n    return [];\n  }\n\n  const result: Transaction[] = transactions.map(rawTransaction => {\n    const date = moment(rawTransaction.DateUTC).milliseconds(0).toISOString();\n    const newTransaction: Transaction = {\n      status,\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      description: rawTransaction.Description || '',\n      identifier: rawTransaction.ReferenceNumberLong,\n      memo: rawTransaction.AdditionalData || '',\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: rawTransaction.Amount,\n      originalAmount: rawTransaction.Amount,\n    };\n\n    return newTransaction;\n  });\n\n  return result;\n}\n\nfunction hangProcess(timeout: number) {\n  return new Promise<void>(resolve => {\n    setTimeout(() => {\n      resolve();\n    }, timeout);\n  });\n}\n\nasync function clickByXPath(page: Page, xpath: string): Promise<void> {\n  await page.waitForSelector(xpath, { timeout: 30000, visible: true });\n  const elm = await page.$$(xpath);\n  await elm[0].click();\n}\n\nfunction removeSpecialCharacters(str: string): string {\n  return str.replace(/[^0-9/-]/g, '');\n}\n\nasync function fetchTransactionsForAccount(\n  page: Page,\n  startDate: Moment,\n  accountId: string,\n): Promise<TransactionsAccount> {\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  await waitUntilElementFound(page, 'button[title=\"חיפוש מתקדם\"]', true);\n  await clickButton(page, 'button[title=\"חיפוש מתקדם\"]');\n  await waitUntilElementFound(page, 'bll-radio-button', true);\n  await clickButton(page, 'bll-radio-button:not([checked])');\n\n  await waitUntilElementFound(page, 'input[formcontrolname=\"txtInputFrom\"]', true);\n\n  await fillInput(page, 'input[formcontrolname=\"txtInputFrom\"]', startDate.format(DATE_FORMAT));\n\n  // we must blur the from control otherwise the search will use the previous value\n  await page.focus(\"button[aria-label='סנן']\");\n\n  await clickButton(page, \"button[aria-label='סנן']\");\n  const finalResponse = await page.waitForResponse(response => {\n    return response.url() === FILTERED_TRANSACTIONS_URL && response.request().method() === 'POST';\n  });\n\n  const responseJson: any = await finalResponse.json();\n\n  const accountNumber = accountId.replace('/', '_').replace(/[^\\d-_]/g, '');\n\n  const response = JSON.parse(responseJson.jsonResp);\n\n  const pendingTransactions = response.TodayTransactionsItems;\n  const transactions = response.HistoryTransactionsItems;\n  const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;\n\n  const pendingTxns = extractTransactionsFromPage(pendingTransactions, TransactionStatuses.Pending);\n  const completedTxns = extractTransactionsFromPage(transactions, TransactionStatuses.Completed);\n  const txns = [...pendingTxns, ...completedTxns];\n\n  return {\n    accountNumber,\n    balance,\n    txns,\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment): Promise<TransactionsAccount[]> {\n  const accounts: TransactionsAccount[] = [];\n\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  const accountsIds = (await page.evaluate(() =>\n    Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), e => e.textContent),\n  )) as string[];\n\n  // due to a bug, the altered value might include undesired signs like & that should be removed\n\n  if (!accountsIds.length) {\n    throw new Error('Failed to extract or parse the account number');\n  }\n\n  for (const accountId of accountsIds) {\n    if (accountsIds.length > 1) {\n      // get list of accounts and check accountId\n      await clickByXPath(page, 'xpath///*[contains(@class, \"number\") and contains(@class, \"combo-inner\")]');\n      await clickByXPath(page, `xpath///span[contains(text(), '${accountId}')]`);\n    }\n\n    accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));\n  }\n\n  return accounts;\n}\n\nasync function navigateToLogin(page: Page): Promise<void> {\n  const loginButtonSelector = '.enter-account a[originaltitle=\"כניסה לחשבונך\"]';\n  debug('wait for homepage to click on login button');\n  await waitUntilElementFound(page, loginButtonSelector);\n  debug('navigate to login page');\n  const loginUrl = await pageEval(page, loginButtonSelector, null, element => {\n    return (element as any).href;\n  });\n  debug(`navigating to page (${loginUrl})`);\n  await page.goto(loginUrl);\n  debug('waiting for page to be loaded (networkidle2)');\n  await waitForNavigation(page, { waitUntil: 'networkidle2' });\n  debug('waiting for components of login to enter credentials');\n  await Promise.all([\n    waitUntilElementFound(page, 'input[placeholder=\"שם משתמש\"]', true),\n    waitUntilElementFound(page, 'input[placeholder=\"סיסמה\"]', true),\n    waitUntilElementFound(page, 'button[type=\"submit\"]', true),\n  ]);\n}\n\nasync function waitForPostLogin(page: Page): Promise<void> {\n  await Promise.race([\n    waitUntilElementFound(page, 'a[title=\"דלג לחשבון\"]', true, 60000),\n    waitUntilElementFound(page, 'div.main-content', false, 60000),\n    page.waitForSelector(`xpath//div[contains(string(),\"${INVALID_PASSWORD_MSG}\")]`),\n    waitUntilElementFound(page, 'form[action=\"/changepassword\"]', true, 60000), // not sure if they kept this one\n  ]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass LeumiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: \"button[type='submit']\",\n      checkReadiness: async () => navigateToLogin(this.page),\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const minimumStartMoment = moment().subtract(3, 'years').add(1, 'day');\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(minimumStartMoment, moment(startDate));\n\n    await this.navigateTo(TRANSACTIONS_URL);\n\n    const accounts = await fetchTransactions(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default LeumiScraper;\n"]}
@@ -1,22 +1,13 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.default = void 0;
7
- require("core-js/modules/es.array.iterator.js");
8
- var _baseBeinleumiGroup = _interopRequireDefault(require("./base-beinleumi-group"));
9
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
10
- function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
11
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
12
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
13
- class MassadScraper extends _baseBeinleumiGroup.default {
14
- constructor(...args) {
15
- super(...args);
16
- _defineProperty(this, "BASE_URL", 'https://online.bankmassad.co.il');
17
- _defineProperty(this, "LOGIN_URL", `${this.BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=MASADPRTAL&site=Private&KODSAFA=HE`);
18
- _defineProperty(this, "TRANSACTIONS_URL", `${this.BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`);
19
- }
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
+ const base_beinleumi_group_1 = __importDefault(require("./base-beinleumi-group"));
7
+ class MassadScraper extends base_beinleumi_group_1.default {
8
+ BASE_URL = 'https://online.bankmassad.co.il';
9
+ LOGIN_URL = `${this.BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=MASADPRTAL&site=Private&KODSAFA=HE`;
10
+ TRANSACTIONS_URL = `${this.BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;
20
11
  }
21
- var _default = exports.default = MassadScraper;
22
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfYmFzZUJlaW5sZXVtaUdyb3VwIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsInJlcXVpcmUiLCJlIiwiX19lc01vZHVsZSIsImRlZmF1bHQiLCJfZGVmaW5lUHJvcGVydHkiLCJyIiwidCIsIl90b1Byb3BlcnR5S2V5IiwiT2JqZWN0IiwiZGVmaW5lUHJvcGVydHkiLCJ2YWx1ZSIsImVudW1lcmFibGUiLCJjb25maWd1cmFibGUiLCJ3cml0YWJsZSIsImkiLCJfdG9QcmltaXRpdmUiLCJTeW1ib2wiLCJ0b1ByaW1pdGl2ZSIsImNhbGwiLCJUeXBlRXJyb3IiLCJTdHJpbmciLCJOdW1iZXIiLCJNYXNzYWRTY3JhcGVyIiwiQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciIsImNvbnN0cnVjdG9yIiwiYXJncyIsIkJBU0VfVVJMIiwiX2RlZmF1bHQiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmFwZXJzL21hc3NhZC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciBmcm9tICcuL2Jhc2UtYmVpbmxldW1pLWdyb3VwJztcblxuY2xhc3MgTWFzc2FkU2NyYXBlciBleHRlbmRzIEJlaW5sZXVtaUdyb3VwQmFzZVNjcmFwZXIge1xuICBCQVNFX1VSTCA9ICdodHRwczovL29ubGluZS5iYW5rbWFzc2FkLmNvLmlsJztcblxuICBMT0dJTl9VUkwgPSBgJHt0aGlzLkJBU0VfVVJMfS9NYXRhZkxvZ2luU2VydmljZS9NYXRhZkxvZ2luU2VydmxldD9iYW5rSWQ9TUFTQURQUlRBTCZzaXRlPVByaXZhdGUmS09EU0FGQT1IRWA7XG5cbiAgVFJBTlNBQ1RJT05TX1VSTCA9IGAke3RoaXMuQkFTRV9VUkx9L3dwcy9teXBvcnRhbC9GaWJpTWVudS9PbmxpbmUvT25BY2NvdW50TW5nbWVudC9PbkJhbGFuY2VUcmFucy9Qcml2YXRlQWNjb3VudEZsb3dgO1xufVxuXG5leHBvcnQgZGVmYXVsdCBNYXNzYWRTY3JhcGVyO1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUEsSUFBQUEsbUJBQUEsR0FBQUMsc0JBQUEsQ0FBQUMsT0FBQTtBQUErRCxTQUFBRCx1QkFBQUUsQ0FBQSxXQUFBQSxDQUFBLElBQUFBLENBQUEsQ0FBQUMsVUFBQSxHQUFBRCxDQUFBLEtBQUFFLE9BQUEsRUFBQUYsQ0FBQTtBQUFBLFNBQUFHLGdCQUFBSCxDQUFBLEVBQUFJLENBQUEsRUFBQUMsQ0FBQSxZQUFBRCxDQUFBLEdBQUFFLGNBQUEsQ0FBQUYsQ0FBQSxNQUFBSixDQUFBLEdBQUFPLE1BQUEsQ0FBQUMsY0FBQSxDQUFBUixDQUFBLEVBQUFJLENBQUEsSUFBQUssS0FBQSxFQUFBSixDQUFBLEVBQUFLLFVBQUEsTUFBQUMsWUFBQSxNQUFBQyxRQUFBLFVBQUFaLENBQUEsQ0FBQUksQ0FBQSxJQUFBQyxDQUFBLEVBQUFMLENBQUE7QUFBQSxTQUFBTSxlQUFBRCxDQUFBLFFBQUFRLENBQUEsR0FBQUMsWUFBQSxDQUFBVCxDQUFBLHVDQUFBUSxDQUFBLEdBQUFBLENBQUEsR0FBQUEsQ0FBQTtBQUFBLFNBQUFDLGFBQUFULENBQUEsRUFBQUQsQ0FBQSwyQkFBQUMsQ0FBQSxLQUFBQSxDQUFBLFNBQUFBLENBQUEsTUFBQUwsQ0FBQSxHQUFBSyxDQUFBLENBQUFVLE1BQUEsQ0FBQUMsV0FBQSxrQkFBQWhCLENBQUEsUUFBQWEsQ0FBQSxHQUFBYixDQUFBLENBQUFpQixJQUFBLENBQUFaLENBQUEsRUFBQUQsQ0FBQSx1Q0FBQVMsQ0FBQSxTQUFBQSxDQUFBLFlBQUFLLFNBQUEseUVBQUFkLENBQUEsR0FBQWUsTUFBQSxHQUFBQyxNQUFBLEVBQUFmLENBQUE7QUFFL0QsTUFBTWdCLGFBQWEsU0FBU0MsMkJBQXlCLENBQUM7RUFBQUMsWUFBQSxHQUFBQyxJQUFBO0lBQUEsU0FBQUEsSUFBQTtJQUFBckIsZUFBQSxtQkFDekMsaUNBQWlDO0lBQUFBLGVBQUEsb0JBRWhDLEdBQUcsSUFBSSxDQUFDc0IsUUFBUSxnRkFBZ0Y7SUFBQXRCLGVBQUEsMkJBRXpGLEdBQUcsSUFBSSxDQUFDc0IsUUFBUSxrRkFBa0Y7RUFBQTtBQUN2SDtBQUFDLElBQUFDLFFBQUEsR0FBQUMsT0FBQSxDQUFBekIsT0FBQSxHQUVjbUIsYUFBYSIsImlnbm9yZUxpc3QiOltdfQ==
12
+ exports.default = MassadScraper;
13
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFzc2FkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmFwZXJzL21hc3NhZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLGtGQUErRDtBQUUvRCxNQUFNLGFBQWMsU0FBUSw4QkFBeUI7SUFDbkQsUUFBUSxHQUFHLGlDQUFpQyxDQUFDO0lBRTdDLFNBQVMsR0FBRyxHQUFHLElBQUksQ0FBQyxRQUFRLGdGQUFnRixDQUFDO0lBRTdHLGdCQUFnQixHQUFHLEdBQUcsSUFBSSxDQUFDLFFBQVEsa0ZBQWtGLENBQUM7Q0FDdkg7QUFFRCxrQkFBZSxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciBmcm9tICcuL2Jhc2UtYmVpbmxldW1pLWdyb3VwJztcblxuY2xhc3MgTWFzc2FkU2NyYXBlciBleHRlbmRzIEJlaW5sZXVtaUdyb3VwQmFzZVNjcmFwZXIge1xuICBCQVNFX1VSTCA9ICdodHRwczovL29ubGluZS5iYW5rbWFzc2FkLmNvLmlsJztcblxuICBMT0dJTl9VUkwgPSBgJHt0aGlzLkJBU0VfVVJMfS9NYXRhZkxvZ2luU2VydmljZS9NYXRhZkxvZ2luU2VydmxldD9iYW5rSWQ9TUFTQURQUlRBTCZzaXRlPVByaXZhdGUmS09EU0FGQT1IRWA7XG5cbiAgVFJBTlNBQ1RJT05TX1VSTCA9IGAke3RoaXMuQkFTRV9VUkx9L3dwcy9teXBvcnRhbC9GaWJpTWVudS9PbmxpbmUvT25BY2NvdW50TW5nbWVudC9PbkJhbGFuY2VUcmFucy9Qcml2YXRlQWNjb3VudEZsb3dgO1xufVxuXG5leHBvcnQgZGVmYXVsdCBNYXNzYWRTY3JhcGVyO1xuIl19