israeli-bank-scrapers 5.4.2 → 5.4.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.
@@ -42,9 +42,13 @@ const COMPLETED_TRANSACTIONS_TABLE = 'table#dataTable077';
42
42
  const PENDING_TRANSACTIONS_TABLE = 'table#dataTable023';
43
43
  const NEXT_PAGE_LINK = 'a#Npage.paging';
44
44
  const CURRENT_BALANCE = '.main_balance';
45
+ const IFRAME_NAME = 'iframe-old-pages';
45
46
  function getPossibleLoginResults() {
46
47
  const urls = {};
47
- urls[_baseScraperWithBrowser.LoginResults.Success] = [/fibi.*accountSummary/];
48
+ urls[_baseScraperWithBrowser.LoginResults.Success] = [/fibi.*accountSummary/,
49
+ // New UI pattern
50
+ /FibiMenu\/Online/ // Old UI pattern
51
+ ];
48
52
  urls[_baseScraperWithBrowser.LoginResults.InvalidPassword] = [/FibiMenu\/Marketing\/Private\/Home/];
49
53
  return urls;
50
54
  }
@@ -223,7 +227,14 @@ async function getCurrentBalance(page) {
223
227
  return getAmountData(balanceStr);
224
228
  }
225
229
  async function waitForPostLogin(page) {
226
- return Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, '#card-header', true), (0, _elementsInteractions.waitUntilElementFound)(page, '#account_num', true)]);
230
+ return Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, '#card-header', true),
231
+ // New UI
232
+ (0, _elementsInteractions.waitUntilElementFound)(page, '#account_num', true),
233
+ // New UI
234
+ (0, _elementsInteractions.waitUntilElementFound)(page, '#matafLogoutLink', true),
235
+ // Old UI
236
+ (0, _elementsInteractions.waitUntilElementFound)(page, '#validationMsg', true) // Old UI
237
+ ]);
227
238
  }
228
239
  async function fetchAccountData(page, startDate) {
229
240
  await searchByDates(page, startDate);
@@ -246,28 +257,40 @@ async function getAccountIdsBySelector(page) {
246
257
  return accountsIds;
247
258
  }
248
259
  async function getTransactionsFrame(page) {
249
- await (0, _waiting.sleep)(5000);
250
- const frames = page.frames();
251
- const targetFrame = frames.find(f => f.name() === 'iframe-old-pages');
252
- if (!targetFrame) {
253
- throw new Error('iframe: "iframe-old-pages" (used for transactions page) was not found on the page');
260
+ // Try a few times to find the iframe, as it might not be immediately available
261
+ for (let attempt = 0; attempt < 3; attempt++) {
262
+ await (0, _waiting.sleep)(2000);
263
+ const frames = page.frames();
264
+ const targetFrame = frames.find(f => f.name() === IFRAME_NAME);
265
+ if (targetFrame) {
266
+ return targetFrame;
267
+ }
254
268
  }
255
- return targetFrame;
269
+ return null;
270
+ }
271
+ async function selectAccount(page, accountId) {
272
+ await page.select('#account_num_select', accountId);
273
+ await (0, _elementsInteractions.waitUntilElementFound)(page, '#account_num_select', true);
274
+ }
275
+ async function fetchAccountDataBothUIs(page, startDate) {
276
+ // Try to get the iframe for the new UI
277
+ const frame = await getTransactionsFrame(page);
278
+
279
+ // Use the frame if available (new UI), otherwise use the page directly (old UI)
280
+ const targetPage = frame || page;
281
+ return fetchAccountData(targetPage, startDate);
256
282
  }
257
283
  async function fetchAccounts(page, startDate) {
258
- const accounts = [];
259
284
  const accountsIds = await getAccountIdsBySelector(page);
260
285
  if (accountsIds.length <= 1) {
261
- const frame = await getTransactionsFrame(page);
262
- const accountData = await fetchAccountData(frame, startDate);
286
+ const accountData = await fetchAccountDataBothUIs(page, startDate);
287
+ return [accountData];
288
+ }
289
+ const accounts = [];
290
+ for (const accountId of accountsIds) {
291
+ await selectAccount(page, accountId);
292
+ const accountData = await fetchAccountDataBothUIs(page, startDate);
263
293
  accounts.push(accountData);
264
- } else {
265
- for (const accountId of accountsIds) {
266
- await page.select('#account_num_select', accountId);
267
- await (0, _elementsInteractions.waitUntilElementFound)(page, '#account_num_select', true);
268
- const accountData = await fetchAccountData(page, startDate);
269
- accounts.push(accountData);
270
- }
271
294
  }
272
295
  return accounts;
273
296
  }
@@ -308,4 +331,4 @@ class BeinleumiGroupBaseScraper extends _baseScraperWithBrowser.BaseScraperWithB
308
331
  }
309
332
  }
