israeli-bank-scrapers-core 6.4.2 → 6.5.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.
@@ -18,7 +18,7 @@ const DATE_FORMAT = 'YYYYMMDD';
18
18
 
19
19
  // eslint-disable-next-line @typescript-eslint/no-namespace
20
20
 
21
- function convertTransactions(txns) {
21
+ function convertTransactions(txns, options) {
22
22
  return txns.map(txn => {
23
23
  const isOutbound = txn.eventActivityTypeCode === 2;
24
24
  let memo = '';
@@ -58,6 +58,9 @@ function convertTransactions(txns) {
58
58
  status: txn.serialNumber === 0 ? _transactions.TransactionStatuses.Pending : _transactions.TransactionStatuses.Completed,
59
59
  memo
60
60
  };
61
+ if (options?.includeRawTransaction) {
62
+ result.rawTransaction = txn;
63
+ }
61
64
  return result;
62
65
  });
63
66
  }
@@ -110,11 +113,11 @@ async function getExtraScrap(txnsResult, baseUrl, page, accountNumber) {
110
113
  transactions: res
111
114
  };
112
115
  }
113
- async function getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDate, endDate, additionalTransactionInformation = false) {
116
+ async function getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDate, endDate, additionalTransactionInformation = false, options) {
114
117
  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;
115
118
  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');
116
119
  const finalResult = additionalTransactionInformation && txnsResult?.transactions.length ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber) : txnsResult;
117
- return convertTransactions(finalResult?.transactions ?? []);
120
+ return convertTransactions(finalResult?.transactions ?? [], options);
118
121
  }
