@tomerh2001/israeli-bank-scrapers 6.3.11
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/LICENSE +21 -0
- package/README.md +425 -0
- package/lib/assertNever.d.ts +1 -0
- package/lib/assertNever.js +10 -0
- package/lib/constants.d.ts +10 -0
- package/lib/constants.js +17 -0
- package/lib/definitions.d.ts +105 -0
- package/lib/definitions.js +116 -0
- package/lib/helpers/browser.d.ts +10 -0
- package/lib/helpers/browser.js +21 -0
- package/lib/helpers/dates.d.ts +2 -0
- package/lib/helpers/dates.js +22 -0
- package/lib/helpers/debug.d.ts +2 -0
- package/lib/helpers/debug.js +12 -0
- package/lib/helpers/elements-interactions.d.ts +17 -0
- package/lib/helpers/elements-interactions.js +111 -0
- package/lib/helpers/fetch.d.ts +6 -0
- package/lib/helpers/fetch.js +111 -0
- package/lib/helpers/navigation.d.ts +6 -0
- package/lib/helpers/navigation.js +39 -0
- package/lib/helpers/storage.d.ts +2 -0
- package/lib/helpers/storage.js +14 -0
- package/lib/helpers/transactions.d.ts +5 -0
- package/lib/helpers/transactions.js +47 -0
- package/lib/helpers/waiting.d.ts +13 -0
- package/lib/helpers/waiting.js +58 -0
- package/lib/index.d.ts +7 -0
- package/lib/index.js +85 -0
- package/lib/scrapers/amex.d.ts +6 -0
- package/lib/scrapers/amex.js +17 -0
- package/lib/scrapers/amex.test.d.ts +1 -0
- package/lib/scrapers/amex.test.js +49 -0
- package/lib/scrapers/base-beinleumi-group.d.ts +66 -0
- package/lib/scrapers/base-beinleumi-group.js +428 -0
- package/lib/scrapers/base-isracard-amex.d.ts +23 -0
- package/lib/scrapers/base-isracard-amex.js +324 -0
- package/lib/scrapers/base-scraper-with-browser.d.ts +57 -0
- package/lib/scrapers/base-scraper-with-browser.js +291 -0
- package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
- package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
- package/lib/scrapers/base-scraper.d.ts +19 -0
- package/lib/scrapers/base-scraper.js +91 -0
- package/lib/scrapers/behatsdaa.d.ts +11 -0
- package/lib/scrapers/behatsdaa.js +113 -0
- package/lib/scrapers/behatsdaa.test.d.ts +1 -0
- package/lib/scrapers/behatsdaa.test.js +46 -0
- package/lib/scrapers/beinleumi.d.ts +7 -0
- package/lib/scrapers/beinleumi.js +15 -0
- package/lib/scrapers/beinleumi.test.d.ts +1 -0
- package/lib/scrapers/beinleumi.test.js +47 -0
- package/lib/scrapers/beyahad-bishvilha.d.ts +30 -0
- package/lib/scrapers/beyahad-bishvilha.js +149 -0
- package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
- package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
- package/lib/scrapers/discount.d.ts +22 -0
- package/lib/scrapers/discount.js +120 -0
- package/lib/scrapers/discount.test.d.ts +1 -0
- package/lib/scrapers/discount.test.js +49 -0
- package/lib/scrapers/errors.d.ts +16 -0
- package/lib/scrapers/errors.js +32 -0
- package/lib/scrapers/factory.d.ts +2 -0
- package/lib/scrapers/factory.js +70 -0
- package/lib/scrapers/factory.test.d.ts +1 -0
- package/lib/scrapers/factory.test.js +19 -0
- package/lib/scrapers/hapoalim.d.ts +24 -0
- package/lib/scrapers/hapoalim.js +198 -0
- package/lib/scrapers/hapoalim.test.d.ts +1 -0
- package/lib/scrapers/hapoalim.test.js +47 -0
- package/lib/scrapers/interface.d.ts +186 -0
- package/lib/scrapers/interface.js +6 -0
- package/lib/scrapers/isracard.d.ts +6 -0
- package/lib/scrapers/isracard.js +17 -0
- package/lib/scrapers/isracard.test.d.ts +1 -0
- package/lib/scrapers/isracard.test.js +49 -0
- package/lib/scrapers/leumi.d.ts +21 -0
- package/lib/scrapers/leumi.js +200 -0
- package/lib/scrapers/leumi.test.d.ts +1 -0
- package/lib/scrapers/leumi.test.js +47 -0
- package/lib/scrapers/massad.d.ts +7 -0
- package/lib/scrapers/massad.js +15 -0
- package/lib/scrapers/max.d.ts +37 -0
- package/lib/scrapers/max.js +299 -0
- package/lib/scrapers/max.test.d.ts +1 -0
- package/lib/scrapers/max.test.js +64 -0
- package/lib/scrapers/mercantile.d.ts +20 -0
- package/lib/scrapers/mercantile.js +18 -0
- package/lib/scrapers/mercantile.test.d.ts +1 -0
- package/lib/scrapers/mercantile.test.js +45 -0
- package/lib/scrapers/mizrahi.d.ts +35 -0
- package/lib/scrapers/mizrahi.js +265 -0
- package/lib/scrapers/mizrahi.test.d.ts +1 -0
- package/lib/scrapers/mizrahi.test.js +56 -0
- package/lib/scrapers/one-zero-queries.d.ts +2 -0
- package/lib/scrapers/one-zero-queries.js +560 -0
- package/lib/scrapers/one-zero.d.ts +36 -0
- package/lib/scrapers/one-zero.js +238 -0
- package/lib/scrapers/one-zero.test.d.ts +1 -0
- package/lib/scrapers/one-zero.test.js +51 -0
- package/lib/scrapers/otsar-hahayal.d.ts +7 -0
- package/lib/scrapers/otsar-hahayal.js +15 -0
- package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
- package/lib/scrapers/otsar-hahayal.test.js +47 -0
- package/lib/scrapers/pagi.d.ts +7 -0
- package/lib/scrapers/pagi.js +15 -0
- package/lib/scrapers/pagi.test.d.ts +1 -0
- package/lib/scrapers/pagi.test.js +47 -0
- package/lib/scrapers/union-bank.d.ts +23 -0
- package/lib/scrapers/union-bank.js +242 -0
- package/lib/scrapers/union-bank.test.d.ts +1 -0
- package/lib/scrapers/union-bank.test.js +47 -0
- package/lib/scrapers/visa-cal.d.ts +20 -0
- package/lib/scrapers/visa-cal.js +318 -0
- package/lib/scrapers/visa-cal.test.d.ts +1 -0
- package/lib/scrapers/visa-cal.test.js +49 -0
- package/lib/scrapers/yahav.d.ts +25 -0
- package/lib/scrapers/yahav.js +247 -0
- package/lib/scrapers/yahav.test.d.ts +1 -0
- package/lib/scrapers/yahav.test.js +49 -0
- package/lib/transactions.d.ts +47 -0
- package/lib/transactions.js +17 -0
- package/package.json +91 -0
|
@@ -0,0 +1,200 @@
|
|
|
1
|
+
"use strict";
|
|
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 _debug = require("../helpers/debug");
|
|
10
|
+
var _elementsInteractions = require("../helpers/elements-interactions");
|
|
11
|
+
var _navigation = require("../helpers/navigation");
|
|
12
|
+
var _transactions = require("../transactions");
|
|
13
|
+
var _baseScraperWithBrowser = require("./base-scraper-with-browser");
|
|
14
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
15
|
+
const debug = (0, _debug.getDebug)('leumi');
|
|
16
|
+
const BASE_URL = 'https://hb2.bankleumi.co.il';
|
|
17
|
+
const LOGIN_URL = 'https://www.leumi.co.il/';
|
|
18
|
+
const TRANSACTIONS_URL = `${BASE_URL}/eBanking/SO/SPA.aspx#/ts/BusinessAccountTrx?WidgetPar=1`;
|
|
19
|
+
const FILTERED_TRANSACTIONS_URL = `${BASE_URL}/ChannelWCF/Broker.svc/ProcessRequest?moduleName=UC_SO_27_GetBusinessAccountTrx`;
|
|
20
|
+
const DATE_FORMAT = 'DD.MM.YY';
|
|
21
|
+
const ACCOUNT_BLOCKED_MSG = 'המנוי חסום';
|
|
22
|
+
const INVALID_PASSWORD_MSG = 'אחד או יותר מפרטי ההזדהות שמסרת שגויים. ניתן לנסות שוב';
|
|
23
|
+
function getPossibleLoginResults() {
|
|
24
|
+
const urls = {
|
|
25
|
+
[_baseScraperWithBrowser.LoginResults.Success]: [/ebanking\/SO\/SPA.aspx/i],
|
|
26
|
+
[_baseScraperWithBrowser.LoginResults.InvalidPassword]: [async options => {
|
|
27
|
+
if (!options || !options.page) {
|
|
28
|
+
throw new Error('missing page options argument');
|
|
29
|
+
}
|
|
30
|
+
const errorMessage = await (0, _elementsInteractions.pageEvalAll)(options.page, 'svg#Capa_1', '', element => {
|
|
31
|
+
return element[0]?.parentElement?.children[1]?.innerText;
|
|
32
|
+
});
|
|
33
|
+
return errorMessage?.startsWith(INVALID_PASSWORD_MSG);
|
|
34
|
+
}],
|
|
35
|
+
[_baseScraperWithBrowser.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, _elementsInteractions.pageEvalAll)(options.page, '.errHeader', '', label => {
|
|
42
|
+
return label[0]?.innerText;
|
|
43
|
+
});
|
|
44
|
+
return errorMessage?.startsWith(ACCOUNT_BLOCKED_MSG);
|
|
45
|
+
}],
|
|
46
|
+
[_baseScraperWithBrowser.LoginResults.ChangePassword]: ['https://hb2.bankleumi.co.il/authenticate'] // NOTICE - might not be relevant starting the Leumi re-design during 2022 Sep
|
|
47
|
+
};
|
|
48
|
+
return urls;
|
|
49
|
+
}
|
|
50
|
+
function createLoginFields(credentials) {
|
|
51
|
+
return [{
|
|
52
|
+
selector: 'input[placeholder="שם משתמש"]',
|
|
53
|
+
value: credentials.username
|
|
54
|
+
}, {
|
|
55
|
+
selector: 'input[placeholder="סיסמה"]',
|
|
56
|
+
value: credentials.password
|
|
57
|
+
}];
|
|
58
|
+
}
|
|
59
|
+
function extractTransactionsFromPage(transactions, status) {
|
|
60
|
+
if (transactions === null || transactions.length === 0) {
|
|
61
|
+
return [];
|
|
62
|
+
}
|
|
63
|
+
const result = transactions.map(rawTransaction => {
|
|
64
|
+
const date = (0, _moment.default)(rawTransaction.DateUTC).milliseconds(0).toISOString();
|
|
65
|
+
const newTransaction = {
|
|
66
|
+
status,
|
|
67
|
+
type: _transactions.TransactionTypes.Normal,
|
|
68
|
+
date,
|
|
69
|
+
processedDate: date,
|
|
70
|
+
description: rawTransaction.Description || '',
|
|
71
|
+
identifier: rawTransaction.ReferenceNumberLong,
|
|
72
|
+
memo: rawTransaction.AdditionalData || '',
|
|
73
|
+
originalCurrency: _constants.SHEKEL_CURRENCY,
|
|
74
|
+
chargedAmount: rawTransaction.Amount,
|
|
75
|
+
originalAmount: rawTransaction.Amount
|
|
76
|
+
};
|
|
77
|
+
return newTransaction;
|
|
78
|
+
});
|
|
79
|
+
return result;
|
|
80
|
+
}
|
|
81
|
+
function hangProcess(timeout) {
|
|
82
|
+
return new Promise(resolve => {
|
|
83
|
+
setTimeout(() => {
|
|
84
|
+
resolve();
|
|
85
|
+
}, timeout);
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
async function clickByXPath(page, xpath) {
|
|
89
|
+
await page.waitForSelector(xpath, {
|
|
90
|
+
timeout: 30000,
|
|
91
|
+
visible: true
|
|
92
|
+
});
|
|
93
|
+
const elm = await page.$$(xpath);
|
|
94
|
+
await elm[0].click();
|
|
95
|
+
}
|
|
96
|
+
function removeSpecialCharacters(str) {
|
|
97
|
+
return str.replace(/[^0-9/-]/g, '');
|
|
98
|
+
}
|
|
99
|
+
async function fetchTransactionsForAccount(page, startDate, accountId) {
|
|
100
|
+
// DEVELOPER NOTICE the account number received from the server is being altered at
|
|
101
|
+
// runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
|
|
102
|
+
await hangProcess(4000);
|
|
103
|
+
await (0, _elementsInteractions.waitUntilElementFound)(page, 'button[title="חיפוש מתקדם"]', true);
|
|
104
|
+
await (0, _elementsInteractions.clickButton)(page, 'button[title="חיפוש מתקדם"]');
|
|
105
|
+
await (0, _elementsInteractions.waitUntilElementFound)(page, 'bll-radio-button', true);
|
|
106
|
+
await (0, _elementsInteractions.clickButton)(page, 'bll-radio-button:not([checked])');
|
|
107
|
+
await (0, _elementsInteractions.waitUntilElementFound)(page, 'input[formcontrolname="txtInputFrom"]', true);
|
|
108
|
+
await (0, _elementsInteractions.fillInput)(page, 'input[formcontrolname="txtInputFrom"]', startDate.format(DATE_FORMAT));
|
|
109
|
+
|
|
110
|
+
// we must blur the from control otherwise the search will use the previous value
|
|
111
|
+
await page.focus("button[aria-label='סנן']");
|
|
112
|
+
await (0, _elementsInteractions.clickButton)(page, "button[aria-label='סנן']");
|
|
113
|
+
const finalResponse = await page.waitForResponse(response => {
|
|
114
|
+
return response.url() === FILTERED_TRANSACTIONS_URL && response.request().method() === 'POST';
|
|
115
|
+
});
|
|
116
|
+
const responseJson = await finalResponse.json();
|
|
117
|
+
const accountNumber = accountId.replace('/', '_').replace(/[^\d-_]/g, '');
|
|
118
|
+
const response = JSON.parse(responseJson.jsonResp);
|
|
119
|
+
const pendingTransactions = response.TodayTransactionsItems;
|
|
120
|
+
const transactions = response.HistoryTransactionsItems;
|
|
121
|
+
const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;
|
|
122
|
+
const pendingTxns = extractTransactionsFromPage(pendingTransactions, _transactions.TransactionStatuses.Pending);
|
|
123
|
+
const completedTxns = extractTransactionsFromPage(transactions, _transactions.TransactionStatuses.Completed);
|
|
124
|
+
const txns = [...pendingTxns, ...completedTxns];
|
|
125
|
+
return {
|
|
126
|
+
accountNumber,
|
|
127
|
+
balance,
|
|
128
|
+
txns
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
async function fetchTransactions(page, startDate) {
|
|
132
|
+
const accounts = [];
|
|
133
|
+
|
|
134
|
+
// DEVELOPER NOTICE the account number received from the server is being altered at
|
|
135
|
+
// runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
|
|
136
|
+
await hangProcess(4000);
|
|
137
|
+
const accountsIds = await page.evaluate(() => Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), e => e.textContent));
|
|
138
|
+
|
|
139
|
+
// due to a bug, the altered value might include undesired signs like & that should be removed
|
|
140
|
+
|
|
141
|
+
if (!accountsIds.length) {
|
|
142
|
+
throw new Error('Failed to extract or parse the account number');
|
|
143
|
+
}
|
|
144
|
+
for (const accountId of accountsIds) {
|
|
145
|
+
if (accountsIds.length > 1) {
|
|
146
|
+
// get list of accounts and check accountId
|
|
147
|
+
await clickByXPath(page, 'xpath///*[contains(@class, "number") and contains(@class, "combo-inner")]');
|
|
148
|
+
await clickByXPath(page, `xpath///span[contains(text(), '${accountId}')]`);
|
|
149
|
+
}
|
|
150
|
+
accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));
|
|
151
|
+
}
|
|
152
|
+
return accounts;
|
|
153
|
+
}
|
|
154
|
+
async function navigateToLogin(page) {
|
|
155
|
+
const loginButtonSelector = '.enter-account a[originaltitle="כניסה לחשבונך"]';
|
|
156
|
+
debug('wait for homepage to click on login button');
|
|
157
|
+
await (0, _elementsInteractions.waitUntilElementFound)(page, loginButtonSelector);
|
|
158
|
+
debug('navigate to login page');
|
|
159
|
+
const loginUrl = await (0, _elementsInteractions.pageEval)(page, loginButtonSelector, null, element => {
|
|
160
|
+
return element.href;
|
|
161
|
+
});
|
|
162
|
+
debug(`navigating to page (${loginUrl})`);
|
|
163
|
+
await page.goto(loginUrl);
|
|
164
|
+
debug('waiting for page to be loaded (networkidle2)');
|
|
165
|
+
await (0, _navigation.waitForNavigation)(page, {
|
|
166
|
+
waitUntil: 'networkidle2'
|
|
167
|
+
});
|
|
168
|
+
debug('waiting for components of login to enter credentials');
|
|
169
|
+
await Promise.all([(0, _elementsInteractions.waitUntilElementFound)(page, 'input[placeholder="שם משתמש"]', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'input[placeholder="סיסמה"]', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'button[type="submit"]', true)]);
|
|
170
|
+
}
|
|
171
|
+
async function waitForPostLogin(page) {
|
|
172
|
+
await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, 'a[title="דלג לחשבון"]', true, 60000), (0, _elementsInteractions.waitUntilElementFound)(page, 'div.main-content', false, 60000), page.waitForSelector(`xpath//div[contains(string(),"${INVALID_PASSWORD_MSG}")]`), (0, _elementsInteractions.waitUntilElementFound)(page, 'form[action="/changepassword"]', true, 60000) // not sure if they kept this one
|
|
173
|
+
]);
|
|
174
|
+
}
|
|
175
|
+
class LeumiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
176
|
+
getLoginOptions(credentials) {
|
|
177
|
+
return {
|
|
178
|
+
loginUrl: LOGIN_URL,
|
|
179
|
+
fields: createLoginFields(credentials),
|
|
180
|
+
submitButtonSelector: "button[type='submit']",
|
|
181
|
+
checkReadiness: async () => navigateToLogin(this.page),
|
|
182
|
+
postAction: async () => waitForPostLogin(this.page),
|
|
183
|
+
possibleResults: getPossibleLoginResults()
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
async fetchData() {
|
|
187
|
+
const minimumStartMoment = (0, _moment.default)().subtract(3, 'years').add(1, 'day');
|
|
188
|
+
const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
|
|
189
|
+
const startDate = this.options.startDate || defaultStartMoment.toDate();
|
|
190
|
+
const startMoment = _moment.default.max(minimumStartMoment, (0, _moment.default)(startDate));
|
|
191
|
+
await this.navigateTo(TRANSACTIONS_URL);
|
|
192
|
+
const accounts = await fetchTransactions(this.page, startMoment);
|
|
193
|
+
return {
|
|
194
|
+
success: true,
|
|
195
|
+
accounts
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
var _default = exports.default = LeumiScraper;
|
|
200
|
+
//# 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","parentElement","children","innerText","startsWith","AccountBlocked","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 { 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"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _leumi = _interopRequireDefault(require("./leumi"));
|
|
4
|
+
var _testsUtils = require("../tests/tests-utils");
|
|
5
|
+
var _definitions = require("../definitions");
|
|
6
|
+
var _baseScraperWithBrowser = require("./base-scraper-with-browser");
|
|
7
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
8
|
+
const COMPANY_ID = 'leumi'; // TODO this property should be hard-coded in the provider
|
|
9
|
+
const testsConfig = (0, _testsUtils.getTestsConfig)();
|
|
10
|
+
describe('Leumi legacy scraper', () => {
|
|
11
|
+
beforeAll(() => {
|
|
12
|
+
(0, _testsUtils.extendAsyncTimeout)(); // The default timeout is 5 seconds per async test, this function extends the timeout value
|
|
13
|
+
});
|
|
14
|
+
test('should expose login fields in scrapers constant', () => {
|
|
15
|
+
expect(_definitions.SCRAPERS.leumi).toBeDefined();
|
|
16
|
+
expect(_definitions.SCRAPERS.leumi.loginFields).toContain('username');
|
|
17
|
+
expect(_definitions.SCRAPERS.leumi.loginFields).toContain('password');
|
|
18
|
+
});
|
|
19
|
+
(0, _testsUtils.maybeTestCompanyAPI)(COMPANY_ID, config => config.companyAPI.invalidPassword)('should fail on invalid user/password"', async () => {
|
|
20
|
+
const options = {
|
|
21
|
+
...testsConfig.options,
|
|
22
|
+
companyId: COMPANY_ID
|
|
23
|
+
};
|
|
24
|
+
const scraper = new _leumi.default(options);
|
|
25
|
+
const result = await scraper.scrape({
|
|
26
|
+
username: 'e10s12',
|
|
27
|
+
password: '3f3ss3d'
|
|
28
|
+
});
|
|
29
|
+
expect(result).toBeDefined();
|
|
30
|
+
expect(result.success).toBeFalsy();
|
|
31
|
+
expect(result.errorType).toBe(_baseScraperWithBrowser.LoginResults.InvalidPassword);
|
|
32
|
+
});
|
|
33
|
+
(0, _testsUtils.maybeTestCompanyAPI)(COMPANY_ID)('should scrape transactions', async () => {
|
|
34
|
+
const options = {
|
|
35
|
+
...testsConfig.options,
|
|
36
|
+
companyId: COMPANY_ID
|
|
37
|
+
};
|
|
38
|
+
const scraper = new _leumi.default(options);
|
|
39
|
+
const result = await scraper.scrape(testsConfig.credentials.leumi);
|
|
40
|
+
expect(result).toBeDefined();
|
|
41
|
+
const error = `${result.errorType || ''} ${result.errorMessage || ''}`.trim();
|
|
42
|
+
expect(error).toBe('');
|
|
43
|
+
expect(result.success).toBeTruthy();
|
|
44
|
+
(0, _testsUtils.exportTransactions)(COMPANY_ID, result.accounts || []);
|
|
45
|
+
});
|
|
46
|
+
});
|
|
47
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfbGV1bWkiLCJfaW50ZXJvcFJlcXVpcmVEZWZhdWx0IiwicmVxdWlyZSIsIl90ZXN0c1V0aWxzIiwiX2RlZmluaXRpb25zIiwiX2Jhc2VTY3JhcGVyV2l0aEJyb3dzZXIiLCJlIiwiX19lc01vZHVsZSIsImRlZmF1bHQiLCJDT01QQU5ZX0lEIiwidGVzdHNDb25maWciLCJnZXRUZXN0c0NvbmZpZyIsImRlc2NyaWJlIiwiYmVmb3JlQWxsIiwiZXh0ZW5kQXN5bmNUaW1lb3V0IiwidGVzdCIsImV4cGVjdCIsIlNDUkFQRVJTIiwibGV1bWkiLCJ0b0JlRGVmaW5lZCIsImxvZ2luRmllbGRzIiwidG9Db250YWluIiwibWF5YmVUZXN0Q29tcGFueUFQSSIsImNvbmZpZyIsImNvbXBhbnlBUEkiLCJpbnZhbGlkUGFzc3dvcmQiLCJvcHRpb25zIiwiY29tcGFueUlkIiwic2NyYXBlciIsIkxldW1pU2NyYXBlciIsInJlc3VsdCIsInNjcmFwZSIsInVzZXJuYW1lIiwicGFzc3dvcmQiLCJzdWNjZXNzIiwidG9CZUZhbHN5IiwiZXJyb3JUeXBlIiwidG9CZSIsIkxvZ2luUmVzdWx0cyIsIkludmFsaWRQYXNzd29yZCIsImNyZWRlbnRpYWxzIiwiZXJyb3IiLCJlcnJvck1lc3NhZ2UiLCJ0cmltIiwidG9CZVRydXRoeSIsImV4cG9ydFRyYW5zYWN0aW9ucyIsImFjY291bnRzIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmFwZXJzL2xldW1pLnRlc3QudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IExldW1pU2NyYXBlciBmcm9tICcuL2xldW1pJztcbmltcG9ydCB7IG1heWJlVGVzdENvbXBhbnlBUEksIGV4dGVuZEFzeW5jVGltZW91dCwgZ2V0VGVzdHNDb25maWcsIGV4cG9ydFRyYW5zYWN0aW9ucyB9IGZyb20gJy4uL3Rlc3RzL3Rlc3RzLXV0aWxzJztcbmltcG9ydCB7IFNDUkFQRVJTIH0gZnJvbSAnLi4vZGVmaW5pdGlvbnMnO1xuaW1wb3J0IHsgTG9naW5SZXN1bHRzIH0gZnJvbSAnLi9iYXNlLXNjcmFwZXItd2l0aC1icm93c2VyJztcblxuY29uc3QgQ09NUEFOWV9JRCA9ICdsZXVtaSc7IC8vIFRPRE8gdGhpcyBwcm9wZXJ0eSBzaG91bGQgYmUgaGFyZC1jb2RlZCBpbiB0aGUgcHJvdmlkZXJcbmNvbnN0IHRlc3RzQ29uZmlnID0gZ2V0VGVzdHNDb25maWcoKTtcblxuZGVzY3JpYmUoJ0xldW1pIGxlZ2FjeSBzY3JhcGVyJywgKCkgPT4ge1xuICBiZWZvcmVBbGwoKCkgPT4ge1xuICAgIGV4dGVuZEFzeW5jVGltZW91dCgpOyAvLyBUaGUgZGVmYXVsdCB0aW1lb3V0IGlzIDUgc2Vjb25kcyBwZXIgYXN5bmMgdGVzdCwgdGhpcyBmdW5jdGlvbiBleHRlbmRzIHRoZSB0aW1lb3V0IHZhbHVlXG4gIH0pO1xuXG4gIHRlc3QoJ3Nob3VsZCBleHBvc2UgbG9naW4gZmllbGRzIGluIHNjcmFwZXJzIGNvbnN0YW50JywgKCkgPT4ge1xuICAgIGV4cGVjdChTQ1JBUEVSUy5sZXVtaSkudG9CZURlZmluZWQoKTtcbiAgICBleHBlY3QoU0NSQVBFUlMubGV1bWkubG9naW5GaWVsZHMpLnRvQ29udGFpbigndXNlcm5hbWUnKTtcbiAgICBleHBlY3QoU0NSQVBFUlMubGV1bWkubG9naW5GaWVsZHMpLnRvQ29udGFpbigncGFzc3dvcmQnKTtcbiAgfSk7XG5cbiAgbWF5YmVUZXN0Q29tcGFueUFQSShDT01QQU5ZX0lELCBjb25maWcgPT4gY29uZmlnLmNvbXBhbnlBUEkuaW52YWxpZFBhc3N3b3JkKShcbiAgICAnc2hvdWxkIGZhaWwgb24gaW52YWxpZCB1c2VyL3Bhc3N3b3JkXCInLFxuICAgIGFzeW5jICgpID0+IHtcbiAgICAgIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgICAgIC4uLnRlc3RzQ29uZmlnLm9wdGlvbnMsXG4gICAgICAgIGNvbXBhbnlJZDogQ09NUEFOWV9JRCxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHNjcmFwZXIgPSBuZXcgTGV1bWlTY3JhcGVyKG9wdGlvbnMpO1xuXG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzY3JhcGVyLnNjcmFwZSh7IHVzZXJuYW1lOiAnZTEwczEyJywgcGFzc3dvcmQ6ICczZjNzczNkJyB9KTtcblxuICAgICAgZXhwZWN0KHJlc3VsdCkudG9CZURlZmluZWQoKTtcbiAgICAgIGV4cGVjdChyZXN1bHQuc3VjY2VzcykudG9CZUZhbHN5KCk7XG4gICAgICBleHBlY3QocmVzdWx0LmVycm9yVHlwZSkudG9CZShMb2dpblJlc3VsdHMuSW52YWxpZFBhc3N3b3JkKTtcbiAgICB9LFxuICApO1xuXG4gIG1heWJlVGVzdENvbXBhbnlBUEkoQ09NUEFOWV9JRCkoJ3Nob3VsZCBzY3JhcGUgdHJhbnNhY3Rpb25zJywgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgICAuLi50ZXN0c0NvbmZpZy5vcHRpb25zLFxuICAgICAgY29tcGFueUlkOiBDT01QQU5ZX0lELFxuICAgIH07XG5cbiAgICBjb25zdCBzY3JhcGVyID0gbmV3IExldW1pU2NyYXBlcihvcHRpb25zKTtcbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzY3JhcGVyLnNjcmFwZSh0ZXN0c0NvbmZpZy5jcmVkZW50aWFscy5sZXVtaSk7XG4gICAgZXhwZWN0KHJlc3VsdCkudG9CZURlZmluZWQoKTtcbiAgICBjb25zdCBlcnJvciA9IGAke3Jlc3VsdC5lcnJvclR5cGUgfHwgJyd9ICR7cmVzdWx0LmVycm9yTWVzc2FnZSB8fCAnJ31gLnRyaW0oKTtcbiAgICBleHBlY3QoZXJyb3IpLnRvQmUoJycpO1xuICAgIGV4cGVjdChyZXN1bHQuc3VjY2VzcykudG9CZVRydXRoeSgpO1xuXG4gICAgZXhwb3J0VHJhbnNhY3Rpb25zKENPTVBBTllfSUQsIHJlc3VsdC5hY2NvdW50cyB8fCBbXSk7XG4gIH0pO1xufSk7XG4iXSwibWFwcGluZ3MiOiI7O0FBQUEsSUFBQUEsTUFBQSxHQUFBQyxzQkFBQSxDQUFBQyxPQUFBO0FBQ0EsSUFBQUMsV0FB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
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.default = void 0;
|
|
7
|
+
var _baseBeinleumiGroup = _interopRequireDefault(require("./base-beinleumi-group"));
|
|
8
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
9
|
+
class MassadScraper extends _baseBeinleumiGroup.default {
|
|
10
|
+
BASE_URL = 'https://online.bankmassad.co.il';
|
|
11
|
+
LOGIN_URL = `${this.BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=MASADPRTAL&site=Private&KODSAFA=HE`;
|
|
12
|
+
TRANSACTIONS_URL = `${this.BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;
|
|
13
|
+
}
|
|
14
|
+
var _default = exports.default = MassadScraper;
|
|
15
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfYmFzZUJlaW5sZXVtaUdyb3VwIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsInJlcXVpcmUiLCJlIiwiX19lc01vZHVsZSIsImRlZmF1bHQiLCJNYXNzYWRTY3JhcGVyIiwiQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciIsIkJBU0VfVVJMIiwiTE9HSU5fVVJMIiwiVFJBTlNBQ1RJT05TX1VSTCIsIl9kZWZhdWx0IiwiZXhwb3J0cyJdLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zY3JhcGVycy9tYXNzYWQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEJlaW5sZXVtaUdyb3VwQmFzZVNjcmFwZXIgZnJvbSAnLi9iYXNlLWJlaW5sZXVtaS1ncm91cCc7XG5cbmNsYXNzIE1hc3NhZFNjcmFwZXIgZXh0ZW5kcyBCZWlubGV1bWlHcm91cEJhc2VTY3JhcGVyIHtcbiAgQkFTRV9VUkwgPSAnaHR0cHM6Ly9vbmxpbmUuYmFua21hc3NhZC5jby5pbCc7XG5cbiAgTE9HSU5fVVJMID0gYCR7dGhpcy5CQVNFX1VSTH0vTWF0YWZMb2dpblNlcnZpY2UvTWF0YWZMb2dpblNlcnZsZXQ/YmFua0lkPU1BU0FEUFJUQUwmc2l0ZT1Qcml2YXRlJktPRFNBRkE9SEVgO1xuXG4gIFRSQU5TQUNUSU9OU19VUkwgPSBgJHt0aGlzLkJBU0VfVVJMfS93cHMvbXlwb3J0YWwvRmliaU1lbnUvT25saW5lL09uQWNjb3VudE1uZ21lbnQvT25CYWxhbmNlVHJhbnMvUHJpdmF0ZUFjY291bnRGbG93YDtcbn1cblxuZXhwb3J0IGRlZmF1bHQgTWFzc2FkU2NyYXBlcjtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQUEsSUFBQUEsbUJBQUEsR0FBQUMsc0JBQUEsQ0FBQUMsT0FBQTtBQUErRCxTQUFBRCx1QkFBQUUsQ0FBQSxXQUFBQSxDQUFBLElBQUFBLENBQUEsQ0FBQUMsVUFBQSxHQUFBRCxDQUFBLEtBQUFFLE9BQUEsRUFBQUYsQ0FBQTtBQUUvRCxNQUFNRyxhQUFhLFNBQVNDLDJCQUF5QixDQUFDO0VBQ3BEQyxRQUFRLEdBQUcsaUNBQWlDO0VBRTVDQyxTQUFTLEdBQUcsR0FBRyxJQUFJLENBQUNELFFBQVEsZ0ZBQWdGO0VBRTVHRSxnQkFBZ0IsR0FBRyxHQUFHLElBQUksQ0FBQ0YsUUFBUSxrRkFBa0Y7QUFDdkg7QUFBQyxJQUFBRyxRQUFBLEdBQUFDLE9BQUEsQ0FBQVAsT0FBQSxHQUVjQyxhQUFhIiwiaWdub3JlTGlzdCI6W119
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { type Transaction } from '../transactions';
|
|
2
|
+
import { BaseScraperWithBrowser, type LoginOptions } from './base-scraper-with-browser';
|
|
3
|
+
export interface ScrapedTransaction {
|
|
4
|
+
shortCardNumber: string;
|
|
5
|
+
paymentDate?: string;
|
|
6
|
+
purchaseDate: string;
|
|
7
|
+
actualPaymentAmount: string;
|
|
8
|
+
paymentCurrency: number | null;
|
|
9
|
+
originalCurrency: string;
|
|
10
|
+
originalAmount: number;
|
|
11
|
+
planName: string;
|
|
12
|
+
planTypeId: number;
|
|
13
|
+
comments: string;
|
|
14
|
+
merchantName: string;
|
|
15
|
+
categoryId: number;
|
|
16
|
+
fundsTransferComment?: string;
|
|
17
|
+
fundsTransferReceiverOrTransfer?: string;
|
|
18
|
+
dealData?: {
|
|
19
|
+
arn: string;
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
export declare function getMemo({ comments, fundsTransferReceiverOrTransfer, fundsTransferComment, }: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>): string;
|
|
23
|
+
type ScraperSpecificCredentials = {
|
|
24
|
+
username: string;
|
|
25
|
+
password: string;
|
|
26
|
+
};
|
|
27
|
+
declare class MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {
|
|
28
|
+
getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions;
|
|
29
|
+
fetchData(): Promise<{
|
|
30
|
+
success: boolean;
|
|
31
|
+
accounts: {
|
|
32
|
+
accountNumber: string;
|
|
33
|
+
txns: Transaction[];
|
|
34
|
+
}[];
|
|
35
|
+
}>;
|
|
36
|
+
}
|
|
37
|
+
export default MaxScraper;
|