israeli-bank-scrapers 1.14.2 → 1.15.0

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.
@@ -22,14 +22,14 @@ exports.setValue = setValue;
22
22
 
23
23
  var _waiting = require("./waiting");
24
24
 
25
- async function waitUntilElementFound(page, elementSelector, onlyVisible = false, timeout = 30000) {
25
+ async function waitUntilElementFound(page, elementSelector, onlyVisible = false, timeout) {
26
26
  await page.waitForSelector(elementSelector, {
27
27
  visible: onlyVisible,
28
28
  timeout
29
29
  });
30
30
  }
31
31
 
32
- async function waitUntilElementDisappear(page, elementSelector, timeout = 30000) {
32
+ async function waitUntilElementDisappear(page, elementSelector, timeout) {
33
33
  await page.waitForSelector(elementSelector, {
34
34
  hidden: true,
35
35
  timeout
@@ -130,4 +130,4 @@ async function dropdownElements(page, selector) {
130
130
  }, `${selector} > option`);
131
131
  return options;
132
132
  }
133
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/helpers/elements-interactions.ts"],"names":["waitUntilElementFound","page","elementSelector","onlyVisible","timeout","waitForSelector","visible","waitUntilElementDisappear","hidden","waitUntilIframeFound","framePredicate","description","frame","frames","find","Promise","resolve","Error","fillInput","pageOrFrame","inputSelector","inputValue","$eval","input","inputElement","value","type","setValue","clickButton","buttonSelector","el","click","clickLink","aSelector","pageEvalAll","selector","defaultResult","callback","args","result","$$eval","e","message","indexOf","pageEval","elementPresentOnPage","$","dropdownSelect","selectSelector","select","dropdownElements","options","evaluate","optionSelector","Array","from","document","querySelectorAll","filter","o","map","name","text"],"mappings":"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","sourcesContent":["import { Frame, Page } from 'puppeteer';\nimport { waitUntil } from './waiting';\n\nasync function waitUntilElementFound(page: Page | Frame, elementSelector: string,\n  onlyVisible = false, timeout = 30000) {\n  await page.waitForSelector(elementSelector, { visible: onlyVisible, timeout });\n}\n\nasync function waitUntilElementDisappear(page: Page, elementSelector: string, timeout = 30000) {\n  await page.waitForSelector(elementSelector, { hidden: true, timeout });\n}\n\nasync function waitUntilIframeFound(page: Page, framePredicate: (frame: Frame) => boolean, description = '', timeout = 30000) {\n  let frame: Frame | undefined;\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find(framePredicate);\n    return Promise.resolve(!!frame);\n  }, description, timeout, 1000);\n\n  if (!frame) {\n    throw new Error('failed to find iframe');\n  }\n\n  return frame;\n}\n\nasync function fillInput(pageOrFrame: Page | Frame, inputSelector: string, inputValue: string): Promise<void> {\n  await pageOrFrame.$eval(inputSelector, (input: Element) => {\n    const inputElement = input;\n    // @ts-ignore\n    inputElement.value = '';\n  });\n  await pageOrFrame.type(inputSelector, inputValue);\n}\n\nasync function setValue(pageOrFrame: Page | Frame, inputSelector: string, inputValue: string): Promise<void> {\n  await pageOrFrame.$eval(inputSelector, (input: Element, inputValue) => {\n    const inputElement = input;\n    // @ts-ignore\n    inputElement.value = inputValue;\n  }, [inputValue]);\n}\n\nasync function clickButton(page: Page | Frame, buttonSelector: string) {\n  await page.$eval(buttonSelector, (el) => (el as HTMLElement).click());\n}\n\nasync function clickLink(page: Page, aSelector: string) {\n  await page.$eval(aSelector, (el: any) => {\n    if (!el || typeof el.click === 'undefined') {\n      return;\n    }\n\n    el.click();\n  });\n}\n\nasync function pageEvalAll<R>(page: Page | Frame, selector: string,\n  defaultResult: any, callback: (elements: Element[], ...args: any) => R, ...args: any[]): Promise<R> {\n  let result = defaultResult;\n  try {\n    result = await page.$$eval(selector, callback, ...args);\n  } catch (e) {\n    // TODO temporary workaround to puppeteer@1.5.0 which breaks $$eval bevahvior until they will release a new version.\n    if (e.message.indexOf('Error: failed to find elements matching selector') !== 0) {\n      throw e;\n    }\n  }\n\n  return result;\n}\n\nasync function pageEval<R>(pageOrFrame: Page | Frame, selector: string,\n  defaultResult: any, callback: (elements: Element, ...args: any) => R, ...args: any[]): Promise<R> {\n  let result = defaultResult;\n  try {\n    result = await pageOrFrame.$eval(selector, callback, ...args);\n  } catch (e) {\n    // TODO temporary workaround to puppeteer@1.5.0 which breaks $$eval bevahvior until they will release a new version.\n    if (e.message.indexOf('Error: failed to find element matching selector') !== 0) {\n      throw e;\n    }\n  }\n\n  return result;\n}\n\nasync function elementPresentOnPage(pageOrFrame: Page | Frame, selector: string) {\n  return await pageOrFrame.$(selector) !== null;\n}\n\nasync function dropdownSelect(page: Page, selectSelector: string, value: string) {\n  await page.select(selectSelector, value);\n}\n\nasync function dropdownElements(page: Page, selector: string) {\n  const options = await page.evaluate((optionSelector) => {\n    return Array.from(document.querySelectorAll(optionSelector))\n      .filter((o) => o.value)\n      .map((o) => {\n        return {\n          name: o.text,\n          value: o.value,\n        };\n      });\n  }, `${selector} > option`);\n  return options;\n}\n\nexport {\n  waitUntilElementFound,\n  waitUntilElementDisappear,\n  waitUntilIframeFound,\n  fillInput,\n  clickButton,\n  clickLink,\n  dropdownSelect,\n  dropdownElements,\n  pageEval,\n  pageEvalAll,\n  elementPresentOnPage,\n  setValue,\n};\n"]}
133
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/helpers/elements-interactions.ts"],"names":["waitUntilElementFound","page","elementSelector","onlyVisible","timeout","waitForSelector","visible","waitUntilElementDisappear","hidden","waitUntilIframeFound","framePredicate","description","frame","frames","find","Promise","resolve","Error","fillInput","pageOrFrame","inputSelector","inputValue","$eval","input","inputElement","value","type","setValue","clickButton","buttonSelector","el","click","clickLink","aSelector","pageEvalAll","selector","defaultResult","callback","args","result","$$eval","e","message","indexOf","pageEval","elementPresentOnPage","$","dropdownSelect","selectSelector","select","dropdownElements","options","evaluate","optionSelector","Array","from","document","querySelectorAll","filter","o","map","name","text"],"mappings":";;;;;;;;;;;;;;;;;;;;;;AACA;;AAEA,eAAeA,qBAAf,CAAqCC,IAArC,EAAyDC,eAAzD,EACEC,WAAW,GAAG,KADhB,EACuBC,OADvB,EACyC;AACvC,QAAMH,IAAI,CAACI,eAAL,CAAqBH,eAArB,EAAsC;AAAEI,IAAAA,OAAO,EAAEH,WAAX;AAAwBC,IAAAA;AAAxB,GAAtC,CAAN;AACD;;AAED,eAAeG,yBAAf,CAAyCN,IAAzC,EAAqDC,eAArD,EAA8EE,OAA9E,EAAgG;AAC9F,QAAMH,IAAI,CAACI,eAAL,CAAqBH,eAArB,EAAsC;AAAEM,IAAAA,MAAM,EAAE,IAAV;AAAgBJ,IAAAA;AAAhB,GAAtC,CAAN;AACD;;AAED,eAAeK,oBAAf,CAAoCR,IAApC,EAAgDS,cAAhD,EAA2FC,WAAW,GAAG,EAAzG,EAA6GP,OAAO,GAAG,KAAvH,EAA8H;AAC5H,MAAIQ,KAAJ;AACA,QAAM,wBAAU,MAAM;AACpBA,IAAAA,KAAK,GAAGX,IAAI,CACTY,MADK,GAELC,IAFK,CAEAJ,cAFA,CAAR;AAGA,WAAOK,OAAO,CAACC,OAAR,CAAgB,CAAC,CAACJ,KAAlB,CAAP;AACD,GALK,EAKHD,WALG,EAKUP,OALV,EAKmB,IALnB,CAAN;;AAOA,MAAI,CAACQ,KAAL,EAAY;AACV,UAAM,IAAIK,KAAJ,CAAU,uBAAV,CAAN;AACD;;AAED,SAAOL,KAAP;AACD;;AAED,eAAeM,SAAf,CAAyBC,WAAzB,EAAoDC,aAApD,EAA2EC,UAA3E,EAA8G;AAC5G,QAAMF,WAAW,CAACG,KAAZ,CAAkBF,aAAlB,EAAkCG,KAAD,IAAoB;AACzD,UAAMC,YAAY,GAAGD,KAArB,CADyD,CAEzD;;AACAC,IAAAA,YAAY,CAACC,KAAb,GAAqB,EAArB;AACD,GAJK,CAAN;AAKA,QAAMN,WAAW,CAACO,IAAZ,CAAiBN,aAAjB,EAAgCC,UAAhC,CAAN;AACD;;AAED,eAAeM,QAAf,CAAwBR,WAAxB,EAAmDC,aAAnD,EAA0EC,UAA1E,EAA6G;AAC3G,QAAMF,WAAW,CAACG,KAAZ,CAAkBF,aAAlB,EAAiC,CAACG,KAAD,EAAiBF,UAAjB,KAAgC;AACrE,UAAMG,YAAY,GAAGD,KAArB,CADqE,CAErE;;AACAC,IAAAA,YAAY,CAACC,KAAb,GAAqBJ,UAArB;AACD,GAJK,EAIH,CAACA,UAAD,CAJG,CAAN;AAKD;;AAED,eAAeO,WAAf,CAA2B3B,IAA3B,EAA+C4B,cAA/C,EAAuE;AACrE,QAAM5B,IAAI,CAACqB,KAAL,CAAWO,cAAX,EAA4BC,EAAD,IAASA,EAAD,CAAoBC,KAApB,EAAnC,CAAN;AACD;;AAED,eAAeC,SAAf,CAAyB/B,IAAzB,EAAqCgC,SAArC,EAAwD;AACtD,QAAMhC,IAAI,CAACqB,KAAL,CAAWW,SAAX,EAAuBH,EAAD,IAAa;AACvC,QAAI,CAACA,EAAD,IAAO,OAAOA,EAAE,CAACC,KAAV,KAAoB,WAA/B,EAA4C;AAC1C;AACD;;AAEDD,IAAAA,EAAE,CAACC,KAAH;AACD,GANK,CAAN;AAOD;;AAED,eAAeG,WAAf,CAA8BjC,IAA9B,EAAkDkC,QAAlD,EACEC,aADF,EACsBC,QADtB,EAC0E,GAAGC,IAD7E,EACsG;AACpG,MAAIC,MAAM,GAAGH,aAAb;;AACA,MAAI;AACFG,IAAAA,MAAM,GAAG,MAAMtC,IAAI,CAACuC,MAAL,CAAYL,QAAZ,EAAsBE,QAAtB,EAAgC,GAAGC,IAAnC,CAAf;AACD,GAFD,CAEE,OAAOG,CAAP,EAAU;AACV;AACA,QAAIA,CAAC,CAACC,OAAF,CAAUC,OAAV,CAAkB,kDAAlB,MAA0E,CAA9E,EAAiF;AAC/E,YAAMF,CAAN;AACD;AACF;;AAED,SAAOF,MAAP;AACD;;AAED,eAAeK,QAAf,CAA2BzB,WAA3B,EAAsDgB,QAAtD,EACEC,aADF,EACsBC,QADtB,EACwE,GAAGC,IAD3E,EACoG;AAClG,MAAIC,MAAM,GAAGH,aAAb;;AACA,MAAI;AACFG,IAAAA,MAAM,GAAG,MAAMpB,WAAW,CAACG,KAAZ,CAAkBa,QAAlB,EAA4BE,QAA5B,EAAsC,GAAGC,IAAzC,CAAf;AACD,GAFD,CAEE,OAAOG,CAAP,EAAU;AACV;AACA,QAAIA,CAAC,CAACC,OAAF,CAAUC,OAAV,CAAkB,iDAAlB,MAAyE,CAA7E,EAAgF;AAC9E,YAAMF,CAAN;AACD;AACF;;AAED,SAAOF,MAAP;AACD;;AAED,eAAeM,oBAAf,CAAoC1B,WAApC,EAA+DgB,QAA/D,EAAiF;AAC/E,SAAO,OAAMhB,WAAW,CAAC2B,CAAZ,CAAcX,QAAd,CAAN,MAAkC,IAAzC;AACD;;AAED,eAAeY,cAAf,CAA8B9C,IAA9B,EAA0C+C,cAA1C,EAAkEvB,KAAlE,EAAiF;AAC/E,QAAMxB,IAAI,CAACgD,MAAL,CAAYD,cAAZ,EAA4BvB,KAA5B,CAAN;AACD;;AAED,eAAeyB,gBAAf,CAAgCjD,IAAhC,EAA4CkC,QAA5C,EAA8D;AAC5D,QAAMgB,OAAO,GAAG,MAAMlD,IAAI,CAACmD,QAAL,CAAeC,cAAD,IAAoB;AACtD,WAAOC,KAAK,CAACC,IAAN,CAAWC,QAAQ,CAACC,gBAAT,CAA0BJ,cAA1B,CAAX,EACJK,MADI,CACIC,CAAD,IAAOA,CAAC,CAAClC,KADZ,EAEJmC,GAFI,CAECD,CAAD,IAAO;AACV,aAAO;AACLE,QAAAA,IAAI,EAAEF,CAAC,CAACG,IADH;AAELrC,QAAAA,KAAK,EAAEkC,CAAC,CAAClC;AAFJ,OAAP;AAID,KAPI,CAAP;AAQD,GATqB,EASlB,GAAEU,QAAS,WATO,CAAtB;AAUA,SAAOgB,OAAP;AACD","sourcesContent":["import { Frame, Page } from 'puppeteer';\nimport { waitUntil } from './waiting';\n\nasync function waitUntilElementFound(page: Page | Frame, elementSelector: string,\n  onlyVisible = false, timeout?: number) {\n  await page.waitForSelector(elementSelector, { visible: onlyVisible, timeout });\n}\n\nasync function waitUntilElementDisappear(page: Page, elementSelector: string, timeout?: number) {\n  await page.waitForSelector(elementSelector, { hidden: true, timeout });\n}\n\nasync function waitUntilIframeFound(page: Page, framePredicate: (frame: Frame) => boolean, description = '', timeout = 30000) {\n  let frame: Frame | undefined;\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find(framePredicate);\n    return Promise.resolve(!!frame);\n  }, description, timeout, 1000);\n\n  if (!frame) {\n    throw new Error('failed to find iframe');\n  }\n\n  return frame;\n}\n\nasync function fillInput(pageOrFrame: Page | Frame, inputSelector: string, inputValue: string): Promise<void> {\n  await pageOrFrame.$eval(inputSelector, (input: Element) => {\n    const inputElement = input;\n    // @ts-ignore\n    inputElement.value = '';\n  });\n  await pageOrFrame.type(inputSelector, inputValue);\n}\n\nasync function setValue(pageOrFrame: Page | Frame, inputSelector: string, inputValue: string): Promise<void> {\n  await pageOrFrame.$eval(inputSelector, (input: Element, inputValue) => {\n    const inputElement = input;\n    // @ts-ignore\n    inputElement.value = inputValue;\n  }, [inputValue]);\n}\n\nasync function clickButton(page: Page | Frame, buttonSelector: string) {\n  await page.$eval(buttonSelector, (el) => (el as HTMLElement).click());\n}\n\nasync function clickLink(page: Page, aSelector: string) {\n  await page.$eval(aSelector, (el: any) => {\n    if (!el || typeof el.click === 'undefined') {\n      return;\n    }\n\n    el.click();\n  });\n}\n\nasync function pageEvalAll<R>(page: Page | Frame, selector: string,\n  defaultResult: any, callback: (elements: Element[], ...args: any) => R, ...args: any[]): Promise<R> {\n  let result = defaultResult;\n  try {\n    result = await page.$$eval(selector, callback, ...args);\n  } catch (e) {\n    // TODO temporary workaround to puppeteer@1.5.0 which breaks $$eval bevahvior until they will release a new version.\n    if (e.message.indexOf('Error: failed to find elements matching selector') !== 0) {\n      throw e;\n    }\n  }\n\n  return result;\n}\n\nasync function pageEval<R>(pageOrFrame: Page | Frame, selector: string,\n  defaultResult: any, callback: (elements: Element, ...args: any) => R, ...args: any[]): Promise<R> {\n  let result = defaultResult;\n  try {\n    result = await pageOrFrame.$eval(selector, callback, ...args);\n  } catch (e) {\n    // TODO temporary workaround to puppeteer@1.5.0 which breaks $$eval bevahvior until they will release a new version.\n    if (e.message.indexOf('Error: failed to find element matching selector') !== 0) {\n      throw e;\n    }\n  }\n\n  return result;\n}\n\nasync function elementPresentOnPage(pageOrFrame: Page | Frame, selector: string) {\n  return await pageOrFrame.$(selector) !== null;\n}\n\nasync function dropdownSelect(page: Page, selectSelector: string, value: string) {\n  await page.select(selectSelector, value);\n}\n\nasync function dropdownElements(page: Page, selector: string) {\n  const options = await page.evaluate((optionSelector) => {\n    return Array.from(document.querySelectorAll(optionSelector))\n      .filter((o) => o.value)\n      .map((o) => {\n        return {\n          name: o.text,\n          value: o.value,\n        };\n      });\n  }, `${selector} > option`);\n  return options;\n}\n\nexport {\n  waitUntilElementFound,\n  waitUntilElementDisappear,\n  waitUntilIframeFound,\n  fillInput,\n  clickButton,\n  clickLink,\n  dropdownSelect,\n  dropdownElements,\n  pageEval,\n  pageEvalAll,\n  elementPresentOnPage,\n  setValue,\n};\n"]}
@@ -183,6 +183,10 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
183
183
  this.page = await this.browser.newPage();
184
184
  }
185
185
 
186
+ if (this.options.defaultTimeout) {
187
+ this.page.setDefaultTimeout(this.options.defaultTimeout);
188
+ }
189
+
186
190
  if (this.options.preparePage) {
187
191
  debug('execute \'preparePage\' interceptor provided in options');
188
192
  await this.options.preparePage(this.page);
@@ -319,4 +323,4 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
319
323
  }
320
324
 
321
325
  exports.BaseScraperWithBrowser = BaseScraperWithBrowser;
322
- //# 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","getViewPort","width","height","initialize","Initializing","env","options","verbose","DEBUG","process","browser","executablePath","undefined","args","headless","showBrowser","puppeteer","launch","prepareBrowser","pages","length","newPage","preparePage","viewport","setViewport","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 | (() => Promise<void>);\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  protected getViewPort() {\n    return {\n      width: VIEWPORT_WIDTH,\n      height: VIEWPORT_HEIGHT,\n    };\n  }\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    const viewport = this.getViewPort();\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 if (typeof loginOptions.submitButtonSelector === 'string') {\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    if (typeof loginOptions.submitButtonSelector === 'string') {\n      await clickButton(loginFrameOrPage, loginOptions.submitButtonSelector);\n    } else {\n      await loginOptions.submitButtonSelector();\n    }\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"]}
326
+ //# 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","getViewPort","width","height","initialize","Initializing","env","options","verbose","DEBUG","process","browser","executablePath","undefined","args","headless","showBrowser","puppeteer","launch","prepareBrowser","pages","length","newPage","defaultTimeout","setDefaultTimeout","preparePage","viewport","setViewport","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 | (() => Promise<void>);\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  protected getViewPort() {\n    return {\n      width: VIEWPORT_WIDTH,\n      height: VIEWPORT_HEIGHT,\n    };\n  }\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.defaultTimeout) {\n      this.page.setDefaultTimeout(this.options.defaultTimeout);\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    const viewport = this.getViewPort();\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 if (typeof loginOptions.submitButtonSelector === 'string') {\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    if (typeof loginOptions.submitButtonSelector === 'string') {\n      await clickButton(loginFrameOrPage, loginOptions.submitButtonSelector);\n    } else {\n      await loginOptions.submitButtonSelector();\n    }\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"]}
@@ -86,6 +86,10 @@ export interface ScraperOptions {
86
86
  * if set, store a screenshot if failed to scrape. Used for debug purposes
87
87
  */
88
88
  storeFailureScreenShotPath?: string;
89
+ /**
90
+ * if set, will set the timeout in milliseconds of puppeteer's `page.setDefaultTimeout`.
91
+ */
92
+ defaultTimeout?: number;
89
93
  }
90
94
  export declare enum ScaperProgressTypes {
91
95
  Initializing = "INITIALIZING",
@@ -134,4 +134,4 @@ class BaseScraper {
134
134
  }
135
135
 
136
136
  exports.BaseScraper = BaseScraper;
137
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper.ts"],"names":["SCRAPE_PROGRESS","ScraperErrorTypes","ScaperProgressTypes","createErrorResult","errorType","errorMessage","success","createTimeoutError","Timeout","createGenericError","Generic","BaseScraper","constructor","options","EventEmitter","initialize","emitProgress","Initializing","scrape","credentials","StartScraping","loginResult","login","e","TimeoutError","message","scrapeResult","fetchData","terminate","EndScraping","_credentials","Error","companyId","_success","Terminating","type","emit","eventName","payload","eventEmitter","onProgress","func","on"],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport { Browser, Page } from 'puppeteer';\nimport { TimeoutError } from '../helpers/waiting';\nimport { TransactionsAccount } from '../transactions';\nimport { CompanyTypes } from '../definitions';\n\nconst SCRAPE_PROGRESS = 'SCRAPE_PROGRESS';\n\nexport enum ScraperErrorTypes {\n  InvalidPassword ='INVALID_PASSWORD',\n  ChangePassword = 'CHANGE_PASSWORD',\n  Timeout = 'TIMEOUT',\n  AccountBlocked = 'ACCOUNT_BLOCKED',\n  Generic = 'GENERIC',\n  General = 'GENERAL_ERROR'\n}\n\nexport interface ScaperLoginResult {\n  success: boolean;\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport interface FutureDebit {\n  amount: number;\n  amountCurrency: string;\n  chargeDate?: string;\n  bankAccountNumber?: string;\n}\n\nexport interface ScaperScrapingResult {\n  success: boolean;\n  accounts?: TransactionsAccount[];\n  futureDebits?: FutureDebit[];\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport type ScraperCredentials = Record<string, string>;\n\nexport interface ScraperOptions {\n  /**\n   * The company you want to scrape\n   */\n  companyId: CompanyTypes;\n\n  /**\n   * include more debug info about in the output\n   */\n  verbose?: boolean;\n\n  /**\n   * the date to fetch transactions from (can't be before the minimum allowed time difference for the scraper)\n   */\n  startDate: Date;\n\n  /**\n   * shows the browser while scraping, good for debugging (default false)\n   */\n  showBrowser?: boolean;\n\n\n  /**\n   * scrape transactions to be processed X months in the future\n   */\n  futureMonthsToScrape?: number;\n\n  /**\n   * option from init puppeteer browser instance outside the libary scope. you can get\n   * browser diretly from puppeteer via `puppeteer.launch()`\n   */\n  browser?: any;\n\n  /**\n   * provide a patch to local chromium to be used by puppeteer. Relevant when using\n   * `israeli-bank-scrapers-core` library\n   */\n  executablePath?: string;\n\n  /**\n   * if set to true, all installment transactions will be combine into the first one\n   */\n  combineInstallments?: boolean;\n\n  /**\n   * additional arguments to pass to the browser instance. The list of flags can be found in\n   *\n   * https://developer.mozilla.org/en-US/docs/Mozilla/Command_Line_Options\n   * https://peter.sh/experiments/chromium-command-line-switches/\n   */\n  args?: string[];\n\n  /**\n   * adjust the browser instance before it is being used\n   *\n   * @param browser\n   */\n  prepareBrowser?: (browser: Browser) => Promise<void>;\n\n  /**\n   * adjust the page instance before it is being used.\n   *\n   * @param page\n   */\n  preparePage?: (page: Page) => Promise<void>;\n\n  /**\n   * if set, store a screenshot if failed to scrape. Used for debug purposes\n   */\n  storeFailureScreenShotPath?: string;\n\n}\n\nexport enum ScaperProgressTypes {\n  Initializing = 'INITIALIZING',\n  StartScraping = 'START_SCRAPING',\n  LoggingIn = 'LOGGING_IN',\n  LoginSuccess = 'LOGIN_SUCCESS',\n  LoginFailed = 'LOGIN_FAILED',\n  ChangePassword = 'CHANGE_PASSWORD',\n  EndScraping = 'END_SCRAPING',\n  Terminating = 'TERMINATING',\n}\n\nfunction createErrorResult(errorType: ScraperErrorTypes, errorMessage: string) {\n  return {\n    success: false,\n    errorType,\n    errorMessage,\n  };\n}\n\nfunction createTimeoutError(errorMessage: string) {\n  return createErrorResult(ScraperErrorTypes.Timeout, errorMessage);\n}\n\nfunction createGenericError(errorMessage: string) {\n  return createErrorResult(ScraperErrorTypes.Generic, errorMessage);\n}\n\nexport class BaseScraper {\n  private eventEmitter = new EventEmitter();\n\n  constructor(public options: ScraperOptions) {\n  }\n\n  // eslint-disable-next-line  @typescript-eslint/require-await\n  async initialize() {\n    this.emitProgress(ScaperProgressTypes.Initializing);\n  }\n\n  async scrape(credentials: ScraperCredentials): Promise<ScaperScrapingResult> {\n    this.emitProgress(ScaperProgressTypes.StartScraping);\n    await this.initialize();\n\n    let loginResult;\n    try {\n      loginResult = await this.login(credentials);\n    } catch (e) {\n      loginResult = e instanceof TimeoutError ?\n        createTimeoutError(e.message) :\n        createGenericError(e.message);\n    }\n\n    let scrapeResult;\n    if (loginResult.success) {\n      try {\n        scrapeResult = await this.fetchData();\n      } catch (e) {\n        scrapeResult =\n          e instanceof TimeoutError ?\n            createTimeoutError(e.message) :\n            createGenericError(e.message);\n      }\n    } else {\n      scrapeResult = loginResult;\n    }\n\n    try {\n      const success = scrapeResult && scrapeResult.success === true;\n      await this.terminate(success);\n    } catch (e) {\n      scrapeResult = createGenericError(e.message);\n    }\n    this.emitProgress(ScaperProgressTypes.EndScraping);\n\n    return scrapeResult;\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  async login(_credentials: Record<string, string>): Promise<ScaperLoginResult> {\n    throw new Error(`login() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line  @typescript-eslint/require-await\n  async fetchData(): Promise<ScaperScrapingResult> {\n    throw new Error(`fetchData() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  async terminate(_success: boolean) {\n    this.emitProgress(ScaperProgressTypes.Terminating);\n  }\n\n  emitProgress(type: ScaperProgressTypes) {\n    this.emit(SCRAPE_PROGRESS, { type });\n  }\n\n  emit(eventName: string, payload: Record<string, any>) {\n    this.eventEmitter.emit(eventName, this.options.companyId, payload);\n  }\n\n  onProgress(func: (...args: any[]) => void) {\n    this.eventEmitter.on(SCRAPE_PROGRESS, func);\n  }\n}\n"]}
137
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper.ts"],"names":["SCRAPE_PROGRESS","ScraperErrorTypes","ScaperProgressTypes","createErrorResult","errorType","errorMessage","success","createTimeoutError","Timeout","createGenericError","Generic","BaseScraper","constructor","options","EventEmitter","initialize","emitProgress","Initializing","scrape","credentials","StartScraping","loginResult","login","e","TimeoutError","message","scrapeResult","fetchData","terminate","EndScraping","_credentials","Error","companyId","_success","Terminating","type","emit","eventName","payload","eventEmitter","onProgress","func","on"],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport { Browser, Page } from 'puppeteer';\nimport { TimeoutError } from '../helpers/waiting';\nimport { TransactionsAccount } from '../transactions';\nimport { CompanyTypes } from '../definitions';\n\nconst SCRAPE_PROGRESS = 'SCRAPE_PROGRESS';\n\nexport enum ScraperErrorTypes {\n  InvalidPassword ='INVALID_PASSWORD',\n  ChangePassword = 'CHANGE_PASSWORD',\n  Timeout = 'TIMEOUT',\n  AccountBlocked = 'ACCOUNT_BLOCKED',\n  Generic = 'GENERIC',\n  General = 'GENERAL_ERROR'\n}\n\nexport interface ScaperLoginResult {\n  success: boolean;\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport interface FutureDebit {\n  amount: number;\n  amountCurrency: string;\n  chargeDate?: string;\n  bankAccountNumber?: string;\n}\n\nexport interface ScaperScrapingResult {\n  success: boolean;\n  accounts?: TransactionsAccount[];\n  futureDebits?: FutureDebit[];\n  errorType?: ScraperErrorTypes;\n  errorMessage?: string; // only on success=false\n}\n\nexport type ScraperCredentials = Record<string, string>;\n\nexport interface ScraperOptions {\n  /**\n   * The company you want to scrape\n   */\n  companyId: CompanyTypes;\n\n  /**\n   * include more debug info about in the output\n   */\n  verbose?: boolean;\n\n  /**\n   * the date to fetch transactions from (can't be before the minimum allowed time difference for the scraper)\n   */\n  startDate: Date;\n\n  /**\n   * shows the browser while scraping, good for debugging (default false)\n   */\n  showBrowser?: boolean;\n\n\n  /**\n   * scrape transactions to be processed X months in the future\n   */\n  futureMonthsToScrape?: number;\n\n  /**\n   * option from init puppeteer browser instance outside the libary scope. you can get\n   * browser diretly from puppeteer via `puppeteer.launch()`\n   */\n  browser?: any;\n\n  /**\n   * provide a patch to local chromium to be used by puppeteer. Relevant when using\n   * `israeli-bank-scrapers-core` library\n   */\n  executablePath?: string;\n\n  /**\n   * if set to true, all installment transactions will be combine into the first one\n   */\n  combineInstallments?: boolean;\n\n  /**\n   * additional arguments to pass to the browser instance. The list of flags can be found in\n   *\n   * https://developer.mozilla.org/en-US/docs/Mozilla/Command_Line_Options\n   * https://peter.sh/experiments/chromium-command-line-switches/\n   */\n  args?: string[];\n\n  /**\n   * adjust the browser instance before it is being used\n   *\n   * @param browser\n   */\n  prepareBrowser?: (browser: Browser) => Promise<void>;\n\n  /**\n   * adjust the page instance before it is being used.\n   *\n   * @param page\n   */\n  preparePage?: (page: Page) => Promise<void>;\n\n  /**\n   * if set, store a screenshot if failed to scrape. Used for debug purposes\n   */\n  storeFailureScreenShotPath?: string;\n\n  /**\n   * if set, will set the timeout in milliseconds of puppeteer's `page.setDefaultTimeout`.\n   */\n  defaultTimeout?: number;\n}\n\nexport enum ScaperProgressTypes {\n  Initializing = 'INITIALIZING',\n  StartScraping = 'START_SCRAPING',\n  LoggingIn = 'LOGGING_IN',\n  LoginSuccess = 'LOGIN_SUCCESS',\n  LoginFailed = 'LOGIN_FAILED',\n  ChangePassword = 'CHANGE_PASSWORD',\n  EndScraping = 'END_SCRAPING',\n  Terminating = 'TERMINATING',\n}\n\nfunction createErrorResult(errorType: ScraperErrorTypes, errorMessage: string) {\n  return {\n    success: false,\n    errorType,\n    errorMessage,\n  };\n}\n\nfunction createTimeoutError(errorMessage: string) {\n  return createErrorResult(ScraperErrorTypes.Timeout, errorMessage);\n}\n\nfunction createGenericError(errorMessage: string) {\n  return createErrorResult(ScraperErrorTypes.Generic, errorMessage);\n}\n\nexport class BaseScraper {\n  private eventEmitter = new EventEmitter();\n\n  constructor(public options: ScraperOptions) {\n  }\n\n  // eslint-disable-next-line  @typescript-eslint/require-await\n  async initialize() {\n    this.emitProgress(ScaperProgressTypes.Initializing);\n  }\n\n  async scrape(credentials: ScraperCredentials): Promise<ScaperScrapingResult> {\n    this.emitProgress(ScaperProgressTypes.StartScraping);\n    await this.initialize();\n\n    let loginResult;\n    try {\n      loginResult = await this.login(credentials);\n    } catch (e) {\n      loginResult = e instanceof TimeoutError ?\n        createTimeoutError(e.message) :\n        createGenericError(e.message);\n    }\n\n    let scrapeResult;\n    if (loginResult.success) {\n      try {\n        scrapeResult = await this.fetchData();\n      } catch (e) {\n        scrapeResult =\n          e instanceof TimeoutError ?\n            createTimeoutError(e.message) :\n            createGenericError(e.message);\n      }\n    } else {\n      scrapeResult = loginResult;\n    }\n\n    try {\n      const success = scrapeResult && scrapeResult.success === true;\n      await this.terminate(success);\n    } catch (e) {\n      scrapeResult = createGenericError(e.message);\n    }\n    this.emitProgress(ScaperProgressTypes.EndScraping);\n\n    return scrapeResult;\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  async login(_credentials: Record<string, string>): Promise<ScaperLoginResult> {\n    throw new Error(`login() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line  @typescript-eslint/require-await\n  async fetchData(): Promise<ScaperScrapingResult> {\n    throw new Error(`fetchData() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  async terminate(_success: boolean) {\n    this.emitProgress(ScaperProgressTypes.Terminating);\n  }\n\n  emitProgress(type: ScaperProgressTypes) {\n    this.emit(SCRAPE_PROGRESS, { type });\n  }\n\n  emit(eventName: string, payload: Record<string, any>) {\n    this.eventEmitter.emit(eventName, this.options.companyId, payload);\n  }\n\n  onProgress(func: (...args: any[]) => void) {\n    this.eventEmitter.on(SCRAPE_PROGRESS, func);\n  }\n}\n"]}
@@ -66,6 +66,7 @@ const ACCUMULATING_BASKET = 'סל מצטבר';
66
66
  const POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';
67
67
  const REPLACEMENT_CARD = 'כרטיס חליפי';
68
68
  const EARLY_REPAYMENT = 'פרעון מוקדם';
69
+ const MONTHLY_CARD_FEE = 'דמי כרטיס';
69
70
  const INVALID_DETAILS_SELECTOR = '#popupWrongDetails';
70
71
  const LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';
71
72
  const categories = new Map();
@@ -125,6 +126,7 @@ function getTransactionType(txnTypeStr) {
125
126
  case POSTPONED_TRANSACTION_INSTALLMENTS:
126
127
  case REPLACEMENT_CARD:
127
128
  case EARLY_REPAYMENT:
129
+ case MONTHLY_CARD_FEE:
128
130
  return _transactions2.TransactionTypes.Normal;
129
131
 
130
132
  case INSTALLMENTS_TYPE_NAME:
@@ -310,4 +312,4 @@ class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
310
312
 
311
313
  var _default = MaxScraper;
312
314
  exports.default = _default;
313
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["debug","BASE_ACTIONS_URL","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","NORMAL_TYPE_NAME","ATM_TYPE_NAME","INTERNET_SHOPPING_TYPE_NAME","INSTALLMENTS_TYPE_NAME","MONTHLY_CHARGE_TYPE_NAME","ONE_MONTH_POSTPONED_TYPE_NAME","MONTHLY_POSTPONED_TYPE_NAME","MONTHLY_PAYMENT_TYPE_NAME","FUTURE_PURCHASE_FINANCING","MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME","THIRTY_DAYS_PLUS_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME","MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME","CREDIT_TYPE_NAME","ACCUMULATING_BASKET","POSTPONED_TRANSACTION_INSTALLMENTS","REPLACEMENT_CARD","EARLY_REPAYMENT","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","loadCategories","res","Array","isArray","result","length","forEach","id","name","set","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","category","get","categoryId","fetchTransactionsForMonth","url","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","clonedTxns","from","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startDate","toDate","moment","max","allMonths","i","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","checkReadiness","postAction","possibleResults","fetchData","results","accounts","map","success"],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntilElementFound, elementPresentOnPage, clickButton } from '../helpers/elements-interactions';\nimport getAllMonthMoments from '../helpers/dates';\nimport { fixInstallments, sortTransactionsByDate, filterOldTransactions } from '../helpers/transactions';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScraperOptions, ScraperCredentials } from './base-scraper';\nimport { getDebug } from '../helpers/debug';\n\nconst debug = getDebug('max');\n\ninterface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_ACTIONS_URL = 'https://online.max.co.il';\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_ACTIONS_URL}/Anonymous/Login/PasswordExpired.aspx`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nconst NORMAL_TYPE_NAME = 'רגילה';\nconst ATM_TYPE_NAME = 'חיוב עסקות מיידי';\nconst INTERNET_SHOPPING_TYPE_NAME = 'אינטרנט/חו\"ל';\nconst INSTALLMENTS_TYPE_NAME = 'תשלומים';\nconst MONTHLY_CHARGE_TYPE_NAME = 'חיוב חודשי';\nconst ONE_MONTH_POSTPONED_TYPE_NAME = 'דחוי חודש';\nconst MONTHLY_POSTPONED_TYPE_NAME = 'דחוי לחיוב החודשי';\nconst MONTHLY_PAYMENT_TYPE_NAME = 'תשלום חודשי';\nconst FUTURE_PURCHASE_FINANCING = 'מימון לרכישה עתידית';\nconst MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME = 'דחוי חודש תשלומים';\nconst THIRTY_DAYS_PLUS_TYPE_NAME = 'עסקת 30 פלוס';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME = 'דחוי חודשיים';\nconst MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME = 'חודשי + ריבית';\nconst CREDIT_TYPE_NAME = 'קרדיט';\nconst ACCUMULATING_BASKET = 'סל מצטבר';\nconst POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';\nconst REPLACEMENT_CARD = 'כרטיס חליפי';\nconst EARLY_REPAYMENT = 'פרעון מוקדם';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n     * url explanation:\n     * userIndex: -1 for all account owners\n     * cardIndex: -1 for all cards under the account\n     * all other query params are static, beside the date which changes for request per month\n     */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result? : Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n      res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(txnTypeStr: string) {\n  const cleanedUpTxnTypeStr = txnTypeStr.replace('\\t', ' ').trim();\n  switch (cleanedUpTxnTypeStr) {\n    case ATM_TYPE_NAME:\n    case NORMAL_TYPE_NAME:\n    case MONTHLY_CHARGE_TYPE_NAME:\n    case ONE_MONTH_POSTPONED_TYPE_NAME:\n    case MONTHLY_POSTPONED_TYPE_NAME:\n    case FUTURE_PURCHASE_FINANCING:\n    case MONTHLY_PAYMENT_TYPE_NAME:\n    case MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME:\n    case THIRTY_DAYS_PLUS_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME:\n    case ACCUMULATING_BASKET:\n    case INTERNET_SHOPPING_TYPE_NAME:\n    case MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME:\n    case POSTPONED_TRANSACTION_INSTALLMENTS:\n    case REPLACEMENT_CARD:\n    case EARLY_REPAYMENT:\n      return TransactionTypes.Normal;\n    case INSTALLMENTS_TYPE_NAME:\n    case CREDIT_TYPE_NAME:\n      return TransactionTypes.Installments;\n    default:\n      throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ?\n    rawTransaction.purchaseDate :\n    rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments ?\n    `${rawTransaction.dealData?.arn}_${installments.number}` :\n    rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    description: rawTransaction.merchantName.trim(),\n    memo: rawTransaction.comments,\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult{\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter((transaction) => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach((accountNumber) => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = filterOldTransactions(clonedTxns, startMoment, combineInstallments || false);\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach((accountNumber) => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(txns, startMoment, options.combineInstallments || false);\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [async () => {\n    return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n  }];\n  urls[LoginResults.UnknownError] = [async () => {\n    return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n  }];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass MaxScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#login-password #send-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map((accountNumber) => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"]}
315
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["debug","BASE_ACTIONS_URL","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","NORMAL_TYPE_NAME","ATM_TYPE_NAME","INTERNET_SHOPPING_TYPE_NAME","INSTALLMENTS_TYPE_NAME","MONTHLY_CHARGE_TYPE_NAME","ONE_MONTH_POSTPONED_TYPE_NAME","MONTHLY_POSTPONED_TYPE_NAME","MONTHLY_PAYMENT_TYPE_NAME","FUTURE_PURCHASE_FINANCING","MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME","THIRTY_DAYS_PLUS_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME","MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME","CREDIT_TYPE_NAME","ACCUMULATING_BASKET","POSTPONED_TRANSACTION_INSTALLMENTS","REPLACEMENT_CARD","EARLY_REPAYMENT","MONTHLY_CARD_FEE","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","loadCategories","res","Array","isArray","result","length","forEach","id","name","set","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","category","get","categoryId","fetchTransactionsForMonth","url","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","clonedTxns","from","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startDate","toDate","moment","max","allMonths","i","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","checkReadiness","postAction","possibleResults","fetchData","results","accounts","map","success"],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\nimport { Page } from 'puppeteer';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { waitUntilElementFound, elementPresentOnPage, clickButton } from '../helpers/elements-interactions';\nimport getAllMonthMoments from '../helpers/dates';\nimport { fixInstallments, sortTransactionsByDate, filterOldTransactions } from '../helpers/transactions';\nimport { Transaction, TransactionStatuses, TransactionTypes } from '../transactions';\nimport { ScraperOptions, ScraperCredentials } from './base-scraper';\nimport { getDebug } from '../helpers/debug';\n\nconst debug = getDebug('max');\n\ninterface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_ACTIONS_URL = 'https://online.max.co.il';\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_ACTIONS_URL}/Anonymous/Login/PasswordExpired.aspx`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nconst NORMAL_TYPE_NAME = 'רגילה';\nconst ATM_TYPE_NAME = 'חיוב עסקות מיידי';\nconst INTERNET_SHOPPING_TYPE_NAME = 'אינטרנט/חו\"ל';\nconst INSTALLMENTS_TYPE_NAME = 'תשלומים';\nconst MONTHLY_CHARGE_TYPE_NAME = 'חיוב חודשי';\nconst ONE_MONTH_POSTPONED_TYPE_NAME = 'דחוי חודש';\nconst MONTHLY_POSTPONED_TYPE_NAME = 'דחוי לחיוב החודשי';\nconst MONTHLY_PAYMENT_TYPE_NAME = 'תשלום חודשי';\nconst FUTURE_PURCHASE_FINANCING = 'מימון לרכישה עתידית';\nconst MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME = 'דחוי חודש תשלומים';\nconst THIRTY_DAYS_PLUS_TYPE_NAME = 'עסקת 30 פלוס';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME = 'דחוי חודשיים';\nconst MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME = 'חודשי + ריבית';\nconst CREDIT_TYPE_NAME = 'קרדיט';\nconst ACCUMULATING_BASKET = 'סל מצטבר';\nconst POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';\nconst REPLACEMENT_CARD = 'כרטיס חליפי';\nconst EARLY_REPAYMENT = 'פרעון מוקדם';\nconst MONTHLY_CARD_FEE = 'דמי כרטיס';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n     * url explanation:\n     * userIndex: -1 for all account owners\n     * cardIndex: -1 for all cards under the account\n     * all other query params are static, beside the date which changes for request per month\n     */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result? : Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n      res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(txnTypeStr: string) {\n  const cleanedUpTxnTypeStr = txnTypeStr.replace('\\t', ' ').trim();\n  switch (cleanedUpTxnTypeStr) {\n    case ATM_TYPE_NAME:\n    case NORMAL_TYPE_NAME:\n    case MONTHLY_CHARGE_TYPE_NAME:\n    case ONE_MONTH_POSTPONED_TYPE_NAME:\n    case MONTHLY_POSTPONED_TYPE_NAME:\n    case FUTURE_PURCHASE_FINANCING:\n    case MONTHLY_PAYMENT_TYPE_NAME:\n    case MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME:\n    case THIRTY_DAYS_PLUS_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME:\n    case ACCUMULATING_BASKET:\n    case INTERNET_SHOPPING_TYPE_NAME:\n    case MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME:\n    case POSTPONED_TRANSACTION_INSTALLMENTS:\n    case REPLACEMENT_CARD:\n    case EARLY_REPAYMENT:\n    case MONTHLY_CARD_FEE:\n      return TransactionTypes.Normal;\n    case INSTALLMENTS_TYPE_NAME:\n    case CREDIT_TYPE_NAME:\n      return TransactionTypes.Installments;\n    default:\n      throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ?\n    rawTransaction.purchaseDate :\n    rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments ?\n    `${rawTransaction.dealData?.arn}_${installments.number}` :\n    rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    description: rawTransaction.merchantName.trim(),\n    memo: rawTransaction.comments,\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult{\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter((transaction) => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach((accountNumber) => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = filterOldTransactions(clonedTxns, startMoment, combineInstallments || false);\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach((accountNumber) => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(txns, startMoment, options.combineInstallments || false);\n    allResults[accountNumber] = txns;\n  });\n\n  return allResults;\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  const urls: PossibleLoginResults = {};\n  urls[LoginResults.Success] = [SUCCESS_URL];\n  urls[LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];\n  urls[LoginResults.InvalidPassword] = [async () => {\n    return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n  }];\n  urls[LoginResults.UnknownError] = [async () => {\n    return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n  }];\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\nclass MaxScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: '#login-password #send-code',\n      preAction: async () => {\n        if (await elementPresentOnPage(this.page, '#closePopup')) {\n          await clickButton(this.page, '#closePopup');\n        }\n        await clickButton(this.page, '.personal-area > a.go-to-personal-area');\n        await waitUntilElementFound(this.page, '#login-password-link', true);\n        await clickButton(this.page, '#login-password-link');\n        await waitUntilElementFound(this.page, '#login-password.tab-pane.active app-user-login-form', true);\n      },\n      checkReadiness: async () => {\n        await waitUntilElementFound(this.page, '.personal-area > a.go-to-personal-area', true);\n      },\n      postAction: async () => redirectOrDialog(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const results = await fetchTransactions(this.page, this.options);\n    const accounts = Object.keys(results).map((accountNumber) => {\n      return {\n        accountNumber,\n        txns: results[accountNumber],\n      };\n    });\n\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default MaxScraper;\n"]}
@@ -46,7 +46,7 @@ async function getLoginFrame(page) {
46
46
  let frame = null;
47
47
  debug('wait until login frame found');
48
48
  await (0, _waiting.waitUntil)(() => {
49
- frame = page.frames().find(f => f.url().includes('connect.cal-online')) || null;
49
+ frame = page.frames().find(f => f.url().includes('calconnect')) || null;
50
50
  return Promise.resolve(!!frame);
51
51
  }, 'wait for iframe with login form', 10000, 1000);
52
52
 
@@ -412,4 +412,4 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
412
412
 
413
413
  var _default = VisaCalScraper;
414
414
  exports.default = _default;
415
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_URL","LONG_DATE_FORMAT","DATE_FORMAT","InvalidPasswordMessage","debug","getLoginFrame","page","frame","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","errorMessage","item","innerText","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","amountStrCln","replace","currency","amount","SHEKEL_CURRENCY_SYMBOL","parseFloat","SHEKEL_CURRENCY","DOLLAR_CURRENCY_SYMBOL","DOLLAR_CURRENCY","EURO_CURRENCY_SYMBOL","EURO_CURRENCY","parts","split","getTransactionInstallments","memo","parsedMemo","exec","length","number","parseInt","total","convertTransactions","txns","map","txn","originalAmountTuple","originalAmount","chargedAmountTuple","chargedAmount","installments","txnDate","date","processedDateFormat","processedDate","txnProcessedDate","result","type","TransactionTypes","Installments","Normal","status","TransactionStatuses","Completed","add","toISOString","originalCurrency","chargedCurrency","description","fetchTransactionsForAccount","startDate","accountNumber","scraperOptions","startDateValue","format","dateSelector","dateHiddenFieldSelector","buttonSelector","nextPageSelector","billingLabelSelector","secondaryBillingLabelSelector","noDataSelector","items","el","startDateIndex","findIndex","option","accountTransactions","currentDateIndex","waitForTimeout","all","waitForNavigation","waitUntil","pageHasNoTransactions","element","siteValue","billingDateLabel","settlementDateRegex","billingDate","hasNextPage","rawTransactions","columns","getElementsByTagName","push","filter","combineInstallments","substring","getAccountNumbers","elements","e","text","then","res","trim","setAccount","account","elem","a","click","fetchTransactions","accountNumbers","accounts","fetchFutureDebits","futureDebitsSelector","debitMountClass","debitWhenChargeClass","debitBankNumberClass","currBankEl","getElementsByClassName","whenCharge","bankNumber","futureDebits","amountData","chargeDate","bankAccountNumber","amountCurrency","VisaCalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","userAgent","fetchData","defaultStartMoment","subtract","toDate","startMoment","moment","max","navigateTo","undefined","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Frame, Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport {\n  clickButton, elementPresentOnPage, pageEval, pageEvalAll, setValue, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport {\n  Transaction,\n  TransactionInstallments,\n  TransactionsAccount,\n  TransactionStatuses,\n  TransactionTypes,\n} from '../transactions';\nimport { ScraperOptions, ScaperScrapingResult, ScraperCredentials } from './base-scraper';\nimport {\n  DOLLAR_CURRENCY, DOLLAR_CURRENCY_SYMBOL, EURO_CURRENCY, EURO_CURRENCY_SYMBOL, SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL,\n} from '../constants';\nimport { waitUntil } from '../helpers/waiting';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { getDebug } from '../helpers/debug';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_URL = 'https://services.cal-online.co.il/Card-Holders/Screens/Transactions/Transactions.aspx';\nconst LONG_DATE_FORMAT = 'DD/MM/YYYY';\nconst DATE_FORMAT = 'DD/MM/YY';\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\ninterface ScrapedTransaction {\n  date: string;\n  processedDate: string;\n  description: string;\n  originalAmount: string;\n  chargedAmount: string;\n  memo: string;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('connect.cal-online')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/AccountManagement/i],\n    [LoginResults.InvalidPassword]: [async (options?: { page?: Page}) => {\n      const page = options?.page;\n      if (!page) {\n        return false;\n      }\n      return hasInvalidPasswordError(page);\n    }],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    // [LoginResults.ChangePassword]: [], // TODO add when reaching this scenario\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(DOLLAR_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(DOLLAR_CURRENCY_SYMBOL, ''));\n    currency = DOLLAR_CURRENCY;\n  } else if (amountStrCln.includes(EURO_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(EURO_CURRENCY_SYMBOL, ''));\n    currency = EURO_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    [currency] = parts;\n    amount = -parseFloat(parts[1]);\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction getTransactionInstallments(memo: string): TransactionInstallments | null {\n  const parsedMemo = (/תשלום (\\d+) מתוך (\\d+)/).exec(memo || '');\n\n  if (!parsedMemo || parsedMemo.length === 0) {\n    return null;\n  }\n\n  return {\n    number: parseInt(parsedMemo[1], 10),\n    total: parseInt(parsedMemo[2], 10),\n  };\n}\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  debug(`convert ${txns.length} raw transactions to official Transaction structure`);\n  return txns.map((txn) => {\n    const originalAmountTuple = getAmountData(txn.originalAmount || '');\n    const chargedAmountTuple = getAmountData(txn.chargedAmount || '');\n\n    const installments = getTransactionInstallments(txn.memo);\n    const txnDate = moment(txn.date, DATE_FORMAT);\n    const processedDateFormat =\n      txn.processedDate.length === 8 ?\n        DATE_FORMAT :\n        txn.processedDate.length === 9 || txn.processedDate.length === 10 ?\n          LONG_DATE_FORMAT :\n          null;\n    if (!processedDateFormat) {\n      throw new Error('invalid processed date');\n    }\n    const txnProcessedDate = moment(txn.processedDate, processedDateFormat);\n\n    const result: Transaction = {\n      type: installments ? TransactionTypes.Installments : TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      date: installments ? txnDate.add(installments.number - 1, 'month').toISOString() : txnDate.toISOString(),\n      processedDate: txnProcessedDate.toISOString(),\n      originalAmount: originalAmountTuple.amount,\n      originalCurrency: originalAmountTuple.currency,\n      chargedAmount: chargedAmountTuple.amount,\n      chargedCurrency: chargedAmountTuple.currency,\n      description: txn.description || '',\n      memo: txn.memo || '',\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment, accountNumber: string, scraperOptions: ScraperOptions): Promise<TransactionsAccount> {\n  const startDateValue = startDate.format('MM/YYYY');\n  const dateSelector = '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_TextBox\"]';\n  const dateHiddenFieldSelector = '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_HiddenField\"]';\n  const buttonSelector = '[id$=\"FormAreaNoBorder_FormArea_ctlSubmitRequest\"]';\n  const nextPageSelector = '[id$=\"FormAreaNoBorder_FormArea_ctlGridPager_btnNext\"]';\n  const billingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlMainToolBar_lblCaption]';\n  const secondaryBillingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlSecondaryToolBar_lblCaption]';\n  const noDataSelector = '[id$=FormAreaNoBorder_FormArea_msgboxErrorMessages]';\n\n  debug('find the start date index in the dropbox');\n  const options = await pageEvalAll(page, '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_OptionList\"] li', [], (items) => {\n    return items.map((el: any) => el.innerText);\n  });\n  const startDateIndex = options.findIndex((option) => option === startDateValue);\n\n  debug(`scrape ${options.length - startDateIndex} billing cycles`);\n  const accountTransactions: Transaction[] = [];\n  for (let currentDateIndex = startDateIndex; currentDateIndex < options.length; currentDateIndex += 1) {\n    debug('wait for date selector to be found');\n    await waitUntilElementFound(page, dateSelector, true);\n    debug(`set hidden value of the date selector to be the index ${currentDateIndex}`);\n    await setValue(page, dateHiddenFieldSelector, `${currentDateIndex}`);\n    debug('wait a second to workaround navigation issue in headless browser mode');\n    await page.waitForTimeout(1000);\n    debug('click on the filter submit button and wait for navigation');\n    await Promise.all([\n      page.waitForNavigation({ waitUntil: 'domcontentloaded' }),\n      clickButton(page, buttonSelector),\n    ]);\n    debug('check if month has no transactions');\n    const pageHasNoTransactions = await pageEval(page, noDataSelector, false, ((element) => {\n      const siteValue = ((element as HTMLSpanElement).innerText || '').replace(/[^ א-ת]/g, '');\n      return siteValue === 'לא נמצאו נתונים';\n    }));\n\n    if (pageHasNoTransactions) {\n      debug('page has no transactions');\n    } else {\n      debug('find the billing date');\n      let billingDateLabel = await pageEval(page, billingLabelSelector, '', ((element) => {\n        return (element as HTMLSpanElement).innerText;\n      }));\n      let settlementDateRegex = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/;\n\n      if (billingDateLabel === '') {\n        billingDateLabel = await pageEval(page, secondaryBillingLabelSelector, '', ((element) => {\n          return (element as HTMLSpanElement).innerText;\n        }));\n        settlementDateRegex = /\\d{1,2}[/]\\d{2,4}/;\n      }\n\n      const billingDate = settlementDateRegex.exec(billingDateLabel)?.[0];\n\n      if (!billingDate) {\n        throw new Error('failed to fetch process date');\n      }\n\n      debug(`found the billing date for that month ${billingDate}`);\n      let hasNextPage = false;\n      do {\n        debug('fetch raw transactions from page');\n        const rawTransactions = await pageEvalAll<(ScrapedTransaction | null)[]>(page, '#ctlMainGrid > tbody tr, #ctlSecondaryGrid > tbody tr', [], (items, billingDate) => {\n          return (items).map((el) => {\n            const columns = el.getElementsByTagName('td');\n            if (columns.length === 6) {\n              return {\n                processedDate: columns[0].innerText,\n                date: columns[1].innerText,\n                description: columns[2].innerText,\n                originalAmount: columns[3].innerText,\n                chargedAmount: columns[4].innerText,\n                memo: columns[5].innerText,\n              };\n            }\n            if (columns.length === 5) {\n              return {\n                processedDate: billingDate,\n                date: columns[0].innerText,\n                description: columns[1].innerText,\n                originalAmount: columns[2].innerText,\n                chargedAmount: columns[3].innerText,\n                memo: columns[4].innerText,\n              };\n            }\n            return null;\n          });\n        }, billingDate);\n        debug(`fetched ${rawTransactions.length} raw transactions from page`);\n        accountTransactions.push(...convertTransactions((rawTransactions as ScrapedTransaction[])\n          .filter((item) => !!item)));\n\n        debug('check for existance of another page');\n        hasNextPage = await elementPresentOnPage(page, nextPageSelector);\n        if (hasNextPage) {\n          debug('has another page, click on button next and wait for page navigation');\n          await Promise.all([\n            page.waitForNavigation({ waitUntil: 'domcontentloaded' }),\n            await clickButton(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]'),\n          ]);\n        }\n      } while (hasNextPage);\n    }\n  }\n\n  debug('filer out old transactions');\n  const txns = filterOldTransactions(accountTransactions, startDate, scraperOptions.combineInstallments || false);\n  debug(`found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`);\n  return {\n    accountNumber,\n    txns,\n  };\n}\n\nasync function getAccountNumbers(page: Page): Promise<string[]> {\n  return pageEvalAll(page, '[id$=lnkItem]', [], (elements) => elements.map((e) => (e as HTMLAnchorElement).text)).then((res) => res.map((text) => /\\d+$/.exec(text.trim())?.[0] ?? ''));\n}\n\nasync function setAccount(page: Page, account: string) {\n  await pageEvalAll(\n    page,\n    '[id$=lnkItem]',\n    null,\n    (elements, account) => {\n      for (const elem of elements) {\n        const a = elem as HTMLAnchorElement;\n        if (a.text.includes(account)) {\n          a.click();\n        }\n      }\n    },\n    account,\n  );\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment, scraperOptions: ScraperOptions): Promise<TransactionsAccount[]> {\n  const accountNumbers: string[] = await getAccountNumbers(page);\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of accountNumbers) {\n    debug(`setting account: ${account}`);\n    await setAccount(page, account);\n    await page.waitForTimeout(1000);\n    accounts.push(\n      await fetchTransactionsForAccount(\n        page,\n        startDate,\n        account,\n        scraperOptions,\n      ),\n    );\n  }\n\n  return accounts;\n}\n\nasync function fetchFutureDebits(page: Page) {\n  const futureDebitsSelector = '.homepage-banks-top';\n\n  const result = await pageEvalAll(page, futureDebitsSelector, [], (items) => {\n    const debitMountClass = 'amount';\n    const debitWhenChargeClass = 'when-charge';\n    const debitBankNumberClass = 'bankDesc';\n\n    return items.map((currBankEl: any) => {\n      const amount = currBankEl.getElementsByClassName(debitMountClass)[0].innerText;\n      const whenCharge = currBankEl.getElementsByClassName(debitWhenChargeClass)[0].innerText;\n      const bankNumber = currBankEl.getElementsByClassName(debitBankNumberClass)[0].innerText;\n      return {\n        amount,\n        whenCharge,\n        bankNumber,\n      };\n    });\n  });\n  const futureDebits = result.map((item) => {\n    const amountData = getAmountData(item.amount);\n    const chargeDate = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/.exec(item.whenCharge)?.[0];\n    const bankAccountNumber = /\\d+-\\d+/.exec(item.bankNumber)?.[0];\n    return {\n      amount: amountData.amount,\n      amountCurrency: amountData.currency,\n      chargeDate,\n      bankAccountNumber,\n    };\n  });\n  return futureDebits;\n}\n\nclass VisaCalScraper extends BaseScraperWithBrowser {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  getLoginOptions(credentials: Record<string, string>) {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\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    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    debug('fetch future debits');\n    const futureDebits = await fetchFutureDebits(this.page);\n\n    debug('navigate to transactions page');\n    await this.navigateTo(TRANSACTIONS_URL, undefined, 60000);\n\n    debug('fetch accounts transactions');\n    const accounts = await fetchTransactions(this.page, startMoment, this.options);\n\n    debug('return the scraped accounts');\n    return {\n      success: true,\n      accounts,\n      futureDebits,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
415
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_URL","LONG_DATE_FORMAT","DATE_FORMAT","InvalidPasswordMessage","debug","getLoginFrame","page","frame","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","errorMessage","item","innerText","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","createLoginFields","credentials","selector","value","username","password","getAmountData","amountStr","amountStrCln","replace","currency","amount","SHEKEL_CURRENCY_SYMBOL","parseFloat","SHEKEL_CURRENCY","DOLLAR_CURRENCY_SYMBOL","DOLLAR_CURRENCY","EURO_CURRENCY_SYMBOL","EURO_CURRENCY","parts","split","getTransactionInstallments","memo","parsedMemo","exec","length","number","parseInt","total","convertTransactions","txns","map","txn","originalAmountTuple","originalAmount","chargedAmountTuple","chargedAmount","installments","txnDate","date","processedDateFormat","processedDate","txnProcessedDate","result","type","TransactionTypes","Installments","Normal","status","TransactionStatuses","Completed","add","toISOString","originalCurrency","chargedCurrency","description","fetchTransactionsForAccount","startDate","accountNumber","scraperOptions","startDateValue","format","dateSelector","dateHiddenFieldSelector","buttonSelector","nextPageSelector","billingLabelSelector","secondaryBillingLabelSelector","noDataSelector","items","el","startDateIndex","findIndex","option","accountTransactions","currentDateIndex","waitForTimeout","all","waitForNavigation","waitUntil","pageHasNoTransactions","element","siteValue","billingDateLabel","settlementDateRegex","billingDate","hasNextPage","rawTransactions","columns","getElementsByTagName","push","filter","combineInstallments","substring","getAccountNumbers","elements","e","text","then","res","trim","setAccount","account","elem","a","click","fetchTransactions","accountNumbers","accounts","fetchFutureDebits","futureDebitsSelector","debitMountClass","debitWhenChargeClass","debitBankNumberClass","currBankEl","getElementsByClassName","whenCharge","bankNumber","futureDebits","amountData","chargeDate","bankAccountNumber","amountCurrency","VisaCalScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","userAgent","fetchData","defaultStartMoment","subtract","toDate","startMoment","moment","max","navigateTo","undefined","success"],"mappings":"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","sourcesContent":["import moment, { Moment } from 'moment';\nimport { Frame, Page } from 'puppeteer';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport {\n  clickButton, elementPresentOnPage, pageEval, pageEvalAll, setValue, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport {\n  Transaction,\n  TransactionInstallments,\n  TransactionsAccount,\n  TransactionStatuses,\n  TransactionTypes,\n} from '../transactions';\nimport { ScraperOptions, ScaperScrapingResult, ScraperCredentials } from './base-scraper';\nimport {\n  DOLLAR_CURRENCY, DOLLAR_CURRENCY_SYMBOL, EURO_CURRENCY, EURO_CURRENCY_SYMBOL, SHEKEL_CURRENCY, SHEKEL_CURRENCY_SYMBOL,\n} from '../constants';\nimport { waitUntil } from '../helpers/waiting';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { getDebug } from '../helpers/debug';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_URL = 'https://services.cal-online.co.il/Card-Holders/Screens/Transactions/Transactions.aspx';\nconst LONG_DATE_FORMAT = 'DD/MM/YYYY';\nconst DATE_FORMAT = 'DD/MM/YY';\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\ninterface ScrapedTransaction {\n  date: string;\n  processedDate: string;\n  description: string;\n  originalAmount: string;\n  chargedAmount: string;\n  memo: string;\n}\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/AccountManagement/i],\n    [LoginResults.InvalidPassword]: [async (options?: { page?: Page}) => {\n      const page = options?.page;\n      if (!page) {\n        return false;\n      }\n      return hasInvalidPasswordError(page);\n    }],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    // [LoginResults.ChangePassword]: [], // TODO add when reaching this scenario\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\n\nfunction getAmountData(amountStr: string) {\n  const amountStrCln = amountStr.replace(',', '');\n  let currency: string | null = null;\n  let amount: number | null = null;\n  if (amountStrCln.includes(SHEKEL_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(SHEKEL_CURRENCY_SYMBOL, ''));\n    currency = SHEKEL_CURRENCY;\n  } else if (amountStrCln.includes(DOLLAR_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(DOLLAR_CURRENCY_SYMBOL, ''));\n    currency = DOLLAR_CURRENCY;\n  } else if (amountStrCln.includes(EURO_CURRENCY_SYMBOL)) {\n    amount = -parseFloat(amountStrCln.replace(EURO_CURRENCY_SYMBOL, ''));\n    currency = EURO_CURRENCY;\n  } else {\n    const parts = amountStrCln.split(' ');\n    [currency] = parts;\n    amount = -parseFloat(parts[1]);\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction getTransactionInstallments(memo: string): TransactionInstallments | null {\n  const parsedMemo = (/תשלום (\\d+) מתוך (\\d+)/).exec(memo || '');\n\n  if (!parsedMemo || parsedMemo.length === 0) {\n    return null;\n  }\n\n  return {\n    number: parseInt(parsedMemo[1], 10),\n    total: parseInt(parsedMemo[2], 10),\n  };\n}\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  debug(`convert ${txns.length} raw transactions to official Transaction structure`);\n  return txns.map((txn) => {\n    const originalAmountTuple = getAmountData(txn.originalAmount || '');\n    const chargedAmountTuple = getAmountData(txn.chargedAmount || '');\n\n    const installments = getTransactionInstallments(txn.memo);\n    const txnDate = moment(txn.date, DATE_FORMAT);\n    const processedDateFormat =\n      txn.processedDate.length === 8 ?\n        DATE_FORMAT :\n        txn.processedDate.length === 9 || txn.processedDate.length === 10 ?\n          LONG_DATE_FORMAT :\n          null;\n    if (!processedDateFormat) {\n      throw new Error('invalid processed date');\n    }\n    const txnProcessedDate = moment(txn.processedDate, processedDateFormat);\n\n    const result: Transaction = {\n      type: installments ? TransactionTypes.Installments : TransactionTypes.Normal,\n      status: TransactionStatuses.Completed,\n      date: installments ? txnDate.add(installments.number - 1, 'month').toISOString() : txnDate.toISOString(),\n      processedDate: txnProcessedDate.toISOString(),\n      originalAmount: originalAmountTuple.amount,\n      originalCurrency: originalAmountTuple.currency,\n      chargedAmount: chargedAmountTuple.amount,\n      chargedCurrency: chargedAmountTuple.currency,\n      description: txn.description || '',\n      memo: txn.memo || '',\n    };\n\n    if (installments) {\n      result.installments = installments;\n    }\n\n    return result;\n  });\n}\n\nasync function fetchTransactionsForAccount(page: Page, startDate: Moment, accountNumber: string, scraperOptions: ScraperOptions): Promise<TransactionsAccount> {\n  const startDateValue = startDate.format('MM/YYYY');\n  const dateSelector = '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_TextBox\"]';\n  const dateHiddenFieldSelector = '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_HiddenField\"]';\n  const buttonSelector = '[id$=\"FormAreaNoBorder_FormArea_ctlSubmitRequest\"]';\n  const nextPageSelector = '[id$=\"FormAreaNoBorder_FormArea_ctlGridPager_btnNext\"]';\n  const billingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlMainToolBar_lblCaption]';\n  const secondaryBillingLabelSelector = '[id$=FormAreaNoBorder_FormArea_ctlSecondaryToolBar_lblCaption]';\n  const noDataSelector = '[id$=FormAreaNoBorder_FormArea_msgboxErrorMessages]';\n\n  debug('find the start date index in the dropbox');\n  const options = await pageEvalAll(page, '[id$=\"FormAreaNoBorder_FormArea_clndrDebitDateScope_OptionList\"] li', [], (items) => {\n    return items.map((el: any) => el.innerText);\n  });\n  const startDateIndex = options.findIndex((option) => option === startDateValue);\n\n  debug(`scrape ${options.length - startDateIndex} billing cycles`);\n  const accountTransactions: Transaction[] = [];\n  for (let currentDateIndex = startDateIndex; currentDateIndex < options.length; currentDateIndex += 1) {\n    debug('wait for date selector to be found');\n    await waitUntilElementFound(page, dateSelector, true);\n    debug(`set hidden value of the date selector to be the index ${currentDateIndex}`);\n    await setValue(page, dateHiddenFieldSelector, `${currentDateIndex}`);\n    debug('wait a second to workaround navigation issue in headless browser mode');\n    await page.waitForTimeout(1000);\n    debug('click on the filter submit button and wait for navigation');\n    await Promise.all([\n      page.waitForNavigation({ waitUntil: 'domcontentloaded' }),\n      clickButton(page, buttonSelector),\n    ]);\n    debug('check if month has no transactions');\n    const pageHasNoTransactions = await pageEval(page, noDataSelector, false, ((element) => {\n      const siteValue = ((element as HTMLSpanElement).innerText || '').replace(/[^ א-ת]/g, '');\n      return siteValue === 'לא נמצאו נתונים';\n    }));\n\n    if (pageHasNoTransactions) {\n      debug('page has no transactions');\n    } else {\n      debug('find the billing date');\n      let billingDateLabel = await pageEval(page, billingLabelSelector, '', ((element) => {\n        return (element as HTMLSpanElement).innerText;\n      }));\n      let settlementDateRegex = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/;\n\n      if (billingDateLabel === '') {\n        billingDateLabel = await pageEval(page, secondaryBillingLabelSelector, '', ((element) => {\n          return (element as HTMLSpanElement).innerText;\n        }));\n        settlementDateRegex = /\\d{1,2}[/]\\d{2,4}/;\n      }\n\n      const billingDate = settlementDateRegex.exec(billingDateLabel)?.[0];\n\n      if (!billingDate) {\n        throw new Error('failed to fetch process date');\n      }\n\n      debug(`found the billing date for that month ${billingDate}`);\n      let hasNextPage = false;\n      do {\n        debug('fetch raw transactions from page');\n        const rawTransactions = await pageEvalAll<(ScrapedTransaction | null)[]>(page, '#ctlMainGrid > tbody tr, #ctlSecondaryGrid > tbody tr', [], (items, billingDate) => {\n          return (items).map((el) => {\n            const columns = el.getElementsByTagName('td');\n            if (columns.length === 6) {\n              return {\n                processedDate: columns[0].innerText,\n                date: columns[1].innerText,\n                description: columns[2].innerText,\n                originalAmount: columns[3].innerText,\n                chargedAmount: columns[4].innerText,\n                memo: columns[5].innerText,\n              };\n            }\n            if (columns.length === 5) {\n              return {\n                processedDate: billingDate,\n                date: columns[0].innerText,\n                description: columns[1].innerText,\n                originalAmount: columns[2].innerText,\n                chargedAmount: columns[3].innerText,\n                memo: columns[4].innerText,\n              };\n            }\n            return null;\n          });\n        }, billingDate);\n        debug(`fetched ${rawTransactions.length} raw transactions from page`);\n        accountTransactions.push(...convertTransactions((rawTransactions as ScrapedTransaction[])\n          .filter((item) => !!item)));\n\n        debug('check for existance of another page');\n        hasNextPage = await elementPresentOnPage(page, nextPageSelector);\n        if (hasNextPage) {\n          debug('has another page, click on button next and wait for page navigation');\n          await Promise.all([\n            page.waitForNavigation({ waitUntil: 'domcontentloaded' }),\n            await clickButton(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]'),\n          ]);\n        }\n      } while (hasNextPage);\n    }\n  }\n\n  debug('filer out old transactions');\n  const txns = filterOldTransactions(accountTransactions, startDate, scraperOptions.combineInstallments || false);\n  debug(`found ${txns.length} valid transactions out of ${accountTransactions.length} transactions for account ending with ${accountNumber.substring(accountNumber.length - 2)}`);\n  return {\n    accountNumber,\n    txns,\n  };\n}\n\nasync function getAccountNumbers(page: Page): Promise<string[]> {\n  return pageEvalAll(page, '[id$=lnkItem]', [], (elements) => elements.map((e) => (e as HTMLAnchorElement).text)).then((res) => res.map((text) => /\\d+$/.exec(text.trim())?.[0] ?? ''));\n}\n\nasync function setAccount(page: Page, account: string) {\n  await pageEvalAll(\n    page,\n    '[id$=lnkItem]',\n    null,\n    (elements, account) => {\n      for (const elem of elements) {\n        const a = elem as HTMLAnchorElement;\n        if (a.text.includes(account)) {\n          a.click();\n        }\n      }\n    },\n    account,\n  );\n}\n\nasync function fetchTransactions(page: Page, startDate: Moment, scraperOptions: ScraperOptions): Promise<TransactionsAccount[]> {\n  const accountNumbers: string[] = await getAccountNumbers(page);\n  const accounts: TransactionsAccount[] = [];\n\n  for (const account of accountNumbers) {\n    debug(`setting account: ${account}`);\n    await setAccount(page, account);\n    await page.waitForTimeout(1000);\n    accounts.push(\n      await fetchTransactionsForAccount(\n        page,\n        startDate,\n        account,\n        scraperOptions,\n      ),\n    );\n  }\n\n  return accounts;\n}\n\nasync function fetchFutureDebits(page: Page) {\n  const futureDebitsSelector = '.homepage-banks-top';\n\n  const result = await pageEvalAll(page, futureDebitsSelector, [], (items) => {\n    const debitMountClass = 'amount';\n    const debitWhenChargeClass = 'when-charge';\n    const debitBankNumberClass = 'bankDesc';\n\n    return items.map((currBankEl: any) => {\n      const amount = currBankEl.getElementsByClassName(debitMountClass)[0].innerText;\n      const whenCharge = currBankEl.getElementsByClassName(debitWhenChargeClass)[0].innerText;\n      const bankNumber = currBankEl.getElementsByClassName(debitBankNumberClass)[0].innerText;\n      return {\n        amount,\n        whenCharge,\n        bankNumber,\n      };\n    });\n  });\n  const futureDebits = result.map((item) => {\n    const amountData = getAmountData(item.amount);\n    const chargeDate = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/.exec(item.whenCharge)?.[0];\n    const bankAccountNumber = /\\d+-\\d+/.exec(item.bankNumber)?.[0];\n    return {\n      amount: amountData.amount,\n      amountCurrency: amountData.currency,\n      chargeDate,\n      bankAccountNumber,\n    };\n  });\n  return futureDebits;\n}\n\nclass VisaCalScraper extends BaseScraperWithBrowser {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  getLoginOptions(credentials: Record<string, string>) {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\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    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    debug('fetch future debits');\n    const futureDebits = await fetchFutureDebits(this.page);\n\n    debug('navigate to transactions page');\n    await this.navigateTo(TRANSACTIONS_URL, undefined, 60000);\n\n    debug('fetch accounts transactions');\n    const accounts = await fetchTransactions(this.page, startMoment, this.options);\n\n    debug('return the scraped accounts');\n    return {\n      success: true,\n      accounts,\n      futureDebits,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "israeli-bank-scrapers",
3
- "version": "1.14.2",
3
+ "version": "1.15.0",
4
4
  "private": false,
5
5
  "description": "Provide scrapers for all major Israeli banks and credit card companies",
6
6
  "engines": {