119
122
  async function getAccountBalance(apiSiteUrl, page, accountNumber) {
120
123
  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;
@@ -142,7 +145,7 @@ async function fetchAccountData(page, baseUrl, options) {
142
145
  debug('getting information for account %s', account.accountNumber);
143
146
  const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;
144
147
  const balance = await getAccountBalance(apiSiteUrl, page, accountNumber);
145
- const txns = await getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDateStr, endDateStr, additionalTransactionInformation);
148
+ const txns = await getAccountTransactions(baseUrl, apiSiteUrl, page, accountNumber, startDateStr, endDateStr, additionalTransactionInformation, options);
146
149
  accounts.push({
147
150
  accountNumber,
148
151
  balance,
@@ -191,4 +194,4 @@ class HapoalimScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
191
194
  }
192
195
  }
193
196
  var _default = exports.default = HapoalimScraper;
194
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_uuid","_debug","_fetch","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","convertTransactions","txns","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","moment","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","waitUntil","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","uuid4","fetchPostWithinPage","getExtraScrap","txnsResult","baseUrl","accountNumber","promises","transactions","transaction","pfmDetails","extraTransactionDetails","fetchGetWithinPage","transactionNumber","res","Promise","all","getAccountTransactions","apiSiteUrl","startDate","endDate","additionalTransactionInformation","txnsUrl","finalResult","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","options","accountDataUrl","accountsInfo","openAccountsInfo","filter","account","accountClosingReasonCode","defaultStartMoment","subtract","add","toDate","startMoment","max","startDateStr","format","endDateStr","accounts","bankNumber","branchNumber","balance","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","waitForRedirect","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/hapoalim.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer-core';\nimport { v4 as uuid4 } from 'uuid';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('hapoalim');\n\nconst DATE_FORMAT = 'YYYYMMDD';\n\n// eslint-disable-next-line @typescript-eslint/no-namespace\ndeclare namespace window {\n  const bnhpApp: any;\n}\n\ninterface ScrapedTransaction {\n  serialNumber?: number;\n  activityDescription?: string;\n  eventAmount: number;\n  valueDate?: string;\n  eventDate?: string;\n  referenceNumber?: number;\n  ScrapedTransaction?: string;\n  eventActivityTypeCode: number;\n  currentBalance: number;\n  pfmDetails: string;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ninterface ScrapedPfmTransaction {\n  transactionNumber: number;\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\n  accountClosingReasonCode: number;\n}[];\n\ntype FetchedAccountTransactionsData = {\n  transactions: ScrapedTransaction[];\n};\n\ntype BalanceAndCreditLimit = {\n  creditLimitAmount: number;\n  creditLimitDescription: string;\n  creditLimitUtilizationAmount: number;\n  creditLimitUtilizationExistanceCode: number;\n  creditLimitUtilizationPercent: number;\n  currentAccountLimitsAmount: number;\n  currentBalance: number;\n  withdrawalBalance: number;\n};\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map(txn => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const { partyHeadline, partyName, messageHeadline, messageDetail } = txn.beneficiaryDetailsData;\n      const memoLines: string[] = [];\n      if (partyHeadline) {\n        memoLines.push(partyHeadline);\n      }\n\n      if (partyName) {\n        memoLines.push(`${partyName}.`);\n      }\n\n      if (messageHeadline) {\n        memoLines.push(messageHeadline);\n      }\n\n      if (messageDetail) {\n        memoLines.push(`${messageDetail}.`);\n      }\n\n      if (memoLines.length) {\n        memo = memoLines.join(' ');\n      }\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber,\n      date: moment(txn.eventDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.valueDate, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: txn.serialNumber === 0 ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      memo,\n    };\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(() => {\n    return page.evaluate(() => !!window.bnhpApp);\n  }, 'waiting for app data load');\n\n  const result = await page.evaluate(() => {\n    return window.bnhpApp.restContext;\n  });\n\n  return result.slice(1);\n}\n\nasync function fetchPoalimXSRFWithinPage(\n  page: Page,\n  url: string,\n  pageUuid: string,\n): Promise<FetchedAccountTransactionsData | null> {\n  const cookies = await page.cookies();\n  const XSRFCookie = cookies.find(cookie => cookie.name === 'XSRF-TOKEN');\n  const headers: Record<string, any> = {};\n  if (XSRFCookie != null) {\n    headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n  }\n  headers.pageUuid = pageUuid;\n  headers.uuid = uuid4();\n  headers['Content-Type'] = 'application/json;charset=UTF-8';\n  return fetchPostWithinPage<FetchedAccountTransactionsData>(page, url, [], headers);\n}\n\nasync function getExtraScrap(\n  txnsResult: FetchedAccountTransactionsData,\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n): Promise<FetchedAccountTransactionsData> {\n  const promises = txnsResult.transactions.map(async (transaction: ScrapedTransaction): Promise<ScrapedTransaction> => {\n    const { pfmDetails, serialNumber } = transaction;\n    if (serialNumber !== 0) {\n      const url = `${baseUrl}${pfmDetails}&accountId=${accountNumber}&lang=he`;\n      const extraTransactionDetails = (await fetchGetWithinPage<ScrapedPfmTransaction[]>(page, url)) || [];\n      if (extraTransactionDetails && extraTransactionDetails.length) {\n        const { transactionNumber } = extraTransactionDetails[0];\n        if (transactionNumber) {\n          return { ...transaction, referenceNumber: transactionNumber };\n        }\n      }\n    }\n    return transaction;\n  });\n  const res = await Promise.all(promises);\n  return { transactions: res };\n}\n\nasync function getAccountTransactions(\n  baseUrl: string,\n  apiSiteUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n  additionalTransactionInformation = false,\n) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  const finalResult =\n    additionalTransactionInformation && txnsResult?.transactions.length\n      ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber)\n      : txnsResult;\n\n  return convertTransactions(finalResult?.transactions ?? []);\n}\n\nasync function getAccountBalance(apiSiteUrl: string, page: Page, accountNumber: string) {\n  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n  const balanceAndCreditLimit = await fetchGetWithinPage<BalanceAndCreditLimit>(page, balanceAndCreditLimitUrl);\n\n  return balanceAndCreditLimit?.currentBalance;\n}\n\nasync function fetchAccountData(page: Page, baseUrl: string, options: ScraperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n\n  debug('fetching accounts data');\n  const accountsInfo = (await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl)) || [];\n  const openAccountsInfo = accountsInfo.filter(account => account.accountClosingReasonCode === 0);\n  debug(\n    'got %d open accounts from %d total accounts, fetching txns and balance',\n    openAccountsInfo.length,\n    accountsInfo.length,\n  );\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const { additionalTransactionInformation } = options;\n\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of openAccountsInfo) {\n    debug('getting information for account %s', account.accountNumber);\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    const txns = await getAccountTransactions(\n      baseUrl,\n      apiSiteUrl,\n      page,\n      accountNumber,\n      startDateStr,\n      endDateStr,\n      additionalTransactionInformation,\n    );\n\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n  debug('fetching ended');\n  return accountData;\n}\n\nfunction getPossibleLoginResults(baseUrl: string) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${baseUrl}/portalserver/HomePage`,\n    `${baseUrl}/ng-portals-bt/rb/he/homepage`,\n    `${baseUrl}/ng-portals/rb/he/homepage`,\n  ];\n  urls[LoginResults.InvalidPassword] = [\n    `${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`,\n  ];\n  urls[LoginResults.ChangePassword] = [\n    `${baseUrl}/MCP/START?flow=MCP&state=START&expiredDate=null`,\n    /\\/ABOUTTOEXPIRE\\/START/i,\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { userCode: string; password: string };\n\nclass HapoalimScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.baseUrl}/cgi-bin/poalwwwc?reqName=getLogonPage`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.login-btn',\n      postAction: async () => waitForRedirect(this.page),\n      possibleResults: getPossibleLoginResults(this.baseUrl),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.baseUrl, this.options);\n  }\n}\n\nexport default HapoalimScraper;\n"],"mappings":";;;;;;AAAA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AAEA,IAAAC,KAAA,GAAAD,OAAA;AACA,IAAAE,MAAA,GAAAF,OAAA;AACA,IAAAG,MAAA,GAAAH,OAAA;AACA,IAAAI,WAAA,GAAAJ,OAAA;AACA,IAAAK,QAAA,GAAAL,OAAA;AACA,IAAAM,aAAA,GAAAN,OAAA;AACA,IAAAO,uBAAA,GAAAP,OAAA;AAA8G,SAAAD,uBAAAS,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAG9G,MAAMG,KAAK,GAAG,IAAAC,eAAQ,EAAC,UAAU,CAAC;AAElC,MAAMC,WAAW,GAAG,UAAU;;AAE9B;;AAkDA,SAASC,mBAAmBA,CAACC,IAA0B,EAAiB;EACtE,OAAOA,IAAI,CAACC,GAAG,CAACC,GAAG,IAAI;IACrB,MAAMC,UAAU,GAAGD,GAAG,CAACE,qBAAqB,KAAK,CAAC;IAElD,IAAIC,IAAI,GAAG,EAAE;IACb,IAAIH,GAAG,CAACI,sBAAsB,EAAE;MAC9B,MAAM;QAAEC,aAAa;QAAEC,SAAS;QAAEC,eAAe;QAAEC;MAAc,CAAC,GAAGR,GAAG,CAACI,sBAAsB;MAC/F,MAAMK,SAAmB,GAAG,EAAE;MAC9B,IAAIJ,aAAa,EAAE;QACjBI,SAAS,CAACC,IAAI,CAACL,aAAa,CAAC;MAC/B;MAEA,IAAIC,SAAS,EAAE;QACbG,SAAS,CAACC,IAAI,CAAC,GAAGJ,SAAS,GAAG,CAAC;MACjC;MAEA,IAAIC,eAAe,EAAE;QACnBE,SAAS,CAACC,IAAI,CAACH,eAAe,CAAC;MACjC;MAEA,IAAIC,aAAa,EAAE;QACjBC,SAAS,CAACC,IAAI,CAAC,GAAGF,aAAa,GAAG,CAAC;MACrC;MAEA,IAAIC,SAAS,CAACE,MAAM,EAAE;QACpBR,IAAI,GAAGM,SAAS,CAACG,IAAI,CAAC,GAAG,CAAC;MAC5B;IACF;IAEA,MAAMC,MAAmB,GAAG;MAC1BC,IAAI,EAAEC,8BAAgB,CAACC,MAAM;MAC7BC,UAAU,EAAEjB,GAAG,CAACkB,eAAe;MAC/BC,IAAI,EAAE,IAAAC,eAAM,EAACpB,GAAG,CAACqB,SAAS,EAAEzB,WAAW,CAAC,CAAC0B,WAAW,CAAC,CAAC;MACtDC,aAAa,EAAE,IAAAH,eAAM,EAACpB,GAAG,CAACwB,SAAS,EAAE5B,WAAW,CAAC,CAAC0B,WAAW,CAAC,CAAC;MAC/DG,cAAc,EAAExB,UAAU,GAAG,CAACD,GAAG,CAAC0B,WAAW,GAAG1B,GAAG,CAAC0B,WAAW;MAC/DC,gBAAgB,EAAE,KAAK;MACvBC,aAAa,EAAE3B,UAAU,GAAG,CAACD,GAAG,CAAC0B,WAAW,GAAG1B,GAAG,CAAC0B,WAAW;MAC9DG,WAAW,EAAE7B,GAAG,CAAC8B,mBAAmB,IAAI,EAAE;MAC1CC,MAAM,EAAE/B,GAAG,CAACgC,YAAY,KAAK,CAAC,GAAGC,iCAAmB,CAACC,OAAO,GAAGD,iCAAmB,CAACE,SAAS;MAC5FhC;IACF,CAAC;IAED,OAAOU,MAAM;EACf,CAAC,CAAC;AACJ;AAEA,eAAeuB,cAAcA,CAACC,IAAU,EAAE;EACxC,MAAM,IAAAC,kBAAS,EAAC,MAAM;IACpB,OAAOD,IAAI,CAACE,QAAQ,CAAC,MAAM,CAAC,CAACC,MAAM,CAACC,OAAO,CAAC;EAC9C,CAAC,EAAE,2BAA2B,CAAC;EAE/B,MAAM5B,MAAM,GAAG,MAAMwB,IAAI,CAACE,QAAQ,CAAC,MAAM;IACvC,OAAOC,MAAM,CAACC,OAAO,CAACC,WAAW;EACnC,CAAC,CAAC;EAEF,OAAO7B,MAAM,CAAC8B,KAAK,CAAC,CAAC,CAAC;AACxB;AAEA,eAAeC,yBAAyBA,CACtCP,IAAU,EACVQ,GAAW,EACXC,QAAgB,EACgC;EAChD,MAAMC,OAAO,GAAG,MAAMV,IAAI,CAACU,OAAO,CAAC,CAAC;EACpC,MAAMC,UAAU,GAAGD,OAAO,CAACE,IAAI,CAACC,MAAM,IAAIA,MAAM,CAACC,IAAI,KAAK,YAAY,CAAC;EACvE,MAAMC,OAA4B,GAAG,CAAC,CAAC;EACvC,IAAIJ,UAAU,IAAI,IAAI,EAAE;IACtBI,OAAO,CAAC,cAAc,CAAC,GAAGJ,UAAU,CAACK,KAAK;EAC5C;EACAD,OAAO,CAACN,QAAQ,GAAGA,QAAQ;EAC3BM,OAAO,CAACE,IAAI,GAAG,IAAAC,QAAK,EAAC,CAAC;EACtBH,OAAO,CAAC,cAAc,CAAC,GAAG,gCAAgC;EAC1D,OAAO,IAAAI,0BAAmB,EAAiCnB,IAAI,EAAEQ,GAAG,EAAE,EAAE,EAAEO,OAAO,CAAC;AACpF;AAEA,eAAeK,aAAaA,CAC1BC,UAA0C,EAC1CC,OAAe,EACftB,IAAU,EACVuB,aAAqB,EACoB;EACzC,MAAMC,QAAQ,GAAGH,UAAU,CAACI,YAAY,CAAC/D,GAAG,CAAC,MAAOgE,WAA+B,IAAkC;IACnH,MAAM;MAAEC,UAAU;MAAEhC;IAAa,CAAC,GAAG+B,WAAW;IAChD,IAAI/B,YAAY,KAAK,CAAC,EAAE;MACtB,MAAMa,GAAG,GAAG,GAAGc,OAAO,GAAGK,UAAU,cAAcJ,aAAa,UAAU;MACxE,MAAMK,uBAAuB,GAAG,CAAC,MAAM,IAAAC,yBAAkB,EAA0B7B,IAAI,EAAEQ,GAAG,CAAC,KAAK,EAAE;MACpG,IAAIoB,uBAAuB,IAAIA,uBAAuB,CAACtD,MAAM,EAAE;QAC7D,MAAM;UAAEwD;QAAkB,CAAC,GAAGF,uBAAuB,CAAC,CAAC,CAAC;QACxD,IAAIE,iBAAiB,EAAE;UACrB,OAAO;YAAE,GAAGJ,WAAW;YAAE7C,eAAe,EAAEiD;UAAkB,CAAC;QAC/D;MACF;IACF;IACA,OAAOJ,WAAW;EACpB,CAAC,CAAC;EACF,MAAMK,GAAG,GAAG,MAAMC,OAAO,CAACC,GAAG,CAACT,QAAQ,CAAC;EACvC,OAAO;IAAEC,YAAY,EAAEM;EAAI,CAAC;AAC9B;AAEA,eAAeG,sBAAsBA,CACnCZ,OAAe,EACfa,UAAkB,EAClBnC,IAAU,EACVuB,aAAqB,EACrBa,SAAiB,EACjBC,OAAe,EACfC,gCAAgC,GAAG,KAAK,EACxC;EACA,MAAMC,OAAO,GAAG,GAAGJ,UAAU,2CAA2CZ,aAAa,0CAA0Cc,OAAO,uBAAuBD,SAAS,aAAa;EACnL,MAAMf,UAAU,GAAG,MAAMd,yBAAyB,CAACP,IAAI,EAAEuC,OAAO,EAAE,+BAA+B,CAAC;EAElG,MAAMC,WAAW,GACfF,gCAAgC,IAAIjB,UAAU,EAAEI,YAAY,CAACnD,MAAM,GAC/D,MAAM8C,aAAa,CAACC,UAAU,EAAEC,OAAO,EAAEtB,IAAI,EAAEuB,aAAa,CAAC,GAC7DF,UAAU;EAEhB,OAAO7D,mBAAmB,CAACgF,WAAW,EAAEf,YAAY,IAAI,EAAE,CAAC;AAC7D;AAEA,eAAegB,iBAAiBA,CAACN,UAAkB,EAAEnC,IAAU,EAAEuB,aAAqB,EAAE;EACtF,MAAMmB,wBAAwB,GAAG,GAAGP,UAAU,8DAA8DZ,aAAa,uBAAuB;EAChJ,MAAMoB,qBAAqB,GAAG,MAAM,IAAAd,yBAAkB,EAAwB7B,IAAI,EAAE0C,wBAAwB,CAAC;EAE7G,OAAOC,qBAAqB,EAAEC,cAAc;AAC9C;AAEA,eAAeC,gBAAgBA,CAAC7C,IAAU,EAAEsB,OAAe,EAAEwB,OAAuB,EAAE;EACpF,MAAMzC,WAAW,GAAG,MAAMN,cAAc,CAACC,IAAI,CAAC;EAC9C,MAAMmC,UAAU,GAAG,GAAGb,OAAO,IAAIjB,WAAW,EAAE;EAC9C,MAAM0C,cAAc,GAAG,GAAGzB,OAAO,kCAAkC;EAEnEjE,KAAK,CAAC,wBAAwB,CAAC;EAC/B,MAAM2F,YAAY,GAAG,CAAC,MAAM,IAAAnB,yBAAkB,EAAqB7B,IAAI,EAAE+C,cAAc,CAAC,KAAK,EAAE;EAC/F,MAAME,gBAAgB,GAAGD,YAAY,CAACE,MAAM,CAACC,OAAO,IAAIA,OAAO,CAACC,wBAAwB,KAAK,CAAC,CAAC;EAC/F/F,KAAK,CACH,wEAAwE,EACxE4F,gBAAgB,CAAC3E,MAAM,EACvB0E,YAAY,CAAC1E,MACf,CAAC;EAED,MAAM+E,kBAAkB,GAAG,IAAAtE,eAAM,EAAC,CAAC,CAACuE,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC,CAACC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC;EACtE,MAAMnB,SAAS,GAAGU,OAAO,CAACV,SAAS,IAAIiB,kBAAkB,CAACG,MAAM,CAAC,CAAC;EAClE,MAAMC,WAAW,GAAG1E,eAAM,CAAC2E,GAAG,CAACL,kBAAkB,EAAE,IAAAtE,eAAM,EAACqD,SAAS,CAAC,CAAC;EACrE,MAAM;IAAEE;EAAiC,CAAC,GAAGQ,OAAO;EAEpD,MAAMa,YAAY,GAAGF,WAAW,CAACG,MAAM,CAACrG,WAAW,CAAC;EACpD,MAAMsG,UAAU,GAAG,IAAA9E,eAAM,EAAC,CAAC,CAAC6E,MAAM,CAACrG,WAAW,CAAC;EAE/C,MAAMuG,QAA+B,GAAG,EAAE;EAE1C,KAAK,MAAMX,OAAO,IAAIF,gBAAgB,EAAE;IACtC5F,KAAK,CAAC,oCAAoC,EAAE8F,OAAO,CAAC5B,aAAa,CAAC;IAClE,MAAMA,aAAa,GAAG,GAAG4B,OAAO,CAACY,UAAU,IAAIZ,OAAO,CAACa,YAAY,IAAIb,OAAO,CAAC5B,aAAa,EAAE;IAE9F,MAAM0C,OAAO,GAAG,MAAMxB,iBAAiB,CAACN,UAAU,EAAEnC,IAAI,EAAEuB,aAAa,CAAC;IACxE,MAAM9D,IAAI,GAAG,MAAMyE,sBAAsB,CACvCZ,OAAO,EACPa,UAAU,EACVnC,IAAI,EACJuB,aAAa,EACboC,YAAY,EACZE,UAAU,EACVvB,gCACF,CAAC;IAEDwB,QAAQ,CAACzF,IAAI,CAAC;MACZkD,aAAa;MACb0C,OAAO;MACPxG;IACF,CAAC,CAAC;EACJ;EAEA,MAAMyG,WAAW,GAAG;IAClBC,OAAO,EAAE,IAAI;IACbL;EACF,CAAC;EACDzG,KAAK,CAAC,gBAAgB,CAAC;EACvB,OAAO6G,WAAW;AACpB;AAEA,SAASE,uBAAuBA,CAAC9C,OAAe,EAAE;EAChD,MAAM+C,IAA0B,GAAG,CAAC,CAAC;EACrCA,IAAI,CAACC,oCAAY,CAACC,OAAO,CAAC,GAAG,CAC3B,GAAGjD,OAAO,wBAAwB,EAClC,GAAGA,OAAO,+BAA+B,EACzC,GAAGA,OAAO,4BAA4B,CACvC;EACD+C,IAAI,CAACC,oCAAY,CAACE,eAAe,CAAC,GAAG,CACnC,GAAGlD,OAAO,8EAA8E,CACzF;EACD+C,IAAI,CAACC,oCAAY,CAACG,cAAc,CAAC,GAAG,CAClC,GAAGnD,OAAO,kDAAkD,EAC5D,yBAAyB,CAC1B;EACD,OAAO+C,IAAI;AACb;AAEA,SAASK,iBAAiBA,CAACC,WAAuC,EAAE;EAClE,OAAO,CACL;IAAEC,QAAQ,EAAE,WAAW;IAAE5D,KAAK,EAAE2D,WAAW,CAACE;EAAS,CAAC,EACtD;IAAED,QAAQ,EAAE,WAAW;IAAE5D,KAAK,EAAE2D,WAAW,CAACG;EAAS,CAAC,CACvD;AACH;AAIA,MAAMC,eAAe,SAASC,8CAAsB,CAA6B;EAC/E;EACA,IAAI1D,OAAOA,CAAA,EAAG;IACZ,OAAO,kCAAkC;EAC3C;EAEA2D,eAAeA,CAACN,WAAuC,EAAE;IACvD,OAAO;MACLO,QAAQ,EAAE,GAAG,IAAI,CAAC5D,OAAO,wCAAwC;MACjE6D,MAAM,EAAET,iBAAiB,CAACC,WAAW,CAAC;MACtCS,oBAAoB,EAAE,YAAY;MAClCC,UAAU,EAAE,MAAAA,CAAA,KAAY,IAAAC,2BAAe,EAAC,IAAI,CAACtF,IAAI,CAAC;MAClDuF,eAAe,EAAEnB,uBAAuB,CAAC,IAAI,CAAC9C,OAAO;IACvD,CAAC;EACH;EAEA,MAAMkE,SAASA,CAAA,EAAG;IAChB,OAAO3C,gBAAgB,CAAC,IAAI,CAAC7C,IAAI,EAAE,IAAI,CAACsB,OAAO,EAAE,IAAI,CAACwB,OAAO,CAAC;EAChE;AACF;AAAC,IAAA2C,QAAA,GAAAC,OAAA,CAAAtI,OAAA,GAEc2H,eAAe","ignoreList":[]}
197
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_uuid","_debug","_fetch","_navigation","_waiting","_transactions","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","DATE_FORMAT","convertTransactions","txns","options","map","txn","isOutbound","eventActivityTypeCode","memo","beneficiaryDetailsData","partyHeadline","partyName","messageHeadline","messageDetail","memoLines","push","length","join","result","type","TransactionTypes","Normal","identifier","referenceNumber","date","moment","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","includeRawTransaction","rawTransaction","getRestContext","page","waitUntil","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","uuid4","fetchPostWithinPage","getExtraScrap","txnsResult","baseUrl","accountNumber","promises","transactions","transaction","pfmDetails","extraTransactionDetails","fetchGetWithinPage","transactionNumber","res","Promise","all","getAccountTransactions","apiSiteUrl","startDate","endDate","additionalTransactionInformation","txnsUrl","finalResult","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","accountDataUrl","accountsInfo","openAccountsInfo","filter","account","accountClosingReasonCode","defaultStartMoment","subtract","add","toDate","startMoment","max","startDateStr","format","endDateStr","accounts","bankNumber","branchNumber","balance","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","waitForRedirect","possibleResults","fetchData","_default","exports"],"sources":["../../src/scrapers/hapoalim.ts"],"sourcesContent":["import moment from 'moment';\nimport { type Page } from 'puppeteer-core';\nimport { v4 as uuid4 } from 'uuid';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { type Transaction, TransactionStatuses, TransactionTypes, type TransactionsAccount } from '../transactions';\nimport { BaseScraperWithBrowser, LoginResults, type PossibleLoginResults } from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('hapoalim');\n\nconst DATE_FORMAT = 'YYYYMMDD';\n\n// eslint-disable-next-line @typescript-eslint/no-namespace\ndeclare namespace window {\n  const bnhpApp: any;\n}\n\ninterface ScrapedTransaction {\n  serialNumber?: number;\n  activityDescription?: string;\n  eventAmount: number;\n  valueDate?: string;\n  eventDate?: string;\n  referenceNumber?: number;\n  ScrapedTransaction?: string;\n  eventActivityTypeCode: number;\n  currentBalance: number;\n  pfmDetails: string;\n  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ninterface ScrapedPfmTransaction {\n  transactionNumber: number;\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\n  accountClosingReasonCode: number;\n}[];\n\ntype FetchedAccountTransactionsData = {\n  transactions: ScrapedTransaction[];\n};\n\ntype BalanceAndCreditLimit = {\n  creditLimitAmount: number;\n  creditLimitDescription: string;\n  creditLimitUtilizationAmount: number;\n  creditLimitUtilizationExistanceCode: number;\n  creditLimitUtilizationPercent: number;\n  currentAccountLimitsAmount: number;\n  currentBalance: number;\n  withdrawalBalance: number;\n};\n\nfunction convertTransactions(txns: ScrapedTransaction[], options?: ScraperOptions): Transaction[] {\n  return txns.map(txn => {\n    const isOutbound = txn.eventActivityTypeCode === 2;\n\n    let memo = '';\n    if (txn.beneficiaryDetailsData) {\n      const { partyHeadline, partyName, messageHeadline, messageDetail } = txn.beneficiaryDetailsData;\n      const memoLines: string[] = [];\n      if (partyHeadline) {\n        memoLines.push(partyHeadline);\n      }\n\n      if (partyName) {\n        memoLines.push(`${partyName}.`);\n      }\n\n      if (messageHeadline) {\n        memoLines.push(messageHeadline);\n      }\n\n      if (messageDetail) {\n        memoLines.push(`${messageDetail}.`);\n      }\n\n      if (memoLines.length) {\n        memo = memoLines.join(' ');\n      }\n    }\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      identifier: txn.referenceNumber,\n      date: moment(txn.eventDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.valueDate, DATE_FORMAT).toISOString(),\n      originalAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: isOutbound ? -txn.eventAmount : txn.eventAmount,\n      description: txn.activityDescription || '',\n      status: txn.serialNumber === 0 ? TransactionStatuses.Pending : TransactionStatuses.Completed,\n      memo,\n    };\n\n    if (options?.includeRawTransaction) {\n      result.rawTransaction = txn;\n    }\n\n    return result;\n  });\n}\n\nasync function getRestContext(page: Page) {\n  await waitUntil(() => {\n    return page.evaluate(() => !!window.bnhpApp);\n  }, 'waiting for app data load');\n\n  const result = await page.evaluate(() => {\n    return window.bnhpApp.restContext;\n  });\n\n  return result.slice(1);\n}\n\nasync function fetchPoalimXSRFWithinPage(\n  page: Page,\n  url: string,\n  pageUuid: string,\n): Promise<FetchedAccountTransactionsData | null> {\n  const cookies = await page.cookies();\n  const XSRFCookie = cookies.find(cookie => cookie.name === 'XSRF-TOKEN');\n  const headers: Record<string, any> = {};\n  if (XSRFCookie != null) {\n    headers['X-XSRF-TOKEN'] = XSRFCookie.value;\n  }\n  headers.pageUuid = pageUuid;\n  headers.uuid = uuid4();\n  headers['Content-Type'] = 'application/json;charset=UTF-8';\n  return fetchPostWithinPage<FetchedAccountTransactionsData>(page, url, [], headers);\n}\n\nasync function getExtraScrap(\n  txnsResult: FetchedAccountTransactionsData,\n  baseUrl: string,\n  page: Page,\n  accountNumber: string,\n): Promise<FetchedAccountTransactionsData> {\n  const promises = txnsResult.transactions.map(async (transaction: ScrapedTransaction): Promise<ScrapedTransaction> => {\n    const { pfmDetails, serialNumber } = transaction;\n    if (serialNumber !== 0) {\n      const url = `${baseUrl}${pfmDetails}&accountId=${accountNumber}&lang=he`;\n      const extraTransactionDetails = (await fetchGetWithinPage<ScrapedPfmTransaction[]>(page, url)) || [];\n      if (extraTransactionDetails && extraTransactionDetails.length) {\n        const { transactionNumber } = extraTransactionDetails[0];\n        if (transactionNumber) {\n          return { ...transaction, referenceNumber: transactionNumber };\n        }\n      }\n    }\n    return transaction;\n  });\n  const res = await Promise.all(promises);\n  return { transactions: res };\n}\n\nasync function getAccountTransactions(\n  baseUrl: string,\n  apiSiteUrl: string,\n  page: Page,\n  accountNumber: string,\n  startDate: string,\n  endDate: string,\n  additionalTransactionInformation = false,\n  options?: ScraperOptions,\n) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=1000&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  const finalResult =\n    additionalTransactionInformation && txnsResult?.transactions.length\n      ? await getExtraScrap(txnsResult, baseUrl, page, accountNumber)\n      : txnsResult;\n\n  return convertTransactions(finalResult?.transactions ?? [], options);\n}\n\nasync function getAccountBalance(apiSiteUrl: string, page: Page, accountNumber: string) {\n  const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n  const balanceAndCreditLimit = await fetchGetWithinPage<BalanceAndCreditLimit>(page, balanceAndCreditLimitUrl);\n\n  return balanceAndCreditLimit?.currentBalance;\n}\n\nasync function fetchAccountData(page: Page, baseUrl: string, options: ScraperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n\n  debug('fetching accounts data');\n  const accountsInfo = (await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl)) || [];\n  const openAccountsInfo = accountsInfo.filter(account => account.accountClosingReasonCode === 0);\n  debug(\n    'got %d open accounts from %d total accounts, fetching txns and balance',\n    openAccountsInfo.length,\n    accountsInfo.length,\n  );\n\n  const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const { additionalTransactionInformation } = options;\n\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of openAccountsInfo) {\n    debug('getting information for account %s', account.accountNumber);\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    const txns = await getAccountTransactions(\n      baseUrl,\n      apiSiteUrl,\n      page,\n      accountNumber,\n      startDateStr,\n      endDateStr,\n      additionalTransactionInformation,\n      options,\n    );\n\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n  debug('fetching ended');\n  return accountData;\n}\n\nfunction getPossibleLoginResults(baseUrl: string) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [\n    `${baseUrl}/portalserver/HomePage`,\n    `${baseUrl}/ng-portals-bt/rb/he/homepage`,\n    `${baseUrl}/ng-portals/rb/he/homepage`,\n  ];\n  urls[LoginResults.InvalidPassword] = [\n    `${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`,\n  ];\n  urls[LoginResults.ChangePassword] = [\n    `${baseUrl}/MCP/START?flow=MCP&state=START&expiredDate=null`,\n    /\\/ABOUTTOEXPIRE\\/START/i,\n  ];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = { userCode: string; password: string };\n\nclass HapoalimScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\n    return {\n      loginUrl: `${this.baseUrl}/cgi-bin/poalwwwc?reqName=getLogonPage`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.login-btn',\n      postAction: async () => waitForRedirect(this.page),\n      possibleResults: getPossibleLoginResults(this.baseUrl),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.baseUrl, this.options);\n  }\n}\n\nexport default HapoalimScraper;\n"],"mappings":"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","ignoreList":[]}
@@ -132,6 +132,11 @@ export type ScraperOptions = ScraperBrowserOptions & {
132
132
  * Please note: It will take more time to finish the process.
133
133
  */
134
134
  additionalTransactionInformation?: boolean;
135
+ /**
136
+ * Include the raw transaction object as received from the scraper source for debugging purposes.
137
+ * @default false
138
+ */
139
+ includeRawTransaction?: boolean;
135
140
  /**
136
141
  * Adjust the viewport size of the browser page.
137
142
  * If not set, the default viewport size of 1024x768 will be used.
@@ -3,4 +3,4 @@
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":[],"sources":["../../src/scrapers/interface.ts"],"sourcesContent":["import { type BrowserContext, type Browser, type Page } from 'puppeteer-core';\nimport { type CompanyTypes, type ScraperProgressTypes } from '../definitions';\nimport { type TransactionsAccount } from '../transactions';\nimport { type ErrorResult, type ScraperErrorTypes } from './errors';\n\n// TODO: Remove this type when the scraper 'factory' will return concrete scraper types\n// Instead of a generic interface (which in turn uses this type)\nexport type ScraperCredentials =\n  | { userCode: string; password: string }\n  | { username: string; password: string }\n  | { id: string; password: string }\n  | { id: string; password: string; num: string }\n  | { id: string; password: string; card6Digits: string }\n  | { username: string; nationalID: string; password: string }\n  | ({ email: string; password: string } & (\n      | {\n          otpCodeRetriever: () => Promise<string>;\n          phoneNumber: string;\n        }\n      | {\n          otpLongTermToken: string;\n        }\n    ));\n\nexport type OptInFeatures =\n  | 'isracard-amex:skipAdditionalTransactionInformation'\n  | 'mizrahi:pendingIfNoIdentifier'\n  | 'mizrahi:pendingIfHasGenericDescription'\n  | 'mizrahi:pendingIfTodayTransaction';\n\nexport interface FutureDebit {\n  amount: number;\n  amountCurrency: string;\n  chargeDate?: string;\n  bankAccountNumber?: string;\n}\n\ninterface ExternalBrowserOptions {\n  /**\n   * An externally created browser instance.\n   * you can get a browser directly from puppeteer via `puppeteer.launch()`\n   *\n   * Note: The browser will be closed by the library after the scraper finishes unless `skipCloseBrowser` is set to true\n   */\n  browser: Browser;\n\n  /**\n   * If true, the browser will not be closed by the library after the scraper finishes\n   */\n  skipCloseBrowser?: boolean;\n}\n\ninterface ExternalBrowserContextOptions {\n  /**\n   * An externally managed browser context. This is useful when you want to manage the browser\n   */\n  browserContext: BrowserContext;\n}\n\ninterface DefaultBrowserOptions {\n  /**\n   * shows the browser while scraping, good for debugging (default false)\n   */\n  showBrowser?: boolean;\n\n  /**\n   * provide a patch to local chromium to be used by puppeteer. Relevant when using\n   * `israeli-bank-scrapers-core` library\n   */\n  executablePath?: string;\n\n  /**\n   * additional arguments to pass to the browser instance. The list of flags can be found in\n   *\n   * https://developer.mozilla.org/en-US/docs/Mozilla/Command_Line_Options\n   * https://peter.sh/experiments/chromium-command-line-switches/\n   */\n  args?: string[];\n\n  /**\n   * Maximum navigation time in milliseconds, pass 0 to disable timeout.\n   * @default 30000\n   */\n  timeout?: number;\n\n  /**\n   * adjust the browser instance before it is being used\n   *\n   * @param browser\n   */\n  prepareBrowser?: (browser: Browser) => Promise<void>;\n}\n\ntype ScraperBrowserOptions = ExternalBrowserOptions | ExternalBrowserContextOptions | DefaultBrowserOptions;\n\nexport type ScraperOptions = ScraperBrowserOptions & {\n  /**\n   * The company you want to scrape\n   */\n  companyId: CompanyTypes;\n\n  /**\n   * include more debug info about in the output\n   */\n  verbose?: boolean;\n\n  /**\n   * the date to fetch transactions from (can't be before the minimum allowed time difference for the scraper)\n   */\n  startDate: Date;\n\n  /**\n   * scrape transactions to be processed X months in the future\n   */\n  futureMonthsToScrape?: number;\n\n  /**\n   * if set to true, all installment transactions will be combine into the first one\n   */\n  combineInstallments?: boolean;\n\n  /**\n   * adjust the page instance before it is being used.\n   *\n   * @param page\n   */\n  preparePage?: (page: Page) => Promise<void>;\n\n  /**\n   * if set, store a screenshot if failed to scrape. Used for debug purposes\n   */\n  storeFailureScreenShotPath?: string;\n\n  /**\n   * if set, will set the timeout in milliseconds of puppeteer's `page.setDefaultTimeout`.\n   */\n  defaultTimeout?: number;\n\n  /**\n   * Options for manipulation of output data\n   */\n  outputData?: OutputDataOptions;\n\n  /**\n   * Perform additional operation for each transaction to get more information (Like category) about it.\n   * Please note: It will take more time to finish the process.\n   */\n  additionalTransactionInformation?: boolean;\n\n  /**\n   * Adjust the viewport size of the browser page.\n   * If not set, the default viewport size of 1024x768 will be used.\n   */\n  viewportSize?: {\n    width: number;\n    height: number;\n  };\n\n  /**\n   * The number of times to retry the navigation in case of a failure (default 0)\n   */\n  navigationRetryCount?: number;\n\n  /**\n   * Opt-in features for the scrapers, allowing safe rollout of new breaking changes.\n   */\n  optInFeatures?: Array<OptInFeatures>;\n};\n\nexport interface OutputDataOptions {\n  /**\n   * if true, the result wouldn't be filtered out by date, and you will return unfiltered scrapped data.\n   */\n  enableTransactionsFilterByDate?: boolean;\n}\n\nexport interface ScraperScrapingResult {\n  success: boolean;\n  accounts?: TransactionsAccount[];\n  futureDebits?: FutureDebit[];\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport interface Scraper<TCredentials extends ScraperCredentials> {\n  scrape(credentials: TCredentials): Promise<ScraperScrapingResult>;\n  onProgress(func: (companyId: CompanyTypes, payload: { type: ScraperProgressTypes }) => void): void;\n  triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult>;\n  getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult>;\n}\n\nexport type ScraperTwoFactorAuthTriggerResult =\n  | ErrorResult\n  | {\n      success: true;\n    };\n\nexport type ScraperGetLongTermTwoFactorTokenResult =\n  | ErrorResult\n  | {\n      success: true;\n      longTermTwoFactorAuthToken: string;\n    };\n\nexport interface ScraperLoginResult {\n  success: boolean;\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n  persistentOtpToken?: string;\n}\n"],"mappings":"","ignoreList":[]}
6
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":[],"sources":["../../src/scrapers/interface.ts"],"sourcesContent":["import { type BrowserContext, type Browser, type Page } from 'puppeteer-core';\nimport { type CompanyTypes, type ScraperProgressTypes } from '../definitions';\nimport { type TransactionsAccount } from '../transactions';\nimport { type ErrorResult, type ScraperErrorTypes } from './errors';\n\n// TODO: Remove this type when the scraper 'factory' will return concrete scraper types\n// Instead of a generic interface (which in turn uses this type)\nexport type ScraperCredentials =\n  | { userCode: string; password: string }\n  | { username: string; password: string }\n  | { id: string; password: string }\n  | { id: string; password: string; num: string }\n  | { id: string; password: string; card6Digits: string }\n  | { username: string; nationalID: string; password: string }\n  | ({ email: string; password: string } & (\n      | {\n          otpCodeRetriever: () => Promise<string>;\n          phoneNumber: string;\n        }\n      | {\n          otpLongTermToken: string;\n        }\n    ));\n\nexport type OptInFeatures =\n  | 'isracard-amex:skipAdditionalTransactionInformation'\n  | 'mizrahi:pendingIfNoIdentifier'\n  | 'mizrahi:pendingIfHasGenericDescription'\n  | 'mizrahi:pendingIfTodayTransaction';\n\nexport interface FutureDebit {\n  amount: number;\n  amountCurrency: string;\n  chargeDate?: string;\n  bankAccountNumber?: string;\n}\n\ninterface ExternalBrowserOptions {\n  /**\n   * An externally created browser instance.\n   * you can get a browser directly from puppeteer via `puppeteer.launch()`\n   *\n   * Note: The browser will be closed by the library after the scraper finishes unless `skipCloseBrowser` is set to true\n   */\n  browser: Browser;\n\n  /**\n   * If true, the browser will not be closed by the library after the scraper finishes\n   */\n  skipCloseBrowser?: boolean;\n}\n\ninterface ExternalBrowserContextOptions {\n  /**\n   * An externally managed browser context. This is useful when you want to manage the browser\n   */\n  browserContext: BrowserContext;\n}\n\ninterface DefaultBrowserOptions {\n  /**\n   * shows the browser while scraping, good for debugging (default false)\n   */\n  showBrowser?: boolean;\n\n  /**\n   * provide a patch to local chromium to be used by puppeteer. Relevant when using\n   * `israeli-bank-scrapers-core` library\n   */\n  executablePath?: string;\n\n  /**\n   * additional arguments to pass to the browser instance. The list of flags can be found in\n   *\n   * https://developer.mozilla.org/en-US/docs/Mozilla/Command_Line_Options\n   * https://peter.sh/experiments/chromium-command-line-switches/\n   */\n  args?: string[];\n\n  /**\n   * Maximum navigation time in milliseconds, pass 0 to disable timeout.\n   * @default 30000\n   */\n  timeout?: number;\n\n  /**\n   * adjust the browser instance before it is being used\n   *\n   * @param browser\n   */\n  prepareBrowser?: (browser: Browser) => Promise<void>;\n}\n\ntype ScraperBrowserOptions = ExternalBrowserOptions | ExternalBrowserContextOptions | DefaultBrowserOptions;\n\nexport type ScraperOptions = ScraperBrowserOptions & {\n  /**\n   * The company you want to scrape\n   */\n  companyId: CompanyTypes;\n\n  /**\n   * include more debug info about in the output\n   */\n  verbose?: boolean;\n\n  /**\n   * the date to fetch transactions from (can't be before the minimum allowed time difference for the scraper)\n   */\n  startDate: Date;\n\n  /**\n   * scrape transactions to be processed X months in the future\n   */\n  futureMonthsToScrape?: number;\n\n  /**\n   * if set to true, all installment transactions will be combine into the first one\n   */\n  combineInstallments?: boolean;\n\n  /**\n   * adjust the page instance before it is being used.\n   *\n   * @param page\n   */\n  preparePage?: (page: Page) => Promise<void>;\n\n  /**\n   * if set, store a screenshot if failed to scrape. Used for debug purposes\n   */\n  storeFailureScreenShotPath?: string;\n\n  /**\n   * if set, will set the timeout in milliseconds of puppeteer's `page.setDefaultTimeout`.\n   */\n  defaultTimeout?: number;\n\n  /**\n   * Options for manipulation of output data\n   */\n  outputData?: OutputDataOptions;\n\n  /**\n   * Perform additional operation for each transaction to get more information (Like category) about it.\n   * Please note: It will take more time to finish the process.\n   */\n  additionalTransactionInformation?: boolean;\n\n  /**\n   * Include the raw transaction object as received from the scraper source for debugging purposes.\n   * @default false\n   */\n  includeRawTransaction?: boolean;\n\n  /**\n   * Adjust the viewport size of the browser page.\n   * If not set, the default viewport size of 1024x768 will be used.\n   */\n  viewportSize?: {\n    width: number;\n    height: number;\n  };\n\n  /**\n   * The number of times to retry the navigation in case of a failure (default 0)\n   */\n  navigationRetryCount?: number;\n\n  /**\n   * Opt-in features for the scrapers, allowing safe rollout of new breaking changes.\n   */\n  optInFeatures?: Array<OptInFeatures>;\n};\n\nexport interface OutputDataOptions {\n  /**\n   * if true, the result wouldn't be filtered out by date, and you will return unfiltered scrapped data.\n   */\n  enableTransactionsFilterByDate?: boolean;\n}\n\nexport interface ScraperScrapingResult {\n  success: boolean;\n  accounts?: TransactionsAccount[];\n  futureDebits?: FutureDebit[];\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport interface Scraper<TCredentials extends ScraperCredentials> {\n  scrape(credentials: TCredentials): Promise<ScraperScrapingResult>;\n  onProgress(func: (companyId: CompanyTypes, payload: { type: ScraperProgressTypes }) => void): void;\n  triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult>;\n  getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult>;\n}\n\nexport type ScraperTwoFactorAuthTriggerResult =\n  | ErrorResult\n  | {\n      success: true;\n    };\n\nexport type ScraperGetLongTermTwoFactorTokenResult =\n  | ErrorResult\n  | {\n      success: true;\n      longTermTwoFactorAuthToken: string;\n    };\n\nexport interface ScraperLoginResult {\n  success: boolean;\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n  persistentOtpToken?: string;\n}\n"],"mappings":"","ignoreList":[]}
@@ -56,7 +56,7 @@ function createLoginFields(credentials) {
56
56
  value: credentials.password
57
57
  }];
