israeli-bank-scrapers 6.1.1 → 6.1.3

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 (81) 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.d.ts +32 -0
  18. package/lib/scrapers/base-beinleumi-group.js +350 -234
  19. package/lib/scrapers/base-isracard-amex.js +273 -273
  20. package/lib/scrapers/base-scraper-with-browser.js +263 -241
  21. package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
  22. package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
  23. package/lib/scrapers/base-scraper.js +82 -82
  24. package/lib/scrapers/behatsdaa.js +103 -98
  25. package/lib/scrapers/behatsdaa.test.d.ts +1 -0
  26. package/lib/scrapers/behatsdaa.test.js +46 -0
  27. package/lib/scrapers/beinleumi.js +13 -11
  28. package/lib/scrapers/beinleumi.test.d.ts +1 -0
  29. package/lib/scrapers/beinleumi.test.js +47 -0
  30. package/lib/scrapers/beyahad-bishvilha.js +132 -132
  31. package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
  32. package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
  33. package/lib/scrapers/discount.js +101 -97
  34. package/lib/scrapers/discount.test.d.ts +1 -0
  35. package/lib/scrapers/discount.test.js +49 -0
  36. package/lib/scrapers/errors.js +25 -22
  37. package/lib/scrapers/factory.js +67 -66
  38. package/lib/scrapers/factory.test.d.ts +1 -0
  39. package/lib/scrapers/factory.test.js +19 -0
  40. package/lib/scrapers/hapoalim.js +175 -162
  41. package/lib/scrapers/hapoalim.test.d.ts +1 -0
  42. package/lib/scrapers/hapoalim.test.js +47 -0
  43. package/lib/scrapers/interface.js +5 -2
  44. package/lib/scrapers/isracard.js +13 -11
  45. package/lib/scrapers/isracard.test.d.ts +1 -0
  46. package/lib/scrapers/isracard.test.js +49 -0
  47. package/lib/scrapers/leumi.js +170 -167
  48. package/lib/scrapers/leumi.test.d.ts +1 -0
  49. package/lib/scrapers/leumi.test.js +47 -0
  50. package/lib/scrapers/massad.js +13 -11
  51. package/lib/scrapers/max.js +261 -261
  52. package/lib/scrapers/max.test.d.ts +1 -0
  53. package/lib/scrapers/max.test.js +65 -0
  54. package/lib/scrapers/mercantile.js +16 -14
  55. package/lib/scrapers/mercantile.test.d.ts +1 -0
  56. package/lib/scrapers/mercantile.test.js +45 -0
  57. package/lib/scrapers/mizrahi.js +154 -158
  58. package/lib/scrapers/mizrahi.test.d.ts +1 -0
  59. package/lib/scrapers/mizrahi.test.js +53 -0
  60. package/lib/scrapers/one-zero-queries.js +7 -4
  61. package/lib/scrapers/one-zero.js +221 -176
  62. package/lib/scrapers/one-zero.test.d.ts +1 -0
  63. package/lib/scrapers/one-zero.test.js +51 -0
  64. package/lib/scrapers/otsar-hahayal.js +13 -11
  65. package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
  66. package/lib/scrapers/otsar-hahayal.test.js +47 -0
  67. package/lib/scrapers/pagi.js +13 -11
  68. package/lib/scrapers/pagi.test.d.ts +1 -0
  69. package/lib/scrapers/pagi.test.js +47 -0
  70. package/lib/scrapers/union-bank.js +173 -172
  71. package/lib/scrapers/union-bank.test.d.ts +1 -0
  72. package/lib/scrapers/union-bank.test.js +47 -0
  73. package/lib/scrapers/visa-cal.d.ts +2 -0
  74. package/lib/scrapers/visa-cal.js +263 -254
  75. package/lib/scrapers/visa-cal.test.d.ts +1 -0
  76. package/lib/scrapers/visa-cal.test.js +49 -0
  77. package/lib/scrapers/yahav.js +206 -190
  78. package/lib/scrapers/yahav.test.d.ts +1 -0
  79. package/lib/scrapers/yahav.test.js +49 -0
  80. package/lib/transactions.js +16 -13
  81. package/package.json +9 -4
@@ -1,26 +1,25 @@
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
- const moment_1 = __importDefault(require("moment"));
7
- const constants_1 = require("../constants");
8
- const elements_interactions_1 = require("../helpers/elements-interactions");
9
- const fetch_1 = require("../helpers/fetch");
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 errors_1 = require("./errors");
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _moment = _interopRequireDefault(require("moment"));
8
+ var _constants = require("../constants");
9
+ var _elementsInteractions = require("../helpers/elements-interactions");
10
+ var _fetch = require("../helpers/fetch");
11
+ var _navigation = require("../helpers/navigation");
12
+ var _transactions = require("../transactions");
13
+ var _baseScraperWithBrowser = require("./base-scraper-with-browser");
14
+ var _errors = require("./errors");
15
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
14
16
  const BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';
15
17
  const LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;
16
18
  const BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';
17
19
  const AFTER_LOGIN_BASE_URL = /https:\/\/mto\.mizrahi-tefahot\.co\.il\/OnlineApp\/.*/;
18
20
  const OSH_PAGE = '/osh/legacy/legacy-Osh-Main';
19
21
  const TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';
20
- const TRANSACTIONS_REQUEST_URLS = [
21
- `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,
22
- `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,
23
- ];
22
+ const TRANSACTIONS_REQUEST_URLS = [`${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`, `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`];
24
23
  const PENDING_TRANSACTIONS_PAGE = '/osh/legacy/legacy-Osh-p420';
25
24
  const PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';
26
25
  const CHANGE_PASSWORD_URL = /https:\/\/www\.mizrahi-tefahot\.co\.il\/login\/index\.html#\/change-pass/;
@@ -37,167 +36,164 @@ const pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';
37
36
  const checkingAccountTabHebrewName = 'עובר ושב';
38
37
  const checkingAccountTabEnglishName = 'Checking Account';
39
38
  function createLoginFields(credentials) {
40
- return [
41
- { selector: usernameSelector, value: credentials.username },
42
- { selector: passwordSelector, value: credentials.password },
43
- ];
39
+ return [{
40
+ selector: usernameSelector,
41
+ value: credentials.username
42
+ }, {
43
+ selector: passwordSelector,
44
+ value: credentials.password
45
+ }];
44
46
  }