310
333
  var _default = exports.default = BeinleumiGroupBaseScraper;
311
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","_defineProperty","r","t","_toPropertyKey","Object","defineProperty","value","enumerable","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","DATE_FORMAT","NO_TRANSACTION_IN_DATE_RANGE_TEXT","DATE_COLUMN_CLASS_COMPLETED","DATE_COLUMN_CLASS_PENDING","DESCRIPTION_COLUMN_CLASS_COMPLETED","DESCRIPTION_COLUMN_CLASS_PENDING","REFERENCE_COLUMN_CLASS","DEBIT_COLUMN_CLASS","CREDIT_COLUMN_CLASS","ERROR_MESSAGE_CLASS","ACCOUNTS_NUMBER","CLOSE_SEARCH_BY_DATES_BUTTON_CLASS","SHOW_SEARCH_BY_DATES_BUTTON_VALUE","COMPLETED_TRANSACTIONS_TABLE","PENDING_TRANSACTIONS_TABLE","NEXT_PAGE_LINK","CURRENT_BALANCE","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","createLoginFields","credentials","selector","username","password","getAmountData","amountStr","amountStrCopy","replace","SHEKEL_CURRENCY_SYMBOL","replaceAll","parseFloat","getTxnAmount","txn","credit","debit","isNaN","convertTransactions","txns","map","convertedDate","moment","date","toISOString","convertedAmount","type","TransactionTypes","Normal","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","status","description","memo","getTransactionDate","tds","transactionType","transactionsColsTypes","trim","getTransactionDescription","getTransactionReference","getTransactionDebit","getTransactionCredit","extractTransactionDetails","txnRow","transactionStatus","innerTds","item","getTransactionsColsTypeClasses","page","tableLocator","result","typeClassesObjs","pageEvalAll","td","index","colClass","getAttribute","typeClassObj","extractTransaction","push","extractTransactions","transactionsRows","trs","tr","Array","from","getElementsByTagName","innerText","isNoTransactionInDateRangeError","hasErrorInfoElement","elementPresentOnPage","errorText","$eval","errorElement","searchByDates","startDate","clickButton","waitUntilElementFound","fillInput","format","waitForNavigation","getAccountNumber","selectedSnifAccount","option","checkIfHasNextPage","navigateToNextPage","scrapeTransactions","needToPaginate","hasNextPage","currentPageTxns","getAccountTransactions","Promise","race","noTransactionInRangeError","pendingTxns","TransactionStatuses","Pending","completedTxns","Completed","getCurrentBalance","balanceElement","$","balanceStr","evaluate","waitForPostLogin","fetchAccountData","accountNumber","balance","getAccountIdsBySelector","accountsIds","selectElement","document","getElementById","options","querySelectorAll","getTransactionsFrame","sleep","frames","targetFrame","find","f","name","Error","fetchAccounts","accounts","length","frame","accountData","accountId","select","BeinleumiGroupBaseScraper","BaseScraperWithBrowser","constructor","args","getLoginOptions","loginUrl","LOGIN_URL","fields","submitButtonSelector","postAction","possibleResults","preAction","fetchData","defaultStartMoment","subtract","add","startMomentLimit","year","toDate","startMoment","max","navigateTo","TRANSACTIONS_URL","success","_default","exports"],"sources":["../../src/scrapers/base-beinleumi-group.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Frame, type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport {\n  clickButton,\n  elementPresentOnPage,\n  fillInput,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { sleep } from '../helpers/waiting';\nimport {\n  TransactionStatuses, TransactionTypes,\n  type Transaction,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\n\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא נמצאו נתונים בנושא המבוקש';\nconst DATE_COLUMN_CLASS_COMPLETED = 'date first';\nconst DATE_COLUMN_CLASS_PENDING = 'first date';\nconst DESCRIPTION_COLUMN_CLASS_COMPLETED = 'reference wrap_normal';\nconst DESCRIPTION_COLUMN_CLASS_PENDING = 'details wrap_normal';\nconst REFERENCE_COLUMN_CLASS = 'details';\nconst DEBIT_COLUMN_CLASS = 'debit';\nconst CREDIT_COLUMN_CLASS = 'credit';\nconst ERROR_MESSAGE_CLASS = 'NO_DATA';\nconst ACCOUNTS_NUMBER = 'div.fibi_account span.acc_num';\nconst CLOSE_SEARCH_BY_DATES_BUTTON_CLASS = 'ui-datepicker-close';\nconst SHOW_SEARCH_BY_DATES_BUTTON_VALUE = 'הצג';\nconst COMPLETED_TRANSACTIONS_TABLE = 'table#dataTable077';\nconst PENDING_TRANSACTIONS_TABLE = 'table#dataTable023';\nconst NEXT_PAGE_LINK = 'a#Npage.paging';\nconst CURRENT_BALANCE = '.main_balance';\n\ntype TransactionsColsTypes = Record<string, number>;\ntype TransactionsTrTds = string[];\ntype TransactionsTr = { innerTds: TransactionsTrTds };\n\ninterface ScrapedTransaction {\n  reference: string;\n  date: string;\n  credit: string;\n  debit: string;\n  memo?: string;\n  description: string;\n  status: TransactionStatuses;\n}\n\nexport function getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [/fibi.*accountSummary/];\n  urls[LoginResults.InvalidPassword] = [/FibiMenu\\/Marketing\\/Private\\/Home/];\n  return urls;\n}\n\nexport function createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string) {\n  let amountStrCopy = amountStr.replace(SHEKEL_CURRENCY_SYMBOL, '');\n  amountStrCopy = amountStrCopy.replaceAll(',', '');\n  return parseFloat(amountStrCopy);\n}\n\nfunction getTxnAmount(txn: ScrapedTransaction) {\n  const credit = getAmountData(txn.credit);\n  const debit = getAmountData(txn.debit);\n  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn): Transaction => {\n    const convertedDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const convertedAmount = getTxnAmount(txn);\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: convertedDate,\n      processedDate: convertedDate,\n      originalAmount: convertedAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: convertedAmount,\n      status: txn.status,\n      description: txn.description,\n      memo: txn.memo,\n    };\n  });\n}\n\nfunction getTransactionDate(tds: TransactionsTrTds, transactionType: string, transactionsColsTypes: TransactionsColsTypes) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionDescription(tds: TransactionsTrTds, transactionType: string, transactionsColsTypes: TransactionsColsTypes) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionReference(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[REFERENCE_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionDebit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[DEBIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionCredit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[CREDIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction extractTransactionDetails(txnRow: TransactionsTr, transactionStatus: TransactionStatuses, transactionsColsTypes: TransactionsColsTypes): ScrapedTransaction {\n  const tds = txnRow.innerTds;\n  const item = {\n    status: transactionStatus,\n    date: getTransactionDate(tds, transactionStatus, transactionsColsTypes),\n    description: getTransactionDescription(tds, transactionStatus, transactionsColsTypes),\n    reference: getTransactionReference(tds, transactionsColsTypes),\n    debit: getTransactionDebit(tds, transactionsColsTypes),\n    credit: getTransactionCredit(tds, transactionsColsTypes),\n  };\n\n  return item;\n}\n\nasync function getTransactionsColsTypeClasses(page: Page | Frame, tableLocator: string): Promise<TransactionsColsTypes> {\n  const result: TransactionsColsTypes = {};\n  const typeClassesObjs = await pageEvalAll(page, `${tableLocator} tbody tr:first-of-type td`, null, (tds) => {\n    return tds.map((td, index) => ({\n      colClass: td.getAttribute('class'),\n      index,\n    }));\n  });\n\n  for (const typeClassObj of typeClassesObjs) {\n    if (typeClassObj.colClass) {\n      result[typeClassObj.colClass] = typeClassObj.index;\n    }\n  }\n  return result;\n}\n\nfunction extractTransaction(txns: ScrapedTransaction[], transactionStatus: TransactionStatuses, txnRow: TransactionsTr, transactionsColsTypes: TransactionsColsTypes) {\n  const txn = extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes);\n  if (txn.date !== '') {\n    txns.push(txn);\n  }\n}\n\nasync function extractTransactions(page: Page | Frame, tableLocator: string, transactionStatus: TransactionStatuses) {\n  const txns: ScrapedTransaction[] = [];\n  const transactionsColsTypes = await getTransactionsColsTypeClasses(page, tableLocator);\n\n  const transactionsRows = await pageEvalAll<TransactionsTr[]>(page, `${tableLocator} tbody tr`, [], (trs) => {\n    return trs.map((tr) => ({\n      innerTds: Array.from(tr.getElementsByTagName('td')).map((td) => td.innerText),\n    }));\n  });\n\n  for (const txnRow of transactionsRows) {\n    extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes);\n  }\n  return txns;\n}\n\nasync function isNoTransactionInDateRangeError(page: Page | Frame) {\n  const hasErrorInfoElement = await elementPresentOnPage(page, `.${ERROR_MESSAGE_CLASS}`);\n  if (hasErrorInfoElement) {\n    const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, (errorElement) => {\n      return (errorElement as HTMLElement).innerText;\n    });\n    return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;\n  }\n  return false;\n}\n\nasync function searchByDates(page: Page | Frame, startDate: Moment) {\n  await clickButton(page, 'a#tabHeader4');\n  await waitUntilElementFound(page, 'div#fibi_dates');\n  await fillInput(\n    page,\n    'input#fromDate',\n    startDate.format(DATE_FORMAT),\n  );\n  await clickButton(page, `button[class*=${CLOSE_SEARCH_BY_DATES_BUTTON_CLASS}]`);\n  await clickButton(page, `input[value=${SHOW_SEARCH_BY_DATES_BUTTON_VALUE}]`);\n  await waitForNavigation(page);\n}\n\nasync function getAccountNumber(page: Page | Frame) {\n  const selectedSnifAccount = await page.$eval(ACCOUNTS_NUMBER, (option) => {\n    return (option as HTMLElement).innerText;\n  });\n\n  return selectedSnifAccount.replace('/', '_').trim();\n}\n\nasync function checkIfHasNextPage(page: Page | Frame) {\n  return elementPresentOnPage(page, NEXT_PAGE_LINK);\n}\n\nasync function navigateToNextPage(page: Page | Frame) {\n  await clickButton(page, NEXT_PAGE_LINK);\n  await waitForNavigation(page);\n}\n\n/* Couldn't reproduce scenario with multiple pages of pending transactions - Should support if exists such case.\n   needToPaginate is false if scraping pending transactions */\nasync function scrapeTransactions(page: Page | Frame, tableLocator: string, transactionStatus: TransactionStatuses, needToPaginate: boolean) {\n  const txns = [];\n  let hasNextPage = false;\n\n  do {\n    const currentPageTxns = await extractTransactions(page, tableLocator, transactionStatus);\n    txns.push(...currentPageTxns);\n    if (needToPaginate) {\n      hasNextPage = await checkIfHasNextPage(page);\n      if (hasNextPage) {\n        await navigateToNextPage(page);\n      }\n    }\n  } while (hasNextPage);\n\n  return convertTransactions(txns);\n}\n\nasync function getAccountTransactions(page: Page | Frame) {\n  await Promise.race([\n    waitUntilElementFound(page, 'div[id*=\\'divTable\\']', false),\n    waitUntilElementFound(page, `.${ERROR_MESSAGE_CLASS}`, false),\n  ]);\n\n  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);\n  if (noTransactionInRangeError) {\n    return [];\n  }\n\n  const pendingTxns = await scrapeTransactions(page, PENDING_TRANSACTIONS_TABLE,\n    TransactionStatuses.Pending, false);\n  const completedTxns = await scrapeTransactions(page, COMPLETED_TRANSACTIONS_TABLE,\n    TransactionStatuses.Completed, true);\n  const txns = [\n    ...pendingTxns,\n    ...completedTxns,\n  ];\n  return txns;\n}\n\nasync function getCurrentBalance(page: Page | Frame) {\n  const balanceElement = await page.$(CURRENT_BALANCE);\n  if (!balanceElement) {\n    return undefined;\n  }\n  const balanceStr = await balanceElement.evaluate((option) => {\n    return (option as HTMLElement).innerText;\n  });\n  return getAmountData(balanceStr);\n}\n\nexport async function waitForPostLogin(page: Page) {\n  return Promise.race([\n    waitUntilElementFound(page, '#card-header', true),\n    waitUntilElementFound(page, '#account_num', true),\n  ]);\n}\n\nasync function fetchAccountData(page: Page | Frame, startDate: Moment) {\n  await searchByDates(page, startDate);\n  const accountNumber = await getAccountNumber(page);\n  const balance = await getCurrentBalance(page);\n  const txns = await getAccountTransactions(page);\n\n  return {\n    accountNumber,\n    txns,\n    balance,\n  };\n}\n\nasync function getAccountIdsBySelector(page: Page): Promise<string[]> {\n  const accountsIds = await page.evaluate(() => {\n    const selectElement = document.getElementById('account_num_select');\n    const options = selectElement ? selectElement.querySelectorAll('option') : [];\n    if (!options) return [];\n    return Array.from(options, (option) => option.value);\n  });\n  return accountsIds;\n}\n\nasync function getTransactionsFrame(page: Page): Promise<Frame> {\n  await sleep(5000);\n  const frames = page.frames();\n  const targetFrame = frames.find(f => f.name() === 'iframe-old-pages');\n\n  if (!targetFrame) {\n    throw new Error('iframe: \"iframe-old-pages\" (used for transactions page) was not found on the page');\n  }\n\n  return targetFrame;\n}\n\nasync function fetchAccounts(page: Page, startDate: Moment) {\n  const accounts: TransactionsAccount[] = [];\n  const accountsIds = await getAccountIdsBySelector(page);\n  if (accountsIds.length <= 1) {\n    const frame = await getTransactionsFrame(page);\n    const accountData = await fetchAccountData(frame, startDate);\n    accounts.push(accountData);\n  } else {\n    for (const accountId of accountsIds) {\n      await page.select('#account_num_select', accountId);\n      await waitUntilElementFound(page, '#account_num_select', true);\n      const accountData = await fetchAccountData(page, startDate);\n      accounts.push(accountData);\n    }\n  }\n  return accounts;\n}\n\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass BeinleumiGroupBaseScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  BASE_URL = '';\n\n  LOGIN_URL = '';\n\n  TRANSACTIONS_URL = '';\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n      // HACK: For some reason, though the login button (#continueBtn) is present and visible, the click action does not perform.\n      // Adding this delay fixes the issue.\n      preAction: async () => {\n        await sleep(1000);\n      },\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startMomentLimit = moment({ year: 1600 });\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(startMomentLimit, moment(startDate));\n\n    await this.navigateTo(this.TRANSACTIONS_URL);\n\n    const accounts = await fetchAccounts(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default BeinleumiGroupBaseScraper;\n"],"mappings":"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","ignoreList":[]}
334
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_constants","_elementsInteractions","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","_defineProperty","r","t","_toPropertyKey","Object","defineProperty","value","enumerable","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","DATE_FORMAT","NO_TRANSACTION_IN_DATE_RANGE_TEXT","DATE_COLUMN_CLASS_COMPLETED","DATE_COLUMN_CLASS_PENDING","DESCRIPTION_COLUMN_CLASS_COMPLETED","DESCRIPTION_COLUMN_CLASS_PENDING","REFERENCE_COLUMN_CLASS","DEBIT_COLUMN_CLASS","CREDIT_COLUMN_CLASS","ERROR_MESSAGE_CLASS","ACCOUNTS_NUMBER","CLOSE_SEARCH_BY_DATES_BUTTON_CLASS","SHOW_SEARCH_BY_DATES_BUTTON_VALUE","COMPLETED_TRANSACTIONS_TABLE","PENDING_TRANSACTIONS_TABLE","NEXT_PAGE_LINK","CURRENT_BALANCE","IFRAME_NAME","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","createLoginFields","credentials","selector","username","password","getAmountData","amountStr","amountStrCopy","replace","SHEKEL_CURRENCY_SYMBOL","replaceAll","parseFloat","getTxnAmount","txn","credit","debit","isNaN","convertTransactions","txns","map","convertedDate","moment","date","toISOString","convertedAmount","type","TransactionTypes","Normal","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","SHEKEL_CURRENCY","chargedAmount","status","description","memo","getTransactionDate","tds","transactionType","transactionsColsTypes","trim","getTransactionDescription","getTransactionReference","getTransactionDebit","getTransactionCredit","extractTransactionDetails","txnRow","transactionStatus","innerTds","item","getTransactionsColsTypeClasses","page","tableLocator","result","typeClassesObjs","pageEvalAll","td","index","colClass","getAttribute","typeClassObj","extractTransaction","push","extractTransactions","transactionsRows","trs","tr","Array","from","getElementsByTagName","innerText","isNoTransactionInDateRangeError","hasErrorInfoElement","elementPresentOnPage","errorText","$eval","errorElement","searchByDates","startDate","clickButton","waitUntilElementFound","fillInput","format","waitForNavigation","getAccountNumber","selectedSnifAccount","option","checkIfHasNextPage","navigateToNextPage","scrapeTransactions","needToPaginate","hasNextPage","currentPageTxns","getAccountTransactions","Promise","race","noTransactionInRangeError","pendingTxns","TransactionStatuses","Pending","completedTxns","Completed","getCurrentBalance","balanceElement","$","balanceStr","evaluate","waitForPostLogin","fetchAccountData","accountNumber","balance","getAccountIdsBySelector","accountsIds","selectElement","document","getElementById","options","querySelectorAll","getTransactionsFrame","attempt","sleep","frames","targetFrame","find","f","name","selectAccount","accountId","select","fetchAccountDataBothUIs","frame","targetPage","fetchAccounts","length","accountData","accounts","BeinleumiGroupBaseScraper","BaseScraperWithBrowser","constructor","args","getLoginOptions","loginUrl","LOGIN_URL","fields","submitButtonSelector","postAction","possibleResults","preAction","fetchData","defaultStartMoment","subtract","add","startMomentLimit","year","toDate","startMoment","max","navigateTo","TRANSACTIONS_URL","success","_default","exports"],"sources":["../../src/scrapers/base-beinleumi-group.ts"],"sourcesContent":["import moment, { type Moment } from 'moment';\nimport { type Frame, type Page } from 'puppeteer';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport {\n  clickButton,\n  elementPresentOnPage,\n  fillInput,\n  pageEvalAll,\n  waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { sleep } from '../helpers/waiting';\nimport {\n  TransactionStatuses, TransactionTypes,\n  type Transaction,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\n\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst NO_TRANSACTION_IN_DATE_RANGE_TEXT = 'לא נמצאו נתונים בנושא המבוקש';\nconst DATE_COLUMN_CLASS_COMPLETED = 'date first';\nconst DATE_COLUMN_CLASS_PENDING = 'first date';\nconst DESCRIPTION_COLUMN_CLASS_COMPLETED = 'reference wrap_normal';\nconst DESCRIPTION_COLUMN_CLASS_PENDING = 'details wrap_normal';\nconst REFERENCE_COLUMN_CLASS = 'details';\nconst DEBIT_COLUMN_CLASS = 'debit';\nconst CREDIT_COLUMN_CLASS = 'credit';\nconst ERROR_MESSAGE_CLASS = 'NO_DATA';\nconst ACCOUNTS_NUMBER = 'div.fibi_account span.acc_num';\nconst CLOSE_SEARCH_BY_DATES_BUTTON_CLASS = 'ui-datepicker-close';\nconst SHOW_SEARCH_BY_DATES_BUTTON_VALUE = 'הצג';\nconst COMPLETED_TRANSACTIONS_TABLE = 'table#dataTable077';\nconst PENDING_TRANSACTIONS_TABLE = 'table#dataTable023';\nconst NEXT_PAGE_LINK = 'a#Npage.paging';\nconst CURRENT_BALANCE = '.main_balance';\nconst IFRAME_NAME = 'iframe-old-pages';\n\ntype TransactionsColsTypes = Record<string, number>;\ntype TransactionsTrTds = string[];\ntype TransactionsTr = { innerTds: TransactionsTrTds };\n\ninterface ScrapedTransaction {\n  reference: string;\n  date: string;\n  credit: string;\n  debit: string;\n  memo?: string;\n  description: string;\n  status: TransactionStatuses;\n}\n\nexport function getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    /fibi.*accountSummary/,  // New UI pattern\n    /FibiMenu\\/Online/,       // Old UI pattern\n  ];\n  urls[LoginResults.InvalidPassword] = [/FibiMenu\\/Marketing\\/Private\\/Home/];\n  return urls;\n}\n\nexport function createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string) {\n  let amountStrCopy = amountStr.replace(SHEKEL_CURRENCY_SYMBOL, '');\n  amountStrCopy = amountStrCopy.replaceAll(',', '');\n  return parseFloat(amountStrCopy);\n}\n\nfunction getTxnAmount(txn: ScrapedTransaction) {\n  const credit = getAmountData(txn.credit);\n  const debit = getAmountData(txn.debit);\n  return (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn): Transaction => {\n    const convertedDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const convertedAmount = getTxnAmount(txn);\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: convertedDate,\n      processedDate: convertedDate,\n      originalAmount: convertedAmount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: convertedAmount,\n      status: txn.status,\n      description: txn.description,\n      memo: txn.memo,\n    };\n  });\n}\n\nfunction getTransactionDate(tds: TransactionsTrTds, transactionType: string, transactionsColsTypes: TransactionsColsTypes) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DATE_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionDescription(tds: TransactionsTrTds, transactionType: string, transactionsColsTypes: TransactionsColsTypes) {\n  if (transactionType === 'completed') {\n    return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_COMPLETED]] || '').trim();\n  }\n  return (tds[transactionsColsTypes[DESCRIPTION_COLUMN_CLASS_PENDING]] || '').trim();\n}\n\nfunction getTransactionReference(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[REFERENCE_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionDebit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[DEBIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction getTransactionCredit(tds: TransactionsTrTds, transactionsColsTypes: TransactionsColsTypes) {\n  return (tds[transactionsColsTypes[CREDIT_COLUMN_CLASS]] || '').trim();\n}\n\nfunction extractTransactionDetails(txnRow: TransactionsTr, transactionStatus: TransactionStatuses, transactionsColsTypes: TransactionsColsTypes): ScrapedTransaction {\n  const tds = txnRow.innerTds;\n  const item = {\n    status: transactionStatus,\n    date: getTransactionDate(tds, transactionStatus, transactionsColsTypes),\n    description: getTransactionDescription(tds, transactionStatus, transactionsColsTypes),\n    reference: getTransactionReference(tds, transactionsColsTypes),\n    debit: getTransactionDebit(tds, transactionsColsTypes),\n    credit: getTransactionCredit(tds, transactionsColsTypes),\n  };\n\n  return item;\n}\n\nasync function getTransactionsColsTypeClasses(page: Page | Frame, tableLocator: string): Promise<TransactionsColsTypes> {\n  const result: TransactionsColsTypes = {};\n  const typeClassesObjs = await pageEvalAll(page, `${tableLocator} tbody tr:first-of-type td`, null, (tds) => {\n    return tds.map((td, index) => ({\n      colClass: td.getAttribute('class'),\n      index,\n    }));\n  });\n\n  for (const typeClassObj of typeClassesObjs) {\n    if (typeClassObj.colClass) {\n      result[typeClassObj.colClass] = typeClassObj.index;\n    }\n  }\n  return result;\n}\n\nfunction extractTransaction(txns: ScrapedTransaction[], transactionStatus: TransactionStatuses, txnRow: TransactionsTr, transactionsColsTypes: TransactionsColsTypes) {\n  const txn = extractTransactionDetails(txnRow, transactionStatus, transactionsColsTypes);\n  if (txn.date !== '') {\n    txns.push(txn);\n  }\n}\n\nasync function extractTransactions(page: Page | Frame, tableLocator: string, transactionStatus: TransactionStatuses) {\n  const txns: ScrapedTransaction[] = [];\n  const transactionsColsTypes = await getTransactionsColsTypeClasses(page, tableLocator);\n\n  const transactionsRows = await pageEvalAll<TransactionsTr[]>(page, `${tableLocator} tbody tr`, [], (trs) => {\n    return trs.map((tr) => ({\n      innerTds: Array.from(tr.getElementsByTagName('td')).map((td) => td.innerText),\n    }));\n  });\n\n  for (const txnRow of transactionsRows) {\n    extractTransaction(txns, transactionStatus, txnRow, transactionsColsTypes);\n  }\n  return txns;\n}\n\nasync function isNoTransactionInDateRangeError(page: Page | Frame) {\n  const hasErrorInfoElement = await elementPresentOnPage(page, `.${ERROR_MESSAGE_CLASS}`);\n  if (hasErrorInfoElement) {\n    const errorText = await page.$eval(`.${ERROR_MESSAGE_CLASS}`, (errorElement) => {\n      return (errorElement as HTMLElement).innerText;\n    });\n    return errorText.trim() === NO_TRANSACTION_IN_DATE_RANGE_TEXT;\n  }\n  return false;\n}\n\nasync function searchByDates(page: Page | Frame, startDate: Moment) {\n  await clickButton(page, 'a#tabHeader4');\n  await waitUntilElementFound(page, 'div#fibi_dates');\n  await fillInput(\n    page,\n    'input#fromDate',\n    startDate.format(DATE_FORMAT),\n  );\n  await clickButton(page, `button[class*=${CLOSE_SEARCH_BY_DATES_BUTTON_CLASS}]`);\n  await clickButton(page, `input[value=${SHOW_SEARCH_BY_DATES_BUTTON_VALUE}]`);\n  await waitForNavigation(page);\n}\n\nasync function getAccountNumber(page: Page | Frame) {\n  const selectedSnifAccount = await page.$eval(ACCOUNTS_NUMBER, (option) => {\n    return (option as HTMLElement).innerText;\n  });\n\n  return selectedSnifAccount.replace('/', '_').trim();\n}\n\nasync function checkIfHasNextPage(page: Page | Frame) {\n  return elementPresentOnPage(page, NEXT_PAGE_LINK);\n}\n\nasync function navigateToNextPage(page: Page | Frame) {\n  await clickButton(page, NEXT_PAGE_LINK);\n  await waitForNavigation(page);\n}\n\n/* Couldn't reproduce scenario with multiple pages of pending transactions - Should support if exists such case.\n   needToPaginate is false if scraping pending transactions */\nasync function scrapeTransactions(page: Page | Frame, tableLocator: string, transactionStatus: TransactionStatuses, needToPaginate: boolean) {\n  const txns = [];\n  let hasNextPage = false;\n\n  do {\n    const currentPageTxns = await extractTransactions(page, tableLocator, transactionStatus);\n    txns.push(...currentPageTxns);\n    if (needToPaginate) {\n      hasNextPage = await checkIfHasNextPage(page);\n      if (hasNextPage) {\n        await navigateToNextPage(page);\n      }\n    }\n  } while (hasNextPage);\n\n  return convertTransactions(txns);\n}\n\nasync function getAccountTransactions(page: Page | Frame) {\n  await Promise.race([\n    waitUntilElementFound(page, 'div[id*=\\'divTable\\']', false),\n    waitUntilElementFound(page, `.${ERROR_MESSAGE_CLASS}`, false),\n  ]);\n\n  const noTransactionInRangeError = await isNoTransactionInDateRangeError(page);\n  if (noTransactionInRangeError) {\n    return [];\n  }\n\n  const pendingTxns = await scrapeTransactions(page, PENDING_TRANSACTIONS_TABLE,\n    TransactionStatuses.Pending, false);\n  const completedTxns = await scrapeTransactions(page, COMPLETED_TRANSACTIONS_TABLE,\n    TransactionStatuses.Completed, true);\n  const txns = [\n    ...pendingTxns,\n    ...completedTxns,\n  ];\n  return txns;\n}\n\nasync function getCurrentBalance(page: Page | Frame) {\n  const balanceElement = await page.$(CURRENT_BALANCE);\n  if (!balanceElement) {\n    return undefined;\n  }\n  const balanceStr = await balanceElement.evaluate((option) => {\n    return (option as HTMLElement).innerText;\n  });\n  return getAmountData(balanceStr);\n}\n\nexport async function waitForPostLogin(page: Page) {\n  return Promise.race([\n    waitUntilElementFound(page, '#card-header', true), // New UI\n    waitUntilElementFound(page, '#account_num', true), // New UI\n    waitUntilElementFound(page, '#matafLogoutLink', true), // Old UI\n    waitUntilElementFound(page, '#validationMsg', true), // Old UI\n  ]);\n}\n\nasync function fetchAccountData(page: Page | Frame, startDate: Moment) {\n  await searchByDates(page, startDate);\n  const accountNumber = await getAccountNumber(page);\n  const balance = await getCurrentBalance(page);\n  const txns = await getAccountTransactions(page);\n\n  return {\n    accountNumber,\n    txns,\n    balance,\n  };\n}\n\nasync function getAccountIdsBySelector(page: Page): Promise<string[]> {\n  const accountsIds = await page.evaluate(() => {\n    const selectElement = document.getElementById('account_num_select');\n    const options = selectElement ? selectElement.querySelectorAll('option') : [];\n    if (!options) return [];\n    return Array.from(options, (option) => option.value);\n  });\n  return accountsIds;\n}\n\nasync function getTransactionsFrame(page: Page): Promise<Frame | null> {\n  // Try a few times to find the iframe, as it might not be immediately available\n  for (let attempt = 0; attempt < 3; attempt++) {\n    await sleep(2000);\n    const frames = page.frames();\n    const targetFrame = frames.find(f => f.name() === IFRAME_NAME);\n    \n    if (targetFrame) {\n      return targetFrame;\n    }\n  }\n  \n  return null;\n}\n\nasync function selectAccount(page: Page, accountId: string) {\n  await page.select('#account_num_select', accountId);\n  await waitUntilElementFound(page, '#account_num_select', true);\n}\n\nasync function fetchAccountDataBothUIs(page: Page, startDate: Moment) {\n  // Try to get the iframe for the new UI\n  const frame = await getTransactionsFrame(page);\n    \n  // Use the frame if available (new UI), otherwise use the page directly (old UI)\n  const targetPage = frame || page;\n  return fetchAccountData(targetPage, startDate);\n}\n\nasync function fetchAccounts(page: Page, startDate: Moment):Promise<TransactionsAccount[]> {\n  const accountsIds = await getAccountIdsBySelector(page);\n\n  if (accountsIds.length <= 1) {\n    const accountData = await fetchAccountDataBothUIs(page, startDate);\n    return [accountData];\n  }\n\n  const accounts: TransactionsAccount[] = [];\n  for (const accountId of accountsIds) {\n    await selectAccount(page, accountId);\n    \n    const accountData = await fetchAccountDataBothUIs(page, startDate);\n    accounts.push(accountData);\n  }\n  \n  return accounts;\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass BeinleumiGroupBaseScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  BASE_URL = '';\n\n  LOGIN_URL = '';\n\n  TRANSACTIONS_URL = '';\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n      // HACK: For some reason, though the login button (#continueBtn) is present and visible, the click action does not perform.\n      // Adding this delay fixes the issue.\n      preAction: async () => {\n        await sleep(1000);\n      },\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startMomentLimit = moment({ year: 1600 });\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(startMomentLimit, moment(startDate));\n\n    await this.navigateTo(this.TRANSACTIONS_URL);\n\n    const accounts = await fetchAccounts(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default BeinleumiGroupBaseScraper;\n"],"mappings":"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","ignoreList":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "israeli-bank-scrapers",
3
- "version": "5.4.2",
3
+ "version": "5.4.3",
4
4
  "private": false,
5
5
  "description": "Provide scrapers for all major Israeli banks and credit card companies",
6
6
  "engines": {