58
58
  }
59
- function extractTransactionsFromPage(transactions, status) {
59
+ function extractTransactionsFromPage(transactions, status, options) {
60
60
  if (transactions === null || transactions.length === 0) {
61
61
  return [];
62
62
  }
@@ -74,6 +74,9 @@ function extractTransactionsFromPage(transactions, status) {
74
74
  chargedAmount: rawTransaction.Amount,
75
75
  originalAmount: rawTransaction.Amount
76
76
  };
77
+ if (options?.includeRawTransaction) {
78
+ newTransaction.rawTransaction = rawTransaction;
79
+ }
77
80
  return newTransaction;
78
81
  });
79
82
  return result;
@@ -96,7 +99,7 @@ async function clickByXPath(page, xpath) {
96
99
  function removeSpecialCharacters(str) {
97
100
  return str.replace(/[^0-9/-]/g, '');
98
101
  }
99
- async function fetchTransactionsForAccount(page, startDate, accountId) {
102
+ async function fetchTransactionsForAccount(page, startDate, accountId, options) {
100
103
  // DEVELOPER NOTICE the account number received from the server is being altered at
101
104
  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.
102
105
  await hangProcess(4000);
@@ -119,8 +122,8 @@ async function fetchTransactionsForAccount(page, startDate, accountId) {
119
122
  const pendingTransactions = response.TodayTransactionsItems;
120
123
  const transactions = response.HistoryTransactionsItems;
121
124
  const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;
122
- const pendingTxns = extractTransactionsFromPage(pendingTransactions, _transactions.TransactionStatuses.Pending);
123
- const completedTxns = extractTransactionsFromPage(transactions, _transactions.TransactionStatuses.Completed);
125
+ const pendingTxns = extractTransactionsFromPage(pendingTransactions, _transactions.TransactionStatuses.Pending, options);
126
+ const completedTxns = extractTransactionsFromPage(transactions, _transactions.TransactionStatuses.Completed, options);
124
127
  const txns = [...pendingTxns, ...completedTxns];
125
128
  return {
126
129
  accountNumber,
@@ -128,7 +131,7 @@ async function fetchTransactionsForAccount(page, startDate, accountId) {
128
131
  txns
129
132
  };
130
133
  }
131
- async function fetchTransactions(page, startDate) {
134
+ async function fetchTransactions(page, startDate, options) {
132
135
  const accounts = [];
133
136
 
134
137
  // DEVELOPER NOTICE the account number received from the server is being altered at
@@ -147,7 +150,7 @@ async function fetchTransactions(page, startDate) {
147
150
  await clickByXPath(page, 'xpath///*[contains(@class, "number") and contains(@class, "combo-inner")]');
148
151
  await clickByXPath(page, `xpath///span[contains(text(), '${accountId}')]`);
149
152
  }
150
- accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));
153
+ accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId), options));
151
154
  }