45
47
  async function isLoggedIn(options) {
46
- if (!options?.page) {
47
- return false;
48
- }
49
- const oshXPath = `//a//span[contains(., "${checkingAccountTabHebrewName}") or contains(., "${checkingAccountTabEnglishName}")]`;
50
- const oshTab = await options.page.$$(`xpath${oshXPath}`);
51
- return oshTab.length > 0;
48
+ if (!options?.page) {
49
+ return false;
50
+ }
51
+ const oshXPath = `//a//span[contains(., "${checkingAccountTabHebrewName}") or contains(., "${checkingAccountTabEnglishName}")]`;
52
+ const oshTab = await options.page.$$(`xpath${oshXPath}`);
53
+ return oshTab.length > 0;
52
54
  }
53
55
  function getPossibleLoginResults(page) {
54
- return {
55
- [base_scraper_with_browser_1.LoginResults.Success]: [AFTER_LOGIN_BASE_URL, isLoggedIn],
56
- [base_scraper_with_browser_1.LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],
57
- [base_scraper_with_browser_1.LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],
58
- };
56
+ return {
57
+ [_baseScraperWithBrowser.LoginResults.Success]: [AFTER_LOGIN_BASE_URL, isLoggedIn],
58
+ [_baseScraperWithBrowser.LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],
59
+ [_baseScraperWithBrowser.LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL]
60
+ };
59
61
  }
60
62
  function getStartMoment(optionsStartDate) {
61
- const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years');
62
- const startDate = optionsStartDate || defaultStartMoment.toDate();
63
- return moment_1.default.max(defaultStartMoment, (0, moment_1.default)(startDate));
63
+ const defaultStartMoment = (0, _moment.default)().subtract(1, 'years');
64
+ const startDate = optionsStartDate || defaultStartMoment.toDate();
65
+ return _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
64
66
  }
65
67
  function createDataFromRequest(request, optionsStartDate) {
66
- const data = JSON.parse(request.postData() || '{}');
67
- data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);
68
- data.inToDate = (0, moment_1.default)().format(DATE_FORMAT);
69
- data.table.maxRow = MAX_ROWS_PER_REQUEST;
70
- return data;
68
+ const data = JSON.parse(request.postData() || '{}');
69
+ data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);
70
+ data.inToDate = (0, _moment.default)().format(DATE_FORMAT);
71
+ data.table.maxRow = MAX_ROWS_PER_REQUEST;
72
+ return data;
71
73
  }
72
74
  function createHeadersFromRequest(request) {
73
- return {
74
- mizrahixsrftoken: request.headers().mizrahixsrftoken,
75
- 'Content-Type': request.headers()['content-type'],
76
- };
75
+ return {
76
+ mizrahixsrftoken: request.headers().mizrahixsrftoken,
77
+ 'Content-Type': request.headers()['content-type']
78
+ };
77
79
  }
78
80
  function convertTransactions(txns) {
79
- return txns.map(row => {
80
- const txnDate = (0, moment_1.default)(row.MC02PeulaTaaEZ, moment_1.default.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();
81
- return {
82
- type: transactions_1.TransactionTypes.Normal,
83
- identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,
84
- date: txnDate,
85
- processedDate: txnDate,
86
- originalAmount: row.MC02SchumEZ,
87
- originalCurrency: constants_1.SHEKEL_CURRENCY,
88
- chargedAmount: row.MC02SchumEZ,
89
- description: row.MC02TnuaTeurEZ,
90
- status: transactions_1.TransactionStatuses.Completed,
91
- };
92
- });
81
+ return txns.map(row => {
82
+ const txnDate = (0, _moment.default)(row.MC02PeulaTaaEZ, _moment.default.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();
83
+ return {
84
+ type: _transactions.TransactionTypes.Normal,
85
+ identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,
86
+ date: txnDate,
87
+ processedDate: txnDate,
88
+ originalAmount: row.MC02SchumEZ,
89
+ originalCurrency: _constants.SHEKEL_CURRENCY,
90
+ chargedAmount: row.MC02SchumEZ,
91
+ description: row.MC02TnuaTeurEZ,
92
+ status: _transactions.TransactionStatuses.Completed
93
+ };
94
+ });
93
95
  }
94
96
  async function extractPendingTransactions(page) {
95
- const pendingTxn = await (0, elements_interactions_1.pageEvalAll)(page, 'tr.rgRow', [], trs => {
96
- return trs.map(tr => Array.from(tr.querySelectorAll('td'), (td) => td.textContent || ''));
97
- });
98
- return pendingTxn.map(txn => {
99
- const date = (0, moment_1.default)(txn[0], 'DD/MM/YY').toISOString();
100
- const amount = parseInt(txn[3], 10);
101
- return {
102
- type: transactions_1.TransactionTypes.Normal,
103
- date,
104
- processedDate: date,
105
- originalAmount: amount,
106
- originalCurrency: constants_1.SHEKEL_CURRENCY,
107
- chargedAmount: amount,
108
- description: txn[1],
109
- status: transactions_1.TransactionStatuses.Pending,
110
- };
111
- });
97
+ const pendingTxn = await (0, _elementsInteractions.pageEvalAll)(page, 'tr.rgRow', [], trs => {
98
+ return trs.map(tr => Array.from(tr.querySelectorAll('td'), td => td.textContent || ''));
99
+ });
100
+ return pendingTxn.map(txn => {
101
+ const date = (0, _moment.default)(txn[0], 'DD/MM/YY').toISOString();
102
+ const amount = parseInt(txn[3], 10);
103
+ return {
104
+ type: _transactions.TransactionTypes.Normal,
105
+ date,
106
+ processedDate: date,
107
+ originalAmount: amount,
108
+ originalCurrency: _constants.SHEKEL_CURRENCY,
109
+ chargedAmount: amount,
110
+ description: txn[1],
111
+ status: _transactions.TransactionStatuses.Pending
112
+ };
113
+ });
112
114
  }
113
115
  async function postLogin(page) {
114
- await Promise.race([
115
- (0, elements_interactions_1.waitUntilElementFound)(page, afterLoginSelector),
116
- (0, elements_interactions_1.waitUntilElementFound)(page, invalidPasswordSelector),
117
- (0, navigation_1.waitForUrl)(page, CHANGE_PASSWORD_URL),
118
- ]);
116
+ await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, afterLoginSelector), (0, _elementsInteractions.waitUntilElementFound)(page, invalidPasswordSelector), (0, _navigation.waitForUrl)(page, CHANGE_PASSWORD_URL)]);
119
117
  }
