israeli-bank-scrapers 6.0.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.
- package/README.md +1 -0
- package/lib/assertNever.js +5 -7
- package/lib/constants.js +13 -16
- package/lib/definitions.js +109 -113
- package/lib/helpers/browser.d.ts +8 -0
- package/lib/helpers/browser.js +14 -11
- package/lib/helpers/dates.js +18 -19
- package/lib/helpers/debug.js +9 -9
- package/lib/helpers/elements-interactions.js +78 -84
- package/lib/helpers/fetch.js +82 -89
- package/lib/helpers/navigation.js +24 -31
- package/lib/helpers/storage.js +10 -12
- package/lib/helpers/transactions.js +33 -35
- package/lib/helpers/waiting.js +45 -44
- package/lib/index.js +15 -82
- package/lib/scrapers/amex.js +11 -13
- package/lib/scrapers/base-beinleumi-group.js +233 -252
- package/lib/scrapers/base-isracard-amex.js +273 -286
- package/lib/scrapers/base-scraper-with-browser.d.ts +2 -1
- package/lib/scrapers/base-scraper-with-browser.js +240 -269
- package/lib/scrapers/base-scraper.js +82 -86
- package/lib/scrapers/behatsdaa.js +98 -107
- package/lib/scrapers/beinleumi.js +11 -20
- package/lib/scrapers/beyahad-bishvilha.js +132 -138
- package/lib/scrapers/discount.js +97 -103
- package/lib/scrapers/errors.js +22 -25
- package/lib/scrapers/factory.js +66 -67
- package/lib/scrapers/hapoalim.js +162 -182
- package/lib/scrapers/interface.d.ts +12 -0
- package/lib/scrapers/interface.js +2 -5
- package/lib/scrapers/isracard.js +11 -13
- package/lib/scrapers/leumi.js +167 -176
- package/lib/scrapers/massad.js +11 -20
- package/lib/scrapers/max.js +256 -268
- package/lib/scrapers/mercantile.js +14 -20
- package/lib/scrapers/mizrahi.js +158 -159
- package/lib/scrapers/one-zero-queries.js +4 -7
- package/lib/scrapers/one-zero.js +176 -240
- package/lib/scrapers/otsar-hahayal.js +11 -20
- package/lib/scrapers/pagi.js +11 -20
- package/lib/scrapers/union-bank.js +172 -179
- package/lib/scrapers/visa-cal.js +254 -263
- package/lib/scrapers/yahav.js +190 -211
- package/lib/transactions.js +13 -16
- package/package.json +12 -14
- package/lib/scrapers/amex.test.d.ts +0 -1
- package/lib/scrapers/amex.test.js +0 -54
- package/lib/scrapers/base-scraper-with-browser.test.d.ts +0 -1
- package/lib/scrapers/base-scraper-with-browser.test.js +0 -58
- package/lib/scrapers/behatsdaa.test.d.ts +0 -1
- package/lib/scrapers/behatsdaa.test.js +0 -50
- package/lib/scrapers/beinleumi.test.d.ts +0 -1
- package/lib/scrapers/beinleumi.test.js +0 -52
- package/lib/scrapers/beyahad-bishvilha.test.d.ts +0 -1
- package/lib/scrapers/beyahad-bishvilha.test.js +0 -52
- package/lib/scrapers/discount.test.d.ts +0 -1
- package/lib/scrapers/discount.test.js +0 -54
- package/lib/scrapers/factory.test.d.ts +0 -1
- package/lib/scrapers/factory.test.js +0 -19
- package/lib/scrapers/hapoalim.test.d.ts +0 -1
- package/lib/scrapers/hapoalim.test.js +0 -52
- package/lib/scrapers/isracard.test.d.ts +0 -1
- package/lib/scrapers/isracard.test.js +0 -54
- package/lib/scrapers/leumi.test.d.ts +0 -1
- package/lib/scrapers/leumi.test.js +0 -52
- package/lib/scrapers/max.test.d.ts +0 -1
- package/lib/scrapers/max.test.js +0 -71
- package/lib/scrapers/mercantile.test.d.ts +0 -1
- package/lib/scrapers/mercantile.test.js +0 -50
- package/lib/scrapers/mizrahi.test.d.ts +0 -1
- package/lib/scrapers/mizrahi.test.js +0 -58
- package/lib/scrapers/one-zero.test.d.ts +0 -1
- package/lib/scrapers/one-zero.test.js +0 -56
- package/lib/scrapers/otsar-hahayal.test.d.ts +0 -1
- package/lib/scrapers/otsar-hahayal.test.js +0 -52
- package/lib/scrapers/pagi.test.d.ts +0 -1
- package/lib/scrapers/pagi.test.js +0 -52
- package/lib/scrapers/union-bank.test.d.ts +0 -1
- package/lib/scrapers/union-bank.test.js +0 -52
- package/lib/scrapers/visa-cal.test.d.ts +0 -1
- package/lib/scrapers/visa-cal.test.js +0 -54
- package/lib/scrapers/yahav.test.d.ts +0 -1
- package/lib/scrapers/yahav.test.js +0 -54
package/lib/scrapers/leumi.js
CHANGED
|
@@ -1,22 +1,16 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
});
|
|
6
|
-
|
|
7
|
-
require("
|
|
8
|
-
require("
|
|
9
|
-
require("
|
|
10
|
-
require("
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
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
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
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
|
-
|
|
58
|
-
|
|
59
|
-
|
|
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
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
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
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
80
|
+
return new Promise(resolve => {
|
|
81
|
+
setTimeout(() => {
|
|
82
|
+
resolve();
|
|
83
|
+
}, timeout);
|
|
84
|
+
});
|
|
93
85
|
}
|
|
94
86
|
async function clickByXPath(page, xpath) {
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
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
|
-
|
|
92
|
+
return str.replace(/[^0-9/-]/g, '');
|
|
104
93
|
}
|
|
105
94
|
async function fetchTransactionsForAccount(page, startDate, accountId) {
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
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
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
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
|
|
157
|
-
}
|
|
158
|
-
return accounts;
|
|
143
|
+
return accounts;
|
|
159
144
|
}
|
|
160
145
|
async function navigateToLogin(page) {
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
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
|
-
|
|
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
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
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
|
-
|
|
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"]}
|
package/lib/scrapers/massad.js
CHANGED
|
@@ -1,22 +1,13 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
});
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
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
|
-
|
|
22
|
-
//# sourceMappingURL=data:application/json;
|
|
12
|
+
exports.default = MassadScraper;
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFzc2FkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmFwZXJzL21hc3NhZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLGtGQUErRDtBQUUvRCxNQUFNLGFBQWMsU0FBUSw4QkFBeUI7SUFDbkQsUUFBUSxHQUFHLGlDQUFpQyxDQUFDO0lBRTdDLFNBQVMsR0FBRyxHQUFHLElBQUksQ0FBQyxRQUFRLGdGQUFnRixDQUFDO0lBRTdHLGdCQUFnQixHQUFHLEdBQUcsSUFBSSxDQUFDLFFBQVEsa0ZBQWtGLENBQUM7Q0FDdkg7QUFFRCxrQkFBZSxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciBmcm9tICcuL2Jhc2UtYmVpbmxldW1pLWdyb3VwJztcblxuY2xhc3MgTWFzc2FkU2NyYXBlciBleHRlbmRzIEJlaW5sZXVtaUdyb3VwQmFzZVNjcmFwZXIge1xuICBCQVNFX1VSTCA9ICdodHRwczovL29ubGluZS5iYW5rbWFzc2FkLmNvLmlsJztcblxuICBMT0dJTl9VUkwgPSBgJHt0aGlzLkJBU0VfVVJMfS9NYXRhZkxvZ2luU2VydmljZS9NYXRhZkxvZ2luU2VydmxldD9iYW5rSWQ9TUFTQURQUlRBTCZzaXRlPVByaXZhdGUmS09EU0FGQT1IRWA7XG5cbiAgVFJBTlNBQ1RJT05TX1VSTCA9IGAke3RoaXMuQkFTRV9VUkx9L3dwcy9teXBvcnRhbC9GaWJpTWVudS9PbmxpbmUvT25BY2NvdW50TW5nbWVudC9PbkJhbGFuY2VUcmFucy9Qcml2YXRlQWNjb3VudEZsb3dgO1xufVxuXG5leHBvcnQgZGVmYXVsdCBNYXNzYWRTY3JhcGVyO1xuIl19
|