152
155
  return accounts;
153
156
  }
@@ -189,7 +192,7 @@ class LeumiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
189
192
  const startDate = this.options.startDate || defaultStartMoment.toDate();
190
193
  const startMoment = _moment.default.max(minimumStartMoment, (0, _moment.default)(startDate));
191
194
  await this.navigateTo(TRANSACTIONS_URL);
192
- const accounts = await fetchTransactions(this.page, startMoment);
195
+ const accounts = await fetchTransactions(this.page, startMoment, this.options);
193
196
  return {
194
197
  success: true,
195
198
  accounts
@@ -197,4 +200,4 @@ class LeumiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
197
200
  }
198
201
  }
199
202
  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-core';\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":[]}
203
+ //# 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","includeRawTransaction","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-core';\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 ScraperOptions, 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(\n  transactions: any[],\n  status: TransactionStatuses,\n  options?: ScraperOptions,\n): 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    if (options?.includeRawTransaction) {\n      newTransaction.rawTransaction = rawTransaction;\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  options: ScraperOptions,\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, options);\n  const completedTxns = extractTransactionsFromPage(transactions, TransactionStatuses.Completed, options);\n  const txns = [...pendingTxns, ...completedTxns];\n\n  return {\n    accountNumber,\n    balance,\n    txns,\n  };\n}\n\nasync function fetchTransactions(\n  page: Page,\n  startDate: Moment,\n  options: ScraperOptions,\n): 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), options));\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, this.options);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default LeumiScraper;\n"],"mappings":"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","ignoreList":[]}