120
- class MizrahiScraper extends base_scraper_with_browser_1.BaseScraperWithBrowser {
121
- getLoginOptions(credentials) {
122
- return {
123
- loginUrl: LOGIN_URL,
124
- fields: createLoginFields(credentials),
125
- submitButtonSelector,
126
- checkReadiness: async () => (0, elements_interactions_1.waitUntilElementDisappear)(this.page, loginSpinnerSelector),
127
- postAction: async () => postLogin(this.page),
128
- possibleResults: getPossibleLoginResults(this.page),
129
- };
130
- }
131
- async fetchData() {
132
- await this.page.$eval('#dropdownBasic, .item', el => el.click());
133
- const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;
134
- try {
135
- const results = [];
136
- for (let i = 0; i < numOfAccounts; i += 1) {
137
- if (i > 0) {
138
- await this.page.$eval('#dropdownBasic, .item', el => el.click());
139
- }
140
- await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, el => el.click());
141
- results.push(await this.fetchAccount());
142
- }
143
- return {
144
- success: true,
145
- accounts: results,
146
- };
147
- }
148
- catch (e) {
149
- return {
150
- success: false,
151
- errorType: errors_1.ScraperErrorTypes.Generic,
152
- errorMessage: e.message,
153
- };
118
+ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
119
+ getLoginOptions(credentials) {
120
+ return {
121
+ loginUrl: LOGIN_URL,
122
+ fields: createLoginFields(credentials),
123
+ submitButtonSelector,
124
+ checkReadiness: async () => (0, _elementsInteractions.waitUntilElementDisappear)(this.page, loginSpinnerSelector),
125
+ postAction: async () => postLogin(this.page),
126
+ possibleResults: getPossibleLoginResults(this.page)
127
+ };
128
+ }
129
+ async fetchData() {
130
+ await this.page.$eval('#dropdownBasic, .item', el => el.click());
131
+ const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;
132
+ try {
133
+ const results = [];
134
+ for (let i = 0; i < numOfAccounts; i += 1) {
135
+ if (i > 0) {
136
+ await this.page.$eval('#dropdownBasic, .item', el => el.click());
154
137
  }
138
+ await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, el => el.click());
139
+ results.push(await this.fetchAccount());
140
+ }
141
+ return {
142
+ success: true,
143
+ accounts: results
144
+ };
145
+ } catch (e) {
146
+ return {
147
+ success: false,
148
+ errorType: _errors.ScraperErrorTypes.Generic,
149
+ errorMessage: e.message
150
+ };
155
151
  }
156
- async getPendingTransactions() {
157
- await this.page.$eval(`a[href*="${PENDING_TRANSACTIONS_PAGE}"]`, el => el.click());
158
- const frame = await (0, elements_interactions_1.waitUntilIframeFound)(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));
159
- const isPending = await (0, elements_interactions_1.waitUntilElementFound)(frame, pendingTrxIdentifierId)
160
- .then(() => true)
161
- .catch(() => false);
162
- if (!isPending) {
163
- return [];
164
- }
165
- const pendingTxn = await extractPendingTransactions(frame);
166
- return pendingTxn;
152
+ }
153
+ async getPendingTransactions() {
154
+ await this.page.$eval(`a[href*="${PENDING_TRANSACTIONS_PAGE}"]`, el => el.click());
155
+ const frame = await (0, _elementsInteractions.waitUntilIframeFound)(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));
156
+ const isPending = await (0, _elementsInteractions.waitUntilElementFound)(frame, pendingTrxIdentifierId).then(() => true).catch(() => false);
157
+ if (!isPending) {
158
+ return [];
167
159
  }
168
- async fetchAccount() {
169
- await this.page.waitForSelector(`a[href*="${OSH_PAGE}"]`);
170
- await this.page.$eval(`a[href*="${OSH_PAGE}"]`, el => el.click());
171
- await (0, elements_interactions_1.waitUntilElementFound)(this.page, `a[href*="${TRANSACTIONS_PAGE}"]`);
172
- await this.page.$eval(`a[href*="${TRANSACTIONS_PAGE}"]`, el => el.click());
173
- const accountNumberElement = await this.page.$('#dropdownBasic b span');
174
- const accountNumberHandle = await accountNumberElement?.getProperty('title');
175
- const accountNumber = (await accountNumberHandle?.jsonValue());
176
- if (!accountNumber) {
177
- throw new Error('Account number not found');
178
- }
179
- const response = await Promise.any(TRANSACTIONS_REQUEST_URLS.map(async (url) => {
180
- const request = await this.page.waitForRequest(url);
181
- const data = createDataFromRequest(request, this.options.startDate);
182
- const headers = createHeadersFromRequest(request);
183
- return (0, fetch_1.fetchPostWithinPage)(this.page, url, data, headers);
184
- }));
185
- if (!response || response.header.success === false) {
186
- throw new Error(`Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`);
187
- }
188
- const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);
189
- const oshTxn = convertTransactions(relevantRows);
190
- // workaround for a bug which the bank's API returns transactions before the requested start date
191
- const startMoment = getStartMoment(this.options.startDate);
192
- const oshTxnAfterStartDate = oshTxn.filter(txn => (0, moment_1.default)(txn.date).isSameOrAfter(startMoment));
193
- const pendingTxn = await this.getPendingTransactions();
194
- const allTxn = oshTxnAfterStartDate.concat(pendingTxn);
195
- return {
196
- accountNumber,
197
- txns: allTxn,
198
- balance: +response.body.fields?.Yitra,
199
- };
160
+ const pendingTxn = await extractPendingTransactions(frame);
161
+ return pendingTxn;
162
+ }
163
+ async fetchAccount() {
164
+ await this.page.waitForSelector(`a[href*="${OSH_PAGE}"]`);
165
+ await this.page.$eval(`a[href*="${OSH_PAGE}"]`, el => el.click());
166
+ await (0, _elementsInteractions.waitUntilElementFound)(this.page, `a[href*="${TRANSACTIONS_PAGE}"]`);
167
+ await this.page.$eval(`a[href*="${TRANSACTIONS_PAGE}"]`, el => el.click());
168
+ const accountNumberElement = await this.page.$('#dropdownBasic b span');
169
+ const accountNumberHandle = await accountNumberElement?.getProperty('title');
170
+ const accountNumber = await accountNumberHandle?.jsonValue();
171
+ if (!accountNumber) {
172
+ throw new Error('Account number not found');
173
+ }
174
+ const response = await Promise.any(TRANSACTIONS_REQUEST_URLS.map(async url => {
175
+ const request = await this.page.waitForRequest(url);
176
+ const data = createDataFromRequest(request, this.options.startDate);
177
+ const headers = createHeadersFromRequest(request);
178
+ return (0, _fetch.fetchPostWithinPage)(this.page, url, data, headers);
179
+ }));
180
+ if (!response || response.header.success === false) {
181
+ throw new Error(`Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`);
200
182
  }
183
+ const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);
184
+ const oshTxn = convertTransactions(relevantRows);
185
+
186
+ // workaround for a bug which the bank's API returns transactions before the requested start date
187
+ const startMoment = getStartMoment(this.options.startDate);
188
+ const oshTxnAfterStartDate = oshTxn.filter(txn => (0, _moment.default)(txn.date).isSameOrAfter(startMoment));
189
+ const pendingTxn = await this.getPendingTransactions();
190
+ const allTxn = oshTxnAfterStartDate.concat(pendingTxn);
191
+ return {
192
+ accountNumber,
193
+ txns: allTxn,
194
+ balance: +response.body.fields?.Yitra
195
+ };
196
+ }
201
197
  }
