israeli-bank-scrapers 1.9.0 → 1.10.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.
package/README.md CHANGED
@@ -287,7 +287,6 @@ These are the projects known to be using this module:
287
287
  - [Israeli YNAB updater](https://github.com/eshaham/israeli-ynab-updater) - A command line tool for exporting banks data to CSVs, formatted specifically for [YNAB](https://www.youneedabudget.com)
288
288
  - [Israel Finance Telegram Bot](https://github.com/GuyLewin/israel-finance-telegram-bot) - A simple telegram bot that sends notifications about new transactions and interacts with them
289
289
  - [Caspion](https://github.com/brafdlog/caspion) - An app for automatically sending transactions from Israeli banks and credit cards to budget tracking apps
290
- - [Oshi](https://github.com/baruchiro/israeli-bank-scrapers-desktop) - Secure desktop app for retriving your transactions from all israeli banks and credit cards
291
290
  - [Finance Notifier](https://github.com/LiranBri/finance-notifier) - A simple script with the ability to send custom financial alerts to multiple contacts and platforms
292
291
 
293
292
  Built something interesting you want to share here? [Let me know](https://goo.gl/forms/5Fb9JAjvzMIpmzqo2).
@@ -186,6 +186,11 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
186
186
  width: VIEWPORT_WIDTH,
187
187
  height: VIEWPORT_HEIGHT
188
188
  });
189
+ this.page.on('requestfailed', request => {
190
+ var _request$failure;
191
+
192
+ debug('Request failed: %s %s', (_request$failure = request.failure()) === null || _request$failure === void 0 ? void 0 : _request$failure.errorText, request.url());
193
+ });
189
194
  }
190
195
 
191
196
  async navigateTo(url, page, timeout) {
@@ -300,4 +305,4 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
300
305
  }
301
306
 
302
307
  exports.BaseScraperWithBrowser = BaseScraperWithBrowser;
303
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper-with-browser.ts"],"names":["VIEWPORT_WIDTH","VIEWPORT_HEIGHT","OK_STATUS","debug","LoginBaseResults","Timeout","Generic","General","ScraperErrorTypes","rest","LoginResults","getKeyByValue","object","value","page","keys","Object","key","conditions","condition","result","RegExp","test","toLowerCase","Promise","resolve","UnknownError","handleLoginResult","scraper","loginResult","Success","emitProgress","ScaperProgressTypes","LoginSuccess","success","InvalidPassword","LoginFailed","errorType","errorMessage","ChangePassword","Error","createGeneralError","BaseScraperWithBrowser","BaseScraper","initialize","Initializing","env","options","verbose","DEBUG","process","browser","executablePath","undefined","args","headless","showBrowser","puppeteer","launch","prepareBrowser","pages","length","newPage","preparePage","setViewport","width","height","navigateTo","url","timeout","pageToUse","response","goto","status","getLoginOptions","_credentials","companyId","fillInputs","pageOrFrame","fields","modified","input","shift","selector","login","credentials","loginOptions","userAgent","setUserAgent","loginUrl","checkReadiness","submitButtonSelector","loginFrameOrPage","preAction","LoggingIn","postAction","current","possibleResults","terminate","_success","Terminating","storeFailureScreenShotPath","screenshot","path","fullPage","close"],"mappings":"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","sourcesContent":["import puppeteer, { Browser, Frame, Page } from 'puppeteer';\n\nimport {\n  ScraperErrorTypes,\n  BaseScraper, ScaperScrapingResult, ScaperProgressTypes,\n  ScraperCredentials,\n} from './base-scraper';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { clickButton, fillInput, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { getDebug } from '../helpers/debug';\n\nconst VIEWPORT_WIDTH = 1024;\nconst VIEWPORT_HEIGHT = 768;\nconst OK_STATUS = 200;\n\nconst debug = getDebug('base-scraper-with-browser');\n\nenum LoginBaseResults {\n  Success = 'SUCCESS',\n  UnknownError = 'UNKNOWN_ERROR'\n}\n\nconst {\n  Timeout, Generic, General, ...rest\n} = ScraperErrorTypes;\nexport const LoginResults = {\n  ...rest,\n  ...LoginBaseResults,\n};\n\nexport type LoginResults = Exclude<ScraperErrorTypes,\nScraperErrorTypes.Timeout\n| ScraperErrorTypes.Generic\n| ScraperErrorTypes.General> | LoginBaseResults;\n\nexport type PossibleLoginResults = {\n  [key in LoginResults]?: (string | RegExp | ((options?: { page?: Page}) => Promise<boolean>))[]\n};\n\nexport interface LoginOptions {\n  loginUrl: string;\n  checkReadiness?: () => Promise<void>;\n  fields: {selector: string, value: string}[];\n  submitButtonSelector: string;\n  preAction?: () => Promise<Frame | void>;\n  postAction?: () => Promise<void>;\n  possibleResults: PossibleLoginResults;\n  userAgent?: string;\n}\n\nasync function getKeyByValue(object: PossibleLoginResults, value: string, page: Page): Promise<LoginResults> {\n  const keys = Object.keys(object);\n  for (const key of keys) {\n    // @ts-ignore\n    const conditions = object[key];\n\n    for (const condition of conditions) {\n      let result = false;\n\n      if (condition instanceof RegExp) {\n        result = condition.test(value);\n      } else if (typeof condition === 'function') {\n        result = await condition({ page, value });\n      } else {\n        result = value.toLowerCase() === condition.toLowerCase();\n      }\n\n      if (result) {\n        // @ts-ignore\n        return Promise.resolve(key);\n      }\n    }\n  }\n\n  return Promise.resolve(LoginResults.UnknownError);\n}\n\nfunction handleLoginResult(scraper: BaseScraperWithBrowser, loginResult: LoginResults) {\n  switch (loginResult) {\n    case LoginResults.Success:\n      scraper.emitProgress(ScaperProgressTypes.LoginSuccess);\n      return { success: true };\n    case LoginResults.InvalidPassword:\n    case LoginResults.UnknownError:\n      scraper.emitProgress(ScaperProgressTypes.LoginFailed);\n      return {\n        success: false,\n        errorType: loginResult === LoginResults.InvalidPassword ? ScraperErrorTypes.InvalidPassword :\n          ScraperErrorTypes.General,\n        errorMessage: `Login failed with ${loginResult} error`,\n      };\n    case LoginResults.ChangePassword:\n      scraper.emitProgress(ScaperProgressTypes.ChangePassword);\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.ChangePassword,\n      };\n    default:\n      throw new Error(`unexpected login result \"${loginResult}\"`);\n  }\n}\n\nfunction createGeneralError(): ScaperScrapingResult {\n  return {\n    success: false,\n    errorType: ScraperErrorTypes.General,\n  };\n}\n\nclass BaseScraperWithBrowser extends BaseScraper {\n  // NOTICE - it is discourage to use bang (!) in general. It is used here because\n  // all the classes that inherit from this base assume is it mandatory.\n  protected browser!: Browser;\n\n  // NOTICE - it is discourage to use bang (!) in general. It is used here because\n  // all the classes that inherit from this base assume is it mandatory.\n  protected page!: Page;\n\n  async initialize() {\n    debug('initialize scraper');\n    this.emitProgress(ScaperProgressTypes.Initializing);\n\n    let env: Record<string, any> | undefined;\n    if (this.options.verbose) {\n      env = { DEBUG: '*', ...process.env };\n    }\n\n    if (typeof this.options.browser !== 'undefined' && this.options.browser !== null) {\n      debug('use custom browser instance provided in options');\n      this.browser = this.options.browser;\n    } else {\n      const executablePath = this.options.executablePath || undefined;\n      const args = this.options.args || [];\n\n      const headless = !this.options.showBrowser;\n      debug(`launch a browser with headless mode = ${headless}`);\n      this.browser = await puppeteer.launch({\n        env,\n        headless,\n        executablePath,\n        args,\n      });\n    }\n\n    if (this.options.prepareBrowser) {\n      debug('execute \\'prepareBrowser\\' interceptor provided in options');\n      await this.options.prepareBrowser(this.browser);\n    }\n\n    if (!this.browser) {\n      debug('failed to initiate a browser, exit');\n      return;\n    }\n\n    const pages = await this.browser.pages();\n    if (pages.length) {\n      debug('browser has already pages open, use the first one');\n      [this.page] = pages;\n    } else {\n      debug('create a new browser page');\n      this.page = await this.browser.newPage();\n    }\n\n    if (this.options.preparePage) {\n      debug('execute \\'preparePage\\' interceptor provided in options');\n      await this.options.preparePage(this.page);\n    }\n\n    debug(`set viewport to width ${VIEWPORT_WIDTH}, height ${VIEWPORT_HEIGHT}`);\n    await this.page.setViewport({\n      width: VIEWPORT_WIDTH,\n      height: VIEWPORT_HEIGHT,\n    });\n  }\n\n  async navigateTo(url: string, page?: Page, timeout?: number): Promise<void> {\n    const pageToUse = page || this.page;\n\n    if (!pageToUse) {\n      return;\n    }\n\n    const options = { ...(timeout === null ? null : { timeout }) };\n    const response = await pageToUse.goto(url, options);\n\n    // note: response will be null when navigating to same url while changing the hash part. the condition below will always accept null as valid result.\n    if (response !== null && (response === undefined || response.status() !== OK_STATUS)) {\n      throw new Error(`Error while trying to navigate to url ${url}`);\n    }\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  getLoginOptions(_credentials: ScraperCredentials): LoginOptions {\n    throw new Error(`getLoginOptions() is not created in ${this.options.companyId}`);\n  }\n\n  async fillInputs(pageOrFrame: Page | Frame, fields: { selector: string, value: string}[]): Promise<void> {\n    const modified = [...fields];\n    const input = modified.shift();\n\n    if (!input) {\n      return;\n    }\n    await fillInput(pageOrFrame, input.selector, input.value);\n    if (modified.length) {\n      await this.fillInputs(pageOrFrame, modified);\n    }\n  }\n\n  async login(credentials: Record<string, string>): Promise<ScaperScrapingResult> {\n    if (!credentials || !this.page) {\n      return createGeneralError();\n    }\n\n    debug('execute login process');\n    const loginOptions = this.getLoginOptions(credentials);\n\n    if (loginOptions.userAgent) {\n      debug('set custom user agent provided in options');\n      await this.page.setUserAgent(loginOptions.userAgent);\n    }\n\n    debug('navigate to login url');\n    await this.navigateTo(loginOptions.loginUrl);\n    if (loginOptions.checkReadiness) {\n      debug('execute \\'checkReadiness\\' interceptor provided in login options');\n      await loginOptions.checkReadiness();\n    } else {\n      debug('wait until submit button is available');\n      await waitUntilElementFound(this.page, loginOptions.submitButtonSelector);\n    }\n\n    let loginFrameOrPage: (Page | Frame | null) = this.page;\n    if (loginOptions.preAction) {\n      debug('execute \\'preAction\\' interceptor provided in login options');\n      loginFrameOrPage = await loginOptions.preAction() || this.page;\n    }\n\n    debug('fill login components input with relevant values');\n    await this.fillInputs(loginFrameOrPage, loginOptions.fields);\n    debug('click on login submit button');\n    await clickButton(loginFrameOrPage, loginOptions.submitButtonSelector);\n    this.emitProgress(ScaperProgressTypes.LoggingIn);\n\n    if (loginOptions.postAction) {\n      debug('execute \\'postAction\\' interceptor provided in login options');\n      await loginOptions.postAction();\n    } else {\n      debug('wait for page navigation');\n      await waitForNavigation(this.page);\n    }\n\n    debug('check login result');\n    const current = await getCurrentUrl(this.page, true);\n    const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);\n    debug(`handle login results ${loginResult}`);\n    return handleLoginResult(this, loginResult);\n  }\n\n  async terminate(_success: boolean) {\n    debug(`terminating browser with success = ${_success}`);\n    this.emitProgress(ScaperProgressTypes.Terminating);\n\n    if (!_success && !!this.options.storeFailureScreenShotPath) {\n      debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);\n      await this.page.screenshot({\n        path: this.options.storeFailureScreenShotPath,\n        fullPage: true,\n      });\n    }\n\n    if (!this.browser) {\n      return;\n    }\n\n    await this.browser.close();\n  }\n}\n\nexport { BaseScraperWithBrowser };\n"]}
308
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper-with-browser.ts"],"names":["VIEWPORT_WIDTH","VIEWPORT_HEIGHT","OK_STATUS","debug","LoginBaseResults","Timeout","Generic","General","ScraperErrorTypes","rest","LoginResults","getKeyByValue","object","value","page","keys","Object","key","conditions","condition","result","RegExp","test","toLowerCase","Promise","resolve","UnknownError","handleLoginResult","scraper","loginResult","Success","emitProgress","ScaperProgressTypes","LoginSuccess","success","InvalidPassword","LoginFailed","errorType","errorMessage","ChangePassword","Error","createGeneralError","BaseScraperWithBrowser","BaseScraper","initialize","Initializing","env","options","verbose","DEBUG","process","browser","executablePath","undefined","args","headless","showBrowser","puppeteer","launch","prepareBrowser","pages","length","newPage","preparePage","setViewport","width","height","on","request","failure","errorText","url","navigateTo","timeout","pageToUse","response","goto","status","getLoginOptions","_credentials","companyId","fillInputs","pageOrFrame","fields","modified","input","shift","selector","login","credentials","loginOptions","userAgent","setUserAgent","loginUrl","checkReadiness","submitButtonSelector","loginFrameOrPage","preAction","LoggingIn","postAction","current","possibleResults","terminate","_success","Terminating","storeFailureScreenShotPath","screenshot","path","fullPage","close"],"mappings":"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","sourcesContent":["import puppeteer, { Browser, Frame, Page } from 'puppeteer';\n\nimport {\n  ScraperErrorTypes,\n  BaseScraper, ScaperScrapingResult, ScaperProgressTypes,\n  ScraperCredentials,\n} from './base-scraper';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { clickButton, fillInput, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { getDebug } from '../helpers/debug';\n\nconst VIEWPORT_WIDTH = 1024;\nconst VIEWPORT_HEIGHT = 768;\nconst OK_STATUS = 200;\n\nconst debug = getDebug('base-scraper-with-browser');\n\nenum LoginBaseResults {\n  Success = 'SUCCESS',\n  UnknownError = 'UNKNOWN_ERROR'\n}\n\nconst {\n  Timeout, Generic, General, ...rest\n} = ScraperErrorTypes;\nexport const LoginResults = {\n  ...rest,\n  ...LoginBaseResults,\n};\n\nexport type LoginResults = Exclude<ScraperErrorTypes,\nScraperErrorTypes.Timeout\n| ScraperErrorTypes.Generic\n| ScraperErrorTypes.General> | LoginBaseResults;\n\nexport type PossibleLoginResults = {\n  [key in LoginResults]?: (string | RegExp | ((options?: { page?: Page}) => Promise<boolean>))[]\n};\n\nexport interface LoginOptions {\n  loginUrl: string;\n  checkReadiness?: () => Promise<void>;\n  fields: {selector: string, value: string}[];\n  submitButtonSelector: string;\n  preAction?: () => Promise<Frame | void>;\n  postAction?: () => Promise<void>;\n  possibleResults: PossibleLoginResults;\n  userAgent?: string;\n}\n\nasync function getKeyByValue(object: PossibleLoginResults, value: string, page: Page): Promise<LoginResults> {\n  const keys = Object.keys(object);\n  for (const key of keys) {\n    // @ts-ignore\n    const conditions = object[key];\n\n    for (const condition of conditions) {\n      let result = false;\n\n      if (condition instanceof RegExp) {\n        result = condition.test(value);\n      } else if (typeof condition === 'function') {\n        result = await condition({ page, value });\n      } else {\n        result = value.toLowerCase() === condition.toLowerCase();\n      }\n\n      if (result) {\n        // @ts-ignore\n        return Promise.resolve(key);\n      }\n    }\n  }\n\n  return Promise.resolve(LoginResults.UnknownError);\n}\n\nfunction handleLoginResult(scraper: BaseScraperWithBrowser, loginResult: LoginResults) {\n  switch (loginResult) {\n    case LoginResults.Success:\n      scraper.emitProgress(ScaperProgressTypes.LoginSuccess);\n      return { success: true };\n    case LoginResults.InvalidPassword:\n    case LoginResults.UnknownError:\n      scraper.emitProgress(ScaperProgressTypes.LoginFailed);\n      return {\n        success: false,\n        errorType: loginResult === LoginResults.InvalidPassword ? ScraperErrorTypes.InvalidPassword :\n          ScraperErrorTypes.General,\n        errorMessage: `Login failed with ${loginResult} error`,\n      };\n    case LoginResults.ChangePassword:\n      scraper.emitProgress(ScaperProgressTypes.ChangePassword);\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.ChangePassword,\n      };\n    default:\n      throw new Error(`unexpected login result \"${loginResult}\"`);\n  }\n}\n\nfunction createGeneralError(): ScaperScrapingResult {\n  return {\n    success: false,\n    errorType: ScraperErrorTypes.General,\n  };\n}\n\nclass BaseScraperWithBrowser extends BaseScraper {\n  // NOTICE - it is discourage to use bang (!) in general. It is used here because\n  // all the classes that inherit from this base assume is it mandatory.\n  protected browser!: Browser;\n\n  // NOTICE - it is discourage to use bang (!) in general. It is used here because\n  // all the classes that inherit from this base assume is it mandatory.\n  protected page!: Page;\n\n  async initialize() {\n    debug('initialize scraper');\n    this.emitProgress(ScaperProgressTypes.Initializing);\n\n    let env: Record<string, any> | undefined;\n    if (this.options.verbose) {\n      env = { DEBUG: '*', ...process.env };\n    }\n\n    if (typeof this.options.browser !== 'undefined' && this.options.browser !== null) {\n      debug('use custom browser instance provided in options');\n      this.browser = this.options.browser;\n    } else {\n      const executablePath = this.options.executablePath || undefined;\n      const args = this.options.args || [];\n\n      const headless = !this.options.showBrowser;\n      debug(`launch a browser with headless mode = ${headless}`);\n      this.browser = await puppeteer.launch({\n        env,\n        headless,\n        executablePath,\n        args,\n      });\n    }\n\n    if (this.options.prepareBrowser) {\n      debug('execute \\'prepareBrowser\\' interceptor provided in options');\n      await this.options.prepareBrowser(this.browser);\n    }\n\n    if (!this.browser) {\n      debug('failed to initiate a browser, exit');\n      return;\n    }\n\n    const pages = await this.browser.pages();\n    if (pages.length) {\n      debug('browser has already pages open, use the first one');\n      [this.page] = pages;\n    } else {\n      debug('create a new browser page');\n      this.page = await this.browser.newPage();\n    }\n\n    if (this.options.preparePage) {\n      debug('execute \\'preparePage\\' interceptor provided in options');\n      await this.options.preparePage(this.page);\n    }\n\n    debug(`set viewport to width ${VIEWPORT_WIDTH}, height ${VIEWPORT_HEIGHT}`);\n    await this.page.setViewport({\n      width: VIEWPORT_WIDTH,\n      height: VIEWPORT_HEIGHT,\n    });\n\n    this.page.on('requestfailed', (request) => {\n      debug('Request failed: %s %s', request.failure()?.errorText, request.url());\n    });\n  }\n\n  async navigateTo(url: string, page?: Page, timeout?: number): Promise<void> {\n    const pageToUse = page || this.page;\n\n    if (!pageToUse) {\n      return;\n    }\n\n    const options = { ...(timeout === null ? null : { timeout }) };\n    const response = await pageToUse.goto(url, options);\n\n    // note: response will be null when navigating to same url while changing the hash part. the condition below will always accept null as valid result.\n    if (response !== null && (response === undefined || response.status() !== OK_STATUS)) {\n      throw new Error(`Error while trying to navigate to url ${url}`);\n    }\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  getLoginOptions(_credentials: ScraperCredentials): LoginOptions {\n    throw new Error(`getLoginOptions() is not created in ${this.options.companyId}`);\n  }\n\n  async fillInputs(pageOrFrame: Page | Frame, fields: { selector: string, value: string}[]): Promise<void> {\n    const modified = [...fields];\n    const input = modified.shift();\n\n    if (!input) {\n      return;\n    }\n    await fillInput(pageOrFrame, input.selector, input.value);\n    if (modified.length) {\n      await this.fillInputs(pageOrFrame, modified);\n    }\n  }\n\n  async login(credentials: Record<string, string>): Promise<ScaperScrapingResult> {\n    if (!credentials || !this.page) {\n      return createGeneralError();\n    }\n\n    debug('execute login process');\n    const loginOptions = this.getLoginOptions(credentials);\n\n    if (loginOptions.userAgent) {\n      debug('set custom user agent provided in options');\n      await this.page.setUserAgent(loginOptions.userAgent);\n    }\n\n    debug('navigate to login url');\n    await this.navigateTo(loginOptions.loginUrl);\n    if (loginOptions.checkReadiness) {\n      debug('execute \\'checkReadiness\\' interceptor provided in login options');\n      await loginOptions.checkReadiness();\n    } else {\n      debug('wait until submit button is available');\n      await waitUntilElementFound(this.page, loginOptions.submitButtonSelector);\n    }\n\n    let loginFrameOrPage: (Page | Frame | null) = this.page;\n    if (loginOptions.preAction) {\n      debug('execute \\'preAction\\' interceptor provided in login options');\n      loginFrameOrPage = await loginOptions.preAction() || this.page;\n    }\n\n    debug('fill login components input with relevant values');\n    await this.fillInputs(loginFrameOrPage, loginOptions.fields);\n    debug('click on login submit button');\n    await clickButton(loginFrameOrPage, loginOptions.submitButtonSelector);\n    this.emitProgress(ScaperProgressTypes.LoggingIn);\n\n    if (loginOptions.postAction) {\n      debug('execute \\'postAction\\' interceptor provided in login options');\n      await loginOptions.postAction();\n    } else {\n      debug('wait for page navigation');\n      await waitForNavigation(this.page);\n    }\n\n    debug('check login result');\n    const current = await getCurrentUrl(this.page, true);\n    const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);\n    debug(`handle login results ${loginResult}`);\n    return handleLoginResult(this, loginResult);\n  }\n\n  async terminate(_success: boolean) {\n    debug(`terminating browser with success = ${_success}`);\n    this.emitProgress(ScaperProgressTypes.Terminating);\n\n    if (!_success && !!this.options.storeFailureScreenShotPath) {\n      debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);\n      await this.page.screenshot({\n        path: this.options.storeFailureScreenShotPath,\n        fullPage: true,\n      });\n    }\n\n    if (!this.browser) {\n      return;\n    }\n\n    await this.browser.close();\n  }\n}\n\nexport { BaseScraperWithBrowser };\n"]}
@@ -107,7 +107,7 @@ async function navigateOrErrorLabel(page) {
107
107
 
108
108
  function getPossibleLoginResults() {
109
109
  const urls = {};
110
- urls[_baseScraperWithBrowser.LoginResults.Success] = [`${BASE_URL}/apollo/core/templates/RETAIL/masterPage.html#/MY_ACCOUNT_HOMEPAGE`];
110
+ urls[_baseScraperWithBrowser.LoginResults.Success] = [`${BASE_URL}/apollo/retail/#/MY_ACCOUNT_HOMEPAGE`];
111
111
  urls[_baseScraperWithBrowser.LoginResults.InvalidPassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`];
112
112
  urls[_baseScraperWithBrowser.LoginResults.ChangePassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/PWD_RENEW`];
113
113
  return urls;
@@ -146,4 +146,4 @@ class DiscountScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
146
146
 
147
147
  var _default = DiscountScraper;
148
148
  exports.default = _default;
149
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/discount.ts"],"names":["BASE_URL","DATE_FORMAT","convertTransactions","txns","txnStatus","map","txn","type","TransactionTypes","Normal","identifier","OperationNumber","date","OperationDate","toISOString","processedDate","ValueDate","originalAmount","OperationAmount","originalCurrency","chargedAmount","description","OperationDescriptionToDisplay","status","fetchAccountData","page","options","apiSiteUrl","accountDataUrl","accountInfo","success","errorType","ScraperErrorTypes","Generic","errorMessage","accountNumber","UserAccountsData","DefaultAccountNumber","defaultStartMoment","subtract","add","startDate","toDate","startMoment","moment","max","startDateStr","format","txnsUrl","txnsResult","Error","CurrentAccountLastTransactions","MsgText","completedTxns","OperationEntry","TransactionStatuses","Completed","rawFutureTxns","_","get","pendingTxns","Pending","accountData","accounts","balance","CurrentAccountInfo","AccountBalance","navigateOrErrorLabel","e","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","value","id","password","num","DiscountScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","checkReadiness","fields","submitButtonSelector","postAction","possibleResults","fetchData"],"mappings":";;;;;;;;;;;AAAA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AAGA;;;;AAKA,MAAMA,QAAQ,GAAG,8BAAjB;AACA,MAAMC,WAAW,GAAG,UAApB;;AA4BA,SAASC,mBAAT,CAA6BC,IAA7B,EAAyDC,SAAzD,EAAwG;AACtG,MAAI,CAACD,IAAL,EAAW;AACT,WAAO,EAAP;AACD;;AACD,SAAOA,IAAI,CAACE,GAAL,CAAUC,GAAD,IAAS;AACvB,WAAO;AACLC,MAAAA,IAAI,EAAEC,+BAAiBC,MADlB;AAELC,MAAAA,UAAU,EAAEJ,GAAG,CAACK,eAFX;AAGLC,MAAAA,IAAI,EAAE,qBAAON,GAAG,CAACO,aAAX,EAA0BZ,WAA1B,EAAuCa,WAAvC,EAHD;AAILC,MAAAA,aAAa,EAAE,qBAAOT,GAAG,CAACU,SAAX,EAAsBf,WAAtB,EAAmCa,WAAnC,EAJV;AAKLG,MAAAA,cAAc,EAAEX,GAAG,CAACY,eALf;AAMLC,MAAAA,gBAAgB,EAAE,KANb;AAOLC,MAAAA,aAAa,EAAEd,GAAG,CAACY,eAPd;AAQLG,MAAAA,WAAW,EAAEf,GAAG,CAACgB,6BARZ;AASLC,MAAAA,MAAM,EAAEnB;AATH,KAAP;AAWD,GAZM,CAAP;AAaD;;AAGD,eAAeoB,gBAAf,CAAgCC,IAAhC,EAA4CC,OAA5C,EAAmG;AACjG,QAAMC,UAAU,GAAI,GAAE3B,QAAS,mBAA/B;AAEA,QAAM4B,cAAc,GAAI,GAAED,UAAW,mBAArC;AACA,QAAME,WAAW,GAAG,MAAM,+BAAuCJ,IAAvC,EAA6CG,cAA7C,CAA1B;;AAEA,MAAI,CAACC,WAAL,EAAkB;AAChB,WAAO;AACLC,MAAAA,OAAO,EAAE,KADJ;AAELC,MAAAA,SAAS,EAAEC,+BAAkBC,OAFxB;AAGLC,MAAAA,YAAY,EAAE;AAHT,KAAP;AAKD;;AACD,QAAMC,aAAa,GAAGN,WAAW,CAACO,gBAAZ,CAA6BC,oBAAnD;AAEA,QAAMC,kBAAkB,GAAG,uBAASC,QAAT,CAAkB,CAAlB,EAAqB,OAArB,EAA8BC,GAA9B,CAAkC,CAAlC,EAAqC,KAArC,CAA3B;AACA,QAAMC,SAAS,GAAGf,OAAO,CAACe,SAAR,IAAqBH,kBAAkB,CAACI,MAAnB,EAAvC;;AACA,QAAMC,WAAW,GAAGC,gBAAOC,GAAP,CAAWP,kBAAX,EAA+B,qBAAOG,SAAP,CAA/B,CAApB;;AAEA,QAAMK,YAAY,GAAGH,WAAW,CAACI,MAAZ,CAAmB9C,WAAnB,CAArB;AACA,QAAM+C,OAAO,GAAI,GAAErB,UAAW,qBAAoBQ,aAAc,mHAAkHW,YAAa,EAA/L;AACA,QAAMG,UAAU,GAAG,MAAM,+BAA2CxB,IAA3C,EAAiDuB,OAAjD,CAAzB;;AACA,MAAI,CAACC,UAAD,IAAeA,UAAU,CAACC,KAA1B,IACF,CAACD,UAAU,CAACE,8BADd,EAC8C;AAC5C,WAAO;AACLrB,MAAAA,OAAO,EAAE,KADJ;AAELC,MAAAA,SAAS,EAAEC,+BAAkBC,OAFxB;AAGLC,MAAAA,YAAY,EAAEe,UAAU,IAAIA,UAAU,CAACC,KAAzB,GAAiCD,UAAU,CAACC,KAAX,CAAiBE,OAAlD,GAA4D;AAHrE,KAAP;AAKD;;AAED,QAAMC,aAAa,GAAGnD,mBAAmB,CACvC+C,UAAU,CAACE,8BAAX,CAA0CG,cADH,EAEvCC,kCAAoBC,SAFmB,CAAzC;;AAIA,QAAMC,aAAa,GAAGC,gBAAEC,GAAF,CAAMV,UAAN,EAAkB,+EAAlB,CAAtB;;AACA,QAAMW,WAAW,GAAG1D,mBAAmB,CAACuD,aAAD,EAAgBF,kCAAoBM,OAApC,CAAvC;AAEA,QAAMC,WAAW,GAAG;AAClBhC,IAAAA,OAAO,EAAE,IADS;AAElBiC,IAAAA,QAAQ,EAAE,CAAC;AACT5B,MAAAA,aADS;AAET6B,MAAAA,OAAO,EAAEf,UAAU,CAACE,8BAAX,CAA0Cc,kBAA1C,CAA6DC,cAF7D;AAGT/D,MAAAA,IAAI,EAAE,CAAC,GAAGkD,aAAJ,EAAmB,GAAGO,WAAtB;AAHG,KAAD;AAFQ,GAApB;AASA,SAAOE,WAAP;AACD;;AAED,eAAeK,oBAAf,CAAoC1C,IAApC,EAAgD;AAC9C,MAAI;AACF,UAAM,mCAAkBA,IAAlB,CAAN;AACD,GAFD,CAEE,OAAO2C,CAAP,EAAU;AACV,UAAM,iDAAsB3C,IAAtB,EAA4B,gBAA5B,EAA8C,KAA9C,EAAqD,GAArD,CAAN;AACD;AACF;;AAED,SAAS4C,uBAAT,GAAyD;AACvD,QAAMC,IAA0B,GAAG,EAAnC;AACAA,EAAAA,IAAI,CAACC,qCAAaC,OAAd,CAAJ,GAA6B,CAAE,GAAExE,QAAS,oEAAb,CAA7B;AACAsE,EAAAA,IAAI,CAACC,qCAAaE,eAAd,CAAJ,GAAqC,CAAE,GAAEzE,QAAS,0DAAb,CAArC;AACAsE,EAAAA,IAAI,CAACC,qCAAaG,cAAd,CAAJ,GAAoC,CAAE,GAAE1E,QAAS,yDAAb,CAApC;AACA,SAAOsE,IAAP;AACD;;AAED,SAASK,iBAAT,CAA2BC,WAA3B,EAA4D;AAC1D,SAAO,CACL;AAAEC,IAAAA,QAAQ,EAAE,OAAZ;AAAqBC,IAAAA,KAAK,EAAEF,WAAW,CAACG;AAAxC,GADK,EAEL;AAAEF,IAAAA,QAAQ,EAAE,aAAZ;AAA2BC,IAAAA,KAAK,EAAEF,WAAW,CAACI;AAA9C,GAFK,EAGL;AAAEH,IAAAA,QAAQ,EAAE,SAAZ;AAAuBC,IAAAA,KAAK,EAAEF,WAAW,CAACK;AAA1C,GAHK,CAAP;AAKD;;AAED,MAAMC,eAAN,SAA8BC,8CAA9B,CAAqD;AACnDC,EAAAA,eAAe,CAACR,WAAD,EAAkC;AAC/C,WAAO;AACLS,MAAAA,QAAQ,EAAG,GAAErF,QAAS,0DADjB;AAELsF,MAAAA,cAAc,EAAE,YAAY,iDAAsB,KAAK7D,IAA3B,EAAiC,OAAjC,CAFvB;AAGL8D,MAAAA,MAAM,EAAEZ,iBAAiB,CAACC,WAAD,CAHpB;AAILY,MAAAA,oBAAoB,EAAE,UAJjB;AAKLC,MAAAA,UAAU,EAAE,YAAYtB,oBAAoB,CAAC,KAAK1C,IAAN,CALvC;AAMLiE,MAAAA,eAAe,EAAErB,uBAAuB;AANnC,KAAP;AAQD;;AAED,QAAMsB,SAAN,GAAkB;AAChB,WAAOnE,gBAAgB,CAAC,KAAKC,IAAN,EAAY,KAAKC,OAAjB,CAAvB;AACD;;AAdkD;;eAiBtCwD,e","sourcesContent":["import _ from 'lodash';\nimport moment from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitUntilElementFound } from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport {\n  Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport {\n  ScaperOptions,\n  ScraperErrorTypes, ScaperScrapingResult, ScraperCredentials,\n} from './base-scraper';\n\nconst BASE_URL = 'https://start.telebank.co.il';\nconst DATE_FORMAT = 'YYYYMMDD';\n\ninterface ScrapedTransaction {\n  OperationNumber: number;\n  OperationDate: string;\n  ValueDate: string;\n  OperationAmount: number;\n  OperationDescriptionToDisplay: string;\n}\n\ninterface CurrentAccountInfo {\n  AccountBalance: number;\n}\n\ninterface ScrapedAccountData {\n  UserAccountsData: {\n    DefaultAccountNumber: string;\n  };\n}\n\ninterface ScrapedTransactionData {\n  Error?: { MsgText: string };\n  CurrentAccountLastTransactions?: {\n    OperationEntry: ScrapedTransaction[];\n    CurrentAccountInfo: CurrentAccountInfo;\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], txnStatus: TransactionStatuses): Transaction[] {\n  if (!txns) {\n    return [];\n  }\n  return txns.map((txn) => {\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.OperationNumber,\n      date: moment(txn.OperationDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.ValueDate, DATE_FORMAT).toISOString(),\n      originalAmount: txn.OperationAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: txn.OperationAmount,\n      description: txn.OperationDescriptionToDisplay,\n      status: txnStatus,\n    };\n  });\n}\n\n\nasync function fetchAccountData(page: Page, options: ScaperOptions): Promise<ScaperScrapingResult> {\n  const apiSiteUrl = `${BASE_URL}/Titan/gatewayAPI`;\n\n  const accountDataUrl = `${apiSiteUrl}/userAccountsData`;\n  const accountInfo = await fetchGetWithinPage<ScrapedAccountData>(page, accountDataUrl);\n\n  if (!accountInfo) {\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.Generic,\n      errorMessage: 'failed to get account data',\n    };\n  }\n  const accountNumber = accountInfo.UserAccountsData.DefaultAccountNumber;\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\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const txnsUrl = `${apiSiteUrl}/lastTransactions/${accountNumber}/Date?IsCategoryDescCode=True&IsTransactionDetails=True&IsEventNames=True&IsFutureTransactionFlag=True&FromDate=${startDateStr}`;\n  const txnsResult = await fetchGetWithinPage<ScrapedTransactionData>(page, txnsUrl);\n  if (!txnsResult || txnsResult.Error ||\n    !txnsResult.CurrentAccountLastTransactions) {\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.Generic,\n      errorMessage: txnsResult && txnsResult.Error ? txnsResult.Error.MsgText : 'unknown error',\n    };\n  }\n\n  const completedTxns = convertTransactions(\n    txnsResult.CurrentAccountLastTransactions.OperationEntry,\n    TransactionStatuses.Completed,\n  );\n  const rawFutureTxns = _.get(txnsResult, 'CurrentAccountLastTransactions.FutureTransactionsBlock.FutureTransactionEntry');\n  const pendingTxns = convertTransactions(rawFutureTxns, TransactionStatuses.Pending);\n\n  const accountData = {\n    success: true,\n    accounts: [{\n      accountNumber,\n      balance: txnsResult.CurrentAccountLastTransactions.CurrentAccountInfo.AccountBalance,\n      txns: [...completedTxns, ...pendingTxns],\n    }],\n  };\n\n  return accountData;\n}\n\nasync function navigateOrErrorLabel(page: Page) {\n  try {\n    await waitForNavigation(page);\n  } catch (e) {\n    await waitUntilElementFound(page, '#general-error', false, 100);\n  }\n}\n\nfunction getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [`${BASE_URL}/apollo/core/templates/RETAIL/masterPage.html#/MY_ACCOUNT_HOMEPAGE`];\n  urls[LoginResults.InvalidPassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`];\n  urls[LoginResults.ChangePassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/PWD_RENEW`];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#tzId', value: credentials.id },\n    { selector: '#tzPassword', value: credentials.password },\n    { selector: '#aidnum', value: credentials.num },\n  ];\n}\n\nclass DiscountScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`,\n      checkReadiness: async () => waitUntilElementFound(this.page, '#tzId'),\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.sendBtn',\n      postAction: async () => navigateOrErrorLabel(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.options);\n  }\n}\n\nexport default DiscountScraper;\n"]}
149
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/discount.ts"],"names":["BASE_URL","DATE_FORMAT","convertTransactions","txns","txnStatus","map","txn","type","TransactionTypes","Normal","identifier","OperationNumber","date","OperationDate","toISOString","processedDate","ValueDate","originalAmount","OperationAmount","originalCurrency","chargedAmount","description","OperationDescriptionToDisplay","status","fetchAccountData","page","options","apiSiteUrl","accountDataUrl","accountInfo","success","errorType","ScraperErrorTypes","Generic","errorMessage","accountNumber","UserAccountsData","DefaultAccountNumber","defaultStartMoment","subtract","add","startDate","toDate","startMoment","moment","max","startDateStr","format","txnsUrl","txnsResult","Error","CurrentAccountLastTransactions","MsgText","completedTxns","OperationEntry","TransactionStatuses","Completed","rawFutureTxns","_","get","pendingTxns","Pending","accountData","accounts","balance","CurrentAccountInfo","AccountBalance","navigateOrErrorLabel","e","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","value","id","password","num","DiscountScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","checkReadiness","fields","submitButtonSelector","postAction","possibleResults","fetchData"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport moment from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitUntilElementFound } from '../helpers/elements-interactions';\nimport { waitForNavigation } from '../helpers/navigation';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport {\n  Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport {\n  ScaperOptions,\n  ScraperErrorTypes, ScaperScrapingResult, ScraperCredentials,\n} from './base-scraper';\n\nconst BASE_URL = 'https://start.telebank.co.il';\nconst DATE_FORMAT = 'YYYYMMDD';\n\ninterface ScrapedTransaction {\n  OperationNumber: number;\n  OperationDate: string;\n  ValueDate: string;\n  OperationAmount: number;\n  OperationDescriptionToDisplay: string;\n}\n\ninterface CurrentAccountInfo {\n  AccountBalance: number;\n}\n\ninterface ScrapedAccountData {\n  UserAccountsData: {\n    DefaultAccountNumber: string;\n  };\n}\n\ninterface ScrapedTransactionData {\n  Error?: { MsgText: string };\n  CurrentAccountLastTransactions?: {\n    OperationEntry: ScrapedTransaction[];\n    CurrentAccountInfo: CurrentAccountInfo;\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], txnStatus: TransactionStatuses): Transaction[] {\n  if (!txns) {\n    return [];\n  }\n  return txns.map((txn) => {\n    return {\n      type: TransactionTypes.Normal,\n      identifier: txn.OperationNumber,\n      date: moment(txn.OperationDate, DATE_FORMAT).toISOString(),\n      processedDate: moment(txn.ValueDate, DATE_FORMAT).toISOString(),\n      originalAmount: txn.OperationAmount,\n      originalCurrency: 'ILS',\n      chargedAmount: txn.OperationAmount,\n      description: txn.OperationDescriptionToDisplay,\n      status: txnStatus,\n    };\n  });\n}\n\n\nasync function fetchAccountData(page: Page, options: ScaperOptions): Promise<ScaperScrapingResult> {\n  const apiSiteUrl = `${BASE_URL}/Titan/gatewayAPI`;\n\n  const accountDataUrl = `${apiSiteUrl}/userAccountsData`;\n  const accountInfo = await fetchGetWithinPage<ScrapedAccountData>(page, accountDataUrl);\n\n  if (!accountInfo) {\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.Generic,\n      errorMessage: 'failed to get account data',\n    };\n  }\n  const accountNumber = accountInfo.UserAccountsData.DefaultAccountNumber;\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\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const txnsUrl = `${apiSiteUrl}/lastTransactions/${accountNumber}/Date?IsCategoryDescCode=True&IsTransactionDetails=True&IsEventNames=True&IsFutureTransactionFlag=True&FromDate=${startDateStr}`;\n  const txnsResult = await fetchGetWithinPage<ScrapedTransactionData>(page, txnsUrl);\n  if (!txnsResult || txnsResult.Error ||\n    !txnsResult.CurrentAccountLastTransactions) {\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.Generic,\n      errorMessage: txnsResult && txnsResult.Error ? txnsResult.Error.MsgText : 'unknown error',\n    };\n  }\n\n  const completedTxns = convertTransactions(\n    txnsResult.CurrentAccountLastTransactions.OperationEntry,\n    TransactionStatuses.Completed,\n  );\n  const rawFutureTxns = _.get(txnsResult, 'CurrentAccountLastTransactions.FutureTransactionsBlock.FutureTransactionEntry');\n  const pendingTxns = convertTransactions(rawFutureTxns, TransactionStatuses.Pending);\n\n  const accountData = {\n    success: true,\n    accounts: [{\n      accountNumber,\n      balance: txnsResult.CurrentAccountLastTransactions.CurrentAccountInfo.AccountBalance,\n      txns: [...completedTxns, ...pendingTxns],\n    }],\n  };\n\n  return accountData;\n}\n\nasync function navigateOrErrorLabel(page: Page) {\n  try {\n    await waitForNavigation(page);\n  } catch (e) {\n    await waitUntilElementFound(page, '#general-error', false, 100);\n  }\n}\n\nfunction getPossibleLoginResults(): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [`${BASE_URL}/apollo/retail/#/MY_ACCOUNT_HOMEPAGE`];\n  urls[LoginResults.InvalidPassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`];\n  urls[LoginResults.ChangePassword] = [`${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/PWD_RENEW`];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#tzId', value: credentials.id },\n    { selector: '#tzPassword', value: credentials.password },\n    { selector: '#aidnum', value: credentials.num },\n  ];\n}\n\nclass DiscountScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/apollo/core/templates/lobby/masterPage.html#/LOGIN_PAGE`,\n      checkReadiness: async () => waitUntilElementFound(this.page, '#tzId'),\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '.sendBtn',\n      postAction: async () => navigateOrErrorLabel(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData() {\n    return fetchAccountData(this.page, this.options);\n  }\n}\n\nexport default DiscountScraper;\n"]}
@@ -1,5 +1,7 @@
1
1
  "use strict";
2
2
 
3
+ require("core-js/modules/es.array.iterator");
4
+
3
5
  require("core-js/modules/es.promise");
4
6
 
5
7
  Object.defineProperty(exports, "__esModule", {
@@ -21,8 +23,11 @@ var _fetch = require("../helpers/fetch");
21
23
 
22
24
  var _transactions = require("../transactions");
23
25
 
26
+ var _debug = require("../helpers/debug");
27
+
24
28
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
25
29
 
30
+ const debug = (0, _debug.getDebug)('hapoalim');
26
31
  const DATE_FORMAT = 'YYYYMMDD'; // eslint-disable-next-line @typescript-eslint/no-namespace
27
32
 
28
33
  function convertTransactions(txns) {
@@ -101,11 +106,27 @@ async function fetchPoalimXSRFWithinPage(page, url, pageUuid) {
101
106
  return (0, _fetch.fetchPostWithinPage)(page, url, [], headers);
102
107
  }
103
108
 
109
+ async function getAccountTransactions(apiSiteUrl, page, accountNumber, startDate, endDate) {
110
+ var _txnsResult$transacti;
111
+
112
+ const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;
113
+ const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');
114
+ return convertTransactions((_txnsResult$transacti = txnsResult === null || txnsResult === void 0 ? void 0 : txnsResult.transactions) !== null && _txnsResult$transacti !== void 0 ? _txnsResult$transacti : []);
115
+ }
116
+
117
+ async function getAccountBalance(apiSiteUrl, page, accountNumber) {
118
+ const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;
119
+ const balanceAndCreditLimit = await (0, _fetch.fetchGetWithinPage)(page, balanceAndCreditLimitUrl);
120
+ return balanceAndCreditLimit === null || balanceAndCreditLimit === void 0 ? void 0 : balanceAndCreditLimit.currentBalance;
121
+ }
122
+
104
123
  async function fetchAccountData(page, baseUrl, options) {
105
124
  const restContext = await getRestContext(page);
106
125
  const apiSiteUrl = `${baseUrl}/${restContext}`;
107
126
  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;
127
+ debug('fetching accounts data');
108
128
  const accountsInfo = (await (0, _fetch.fetchGetWithinPage)(page, accountDataUrl)) || [];
129
+ debug('got %d accounts, fetching txns and balance', accountsInfo.length);
109
130
  const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
110
131
  const startDate = options.startDate || defaultStartMoment.toDate();
111
132
 
@@ -115,19 +136,18 @@ async function fetchAccountData(page, baseUrl, options) {
115
136
  const endDateStr = (0, _moment.default)().format(DATE_FORMAT);
116
137
  const accounts = [];
117
138
 
118
- for (let accountIndex = 0; accountIndex < accountsInfo.length; accountIndex += 1) {
119
- const accountNumber = `${accountsInfo[accountIndex].bankNumber}-${accountsInfo[accountIndex].branchNumber}-${accountsInfo[accountIndex].accountNumber}`;
120
- const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;
121
- const balanceAndCreditLimit = await (0, _fetch.fetchGetWithinPage)(page, balanceAndCreditLimitUrl);
122
- const balance = balanceAndCreditLimit === null || balanceAndCreditLimit === void 0 ? void 0 : balanceAndCreditLimit.currentBalance;
123
- const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDateStr}&retrievalStartDate=${startDateStr}&sortCode=1`;
124
- const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');
125
- let txns = [];
126
-
127
- if (txnsResult) {
128
- txns = convertTransactions(txnsResult.transactions);
139
+ for (const account of accountsInfo) {
140
+ let balance;
141
+ const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;
142
+ const isActiveAccount = account.accountClosingReasonCode === 0;
143
+
144
+ if (isActiveAccount) {
145
+ balance = await getAccountBalance(apiSiteUrl, page, accountNumber);
146
+ } else {
147
+ debug('Skipping balance for a closed account, balance will be undefined');
129
148
  }
130
149
 
150
+ const txns = await getAccountTransactions(apiSiteUrl, page, accountNumber, startDateStr, endDateStr);
131
151
  accounts.push({
132
152
  accountNumber,
133
153
  balance,
@@ -139,6 +159,7 @@ async function fetchAccountData(page, baseUrl, options) {
139
159
  success: true,
140
160
  accounts
141
161
  };
162
+ debug('fetching ended');
142
163
  return accountData;
143
164
  }
144
165
 
@@ -184,4 +205,4 @@ class HapoalimScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
184
205
 
185
206
  var _default = HapoalimScraper;
186
207
  exports.default = _default;
187
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/hapoalim.ts"],"names":["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","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","fetchAccountData","baseUrl","options","apiSiteUrl","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","startDate","toDate","startMoment","moment","max","startDateStr","format","endDateStr","accounts","accountIndex","accountNumber","bankNumber","branchNumber","balanceAndCreditLimitUrl","balanceAndCreditLimit","balance","currentBalance","txnsUrl","txnsResult","transactions","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport uuid4 from 'uuid/v4';\n\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport {\n  TransactionsAccount, Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScaperOptions, ScraperCredentials } from './base-scraper';\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  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\n}\n\ntype FetchedAccountData = {\n  bankNumber: string;\n  accountNumber: string;\n  branchNumber: string;\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 {\n        partyHeadline,\n        partyName,\n        messageHeadline,\n        messageDetail,\n      } = 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(async () => {\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(page: Page, url: string, pageUuid: string): 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 fetchAccountData(page: Page, baseUrl: string, options: ScaperOptions) {\n  const restContext = await getRestContext(page);\n  const apiSiteUrl = `${baseUrl}/${restContext}`;\n  const accountDataUrl = `${baseUrl}/ServerServices/general/accounts`;\n  const accountsInfo = await fetchGetWithinPage<FetchedAccountData>(page, accountDataUrl) || [];\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\n  const startDateStr = startMoment.format(DATE_FORMAT);\n  const endDateStr = moment().format(DATE_FORMAT);\n\n  const accounts: TransactionsAccount[] = [];\n  for (let accountIndex = 0; accountIndex < accountsInfo.length; accountIndex += 1) {\n    const accountNumber = `${accountsInfo[accountIndex].bankNumber}-${accountsInfo[accountIndex].branchNumber}-${accountsInfo[accountIndex].accountNumber}`;\n\n    const balanceAndCreditLimitUrl = `${apiSiteUrl}/current-account/composite/balanceAndCreditLimit?accountId=${accountNumber}&view=details&lang=he`;\n    const balanceAndCreditLimit = await fetchGetWithinPage(page, balanceAndCreditLimitUrl) as BalanceAndCreditLimit;\n    const balance: number = balanceAndCreditLimit?.currentBalance;\n\n    const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDateStr}&retrievalStartDate=${startDateStr}&sortCode=1`;\n    const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n    let txns: Transaction[] = [];\n\n    if (txnsResult) {\n      txns = convertTransactions(txnsResult.transactions);\n    }\n\n    accounts.push({\n      accountNumber,\n      balance,\n      txns,\n    });\n  }\n\n  const accountData = {\n    success: true,\n    accounts,\n  };\n\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  urls[LoginResults.InvalidPassword] = [`${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`];\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: ScraperCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass HapoalimScraper extends BaseScraperWithBrowser {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperCredentials) {\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"]}
208
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/hapoalim.ts"],"names":["debug","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","eventDate","toISOString","processedDate","valueDate","originalAmount","eventAmount","originalCurrency","chargedAmount","description","activityDescription","status","serialNumber","TransactionStatuses","Pending","Completed","getRestContext","page","evaluate","window","bnhpApp","restContext","slice","fetchPoalimXSRFWithinPage","url","pageUuid","cookies","XSRFCookie","find","cookie","name","headers","value","uuid","getAccountTransactions","apiSiteUrl","accountNumber","startDate","endDate","txnsUrl","txnsResult","transactions","getAccountBalance","balanceAndCreditLimitUrl","balanceAndCreditLimit","currentBalance","fetchAccountData","baseUrl","options","accountDataUrl","accountsInfo","defaultStartMoment","subtract","add","toDate","startMoment","moment","max","startDateStr","format","endDateStr","accounts","account","balance","bankNumber","branchNumber","isActiveAccount","accountClosingReasonCode","accountData","success","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","ChangePassword","createLoginFields","credentials","selector","userCode","password","HapoalimScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport uuid4 from 'uuid/v4';\n\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntil } from '../helpers/waiting';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport {\n  TransactionsAccount, Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScaperOptions, ScraperCredentials } from './base-scraper';\nimport { getDebug } from '../helpers/debug';\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  beneficiaryDetailsData?: {\n    partyHeadline?: string;\n    partyName?: string;\n    messageHeadline?: string;\n    messageDetail?: string;\n  };\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 {\n        partyHeadline,\n        partyName,\n        messageHeadline,\n        messageDetail,\n      } = 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(async () => {\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(page: Page, url: string, pageUuid: string): 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 getAccountTransactions(apiSiteUrl: string, page: Page, accountNumber: string, startDate: string, endDate: string) {\n  const txnsUrl = `${apiSiteUrl}/current-account/transactions?accountId=${accountNumber}&numItemsPerPage=150&retrievalEndDate=${endDate}&retrievalStartDate=${startDate}&sortCode=1`;\n  const txnsResult = await fetchPoalimXSRFWithinPage(page, txnsUrl, '/current-account/transactions');\n\n  return convertTransactions(txnsResult?.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: ScaperOptions) {\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  debug('got %d accounts, fetching txns and balance', accountsInfo.length);\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\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 accountsInfo) {\n    let balance: number | undefined;\n    const accountNumber = `${account.bankNumber}-${account.branchNumber}-${account.accountNumber}`;\n\n    const isActiveAccount = account.accountClosingReasonCode === 0;\n    if (isActiveAccount) {\n      balance = await getAccountBalance(apiSiteUrl, page, accountNumber);\n    } else {\n      debug('Skipping balance for a closed account, balance will be undefined');\n    }\n\n    const txns = await getAccountTransactions(apiSiteUrl, page, accountNumber, startDateStr, endDateStr);\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  urls[LoginResults.InvalidPassword] = [`${baseUrl}/AUTHENTICATE/LOGON?flow=AUTHENTICATE&state=LOGON&errorcode=1.6&callme=false`];\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: ScraperCredentials) {\n  return [\n    { selector: '#userCode', value: credentials.userCode },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass HapoalimScraper extends BaseScraperWithBrowser {\n  // eslint-disable-next-line class-methods-use-this\n  get baseUrl() {\n    return 'https://login.bankhapoalim.co.il';\n  }\n\n  getLoginOptions(credentials: ScraperCredentials) {\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"]}
@@ -162,7 +162,7 @@ async function fetchTransactions(page, startDate) {
162
162
 
163
163
  async function waitForPostLogin(page) {
164
164
  // TODO check for condition to provide new password
165
- await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, 'div.leumi-container', true), (0, _elementsInteractions.waitUntilElementFound)(page, '#BodyContent_ctl00_loginErrMsg', true), (0, _elementsInteractions.waitUntilElementFound)(page, '.ErrMsg', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'form[action="/changepassword"]', true)]);
165
+ await Promise.race([(0, _elementsInteractions.waitUntilElementFound)(page, 'a[title="דלג לחשבון"]', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'div.leumi-container', true), (0, _elementsInteractions.waitUntilElementFound)(page, '#BodyContent_ctl00_loginErrMsg', true), (0, _elementsInteractions.waitUntilElementFound)(page, '.ErrMsg', true), (0, _elementsInteractions.waitUntilElementFound)(page, 'form[action="/changepassword"]', true)]);
166
166
  }
167
167
 
168
168
  class LeumiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
@@ -194,4 +194,4 @@ class LeumiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
194
194
 
195
195
  var _default = LeumiScraper;
196
196
  exports.default = _default;
197
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/leumi.ts"],"names":["BASE_URL","TRANSACTIONS_URL","FILTERED_TRANSACTIONS_URL","DATE_FORMAT","ACCOUNT_BLOCKED_MSG","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","AccountBlocked","options","page","Error","errorMessage","label","innerText","startsWith","ChangePassword","createLoginFields","credentials","selector","value","username","password","extractTransactionsFromPage","transactions","status","length","result","map","rawTransaction","newTransaction","type","TransactionTypes","Normal","date","DateUTC","processedDate","description","Description","identifier","ReferenceNumberLong","memo","AdditionalData","originalCurrency","SHEKEL_CURRENCY","chargedAmount","Amount","originalAmount","hangProcess","timeout","Promise","resolve","setTimeout","clickByXPath","xpath","waitForXPath","visible","elm","$x","click","removeSpecialCharacters","str","replace","fetchTransactionsForAccount","startDate","accountId","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","e","textContent","push","waitForPostLogin","race","LeumiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","toDate","startMoment","moment","max","navigateTo","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, LoginOptions } from './base-scraper-with-browser';\nimport {\n  fillInput,\n  clickButton,\n  waitUntilElementFound,\n  pageEvalAll,\n} from '../helpers/elements-interactions';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  TransactionsAccount, Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScaperScrapingResult, ScraperCredentials } from './base-scraper';\n\nconst BASE_URL = 'https://hb2.bankleumi.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 = 'המנוי חסום';\n\n\nfunction getPossibleLoginResults() {\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/ebanking\\/SO\\/SPA.aspx/i],\n    [LoginResults.InvalidPassword]: [/InternalSite\\/CustomUpdate\\/leumi\\/LoginPage.ASP/],\n    [LoginResults.AccountBlocked]: [\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'],\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#wtr_uid', value: credentials.username },\n    { selector: '#wtr_password', 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 newTransaction: Transaction = {\n      status,\n      type: TransactionTypes.Normal,\n      date: rawTransaction.DateUTC,\n      processedDate: rawTransaction.DateUTC,\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.waitForXPath(xpath, { timeout: 30000, visible: true });\n  const elm = await page.$x(xpath);\n  await elm[0].click();\n}\n\nfunction removeSpecialCharacters(str: string): string {\n  return str.replace(/[^0-9/-]/g, '');\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment, accountId: string): Promise<TransactionsAccount> {\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  await waitUntilElementFound(page, 'button[title=\"חיפוש מתקדם\"]', true);\n  await clickButton(page, 'button[title=\"חיפוש מתקדם\"]');\n  await waitUntilElementFound(page, 'bll-radio-button', true);\n  await clickButton(page, 'bll-radio-button:not([checked])');\n\n  await waitUntilElementFound(page, 'input[formcontrolname=\"txtInputFrom\"]', true);\n\n  await fillInput(\n    page,\n    'input[formcontrolname=\"txtInputFrom\"]',\n    startDate.format(DATE_FORMAT),\n  );\n\n  // we must blur the from control otherwise the search will use the previous value\n  await page.focus(\"button[aria-label='סנן']\");\n\n  await clickButton(page, \"button[aria-label='סנן']\");\n  const finalResponse = await page.waitForResponse((response) => {\n    return response.url() === FILTERED_TRANSACTIONS_URL &&\n      response.request().method() === 'POST';\n  });\n\n  const responseJson: any = await finalResponse.json();\n\n  const accountNumber = accountId.replace('/', '_').replace(/[^\\d-_]/g, '');\n\n  const response = JSON.parse(responseJson.jsonResp);\n\n  const pendingTransactions = response.TodayTransactionsItems;\n  const transactions = response.HistoryTransactionsItems;\n  const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;\n\n  const pendingTxns = extractTransactionsFromPage(pendingTransactions, TransactionStatuses.Pending);\n  const completedTxns = extractTransactionsFromPage(transactions, TransactionStatuses.Completed);\n  const txns = [\n    ...pendingTxns,\n    ...completedTxns,\n  ];\n\n  return {\n    accountNumber,\n    balance,\n    txns,\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment): Promise<TransactionsAccount[]> {\n  const accounts: TransactionsAccount[] = [];\n\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  const accountsIds = await page.evaluate(() => Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), (e) => e.textContent)) as string[];\n\n  // due to a bug, the altered value might include undesired signs like & that should be removed\n\n  if (!accountsIds.length) {\n    throw new Error('Failed to extract or parse the account number');\n  }\n\n  for (const accountId of accountsIds) {\n    if (accountsIds.length > 1) {\n      // get list of accounts and check accountId\n      await clickByXPath(page, '//*[contains(@class, \"number\") and contains(@class, \"combo-inner\")]');\n      await clickByXPath(page, `//span[contains(text(), '${accountId}')]`);\n    }\n\n    accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));\n  }\n\n  return accounts;\n}\n\nasync function waitForPostLogin(page: Page): Promise<void> {\n  // TODO check for condition to provide new password\n  await Promise.race([\n    waitUntilElementFound(page, 'div.leumi-container', true),\n    waitUntilElementFound(page, '#BodyContent_ctl00_loginErrMsg', true),\n    waitUntilElementFound(page, '.ErrMsg', true),\n    waitUntilElementFound(page, 'form[action=\"/changepassword\"]', true),\n  ]);\n}\n\nclass LeumiScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: Record<string, string>) {\n    return {\n      loginUrl: `${BASE_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#enter',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData(): Promise<ScaperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, 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"]}
197
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/leumi.ts"],"names":["BASE_URL","TRANSACTIONS_URL","FILTERED_TRANSACTIONS_URL","DATE_FORMAT","ACCOUNT_BLOCKED_MSG","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","AccountBlocked","options","page","Error","errorMessage","label","innerText","startsWith","ChangePassword","createLoginFields","credentials","selector","value","username","password","extractTransactionsFromPage","transactions","status","length","result","map","rawTransaction","newTransaction","type","TransactionTypes","Normal","date","DateUTC","processedDate","description","Description","identifier","ReferenceNumberLong","memo","AdditionalData","originalCurrency","SHEKEL_CURRENCY","chargedAmount","Amount","originalAmount","hangProcess","timeout","Promise","resolve","setTimeout","clickByXPath","xpath","waitForXPath","visible","elm","$x","click","removeSpecialCharacters","str","replace","fetchTransactionsForAccount","startDate","accountId","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","e","textContent","push","waitForPostLogin","race","LeumiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","toDate","startMoment","moment","max","navigateTo","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, LoginOptions } from './base-scraper-with-browser';\nimport {\n  fillInput,\n  clickButton,\n  waitUntilElementFound,\n  pageEvalAll,\n} from '../helpers/elements-interactions';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  TransactionsAccount, Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScaperScrapingResult, ScraperCredentials } from './base-scraper';\n\nconst BASE_URL = 'https://hb2.bankleumi.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 = 'המנוי חסום';\n\n\nfunction getPossibleLoginResults() {\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/ebanking\\/SO\\/SPA.aspx/i],\n    [LoginResults.InvalidPassword]: [/InternalSite\\/CustomUpdate\\/leumi\\/LoginPage.ASP/],\n    [LoginResults.AccountBlocked]: [\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'],\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#wtr_uid', value: credentials.username },\n    { selector: '#wtr_password', 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 newTransaction: Transaction = {\n      status,\n      type: TransactionTypes.Normal,\n      date: rawTransaction.DateUTC,\n      processedDate: rawTransaction.DateUTC,\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.waitForXPath(xpath, { timeout: 30000, visible: true });\n  const elm = await page.$x(xpath);\n  await elm[0].click();\n}\n\nfunction removeSpecialCharacters(str: string): string {\n  return str.replace(/[^0-9/-]/g, '');\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment, accountId: string): Promise<TransactionsAccount> {\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  await waitUntilElementFound(page, 'button[title=\"חיפוש מתקדם\"]', true);\n  await clickButton(page, 'button[title=\"חיפוש מתקדם\"]');\n  await waitUntilElementFound(page, 'bll-radio-button', true);\n  await clickButton(page, 'bll-radio-button:not([checked])');\n\n  await waitUntilElementFound(page, 'input[formcontrolname=\"txtInputFrom\"]', true);\n\n  await fillInput(\n    page,\n    'input[formcontrolname=\"txtInputFrom\"]',\n    startDate.format(DATE_FORMAT),\n  );\n\n  // we must blur the from control otherwise the search will use the previous value\n  await page.focus(\"button[aria-label='סנן']\");\n\n  await clickButton(page, \"button[aria-label='סנן']\");\n  const finalResponse = await page.waitForResponse((response) => {\n    return response.url() === FILTERED_TRANSACTIONS_URL &&\n      response.request().method() === 'POST';\n  });\n\n  const responseJson: any = await finalResponse.json();\n\n  const accountNumber = accountId.replace('/', '_').replace(/[^\\d-_]/g, '');\n\n  const response = JSON.parse(responseJson.jsonResp);\n\n  const pendingTransactions = response.TodayTransactionsItems;\n  const transactions = response.HistoryTransactionsItems;\n  const balance = response.BalanceDisplay ? parseFloat(response.BalanceDisplay) : undefined;\n\n  const pendingTxns = extractTransactionsFromPage(pendingTransactions, TransactionStatuses.Pending);\n  const completedTxns = extractTransactionsFromPage(transactions, TransactionStatuses.Completed);\n  const txns = [\n    ...pendingTxns,\n    ...completedTxns,\n  ];\n\n  return {\n    accountNumber,\n    balance,\n    txns,\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment): Promise<TransactionsAccount[]> {\n  const accounts: TransactionsAccount[] = [];\n\n  // DEVELOPER NOTICE the account number received from the server is being altered at\n  // runtime for some accounts after 1-2 seconds so we need to hang the process for a short while.\n  await hangProcess(4000);\n\n  const accountsIds = await page.evaluate(() => Array.from(document.querySelectorAll('app-masked-number-combo span.display-number-li'), (e) => e.textContent)) as string[];\n\n  // due to a bug, the altered value might include undesired signs like & that should be removed\n\n  if (!accountsIds.length) {\n    throw new Error('Failed to extract or parse the account number');\n  }\n\n  for (const accountId of accountsIds) {\n    if (accountsIds.length > 1) {\n      // get list of accounts and check accountId\n      await clickByXPath(page, '//*[contains(@class, \"number\") and contains(@class, \"combo-inner\")]');\n      await clickByXPath(page, `//span[contains(text(), '${accountId}')]`);\n    }\n\n    accounts.push(await fetchTransactionsForAccount(page, startDate, removeSpecialCharacters(accountId)));\n  }\n\n  return accounts;\n}\n\nasync function waitForPostLogin(page: Page): Promise<void> {\n  // TODO check for condition to provide new password\n  await Promise.race([\n    waitUntilElementFound(page, 'a[title=\"דלג לחשבון\"]', true),\n    waitUntilElementFound(page, 'div.leumi-container', true),\n    waitUntilElementFound(page, '#BodyContent_ctl00_loginErrMsg', true),\n    waitUntilElementFound(page, '.ErrMsg', true),\n    waitUntilElementFound(page, 'form[action=\"/changepassword\"]', true),\n  ]);\n}\n\nclass LeumiScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: Record<string, string>) {\n    return {\n      loginUrl: `${BASE_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#enter',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(),\n    };\n  }\n\n  async fetchData(): Promise<ScaperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, 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"]}
@@ -28,6 +28,7 @@ var _transactions = require("../transactions");
28
28
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
29
29
 
30
30
  const BASE_URL = 'https://online.bankotsar.co.il';
31
+ const LONG_DATE_FORMAT = 'DD/MM/YYYY';
31
32
  const DATE_FORMAT = 'DD/MM/YY';
32
33
 
33
34
  function getPossibleLoginResults(page) {
@@ -79,7 +80,13 @@ function getAmountData(amountStr, hasCurrency = false) {
79
80
 
80
81
  function convertTransactions(txns) {
81
82
  return txns.map(txn => {
82
- const txnDate = (0, _moment.default)(txn.date, DATE_FORMAT).toISOString();
83
+ const dateFormat = txn.date.length === 8 ? DATE_FORMAT : txn.date.length === 10 ? LONG_DATE_FORMAT : null;
84
+
85
+ if (!dateFormat) {
86
+ throw new Error('invalid date format');
87
+ }
88
+
89
+ const txnDate = (0, _moment.default)(txn.date, dateFormat).toISOString();
83
90
  const credit = getAmountData(txn.credit || '').amount;
84
91
  const debit = getAmountData(txn.debit || '').amount;
85
92
  const amount = (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);
@@ -217,4 +224,4 @@ class OtsarHahayalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser
217
224
 
218
225
  var _default = OtsarHahayalScraper;
219
226
  exports.default = _default;
220
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/otsar-hahayal.ts"],"names":["BASE_URL","DATE_FORMAT","getPossibleLoginResults","page","urls","LoginResults","Success","InvalidPassword","getTransactionsUrl","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","hasCurrency","amountStrCln","replace","currency","amount","parseFloat","SHEKEL_CURRENCY","includes","SHEKEL_CURRENCY_SYMBOL","parts","split","convertTransactions","txns","map","txn","txnDate","date","toISOString","credit","debit","Number","isNaN","result","type","TransactionTypes","Normal","status","TransactionStatuses","Completed","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","chargedAmount","description","memo","parseTransactionPage","tdsValues","trs","el","querySelector","innerText","balance","getAccountSummary","balanceElm","$","balanceInnerTextElm","getProperty","balanceText","jsonValue","balanceValue","creditLimit","creditUtilization","balanceCurrency","fetchTransactionsForAccount","startDate","summary","branchNum","$eval","span","accountNmbr","accountNumber","format","hasNextPage","noTransactionElm","pageTxns","concat","button","slice","fetchTransactions","waitForPostLogin","Promise","race","OtsarHahayalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","options","toDate","startMoment","moment","max","url","navigateTo","accounts","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForNavigation } from '../helpers/navigation';\nimport {\n  fillInput,\n  clickButton,\n  waitUntilElementFound,\n  pageEvalAll,\n  elementPresentOnPage,\n} from '../helpers/elements-interactions';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScraperCredentials } from './base-scraper';\n\nconst BASE_URL = 'https://online.bankotsar.co.il';\nconst DATE_FORMAT = 'DD/MM/YY';\n\ninterface ScrapedTransaction {\n  balance?: string;\n  debit?: string;\n  credit?: string;\n  memo?: string;\n  status?: string;\n  reference?: string;\n  description?: string;\n  date: string;\n}\n\nfunction getPossibleLoginResults(page: Page) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [`${BASE_URL}/wps/myportal/FibiMenu/Online`];\n  urls[LoginResults.InvalidPassword] = [() => elementPresentOnPage(page, '#validationMsg')];\n  // TODO: support change password\n  /* urls[LOGIN_RESULT.CHANGE_PASSWORD] = [``]; */\n  return urls;\n}\n\nfunction getTransactionsUrl() {\n  return `${BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string, hasCurrency = false) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (!hasCurrency) {\n    amount = parseFloat(amountStrCln);\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    amount = parseFloat(parts[0]);\n    [, currency] = parts;\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn) => {\n    const txnDate = moment(txn.date, DATE_FORMAT).toISOString();\n    const credit = getAmountData(txn.credit || '').amount;\n    const debit = getAmountData(txn.debit || '').amount;\n    const amount = (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn.description || '',\n      memo: '',\n    };\n\n    return result;\n  });\n}\n\nasync function parseTransactionPage(page: Page): Promise<ScrapedTransaction[]> {\n  const tdsValues = await pageEvalAll(page, '#dataTable077 tbody tr', [], (trs) => {\n    return (trs).map((el) => ({\n      date: (el.querySelector('.date') as HTMLElement).innerText,\n      // reference and description have vice-versa class name\n      description: (el.querySelector('.reference') as HTMLElement).innerText,\n      reference: (el.querySelector('.details') as HTMLElement).innerText,\n      credit: (el.querySelector('.credit') as HTMLElement).innerText,\n      debit: (el.querySelector('.debit') as HTMLElement).innerText,\n      balance: (el.querySelector('.balance') as HTMLElement).innerText,\n    }));\n  });\n\n  return tdsValues;\n}\n\nasync function getAccountSummary(page: Page) {\n  const balanceElm = await page.$('.current_balance');\n  const balanceInnerTextElm = await balanceElm!.getProperty('innerText');\n  const balanceText = await balanceInnerTextElm.jsonValue();\n  const balanceValue = getAmountData(balanceText as string, true);\n  // TODO: Find the credit field in bank website (could see it in my account)\n  return {\n    balance: Number.isNaN(balanceValue.amount) ? 0 : balanceValue.amount,\n    creditLimit: 0.0,\n    creditUtilization: 0.0,\n    balanceCurrency: balanceValue.currency,\n  };\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment) {\n  const summary = await getAccountSummary(page);\n  await waitUntilElementFound(page, 'input#fromDate');\n  // Get account number\n  const branchNum = await page.$eval('.branch_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n\n  const accountNmbr = await page.$eval('.acc_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n  const accountNumber = `14-${branchNum}-${accountNmbr}`;\n  // Search for relavant transaction from startDate\n  await clickButton(page, '#tabHeader4');\n  await fillInput(\n    page,\n    'input#fromDate',\n    startDate.format('DD/MM/YYYY'),\n  );\n\n  await clickButton(page, '#fibi_tab_dates .fibi_btn:nth-child(2)');\n  await waitForNavigation(page);\n  await waitUntilElementFound(page, 'table#dataTable077, #NO_DATA077');\n  let hasNextPage = true;\n  let txns: ScrapedTransaction[] = [];\n\n  const noTransactionElm = await page.$('#NO_DATA077');\n  if (noTransactionElm == null) {\n    // Scape transactions (this maybe spanned on multiple pages)\n    while (hasNextPage) {\n      const pageTxns = await parseTransactionPage(page);\n      txns = txns.concat(pageTxns);\n      const button = await page.$('#Npage');\n      hasNextPage = false;\n      if (button != null) {\n        hasNextPage = true;\n      }\n      if (hasNextPage) {\n        await clickButton(page, '#Npage');\n        await waitForNavigation(page);\n        await waitUntilElementFound(page, 'table#dataTable077');\n      }\n    }\n  }\n\n  return {\n    accountNumber,\n    summary,\n    txns: convertTransactions(txns.slice(1)), // Remove first line which is \"opening balance\"\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment) {\n  // TODO need to extend to support multiple accounts and foreign accounts\n  return [await fetchTransactionsForAccount(page, startDate)];\n}\n\nasync function waitForPostLogin(page: Page) {\n  // TODO check for condition to provide new password\n  return Promise.race([\n    waitUntilElementFound(page, 'div.lotusFrame', true),\n    waitUntilElementFound(page, '#validationMsg'),\n  ]);\n}\n\nclass OtsarHahayalScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=OTSARPRTAL&site=Private&KODSAFA=HE`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    const url = getTransactionsUrl();\n    await this.navigateTo(url);\n\n    const accounts = await fetchTransactions(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default OtsarHahayalScraper;\n"]}
227
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/otsar-hahayal.ts"],"names":["BASE_URL","LONG_DATE_FORMAT","DATE_FORMAT","getPossibleLoginResults","page","urls","LoginResults","Success","InvalidPassword","getTransactionsUrl","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","hasCurrency","amountStrCln","replace","currency","amount","parseFloat","SHEKEL_CURRENCY","includes","SHEKEL_CURRENCY_SYMBOL","parts","split","convertTransactions","txns","map","txn","dateFormat","date","length","Error","txnDate","toISOString","credit","debit","Number","isNaN","result","type","TransactionTypes","Normal","status","TransactionStatuses","Completed","identifier","reference","parseInt","undefined","processedDate","originalAmount","originalCurrency","chargedAmount","description","memo","parseTransactionPage","tdsValues","trs","el","querySelector","innerText","balance","getAccountSummary","balanceElm","$","balanceInnerTextElm","getProperty","balanceText","jsonValue","balanceValue","creditLimit","creditUtilization","balanceCurrency","fetchTransactionsForAccount","startDate","summary","branchNum","$eval","span","accountNmbr","accountNumber","format","hasNextPage","noTransactionElm","pageTxns","concat","button","slice","fetchTransactions","waitForPostLogin","Promise","race","OtsarHahayalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","postAction","possibleResults","fetchData","defaultStartMoment","subtract","add","options","toDate","startMoment","moment","max","url","navigateTo","accounts","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForNavigation } from '../helpers/navigation';\nimport {\n  fillInput,\n  clickButton,\n  waitUntilElementFound,\n  pageEvalAll,\n  elementPresentOnPage,\n} from '../helpers/elements-interactions';\nimport { SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL } from '../constants';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScraperCredentials } from './base-scraper';\n\nconst BASE_URL = 'https://online.bankotsar.co.il';\nconst LONG_DATE_FORMAT = 'DD/MM/YYYY';\nconst DATE_FORMAT = 'DD/MM/YY';\n\ninterface ScrapedTransaction {\n  balance?: string;\n  debit?: string;\n  credit?: string;\n  memo?: string;\n  status?: string;\n  reference?: string;\n  description?: string;\n  date: string;\n}\n\nfunction getPossibleLoginResults(page: Page) {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [`${BASE_URL}/wps/myportal/FibiMenu/Online`];\n  urls[LoginResults.InvalidPassword] = [() => elementPresentOnPage(page, '#validationMsg')];\n  // TODO: support change password\n  /* urls[LOGIN_RESULT.CHANGE_PASSWORD] = [``]; */\n  return urls;\n}\n\nfunction getTransactionsUrl() {\n  return `${BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#username', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nfunction getAmountData(amountStr: string, hasCurrency = false) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (!hasCurrency) {\n    amount = parseFloat(amountStrCln);\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    amount = parseFloat(parts[0]);\n    [, currency] = parts;\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((txn) => {\n    const dateFormat =\n        txn.date.length === 8 ?\n          DATE_FORMAT :\n          txn.date.length === 10 ?\n            LONG_DATE_FORMAT :\n            null;\n    if (!dateFormat) {\n      throw new Error('invalid date format');\n    }\n    const txnDate = moment(txn.date, dateFormat).toISOString();\n    const credit = getAmountData(txn.credit || '').amount;\n    const debit = getAmountData(txn.debit || '').amount;\n    const amount = (Number.isNaN(credit) ? 0 : credit) - (Number.isNaN(debit) ? 0 : debit);\n\n    const result: Transaction = {\n      type: TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      identifier: txn.reference ? parseInt(txn.reference, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn.description || '',\n      memo: '',\n    };\n\n    return result;\n  });\n}\n\nasync function parseTransactionPage(page: Page): Promise<ScrapedTransaction[]> {\n  const tdsValues = await pageEvalAll(page, '#dataTable077 tbody tr', [], (trs) => {\n    return (trs).map((el) => ({\n      date: (el.querySelector('.date') as HTMLElement).innerText,\n      // reference and description have vice-versa class name\n      description: (el.querySelector('.reference') as HTMLElement).innerText,\n      reference: (el.querySelector('.details') as HTMLElement).innerText,\n      credit: (el.querySelector('.credit') as HTMLElement).innerText,\n      debit: (el.querySelector('.debit') as HTMLElement).innerText,\n      balance: (el.querySelector('.balance') as HTMLElement).innerText,\n    }));\n  });\n\n  return tdsValues;\n}\n\nasync function getAccountSummary(page: Page) {\n  const balanceElm = await page.$('.current_balance');\n  const balanceInnerTextElm = await balanceElm!.getProperty('innerText');\n  const balanceText = await balanceInnerTextElm.jsonValue();\n  const balanceValue = getAmountData(balanceText as string, true);\n  // TODO: Find the credit field in bank website (could see it in my account)\n  return {\n    balance: Number.isNaN(balanceValue.amount) ? 0 : balanceValue.amount,\n    creditLimit: 0.0,\n    creditUtilization: 0.0,\n    balanceCurrency: balanceValue.currency,\n  };\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment) {\n  const summary = await getAccountSummary(page);\n  await waitUntilElementFound(page, 'input#fromDate');\n  // Get account number\n  const branchNum = await page.$eval('.branch_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n\n  const accountNmbr = await page.$eval('.acc_num', (span) => {\n    return (span as HTMLElement).innerText;\n  });\n  const accountNumber = `14-${branchNum}-${accountNmbr}`;\n  // Search for relavant transaction from startDate\n  await clickButton(page, '#tabHeader4');\n  await fillInput(\n    page,\n    'input#fromDate',\n    startDate.format('DD/MM/YYYY'),\n  );\n\n  await clickButton(page, '#fibi_tab_dates .fibi_btn:nth-child(2)');\n  await waitForNavigation(page);\n  await waitUntilElementFound(page, 'table#dataTable077, #NO_DATA077');\n  let hasNextPage = true;\n  let txns: ScrapedTransaction[] = [];\n\n  const noTransactionElm = await page.$('#NO_DATA077');\n  if (noTransactionElm == null) {\n    // Scape transactions (this maybe spanned on multiple pages)\n    while (hasNextPage) {\n      const pageTxns = await parseTransactionPage(page);\n      txns = txns.concat(pageTxns);\n      const button = await page.$('#Npage');\n      hasNextPage = false;\n      if (button != null) {\n        hasNextPage = true;\n      }\n      if (hasNextPage) {\n        await clickButton(page, '#Npage');\n        await waitForNavigation(page);\n        await waitUntilElementFound(page, 'table#dataTable077');\n      }\n    }\n  }\n\n  return {\n    accountNumber,\n    summary,\n    txns: convertTransactions(txns.slice(1)), // Remove first line which is \"opening balance\"\n  };\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment) {\n  // TODO need to extend to support multiple accounts and foreign accounts\n  return [await fetchTransactionsForAccount(page, startDate)];\n}\n\nasync function waitForPostLogin(page: Page) {\n  // TODO check for condition to provide new password\n  return Promise.race([\n    waitUntilElementFound(page, 'div.lotusFrame', true),\n    waitUntilElementFound(page, '#validationMsg'),\n  ]);\n}\n\nclass OtsarHahayalScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: `${BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=OTSARPRTAL&site=Private&KODSAFA=HE`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#continueBtn',\n      postAction: async () => waitForPostLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    const url = getTransactionsUrl();\n    await this.navigateTo(url);\n\n    const accounts = await fetchTransactions(this.page, startMoment);\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default OtsarHahayalScraper;\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "israeli-bank-scrapers",
3
- "version": "1.9.0",
3
+ "version": "1.10.3",
4
4
  "private": false,
5
5
  "description": "Provide scrapers for all major Israeli banks and credit card companies",
6
6
  "engines": {