202
- exports.default = MizrahiScraper;
203
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mizrahi.js","sourceRoot":"","sources":["../../src/scrapers/mizrahi.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { type Frame, type HTTPRequest, type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  pageEvalAll,\n  waitUntilElementDisappear,\n  waitUntilElementFound,\n  waitUntilIframeFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForUrl } from '../helpers/navigation';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\n\ninterface ScrapedTransaction {\n  RecTypeSpecified: boolean;\n  MC02PeulaTaaEZ: string;\n  MC02SchumEZ: number;\n  MC02AsmahtaMekoritEZ: string;\n  MC02TnuaTeurEZ: string;\n}\n\ninterface ScrapedTransactionsResult {\n  header: {\n    success: boolean;\n    messages: { text: string }[];\n  };\n  body: {\n    fields: {\n      Yitra: string;\n    };\n    table: {\n      rows: ScrapedTransaction[];\n    };\n  };\n}\n\nconst BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';\nconst LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;\nconst BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';\nconst AFTER_LOGIN_BASE_URL = /https:\\/\\/mto\\.mizrahi-tefahot\\.co\\.il\\/OnlineApp\\/.*/;\nconst OSH_PAGE = '/osh/legacy/legacy-Osh-Main';\nconst TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';\nconst TRANSACTIONS_REQUEST_URLS = [\n  `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,\n  `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,\n];\nconst PENDING_TRANSACTIONS_PAGE = '/osh/legacy/legacy-Osh-p420';\nconst PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';\nconst CHANGE_PASSWORD_URL = /https:\\/\\/www\\.mizrahi-tefahot\\.co\\.il\\/login\\/index\\.html#\\/change-pass/;\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst MAX_ROWS_PER_REQUEST = 10000000000;\n\nconst usernameSelector = '#emailDesktopHeb';\nconst passwordSelector = '#passwordIDDesktopHEB';\nconst submitButtonSelector = '.form-desktop button';\nconst invalidPasswordSelector = 'a[href*=\"https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx\"]';\nconst afterLoginSelector = '#dropdownBasic';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#AccountPicker .item';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\nconst checkingAccountTabHebrewName = 'עובר ושב';\nconst checkingAccountTabEnglishName = 'Checking Account';\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: usernameSelector, value: credentials.username },\n    { selector: passwordSelector, value: credentials.password },\n  ];\n}\n\nasync function isLoggedIn(options: { page?: Page | undefined } | undefined) {\n  if (!options?.page) {\n    return false;\n  }\n  const oshXPath = `//a//span[contains(., \"${checkingAccountTabHebrewName}\") or contains(., \"${checkingAccountTabEnglishName}\")]`;\n  const oshTab = await options.page.$$(`xpath${oshXPath}`);\n  return oshTab.length > 0;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  return {\n    [LoginResults.Success]: [AFTER_LOGIN_BASE_URL, isLoggedIn],\n    [LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],\n    [LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],\n  };\n}\n\nfunction getStartMoment(optionsStartDate: Date) {\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = optionsStartDate || defaultStartMoment.toDate();\n  return moment.max(defaultStartMoment, moment(startDate));\n}\n\nfunction createDataFromRequest(request: HTTPRequest, optionsStartDate: Date) {\n  const data = JSON.parse(request.postData() || '{}');\n\n  data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);\n  data.inToDate = moment().format(DATE_FORMAT);\n  data.table.maxRow = MAX_ROWS_PER_REQUEST;\n\n  return data;\n}\n\nfunction createHeadersFromRequest(request: HTTPRequest) {\n  return {\n    mizrahixsrftoken: request.headers().mizrahixsrftoken,\n    'Content-Type': request.headers()['content-type'],\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map(row => {\n    const txnDate = moment(row.MC02PeulaTaaEZ, moment.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();\n\n    return {\n      type: TransactionTypes.Normal,\n      identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: row.MC02SchumEZ,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: row.MC02SchumEZ,\n      description: row.MC02TnuaTeurEZ,\n      status: TransactionStatuses.Completed,\n    };\n  });\n}\n\nasync function extractPendingTransactions(page: Frame): Promise<Transaction[]> {\n  const pendingTxn = await pageEvalAll(page, 'tr.rgRow', [], trs => {\n    return trs.map(tr => Array.from(tr.querySelectorAll('td'), (td: HTMLTableDataCellElement) => td.textContent || ''));\n  });\n\n  return pendingTxn.map(txn => {\n    const date = moment(txn[0], 'DD/MM/YY').toISOString();\n    const amount = parseInt(txn[3], 10);\n    return {\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn[1],\n      status: TransactionStatuses.Pending,\n    };\n  });\n}\n\nasync function postLogin(page: Page) {\n  await Promise.race([\n    waitUntilElementFound(page, afterLoginSelector),\n    waitUntilElementFound(page, invalidPasswordSelector),\n    waitForUrl(page, CHANGE_PASSWORD_URL),\n  ]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MizrahiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector,\n      checkReadiness: async () => waitUntilElementDisappear(this.page, loginSpinnerSelector),\n      postAction: async () => postLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n\n    try {\n      const results: TransactionsAccount[] = [];\n\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        if (i > 0) {\n          await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n        }\n\n        await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, el => (el as HTMLElement).click());\n        results.push(await this.fetchAccount());\n      }\n\n      return {\n        success: true,\n        accounts: results,\n      };\n    } catch (e) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: (e as Error).message,\n      };\n    }\n  }\n\n  private async getPendingTransactions(): Promise<Transaction[]> {\n    await this.page.$eval(`a[href*=\"${PENDING_TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n    const frame = await waitUntilIframeFound(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    const isPending = await waitUntilElementFound(frame, pendingTrxIdentifierId)\n      .then(() => true)\n      .catch(() => false);\n    if (!isPending) {\n      return [];\n    }\n\n    const pendingTxn = await extractPendingTransactions(frame);\n    return pendingTxn;\n  }\n\n  private async fetchAccount() {\n    await this.page.waitForSelector(`a[href*=\"${OSH_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${OSH_PAGE}\"]`, el => (el as HTMLElement).click());\n    await waitUntilElementFound(this.page, `a[href*=\"${TRANSACTIONS_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n\n    const accountNumberElement = await this.page.$('#dropdownBasic b span');\n    const accountNumberHandle = await accountNumberElement?.getProperty('title');\n    const accountNumber = (await accountNumberHandle?.jsonValue()) as string;\n    if (!accountNumber) {\n      throw new Error('Account number not found');\n    }\n\n    const response = await Promise.any(\n      TRANSACTIONS_REQUEST_URLS.map(async url => {\n        const request = await this.page.waitForRequest(url);\n        const data = createDataFromRequest(request, this.options.startDate);\n        const headers = createHeadersFromRequest(request);\n\n        return fetchPostWithinPage<ScrapedTransactionsResult>(this.page, url, data, headers);\n      }),\n    );\n\n    if (!response || response.header.success === false) {\n      throw new Error(\n        `Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`,\n      );\n    }\n\n    const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);\n    const oshTxn = convertTransactions(relevantRows);\n\n    // workaround for a bug which the bank's API returns transactions before the requested start date\n    const startMoment = getStartMoment(this.options.startDate);\n    const oshTxnAfterStartDate = oshTxn.filter(txn => moment(txn.date).isSameOrAfter(startMoment));\n\n    const pendingTxn = await this.getPendingTransactions();\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber,\n      txns: allTxn,\n      balance: +response.body.fields?.Yitra,\n    };\n  }\n}\n\nexport default MizrahiScraper;\n"]}
198
+ var _default = exports.default = MizrahiScraper;
199
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_fetch","_navigation","_transactions","_baseScraperWithBrowser","_errors","e","__esModule","default","BASE_WEBSITE_URL","LOGIN_URL","BASE_APP_URL","AFTER_LOGIN_BASE_URL","OSH_PAGE","TRANSACTIONS_PAGE","TRANSACTIONS_REQUEST_URLS","PENDING_TRANSACTIONS_PAGE","PENDING_TRANSACTIONS_IFRAME","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","accountDropDownItemSelector","pendingTrxIdentifierId","checkingAccountTabHebrewName","checkingAccountTabEnglishName","createLoginFields","credentials","selector","value","username","password","isLoggedIn","options","page","oshXPath","oshTab","$$","length","getPossibleLoginResults","LoginResults","Success","InvalidPassword","$","ChangePassword","getStartMoment","optionsStartDate","defaultStartMoment","moment","subtract","startDate","toDate","max","createDataFromRequest","request","data","JSON","parse","postData","inFromDate","format","inToDate","table","maxRow","createHeadersFromRequest","mizrahixsrftoken","headers","convertTransactions","txns","map","row","txnDate","MC02PeulaTaaEZ","HTML5_FMT","DATETIME_LOCAL_SECONDS","toISOString","type","TransactionTypes","Normal","identifier","MC02AsmahtaMekoritEZ","parseInt","undefined","date","processedDate","originalAmount","MC02SchumEZ","originalCurrency","SHEKEL_CURRENCY","chargedAmount","description","MC02TnuaTeurEZ","status","TransactionStatuses","Completed","extractPendingTransactions","pendingTxn","pageEvalAll","trs","tr","Array","from","querySelectorAll","td","textContent","txn","amount","Pending","postLogin","Promise","race","waitUntilElementFound","waitForUrl","MizrahiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","checkReadiness","waitUntilElementDisappear","postAction","possibleResults","fetchData","$eval","el","click","numOfAccounts","results","i","push","fetchAccount","success","accounts","errorType","ScraperErrorTypes","Generic","errorMessage","message","getPendingTransactions","frame","waitUntilIframeFound","f","url","includes","isPending","then","catch","waitForSelector","accountNumberElement","accountNumberHandle","getProperty","accountNumber","jsonValue","Error","response","any","waitForRequest","fetchPostWithinPage","header","messages","text","relevantRows","body","rows","filter","RecTypeSpecified","oshTxn","startMoment","oshTxnAfterStartDate","isSameOrAfter","allTxn","concat","balance","Yitra","_default","exports"],"sources":["../../src/scrapers/mizrahi.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Frame, type HTTPRequest, type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  pageEvalAll,\n  waitUntilElementDisappear,\n  waitUntilElementFound,\n  waitUntilIframeFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForUrl } from '../helpers/navigation';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\n\ninterface ScrapedTransaction {\n  RecTypeSpecified: boolean;\n  MC02PeulaTaaEZ: string;\n  MC02SchumEZ: number;\n  MC02AsmahtaMekoritEZ: string;\n  MC02TnuaTeurEZ: string;\n}\n\ninterface ScrapedTransactionsResult {\n  header: {\n    success: boolean;\n    messages: { text: string }[];\n  };\n  body: {\n    fields: {\n      Yitra: string;\n    };\n    table: {\n      rows: ScrapedTransaction[];\n    };\n  };\n}\n\nconst BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';\nconst LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;\nconst BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';\nconst AFTER_LOGIN_BASE_URL = /https:\\/\\/mto\\.mizrahi-tefahot\\.co\\.il\\/OnlineApp\\/.*/;\nconst OSH_PAGE = '/osh/legacy/legacy-Osh-Main';\nconst TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';\nconst TRANSACTIONS_REQUEST_URLS = [\n  `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,\n  `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,\n];\nconst PENDING_TRANSACTIONS_PAGE = '/osh/legacy/legacy-Osh-p420';\nconst PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';\nconst CHANGE_PASSWORD_URL = /https:\\/\\/www\\.mizrahi-tefahot\\.co\\.il\\/login\\/index\\.html#\\/change-pass/;\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst MAX_ROWS_PER_REQUEST = 10000000000;\n\nconst usernameSelector = '#emailDesktopHeb';\nconst passwordSelector = '#passwordIDDesktopHEB';\nconst submitButtonSelector = '.form-desktop button';\nconst invalidPasswordSelector = 'a[href*=\"https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx\"]';\nconst afterLoginSelector = '#dropdownBasic';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#AccountPicker .item';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\nconst checkingAccountTabHebrewName = 'עובר ושב';\nconst checkingAccountTabEnglishName = 'Checking Account';\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: usernameSelector, value: credentials.username },\n    { selector: passwordSelector, value: credentials.password },\n  ];\n}\n\nasync function isLoggedIn(options: { page?: Page | undefined } | undefined) {\n  if (!options?.page) {\n    return false;\n  }\n  const oshXPath = `//a//span[contains(., \"${checkingAccountTabHebrewName}\") or contains(., \"${checkingAccountTabEnglishName}\")]`;\n  const oshTab = await options.page.$$(`xpath${oshXPath}`);\n  return oshTab.length > 0;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  return {\n    [LoginResults.Success]: [AFTER_LOGIN_BASE_URL, isLoggedIn],\n    [LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],\n    [LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],\n  };\n}\n\nfunction getStartMoment(optionsStartDate: Date) {\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = optionsStartDate || defaultStartMoment.toDate();\n  return moment.max(defaultStartMoment, moment(startDate));\n}\n\nfunction createDataFromRequest(request: HTTPRequest, optionsStartDate: Date) {\n  const data = JSON.parse(request.postData() || '{}');\n\n  data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);\n  data.inToDate = moment().format(DATE_FORMAT);\n  data.table.maxRow = MAX_ROWS_PER_REQUEST;\n\n  return data;\n}\n\nfunction createHeadersFromRequest(request: HTTPRequest) {\n  return {\n    mizrahixsrftoken: request.headers().mizrahixsrftoken,\n    'Content-Type': request.headers()['content-type'],\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map(row => {\n    const txnDate = moment(row.MC02PeulaTaaEZ, moment.HTML5_FMT.DATETIME_LOCAL_SECONDS).toISOString();\n\n    return {\n      type: TransactionTypes.Normal,\n      identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: row.MC02SchumEZ,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: row.MC02SchumEZ,\n      description: row.MC02TnuaTeurEZ,\n      status: TransactionStatuses.Completed,\n    };\n  });\n}\n\nasync function extractPendingTransactions(page: Frame): Promise<Transaction[]> {\n  const pendingTxn = await pageEvalAll(page, 'tr.rgRow', [], trs => {\n    return trs.map(tr => Array.from(tr.querySelectorAll('td'), (td: HTMLTableDataCellElement) => td.textContent || ''));\n  });\n\n  return pendingTxn.map(txn => {\n    const date = moment(txn[0], 'DD/MM/YY').toISOString();\n    const amount = parseInt(txn[3], 10);\n    return {\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn[1],\n      status: TransactionStatuses.Pending,\n    };\n  });\n}\n\nasync function postLogin(page: Page) {\n  await Promise.race([\n    waitUntilElementFound(page, afterLoginSelector),\n    waitUntilElementFound(page, invalidPasswordSelector),\n    waitForUrl(page, CHANGE_PASSWORD_URL),\n  ]);\n}\n\ntype ScraperSpecificCredentials = { username: string; password: string };\n\nclass MizrahiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector,\n      checkReadiness: async () => waitUntilElementDisappear(this.page, loginSpinnerSelector),\n      postAction: async () => postLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n\n    try {\n      const results: TransactionsAccount[] = [];\n\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        if (i > 0) {\n          await this.page.$eval('#dropdownBasic, .item', el => (el as HTMLElement).click());\n        }\n\n        await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, el => (el as HTMLElement).click());\n        results.push(await this.fetchAccount());\n      }\n\n      return {\n        success: true,\n        accounts: results,\n      };\n    } catch (e) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: (e as Error).message,\n      };\n    }\n  }\n\n  private async getPendingTransactions(): Promise<Transaction[]> {\n    await this.page.$eval(`a[href*=\"${PENDING_TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n    const frame = await waitUntilIframeFound(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    const isPending = await waitUntilElementFound(frame, pendingTrxIdentifierId)\n      .then(() => true)\n      .catch(() => false);\n    if (!isPending) {\n      return [];\n    }\n\n    const pendingTxn = await extractPendingTransactions(frame);\n    return pendingTxn;\n  }\n\n  private async fetchAccount() {\n    await this.page.waitForSelector(`a[href*=\"${OSH_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${OSH_PAGE}\"]`, el => (el as HTMLElement).click());\n    await waitUntilElementFound(this.page, `a[href*=\"${TRANSACTIONS_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${TRANSACTIONS_PAGE}\"]`, el => (el as HTMLElement).click());\n\n    const accountNumberElement = await this.page.$('#dropdownBasic b span');\n    const accountNumberHandle = await accountNumberElement?.getProperty('title');\n    const accountNumber = (await accountNumberHandle?.jsonValue()) as string;\n    if (!accountNumber) {\n      throw new Error('Account number not found');\n    }\n\n    const response = await Promise.any(\n      TRANSACTIONS_REQUEST_URLS.map(async url => {\n        const request = await this.page.waitForRequest(url);\n        const data = createDataFromRequest(request, this.options.startDate);\n        const headers = createHeadersFromRequest(request);\n\n        return fetchPostWithinPage<ScrapedTransactionsResult>(this.page, url, data, headers);\n      }),\n    );\n\n    if (!response || response.header.success === false) {\n      throw new Error(\n        `Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`,\n      );\n    }\n\n    const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);\n    const oshTxn = convertTransactions(relevantRows);\n\n    // workaround for a bug which the bank's API returns transactions before the requested start date\n    const startMoment = getStartMoment(this.options.startDate);\n    const oshTxnAfterStartDate = oshTxn.filter(txn => moment(txn.date).isSameOrAfter(startMoment));\n\n    const pendingTxn = await this.getPendingTransactions();\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber,\n      txns: allTxn,\n      balance: +response.body.fields?.Yitra,\n    };\n  }\n}\n\nexport default MizrahiScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,53 @@
1
+ "use strict";
2
+
3
+ var _mizrahi = _interopRequireDefault(require("./mizrahi"));
4
+ var _testsUtils = require("../tests/tests-utils");
5
+ var _definitions = require("../definitions");
6
+ var _constants = require("../constants");
7
+ var _baseScraperWithBrowser = require("./base-scraper-with-browser");
8
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
9
+ const COMPANY_ID = 'mizrahi'; // TODO this property should be hard-coded in the provider
10
+ const testsConfig = (0, _testsUtils.getTestsConfig)();
11
+ describe('Mizrahi scraper', () => {
12
+ beforeAll(() => {
13
+ (0, _testsUtils.extendAsyncTimeout)(); // The default timeout is 5 seconds per async test, this function extends the timeout value
14
+ });
15
+ test('should expose login fields in scrapers constant', () => {
16
+ expect(_definitions.SCRAPERS.mizrahi).toBeDefined();
17
+ expect(_definitions.SCRAPERS.mizrahi.loginFields).toContain('username');
18
+ expect(_definitions.SCRAPERS.mizrahi.loginFields).toContain('password');
19
+ });
20
+ (0, _testsUtils.maybeTestCompanyAPI)(COMPANY_ID, config => config.companyAPI.invalidPassword)('should fail on invalid user/password', async () => {
21
+ const options = {
22
+ ...testsConfig.options,
23
+ companyId: COMPANY_ID
24
+ };
25
+ const scraper = new _mizrahi.default(options);
26
+ const result = await scraper.scrape({
27
+ username: 'e10s12',
28
+ password: '3f3ss3d'
29
+ });
30
+ expect(result).toBeDefined();
31
+ expect(result.success).toBeFalsy();
32
+ expect(result.errorType).toBe(_baseScraperWithBrowser.LoginResults.InvalidPassword);
33
+ });
34
+ (0, _testsUtils.maybeTestCompanyAPI)(COMPANY_ID)('should scrape transactions', async () => {
35
+ const options = {
36
+ ...testsConfig.options,
37
+ companyId: COMPANY_ID
38
+ };
39
+ const scraper = new _mizrahi.default(options);
40
+ const result = await scraper.scrape(testsConfig.credentials.mizrahi);
41
+ expect(result).toBeDefined();
42
+ const error = `${result.errorType || ''} ${result.errorMessage || ''}`.trim();
43
+ expect(error).toBe('');
44
+ expect(result.success).toBeTruthy();
45
+ expect(result.accounts).toBeDefined();
46
+ expect(result.accounts.length).toBeGreaterThan(0);
47
+ const account = result.accounts[0];
48
+ expect(account.accountNumber).not.toBe('');
49
+ expect(account.txns[0].date).toMatch(_constants.ISO_DATE_REGEX);
50
+ (0, _testsUtils.exportTransactions)(COMPANY_ID, result.accounts || []);
51
+ });
52
+ });
53
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfbWl6cmFoaSIsIl9pbnRlcm9wUmVxdWlyZURlZmF1bHQiLCJyZXF1aXJlIiwiX3Rlc3RzVXRpbHMiLCJfZGVmaW5pdGlvbnMiLCJfY29uc3RhbnRzIiwiX2Jhc2VTY3JhcGVyV2l0aEJyb3dzZXIiLCJlIiwiX19lc01vZHVsZSIsImRlZmF1bHQiLCJDT01QQU5ZX0lEIiwidGVzdHNDb25maWciLCJnZXRUZXN0c0NvbmZpZyIsImRlc2NyaWJlIiwiYmVmb3JlQWxsIiwiZXh0ZW5kQXN5bmNUaW1lb3V0IiwidGVzdCIsImV4cGVjdCIsIlNDUkFQRVJTIiwibWl6cmFoaSIsInRvQmVEZWZpbmVkIiwibG9naW5GaWVsZHMiLCJ0b0NvbnRhaW4iLCJtYXliZVRlc3RDb21wYW55QVBJIiwiY29uZmlnIiwiY29tcGFueUFQSSIsImludmFsaWRQYXNzd29yZCIsIm9wdGlvbnMiLCJjb21wYW55SWQiLCJzY3JhcGVyIiwiTWl6cmFoaVNjcmFwZXIiLCJyZXN1bHQiLCJzY3JhcGUiLCJ1c2VybmFtZSIsInBhc3N3b3JkIiwic3VjY2VzcyIsInRvQmVGYWxzeSIsImVycm9yVHlwZSIsInRvQmUiLCJMb2dpblJlc3VsdHMiLCJJbnZhbGlkUGFzc3dvcmQiLCJjcmVkZW50aWFscyIsImVycm9yIiwiZXJyb3JNZXNzYWdlIiwidHJpbSIsInRvQmVUcnV0aHkiLCJhY2NvdW50cyIsImxlbmd0aCIsInRvQmVHcmVhdGVyVGhhbiIsImFjY291bnQiLCJhY2NvdW50TnVtYmVyIiwibm90IiwidHhucyIsImRhdGUiLCJ0b01hdGNoIiwiSVNPX0RBVEVfUkVHRVgiLCJleHBvcnRUcmFuc2FjdGlvbnMiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvc2NyYXBlcnMvbWl6cmFoaS50ZXN0LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBNaXpyYWhpU2NyYXBlciBmcm9tICcuL21penJhaGknO1xuaW1wb3J0IHsgbWF5YmVUZXN0Q29tcGFueUFQSSwgZXh0ZW5kQXN5bmNUaW1lb3V0LCBnZXRUZXN0c0NvbmZpZywgZXhwb3J0VHJhbnNhY3Rpb25zIH0gZnJvbSAnLi4vdGVzdHMvdGVzdHMtdXRpbHMnO1xuaW1wb3J0IHsgU0NSQVBFUlMgfSBmcm9tICcuLi9kZWZpbml0aW9ucyc7XG5pbXBvcnQgeyBJU09fREFURV9SRUdFWCB9IGZyb20gJy4uL2NvbnN0YW50cyc7XG5pbXBvcnQgeyBMb2dpblJlc3VsdHMgfSBmcm9tICcuL2Jhc2Utc2NyYXBlci13aXRoLWJyb3dzZXInO1xuaW1wb3J0IHsgdHlwZSBUcmFuc2FjdGlvbnNBY2NvdW50IH0gZnJvbSAnLi4vdHJhbnNhY3Rpb25zJztcblxuY29uc3QgQ09NUEFOWV9JRCA9ICdtaXpyYWhpJzsgLy8gVE9ETyB0aGlzIHByb3BlcnR5IHNob3VsZCBiZSBoYXJkLWNvZGVkIGluIHRoZSBwcm92aWRlclxuY29uc3QgdGVzdHNDb25maWcgPSBnZXRUZXN0c0NvbmZpZygpO1xuXG5kZXNjcmliZSgnTWl6cmFoaSBzY3JhcGVyJywgKCkgPT4ge1xuICBiZWZvcmVBbGwoKCkgPT4ge1xuICAgIGV4dGVuZEFzeW5jVGltZW91dCgpOyAvLyBUaGUgZGVmYXVsdCB0aW1lb3V0IGlzIDUgc2Vjb25kcyBwZXIgYXN5bmMgdGVzdCwgdGhpcyBmdW5jdGlvbiBleHRlbmRzIHRoZSB0aW1lb3V0IHZhbHVlXG4gIH0pO1xuXG4gIHRlc3QoJ3Nob3VsZCBleHBvc2UgbG9naW4gZmllbGRzIGluIHNjcmFwZXJzIGNvbnN0YW50JywgKCkgPT4ge1xuICAgIGV4cGVjdChTQ1JBUEVSUy5taXpyYWhpKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChTQ1JBUEVSUy5taXpyYWhpLmxvZ2luRmllbGRzKS50b0NvbnRhaW4oJ3VzZXJuYW1lJyk7XG4gICAgZXhwZWN0KFNDUkFQRVJTLm1penJhaGkubG9naW5GaWVsZHMpLnRvQ29udGFpbigncGFzc3dvcmQnKTtcbiAgfSk7XG5cbiAgbWF5YmVUZXN0Q29tcGFueUFQSShDT01QQU5ZX0lELCBjb25maWcgPT4gY29uZmlnLmNvbXBhbnlBUEkuaW52YWxpZFBhc3N3b3JkKShcbiAgICAnc2hvdWxkIGZhaWwgb24gaW52YWxpZCB1c2VyL3Bhc3N3b3JkJyxcbiAgICBhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCBvcHRpb25zID0ge1xuICAgICAgICAuLi50ZXN0c0NvbmZpZy5vcHRpb25zLFxuICAgICAgICBjb21wYW55SWQ6IENPTVBBTllfSUQsXG4gICAgICB9O1xuXG4gICAgICBjb25zdCBzY3JhcGVyID0gbmV3IE1penJhaGlTY3JhcGVyKG9wdGlvbnMpO1xuXG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzY3JhcGVyLnNjcmFwZSh7IHVzZXJuYW1lOiAnZTEwczEyJywgcGFzc3dvcmQ6ICczZjNzczNkJyB9KTtcblxuICAgICAgZXhwZWN0KHJlc3VsdCkudG9CZURlZmluZWQoKTtcbiAgICAgIGV4cGVjdChyZXN1bHQuc3VjY2VzcykudG9CZUZhbHN5KCk7XG4gICAgICBleHBlY3QocmVzdWx0LmVycm9yVHlwZSkudG9CZShMb2dpblJlc3VsdHMuSW52YWxpZFBhc3N3b3JkKTtcbiAgICB9LFxuICApO1xuXG4gIG1heWJlVGVzdENvbXBhbnlBUEkoQ09NUEFOWV9JRCkoJ3Nob3VsZCBzY3JhcGUgdHJhbnNhY3Rpb25zJywgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgICAuLi50ZXN0c0NvbmZpZy5vcHRpb25zLFxuICAgICAgY29tcGFueUlkOiBDT01QQU5ZX0lELFxuICAgIH07XG5cbiAgICBjb25zdCBzY3JhcGVyID0gbmV3IE1penJhaGlTY3JhcGVyKG9wdGlvbnMpO1xuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHNjcmFwZXIuc2NyYXBlKHRlc3RzQ29uZmlnLmNyZWRlbnRpYWxzLm1penJhaGkpO1xuICAgIGV4cGVjdChyZXN1bHQpLnRvQmVEZWZpbmVkKCk7XG4gICAgY29uc3QgZXJyb3IgPSBgJHtyZXN1bHQuZXJyb3JUeXBlIHx8ICcnfSAke3Jlc3VsdC5lcnJvck1lc3NhZ2UgfHwgJyd9YC50cmltKCk7XG4gICAgZXhwZWN0KGVycm9yKS50b0JlKCcnKTtcbiAgICBleHBlY3QocmVzdWx0LnN1Y2Nlc3MpLnRvQmVUcnV0aHkoKTtcbiAgICBleHBlY3QocmVzdWx0LmFjY291bnRzKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdCgocmVzdWx0LmFjY291bnRzIGFzIGFueSkubGVuZ3RoKS50b0JlR3JlYXRlclRoYW4oMCk7XG4gICAgY29uc3QgYWNjb3VudDogVHJhbnNhY3Rpb25zQWNjb3VudCA9IChyZXN1bHQgYXMgYW55KS5hY2NvdW50c1swXTtcbiAgICBleHBlY3QoYWNjb3VudC5hY2NvdW50TnVtYmVyKS5ub3QudG9CZSgnJyk7XG4gICAgZXhwZWN0KGFjY291bnQudHhuc1swXS5kYXRlKS50b01hdGNoKElTT19EQVRFX1JFR0VYKTtcblxuICAgIGV4cG9ydFRyYW5zYWN0aW9ucyhDT01QQU5ZX0lELCByZXN1bHQuYWNjb3VudHMgfHwgW10pO1xuICB9KTtcbn0pO1xuIl0sIm1hcHBpbmdzIjoiOztBQUFBLElBQUFBLFFBQUEsR0FBQUMsc0JBQUEsQ0FBQUMsT0FBQTtBQUNBLElBQUFDLFdBQUEsR0FBQUQsT0FBQTtBQUNBLElBQUFFLFlBQUEsR0FBQUYsT0FBQTtBQUNBLElBQUFHLFVBQUEsR0FBQUgsT0FBQTtBQUNBLElBQUFJLHVCQUFBLEdBQUFKLE9BQUE7QUFBMkQsU0FBQUQsdUJBQUFNLENBQUEsV0FBQUEsQ0FBQSxJQUFBQSxDQUFBLENBQUFDLFVBQUEsR0FBQUQsQ0FBQSxLQUFBRSxPQUFBLEVBQUFGLENBQUE7QUFHM0QsTUFBTUcsVUFBVSxHQUFHLFNBQVMsQ0FBQyxDQUFDO0FBQzlCLE1BQU1DLFdBQVcsR0FBRyxJQUFBQywwQkFBYyxFQUFDLENBQUM7QUFFcENDLFFBQVEsQ0FBQyxpQkFBaUIsRUFBRSxNQUFNO0VBQ2hDQyxTQUFTLENBQUMsTUFBTTtJQUNkLElBQUFDLDhCQUFrQixFQUFDLENBQUMsQ0FBQyxDQUFDO0VBQ3hCLENBQUMs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