israeli-bank-scrapers 1.5.1 → 1.6.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -274,7 +274,7 @@ This scraper supports fetching transaction from up to one year.
274
274
  These are the projects known to be using this module:
275
275
  - [Israeli YNAB updater](https://github.com/eshaham/israeli-ynab-updater) - A command line tool for exporting banks data to CSVs, formatted specifically for [YNAB](https://www.youneedabudget.com)
276
276
  - [Israel Finance Telegram Bot](https://github.com/GuyLewin/israel-finance-telegram-bot) - A simple telegram bot that sends notifications about new transactions and interacts with them
277
- - [Automated budget tracking app](https://github.com/brafdlog/budget-tracking) - An app for automatically sending transactions from Israeli banks and credit cards to budget tracking apps
277
+ - [Caspion](https://github.com/brafdlog/caspion) - An app for automatically sending transactions from Israeli banks and credit cards to budget tracking apps
278
278
  - [Oshi](https://github.com/baruchiro/israeli-bank-scrapers-desktop) - Secure desktop app for retriving your transactions from all israeli banks and credit cards
279
279
  - [Finance Notifier](https://github.com/LiranBri/finance-notifier) - A simple script with the ability to send custom financial alerts to multiple contacts and platforms
280
280
 
@@ -1,11 +1,12 @@
1
1
  import { Frame, Page } from 'puppeteer';
2
2
  declare function waitUntilElementFound(page: Page | Frame, elementSelector: string, onlyVisible?: boolean, timeout?: number): Promise<void>;
3
3
  declare function waitUntilElementDisappear(page: Page, elementSelector: string, timeout?: number): Promise<void>;
4
+ declare function waitUntilIframeFound(page: Page, framePredicate: (frame: Frame) => boolean, description?: string, timeout?: number): Promise<Frame>;
4
5
  declare function fillInput(pageOrFrame: Page | Frame, inputSelector: string, inputValue: string): Promise<void>;
5
6
  declare function setValue(pageOrFrame: Page | Frame, inputSelector: string, inputValue: string): Promise<void>;
6
7
  declare function clickButton(page: Page | Frame, buttonSelector: string): Promise<void>;
7
8
  declare function clickLink(page: Page, aSelector: string): Promise<void>;
8
- declare function pageEvalAll<R>(page: Page, selector: string, defaultResult: any, callback: (elements: Element[], ...args: any) => R, ...args: any[]): Promise<R>;
9
+ declare function pageEvalAll<R>(page: Page | Frame, selector: string, defaultResult: any, callback: (elements: Element[], ...args: any) => R, ...args: any[]): Promise<R>;
9
10
  declare function pageEval<R>(pageOrFrame: Page | Frame, selector: string, defaultResult: any, callback: (elements: Element, ...args: any) => R, ...args: any[]): Promise<R>;
10
11
  declare function elementPresentOnPage(pageOrFrame: Page | Frame, selector: string): Promise<boolean>;
11
12
  declare function dropdownSelect(page: Page, selectSelector: string, value: string): Promise<void>;
@@ -13,4 +14,4 @@ declare function dropdownElements(page: Page, selector: string): Promise<{
13
14
  name: any;
14
15
  value: any;
15
16
  }[]>;
16
- export { waitUntilElementFound, waitUntilElementDisappear, fillInput, clickButton, clickLink, dropdownSelect, dropdownElements, pageEval, pageEvalAll, elementPresentOnPage, setValue, };
17
+ export { waitUntilElementFound, waitUntilElementDisappear, waitUntilIframeFound, fillInput, clickButton, clickLink, dropdownSelect, dropdownElements, pageEval, pageEvalAll, elementPresentOnPage, setValue, };
@@ -9,6 +9,7 @@ Object.defineProperty(exports, "__esModule", {
9
9
  });
10
10
  exports.waitUntilElementFound = waitUntilElementFound;
11
11
  exports.waitUntilElementDisappear = waitUntilElementDisappear;
12
+ exports.waitUntilIframeFound = waitUntilIframeFound;
12
13
  exports.fillInput = fillInput;
13
14
  exports.clickButton = clickButton;
14
15
  exports.clickLink = clickLink;
@@ -19,6 +20,8 @@ exports.pageEvalAll = pageEvalAll;
19
20
  exports.elementPresentOnPage = elementPresentOnPage;
20
21
  exports.setValue = setValue;
21
22
 
23
+ var _waiting = require("./waiting");
24
+
22
25
  async function waitUntilElementFound(page, elementSelector, onlyVisible = false, timeout = 30000) {
23
26
  await page.waitForSelector(elementSelector, {
24
27
  visible: onlyVisible,
@@ -33,6 +36,20 @@ async function waitUntilElementDisappear(page, elementSelector, timeout = 30000)
33
36
  });
34
37
  }
35
38
 
39
+ async function waitUntilIframeFound(page, framePredicate, description = '', timeout = 30000) {
40
+ let frame;
41
+ await (0, _waiting.waitUntil)(() => {
42
+ frame = page.frames().find(framePredicate);
43
+ return Promise.resolve(!!frame);
44
+ }, description, timeout, 1000);
45
+
46
+ if (!frame) {
47
+ throw new Error('failed to find iframe');
48
+ }
49
+
50
+ return frame;
51
+ }
52
+
36
53
  async function fillInput(pageOrFrame, inputSelector, inputValue) {
37
54
  await pageOrFrame.$eval(inputSelector, input => {
38
55
  const inputElement = input; // @ts-ignore
@@ -113,4 +130,4 @@ async function dropdownElements(page, selector) {
113
130
  }, `${selector} > option`);
114
131
  return options;
115
132
  }
116
- //# 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","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';\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 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, 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 elements 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  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,OAAO,GAAG,KADjC,EACwC;AACtC,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,OAAO,GAAG,KAAxF,EAA+F;AAC7F,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,kDAAlB,MAA0E,CAA9E,EAAiF;AAC/E,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 = 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 elements 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"]}
@@ -80,7 +80,7 @@ function getTransactionsUrl(monthMoment) {
80
80
  */
81
81
 
82
82
  return (0, _buildUrl.default)(BASE_API_ACTIONS_URL, {
83
- path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={"userIndex":-1,"cardIndex":-1,"monthView":true,"date":"${date}","dates":{"startDate":"0","endDate":"0"}}&v=V3.13-HF.6.26`
83
+ 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&v=V3.69-HF-CarLoanLeviModel.2.57`
84
84
  });
85
85
  }
86
86
 
@@ -277,4 +277,4 @@ class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
277
277
 
278
278
  var _default = MaxScraper;
279
279
  exports.default = _default;
280
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["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","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","matches","match","length","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","installments","undefined","fetchTransactionsForMonth","url","data","transactionsByAccount","result","transactions","filter","transaction","forEach","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","clonedTxns","Array","from","fetchTransactions","options","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 { ScaperOptions, ScraperCredentials } from './base-scraper';\n\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}\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 = 'פריסת העסקה הדחויה';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\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\"}}&v=V3.13-HF.6.26`,\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      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 null;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return null;\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  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    installments: getInstallmentsInfo(rawTransaction.comments) || undefined,\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: ScaperOptions) {\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, true);\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"]}
280
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["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","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","matches","match","length","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","installments","undefined","fetchTransactionsForMonth","url","data","transactionsByAccount","result","transactions","filter","transaction","forEach","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","clonedTxns","Array","from","fetchTransactions","options","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":";;;;;;;;;;;;;;;AAAA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;;;;;;;AAgBA,MAAMA,gBAAgB,GAAG,0BAAzB;AACA,MAAMC,oBAAoB,GAAG,+BAA7B;AACA,MAAMC,gBAAgB,GAAG,uBAAzB;AAEA,MAAMC,SAAS,GAAI,GAAED,gBAAiB,mBAAtC;AACA,MAAME,oBAAoB,GAAI,GAAEJ,gBAAiB,uCAAjD;AACA,MAAMK,WAAW,GAAI,GAAEH,gBAAiB,oBAAxC;AAEA,MAAMI,gBAAgB,GAAG,OAAzB;AACA,MAAMC,aAAa,GAAG,kBAAtB;AACA,MAAMC,2BAA2B,GAAG,cAApC;AACA,MAAMC,sBAAsB,GAAG,SAA/B;AACA,MAAMC,wBAAwB,GAAG,YAAjC;AACA,MAAMC,6BAA6B,GAAG,WAAtC;AACA,MAAMC,2BAA2B,GAAG,mBAApC;AACA,MAAMC,yBAAyB,GAAG,aAAlC;AACA,MAAMC,yBAAyB,GAAG,qBAAlC;AACA,MAAMC,wCAAwC,GAAG,mBAAjD;AACA,MAAMC,0BAA0B,GAAG,cAAnC;AACA,MAAMC,8BAA8B,GAAG,cAAvC;AACA,MAAMC,sCAAsC,GAAG,eAA/C;AACA,MAAMC,gBAAgB,GAAG,OAAzB;AACA,MAAMC,mBAAmB,GAAG,UAA5B;AACA,MAAMC,kCAAkC,GAAG,oBAA3C;AAEA,MAAMC,wBAAwB,GAAG,oBAAjC;AACA,MAAMC,oBAAoB,GAAG,6BAA7B;;AAEA,SAASC,gBAAT,CAA0BC,IAA1B,EAAsC;AACpC,SAAOC,OAAO,CAACC,IAAR,CAAa,CAClB,iCAAgBF,IAAhB,EAAsB,KAAtB,EAA6B,KAA7B,EAAoC,CAACvB,gBAAD,EAAoB,GAAEA,gBAAiB,GAAvC,CAApC,CADkB,EAElB,iDAAsBuB,IAAtB,EAA4BH,wBAA5B,EAAsD,IAAtD,CAFkB,EAGlB,iDAAsBG,IAAtB,EAA4BF,oBAA5B,EAAkD,IAAlD,CAHkB,CAAb,CAAP;AAKD;;AAED,SAASK,kBAAT,CAA4BC,WAA5B,EAAiD;AAC/C,QAAMC,KAAK,GAAGD,WAAW,CAACC,KAAZ,KAAsB,CAApC;AACA,QAAMC,IAAI,GAAGF,WAAW,CAACE,IAAZ,EAAb;AACA,QAAMC,IAAI,GAAI,GAAED,IAAK,IAAGD,KAAM,KAA9B;AAEA;;;;;;;AAMA,SAAO,uBAAS7B,oBAAT,EAA+B;AACpCgC,IAAAA,IAAI,EAAG,kIAAiID,IAAK;AADzG,GAA/B,CAAP;AAGD;;AAED,SAASE,kBAAT,CAA4BC,UAA5B,EAAgD;AAC9C,QAAMC,mBAAmB,GAAGD,UAAU,CAACE,OAAX,CAAmB,IAAnB,EAAyB,GAAzB,EAA8BC,IAA9B,EAA5B;;AACA,UAAQF,mBAAR;AACE,SAAK7B,aAAL;AACA,SAAKD,gBAAL;AACA,SAAKI,wBAAL;AACA,SAAKC,6BAAL;AACA,SAAKC,2BAAL;AACA,SAAKE,yBAAL;AACA,SAAKD,yBAAL;AACA,SAAKE,wCAAL;AACA,SAAKC,0BAAL;AACA,SAAKC,8BAAL;AACA,SAAKG,mBAAL;AACA,SAAKZ,2BAAL;AACA,SAAKU,sCAAL;AACA,SAAKG,kCAAL;AACE,aAAOkB,gCAAiBC,MAAxB;;AACF,SAAK/B,sBAAL;AACA,SAAKU,gBAAL;AACE,aAAOoB,gCAAiBE,YAAxB;;AACF;AACE,YAAM,IAAIC,KAAJ,CAAW,4BAA2BN,mBAAoB,EAA1D,CAAN;AApBJ;AAsBD;;AAED,SAASO,mBAAT,CAA6BC,QAA7B,EAA+C;AAC7C,MAAI,CAACA,QAAL,EAAe;AACb,WAAO,IAAP;AACD;;AACD,QAAMC,OAAO,GAAGD,QAAQ,CAACE,KAAT,CAAe,MAAf,CAAhB;;AACA,MAAI,CAACD,OAAD,IAAYA,OAAO,CAACE,MAAR,GAAiB,CAAjC,EAAoC;AAClC,WAAO,IAAP;AACD;;AAED,SAAO;AACLC,IAAAA,MAAM,EAAEC,QAAQ,CAACJ,OAAO,CAAC,CAAD,CAAR,EAAa,EAAb,CADX;AAELK,IAAAA,KAAK,EAAED,QAAQ,CAACJ,OAAO,CAAC,CAAD,CAAR,EAAa,EAAb;AAFV,GAAP;AAID;;AACD,SAASM,cAAT,CAAwBC,cAAxB,EAAyE;AACvE,QAAMC,SAAS,GAAGD,cAAc,CAACE,WAAf,KAA+B,IAAjD;AACA,QAAMC,aAAa,GAAG,qBAAOF,SAAS,GACpCD,cAAc,CAACI,YADqB,GAEpCJ,cAAc,CAACE,WAFK,EAEQG,WAFR,EAAtB;AAGA,QAAMC,MAAM,GAAGL,SAAS,GAAGM,mCAAoBC,OAAvB,GAAiCD,mCAAoBE,SAA7E;AAEA,SAAO;AACLC,IAAAA,IAAI,EAAE5B,kBAAkB,CAACkB,cAAc,CAACW,QAAhB,CADnB;AAEL/B,IAAAA,IAAI,EAAE,qBAAOoB,cAAc,CAACI,YAAtB,EAAoCC,WAApC,EAFD;AAGLF,IAAAA,aAHK;AAILS,IAAAA,cAAc,EAAE,CAACZ,cAAc,CAACY,cAJ3B;AAKLC,IAAAA,gBAAgB,EAAEb,cAAc,CAACa,gBAL5B;AAMLC,IAAAA,aAAa,EAAE,CAACd,cAAc,CAACe,mBAN1B;AAOLC,IAAAA,WAAW,EAAEhB,cAAc,CAACiB,YAAf,CAA4B/B,IAA5B,EAPR;AAQLgC,IAAAA,IAAI,EAAElB,cAAc,CAACR,QARhB;AASL2B,IAAAA,YAAY,EAAE5B,mBAAmB,CAACS,cAAc,CAACR,QAAhB,CAAnB,IAAgD4B,SATzD;AAULd,IAAAA;AAVK,GAAP;AAYD;;AAOD,eAAee,yBAAf,CAAyChD,IAAzC,EAAqDI,WAArD,EAA0E;AACxE,QAAM6C,GAAG,GAAG9C,kBAAkB,CAACC,WAAD,CAA9B;AAEA,QAAM8C,IAAI,GAAG,MAAM,+BAA8ClD,IAA9C,EAAoDiD,GAApD,CAAnB;AACA,QAAME,qBAAoD,GAAG,EAA7D;AAEA,MAAI,CAACD,IAAD,IAAS,CAACA,IAAI,CAACE,MAAnB,EAA2B,OAAOD,qBAAP;AAE3BD,EAAAA,IAAI,CAACE,MAAL,CAAYC,YAAZ,CACE;AADF,GAEGC,MAFH,CAEWC,WAAD,IAAiB,CAAC,CAACA,WAAW,CAACjB,QAFzC,EAGGkB,OAHH,CAGYD,WAAD,IAAqC;AAC5C,QAAI,CAACJ,qBAAqB,CAACI,WAAW,CAACE,eAAb,CAA1B,EAAyD;AACvDN,MAAAA,qBAAqB,CAACI,WAAW,CAACE,eAAb,CAArB,GAAqD,EAArD;AACD;;AAED,UAAMC,iBAAiB,GAAGhC,cAAc,CAAC6B,WAAD,CAAxC;AACAJ,IAAAA,qBAAqB,CAACI,WAAW,CAACE,eAAb,CAArB,CAAmDE,IAAnD,CAAwDD,iBAAxD;AACD,GAVH;AAYA,SAAOP,qBAAP;AACD;;AAED,SAASS,SAAT,CAAmBC,UAAnB,EAA8DT,MAA9D,EAAqG;AACnG,QAAMU,aAA4C,qBAAQD,UAAR,CAAlD;;AACAE,EAAAA,MAAM,CAACC,IAAP,CAAYZ,MAAZ,EAAoBI,OAApB,CAA6BS,aAAD,IAAmB;AAC7C,QAAI,CAACH,aAAa,CAACG,aAAD,CAAlB,EAAmC;AACjCH,MAAAA,aAAa,CAACG,aAAD,CAAb,GAA+B,EAA/B;AACD;;AACDH,IAAAA,aAAa,CAACG,aAAD,CAAb,CAA6BN,IAA7B,CAAkC,GAAGP,MAAM,CAACa,aAAD,CAA3C;AACD,GALD;AAMA,SAAOH,aAAP;AACD;;AAED,SAASI,mBAAT,CAA6BC,IAA7B,EAAkDC,WAAlD,EAA8EC,mBAA9E,EAA4G;AAC1G,MAAIC,UAAU,GAAGC,KAAK,CAACC,IAAN,CAAWL,IAAX,CAAjB;;AACA,MAAI,CAACE,mBAAL,EAA0B;AACxBC,IAAAA,UAAU,GAAG,mCAAgBA,UAAhB,CAAb;AACD;;AACDA,EAAAA,UAAU,GAAG,0CAAuBA,UAAvB,CAAb;AACAA,EAAAA,UAAU,GAAG,yCAAsBA,UAAtB,EAAkCF,WAAlC,EAA+CC,mBAAmB,IAAI,KAAtE,CAAb;AACA,SAAOC,UAAP;AACD;;AAED,eAAeG,iBAAf,CAAiCzE,IAAjC,EAA6C0E,OAA7C,EAAqE;AACnE,QAAMC,kBAAkB,GAAG,uBAASC,QAAT,CAAkB,CAAlB,EAAqB,OAArB,CAA3B;AACA,QAAMC,SAAS,GAAGH,OAAO,CAACG,SAAR,IAAqBF,kBAAkB,CAACG,MAAnB,EAAvC;;AACA,QAAMV,WAAW,GAAGW,gBAAOC,GAAP,CAAWL,kBAAX,EAA+B,qBAAOE,SAAP,CAA/B,CAApB;;AACA,QAAMI,SAAS,GAAG,oBAAmBb,WAAnB,EAAgC,IAAhC,CAAlB;AAEA,MAAIP,UAAyC,GAAG,EAAhD;;AACA,OAAK,IAAIqB,CAAC,GAAG,CAAb,EAAgBA,CAAC,GAAGD,SAAS,CAAC3D,MAA9B,EAAsC4D,CAAC,IAAI,CAA3C,EAA8C;AAC5C,UAAM9B,MAAM,GAAG,MAAMJ,yBAAyB,CAAChD,IAAD,EAAOiF,SAAS,CAACC,CAAD,CAAhB,CAA9C;AACArB,IAAAA,UAAU,GAAGD,SAAS,CAACC,UAAD,EAAaT,MAAb,CAAtB;AACD;;AAEDW,EAAAA,MAAM,CAACC,IAAP,CAAYH,UAAZ,EAAwBL,OAAxB,CAAiCS,aAAD,IAAmB;AACjD,QAAIE,IAAI,GAAGN,UAAU,CAACI,aAAD,CAArB;AACAE,IAAAA,IAAI,GAAGD,mBAAmB,CAACC,IAAD,EAAOC,WAAP,EAAoBM,OAAO,CAACL,mBAAR,IAA+B,KAAnD,CAA1B;AACAR,IAAAA,UAAU,CAACI,aAAD,CAAV,GAA4BE,IAA5B;AACD,GAJD;AAMA,SAAON,UAAP;AACD;;AAED,SAASsB,uBAAT,CAAiCnF,IAAjC,EAAmE;AACjE,QAAMoF,IAA0B,GAAG,EAAnC;AACAA,EAAAA,IAAI,CAACC,qCAAaC,OAAd,CAAJ,GAA6B,CAAC1G,WAAD,CAA7B;AACAwG,EAAAA,IAAI,CAACC,qCAAaE,cAAd,CAAJ,GAAoC,CAAC5G,oBAAD,CAApC;AACAyG,EAAAA,IAAI,CAACC,qCAAaG,eAAd,CAAJ,GAAqC,CAAC,YAAY;AAChD,WAAO,gDAAqBxF,IAArB,EAA2BH,wBAA3B,CAAP;AACD,GAFoC,CAArC;AAGAuF,EAAAA,IAAI,CAACC,qCAAaI,YAAd,CAAJ,GAAkC,CAAC,YAAY;AAC7C,WAAO,gDAAqBzF,IAArB,EAA2BF,oBAA3B,CAAP;AACD,GAFiC,CAAlC;AAGA,SAAOsF,IAAP;AACD;;AAED,SAASM,iBAAT,CAA2BC,WAA3B,EAA4D;AAC1D,SAAO,CACL;AAAEC,IAAAA,QAAQ,EAAE,YAAZ;AAA0BC,IAAAA,KAAK,EAAEF,WAAW,CAACG;AAA7C,GADK,EAEL;AAAEF,IAAAA,QAAQ,EAAE,WAAZ;AAAyBC,IAAAA,KAAK,EAAEF,WAAW,CAACI;AAA5C,GAFK,CAAP;AAID;;AAED,MAAMC,UAAN,SAAyBC,8CAAzB,CAAgD;AAC9CC,EAAAA,eAAe,CAACP,WAAD,EAAkC;AAC/C,WAAO;AACLQ,MAAAA,QAAQ,EAAEzH,SADL;AAEL0H,MAAAA,MAAM,EAAEV,iBAAiB,CAACC,WAAD,CAFpB;AAGLU,MAAAA,oBAAoB,EAAE,4BAHjB;AAILC,MAAAA,SAAS,EAAE,YAAY;AACrB,YAAI,MAAM,gDAAqB,KAAKtG,IAA1B,EAAgC,aAAhC,CAAV,EAA0D;AACxD,gBAAM,uCAAY,KAAKA,IAAjB,EAAuB,aAAvB,CAAN;AACD;;AACD,cAAM,uCAAY,KAAKA,IAAjB,EAAuB,wCAAvB,CAAN;AACA,cAAM,iDAAsB,KAAKA,IAA3B,EAAiC,sBAAjC,EAAyD,IAAzD,CAAN;AACA,cAAM,uCAAY,KAAKA,IAAjB,EAAuB,sBAAvB,CAAN;AACA,cAAM,iDAAsB,KAAKA,IAA3B,EAAiC,qDAAjC,EAAwF,IAAxF,CAAN;AACD,OAZI;AAaLuG,MAAAA,cAAc,EAAE,YAAY;AAC1B,cAAM,iDAAsB,KAAKvG,IAA3B,EAAiC,wCAAjC,EAA2E,IAA3E,CAAN;AACD,OAfI;AAgBLwG,MAAAA,UAAU,EAAE,YAAYzG,gBAAgB,CAAC,KAAKC,IAAN,CAhBnC;AAiBLyG,MAAAA,eAAe,EAAEtB,uBAAuB,CAAC,KAAKnF,IAAN;AAjBnC,KAAP;AAmBD;;AAED,QAAM0G,SAAN,GAAkB;AAChB,UAAMC,OAAO,GAAG,MAAMlC,iBAAiB,CAAC,KAAKzE,IAAN,EAAY,KAAK0E,OAAjB,CAAvC;AACA,UAAMkC,QAAQ,GAAG7C,MAAM,CAACC,IAAP,CAAY2C,OAAZ,EAAqBE,GAArB,CAA0B5C,aAAD,IAAmB;AAC3D,aAAO;AACLA,QAAAA,aADK;AAELE,QAAAA,IAAI,EAAEwC,OAAO,CAAC1C,aAAD;AAFR,OAAP;AAID,KALgB,CAAjB;AAOA,WAAO;AACL6C,MAAAA,OAAO,EAAE,IADJ;AAELF,MAAAA;AAFK,KAAP;AAID;;AApC6C;;eAuCjCZ,U","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 { ScaperOptions, ScraperCredentials } from './base-scraper';\n\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}\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 = 'פריסת העסקה הדחויה';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\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&v=V3.69-HF-CarLoanLeviModel.2.57`,\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      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 null;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return null;\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  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    installments: getInstallmentsInfo(rawTransaction.comments) || undefined,\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: ScaperOptions) {\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, true);\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"]}
@@ -1,4 +1,4 @@
1
- import { Transaction } from '../transactions';
1
+ import { TransactionsAccount } from '../transactions';
2
2
  import { ScraperCredentials, ScraperErrorTypes } from './base-scraper';
3
3
  import { BaseScraperWithBrowser, PossibleLoginResults } from './base-scraper-with-browser';
4
4
  declare class MizrahiScraper extends BaseScraperWithBrowser {
@@ -14,19 +14,16 @@ declare class MizrahiScraper extends BaseScraperWithBrowser {
14
14
  possibleResults: PossibleLoginResults;
15
15
  };
16
16
  fetchData(): Promise<{
17
+ success: boolean;
18
+ accounts: TransactionsAccount[];
19
+ errorType?: undefined;
20
+ errorMessage?: undefined;
21
+ } | {
17
22
  success: boolean;
18
23
  errorType: ScraperErrorTypes;
19
24
  errorMessage: string;
20
25
  accounts?: undefined;
21
- } | {
22
- success: boolean;
23
- accounts: {
24
- accountNumber: string;
25
- txns: Transaction[];
26
- balance: number;
27
- }[];
28
- errorType?: undefined;
29
- errorMessage?: undefined;
30
26
  }>;
27
+ private fetchAccount;
31
28
  }
32
29
  export default MizrahiScraper;
@@ -33,7 +33,8 @@ const BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';
33
33
  const AFTER_LOGIN_BASE_URL = /https:\/\/mto\.mizrahi-tefahot\.co\.il\/ngOnline\/index\.html#\/main\/uis/;
34
34
  const OSH_PAGE = `${BASE_APP_URL}/ngOnline/index.html#/main/uis/osh/p428/`;
35
35
  const TRANSACTIONS_REQUEST_URL = `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`;
36
- const PENDING_TRANSACTIONS_PAGE = `${BASE_APP_URL}/Online/Osh/p420.aspx`;
36
+ const PENDING_TRANSACTIONS_PAGE = `${BASE_APP_URL}/ngOnline/index.html#/main/uis/legacy/Osh/p420//legacy.Osh.p420`;
37
+ const PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';
37
38
  const CHANGE_PASSWORD_URL = /https:\/\/www\.mizrahi-tefahot\.co\.il\/login\/\w+\/index\.html#\/change-pass/;
38
39
  const DATE_FORMAT = 'DD/MM/YYYY';
39
40
  const MAX_ROWS_PER_REQUEST = 10000000000;
@@ -43,6 +44,8 @@ const submitButtonSelector = '.form-desktop button';
43
44
  const invalidPasswordSelector = 'a[href*="https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx"]';
44
45
  const afterLoginSelector = '#stickyHeaderScrollRegion';
45
46
  const loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';
47
+ const accountDropDownItemSelector = '#sky-account-combo-list ul li .sky-acc-value';
48
+ const pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';
46
49
 
47
50
  function createLoginFields(credentials) {
48
51
  return [{
@@ -137,6 +140,30 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
137
140
  }
138
141
 
139
142
  async fetchData() {
143
+ const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;
144
+
145
+ try {
146
+ const results = [];
147
+
148
+ for (let i = 0; i < numOfAccounts; i += 1) {
149
+ await this.page.$$eval(accountDropDownItemSelector, (els, i) => els[i].click(), i);
150
+ results.push((await this.fetchAccount()));
151
+ }
152
+
153
+ return {
154
+ success: true,
155
+ accounts: results
156
+ };
157
+ } catch (e) {
158
+ return {
159
+ success: false,
160
+ errorType: _baseScraper.ScraperErrorTypes.Generic,
161
+ errorMessage: e.message
162
+ };
163
+ }
164
+ }
165
+
166
+ async fetchAccount() {
140
167
  await this.navigateTo(OSH_PAGE, this.page);
141
168
  const request = await this.page.waitForRequest(TRANSACTIONS_REQUEST_URL);
142
169
  const data = createDataFromRequest(request, this.options.startDate);
@@ -144,11 +171,7 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
144
171
  const response = await (0, _fetch.fetchPostWithinPage)(this.page, TRANSACTIONS_REQUEST_URL, data, headers);
145
172
 
146
173
  if (!response || response.header.success === false) {
147
- return {
148
- success: false,
149
- errorType: _baseScraper.ScraperErrorTypes.Generic,
150
- errorMessage: `Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`
151
- };
174
+ throw new Error(`Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`);
152
175
  }
153
176
 
154
177
  const relevantRows = response.body.table.rows.filter(row => row.RecTypeSpecified);
@@ -157,15 +180,14 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
157
180
  const startMoment = getStartMoment(this.options.startDate);
158
181
  const oshTxnAfterStartDate = oshTxn.filter(txn => (0, _moment.default)(txn.date).isSameOrAfter(startMoment));
159
182
  await this.navigateTo(PENDING_TRANSACTIONS_PAGE, this.page);
160
- const pendingTxn = await extractPendingTransactions(this.page);
183
+ const frame = await (0, _elementsInteractions.waitUntilIframeFound)(this.page, f => f.url().includes(PENDING_TRANSACTIONS_IFRAME));
184
+ await (0, _elementsInteractions.waitUntilElementFound)(frame, pendingTrxIdentifierId);
185
+ const pendingTxn = await extractPendingTransactions(frame);
161
186
  const allTxn = oshTxnAfterStartDate.concat(pendingTxn);
162
187
  return {
163
- success: true,
164
- accounts: [{
165
- accountNumber: response.body.fields.AccountNumber,
166
- txns: allTxn,
167
- balance: +response.body.fields.YitraLeloChekim
168
- }]
188
+ accountNumber: response.body.fields.AccountNumber,
189
+ txns: allTxn,
190
+ balance: +response.body.fields.YitraLeloChekim
169
191
  };
170
192
  }
171
193
 
@@ -173,4 +195,4 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
173
195
 
174
196
  var _default = MizrahiScraper;
175
197
  exports.default = _default;
176
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/mizrahi.ts"],"names":["BASE_WEBSITE_URL","LOGIN_URL","BASE_APP_URL","AFTER_LOGIN_BASE_URL","OSH_PAGE","TRANSACTIONS_REQUEST_URL","PENDING_TRANSACTIONS_PAGE","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","createLoginFields","credentials","selector","value","username","password","getPossibleLoginResults","page","LoginResults","Success","InvalidPassword","$","ChangePassword","getStartMoment","optionsStartDate","defaultStartMoment","subtract","startDate","toDate","moment","max","createDataFromRequest","request","data","JSON","parse","postData","inFromDate","format","inToDate","table","maxRow","createHeadersFromRequest","mizrahixsrftoken","headers","convertTransactions","txns","map","row","txnDate","MC02PeulaTaaEZ","HTML5_FMT","DATETIME_LOCAL_SECONDS","toISOString","type","TransactionTypes","Normal","identifier","MC02AsmahtaMekoritEZ","parseInt","undefined","date","processedDate","originalAmount","MC02SchumEZ","originalCurrency","SHEKEL_CURRENCY","chargedAmount","description","MC02TnuaTeurEZ","status","TransactionStatuses","Completed","extractPendingTransactions","pendingTxn","trs","tr","Array","from","querySelectorAll","td","textContent","txn","amount","Pending","postLogin","Promise","race","MizrahiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","checkReadiness","postAction","possibleResults","fetchData","navigateTo","waitForRequest","options","response","header","success","errorType","ScraperErrorTypes","Generic","errorMessage","messages","text","relevantRows","body","rows","filter","RecTypeSpecified","oshTxn","startMoment","oshTxnAfterStartDate","isSameOrAfter","allTxn","concat","accounts","accountNumber","AccountNumber","balance","YitraLeloChekim"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { Page, Request } from 'puppeteer';\nimport {\n  SHEKEL_CURRENCY,\n} from '../constants';\nimport { pageEvalAll, waitUntilElementDisappear, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForUrl } from '../helpers/navigation';\nimport {\n  Transaction, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScraperCredentials, ScraperErrorTypes } from './base-scraper';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\n\ninterface ScrapedTransaction {\n  RecTypeSpecified: boolean;\n  MC02PeulaTaaEZ: string;\n  MC02SchumEZ: number;\n  MC02AsmahtaMekoritEZ: string;\n  MC02TnuaTeurEZ: string;\n}\n\ninterface ScrapedTransactionsResult {\n  header: {\n    success: boolean;\n    messages: { text: string }[];\n  };\n  body: {\n    fields: {\n      AccountNumber: string;\n      YitraLeloChekim: string;\n    };\n    table: {\n      rows: ScrapedTransaction[];\n    };\n  };\n}\n\nconst BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';\nconst LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;\nconst BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';\nconst AFTER_LOGIN_BASE_URL = /https:\\/\\/mto\\.mizrahi-tefahot\\.co\\.il\\/ngOnline\\/index\\.html#\\/main\\/uis/;\nconst OSH_PAGE = `${BASE_APP_URL}/ngOnline/index.html#/main/uis/osh/p428/`;\nconst TRANSACTIONS_REQUEST_URL = `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`;\nconst PENDING_TRANSACTIONS_PAGE = `${BASE_APP_URL}/Online/Osh/p420.aspx`;\nconst CHANGE_PASSWORD_URL = /https:\\/\\/www\\.mizrahi-tefahot\\.co\\.il\\/login\\/\\w+\\/index\\.html#\\/change-pass/;\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst MAX_ROWS_PER_REQUEST = 10000000000;\n\nconst usernameSelector = '#emailDesktopHeb';\nconst passwordSelector = '#passwordIDDesktopHEB';\nconst submitButtonSelector = '.form-desktop button';\nconst invalidPasswordSelector = 'a[href*=\"https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx\"]';\nconst afterLoginSelector = '#stickyHeaderScrollRegion';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: usernameSelector, value: credentials.username },\n    { selector: passwordSelector, value: credentials.password },\n  ];\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  return {\n    [LoginResults.Success]: [AFTER_LOGIN_BASE_URL],\n    [LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],\n    [LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],\n  };\n}\n\nfunction getStartMoment(optionsStartDate: Date) {\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = optionsStartDate || defaultStartMoment.toDate();\n  return moment.max(defaultStartMoment, moment(startDate));\n}\n\nfunction createDataFromRequest(request: Request, optionsStartDate: Date) {\n  const data = JSON.parse(request.postData() || '{}');\n\n  data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);\n  data.inToDate = moment().format(DATE_FORMAT);\n  data.table.maxRow = MAX_ROWS_PER_REQUEST;\n\n  return data;\n}\n\nfunction createHeadersFromRequest(request: Request) {\n  return {\n    mizrahixsrftoken: request.headers().mizrahixsrftoken,\n    'Content-Type': request.headers()['content-type'],\n  };\n}\n\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((row) => {\n    const txnDate = moment(row.MC02PeulaTaaEZ, moment.HTML5_FMT.DATETIME_LOCAL_SECONDS)\n      .toISOString();\n\n    return {\n      type: TransactionTypes.Normal,\n      identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: row.MC02SchumEZ,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: row.MC02SchumEZ,\n      description: row.MC02TnuaTeurEZ,\n      status: TransactionStatuses.Completed,\n    };\n  });\n}\n\nasync function extractPendingTransactions(page: Page): Promise<Transaction[]> {\n  const pendingTxn = await pageEvalAll(page, 'tr.rgRow', [], (trs) => {\n    return trs.map((tr) => Array.from(tr.querySelectorAll('td'), (td: HTMLTableDataCellElement) => td.textContent || ''));\n  });\n\n  return pendingTxn.map((txn) => {\n    const date = moment(txn[0], 'DD/MM/YY').toISOString();\n    const amount = parseInt(txn[3], 10);\n    return {\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn[1],\n      status: TransactionStatuses.Pending,\n    };\n  });\n}\n\nasync function postLogin(page: Page) {\n  await Promise.race([\n    waitUntilElementFound(page, afterLoginSelector),\n    waitUntilElementFound(page, invalidPasswordSelector),\n    waitForUrl(page, CHANGE_PASSWORD_URL),\n  ]);\n}\n\nclass MizrahiScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector,\n      checkReadiness: async () => waitUntilElementDisappear(this.page, loginSpinnerSelector),\n      postAction: async () => postLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    await this.navigateTo(OSH_PAGE, this.page);\n    const request = await this.page.waitForRequest(TRANSACTIONS_REQUEST_URL);\n    const data = createDataFromRequest(request, this.options.startDate);\n    const headers = createHeadersFromRequest(request);\n\n    const response = await fetchPostWithinPage<ScrapedTransactionsResult>(this.page,\n      TRANSACTIONS_REQUEST_URL, data, headers);\n\n    if (!response || response.header.success === false) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage:\n          `Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`,\n      };\n    }\n\n    const relevantRows = response.body.table.rows.filter((row) => row.RecTypeSpecified);\n    const oshTxn = convertTransactions(relevantRows);\n\n    // workaround for a bug which the bank's API returns transactions before the requested start date\n    const startMoment = getStartMoment(this.options.startDate);\n    const oshTxnAfterStartDate = oshTxn.filter((txn) => moment(txn.date).isSameOrAfter(startMoment));\n\n    await this.navigateTo(PENDING_TRANSACTIONS_PAGE, this.page);\n    const pendingTxn = await extractPendingTransactions(this.page);\n\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      success: true,\n      accounts: [\n        {\n          accountNumber: response.body.fields.AccountNumber,\n          txns: allTxn,\n          balance: +response.body.fields.YitraLeloChekim,\n        },\n      ],\n    };\n  }\n}\n\nexport default MizrahiScraper;\n"]}
198
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/mizrahi.ts"],"names":["BASE_WEBSITE_URL","LOGIN_URL","BASE_APP_URL","AFTER_LOGIN_BASE_URL","OSH_PAGE","TRANSACTIONS_REQUEST_URL","PENDING_TRANSACTIONS_PAGE","PENDING_TRANSACTIONS_IFRAME","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","accountDropDownItemSelector","pendingTrxIdentifierId","createLoginFields","credentials","selector","value","username","password","getPossibleLoginResults","page","LoginResults","Success","InvalidPassword","$","ChangePassword","getStartMoment","optionsStartDate","defaultStartMoment","subtract","startDate","toDate","moment","max","createDataFromRequest","request","data","JSON","parse","postData","inFromDate","format","inToDate","table","maxRow","createHeadersFromRequest","mizrahixsrftoken","headers","convertTransactions","txns","map","row","txnDate","MC02PeulaTaaEZ","HTML5_FMT","DATETIME_LOCAL_SECONDS","toISOString","type","TransactionTypes","Normal","identifier","MC02AsmahtaMekoritEZ","parseInt","undefined","date","processedDate","originalAmount","MC02SchumEZ","originalCurrency","SHEKEL_CURRENCY","chargedAmount","description","MC02TnuaTeurEZ","status","TransactionStatuses","Completed","extractPendingTransactions","pendingTxn","trs","tr","Array","from","querySelectorAll","td","textContent","txn","amount","Pending","postLogin","Promise","race","MizrahiScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","checkReadiness","postAction","possibleResults","fetchData","numOfAccounts","$$","length","results","i","$$eval","els","click","push","fetchAccount","success","accounts","e","errorType","ScraperErrorTypes","Generic","errorMessage","message","navigateTo","waitForRequest","options","response","header","Error","messages","text","relevantRows","body","rows","filter","RecTypeSpecified","oshTxn","startMoment","oshTxnAfterStartDate","isSameOrAfter","frame","f","url","includes","allTxn","concat","accountNumber","AccountNumber","balance","YitraLeloChekim"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { Frame, Page, Request } from 'puppeteer';\nimport { SHEKEL_CURRENCY } from '../constants';\nimport {\n  pageEvalAll, waitUntilElementDisappear, waitUntilElementFound, waitUntilIframeFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { waitForUrl } from '../helpers/navigation';\nimport {\n  Transaction, TransactionsAccount, TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { ScraperCredentials, ScraperErrorTypes } from './base-scraper';\nimport { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\n\ninterface ScrapedTransaction {\n  RecTypeSpecified: boolean;\n  MC02PeulaTaaEZ: string;\n  MC02SchumEZ: number;\n  MC02AsmahtaMekoritEZ: string;\n  MC02TnuaTeurEZ: string;\n}\n\ninterface ScrapedTransactionsResult {\n  header: {\n    success: boolean;\n    messages: { text: string }[];\n  };\n  body: {\n    fields: {\n      AccountNumber: string;\n      YitraLeloChekim: string;\n    };\n    table: {\n      rows: ScrapedTransaction[];\n    };\n  };\n}\n\nconst BASE_WEBSITE_URL = 'https://www.mizrahi-tefahot.co.il';\nconst LOGIN_URL = `${BASE_WEBSITE_URL}/login/index.html#/auth-page-he`;\nconst BASE_APP_URL = 'https://mto.mizrahi-tefahot.co.il';\nconst AFTER_LOGIN_BASE_URL = /https:\\/\\/mto\\.mizrahi-tefahot\\.co\\.il\\/ngOnline\\/index\\.html#\\/main\\/uis/;\nconst OSH_PAGE = `${BASE_APP_URL}/ngOnline/index.html#/main/uis/osh/p428/`;\nconst TRANSACTIONS_REQUEST_URL = `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`;\nconst PENDING_TRANSACTIONS_PAGE = `${BASE_APP_URL}/ngOnline/index.html#/main/uis/legacy/Osh/p420//legacy.Osh.p420`;\nconst PENDING_TRANSACTIONS_IFRAME = 'p420.aspx';\nconst CHANGE_PASSWORD_URL = /https:\\/\\/www\\.mizrahi-tefahot\\.co\\.il\\/login\\/\\w+\\/index\\.html#\\/change-pass/;\nconst DATE_FORMAT = 'DD/MM/YYYY';\nconst MAX_ROWS_PER_REQUEST = 10000000000;\n\nconst usernameSelector = '#emailDesktopHeb';\nconst passwordSelector = '#passwordIDDesktopHEB';\nconst submitButtonSelector = '.form-desktop button';\nconst invalidPasswordSelector = 'a[href*=\"https://sc.mizrahi-tefahot.co.il/SCServices/SC/P010.aspx\"]';\nconst afterLoginSelector = '#stickyHeaderScrollRegion';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#sky-account-combo-list ul li .sky-acc-value';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\n\nfunction createLoginFields(credentials: ScraperCredentials) {\n  return [\n    { selector: usernameSelector, value: credentials.username },\n    { selector: passwordSelector, value: credentials.password },\n  ];\n}\n\nfunction getPossibleLoginResults(page: Page): PossibleLoginResults {\n  return {\n    [LoginResults.Success]: [AFTER_LOGIN_BASE_URL],\n    [LoginResults.InvalidPassword]: [async () => !!(await page.$(invalidPasswordSelector))],\n    [LoginResults.ChangePassword]: [CHANGE_PASSWORD_URL],\n  };\n}\n\nfunction getStartMoment(optionsStartDate: Date) {\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = optionsStartDate || defaultStartMoment.toDate();\n  return moment.max(defaultStartMoment, moment(startDate));\n}\n\nfunction createDataFromRequest(request: Request, optionsStartDate: Date) {\n  const data = JSON.parse(request.postData() || '{}');\n\n  data.inFromDate = getStartMoment(optionsStartDate).format(DATE_FORMAT);\n  data.inToDate = moment().format(DATE_FORMAT);\n  data.table.maxRow = MAX_ROWS_PER_REQUEST;\n\n  return data;\n}\n\nfunction createHeadersFromRequest(request: Request) {\n  return {\n    mizrahixsrftoken: request.headers().mizrahixsrftoken,\n    'Content-Type': request.headers()['content-type'],\n  };\n}\n\n\nfunction convertTransactions(txns: ScrapedTransaction[]): Transaction[] {\n  return txns.map((row) => {\n    const txnDate = moment(row.MC02PeulaTaaEZ, moment.HTML5_FMT.DATETIME_LOCAL_SECONDS)\n      .toISOString();\n\n    return {\n      type: TransactionTypes.Normal,\n      identifier: row.MC02AsmahtaMekoritEZ ? parseInt(row.MC02AsmahtaMekoritEZ, 10) : undefined,\n      date: txnDate,\n      processedDate: txnDate,\n      originalAmount: row.MC02SchumEZ,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: row.MC02SchumEZ,\n      description: row.MC02TnuaTeurEZ,\n      status: TransactionStatuses.Completed,\n    };\n  });\n}\n\nasync function extractPendingTransactions(page: Frame): Promise<Transaction[]> {\n  const pendingTxn = await pageEvalAll(page, 'tr.rgRow', [], (trs) => {\n    return trs.map((tr) => Array.from(tr.querySelectorAll('td'), (td: HTMLTableDataCellElement) => td.textContent || ''));\n  });\n\n  return pendingTxn.map((txn) => {\n    const date = moment(txn[0], 'DD/MM/YY').toISOString();\n    const amount = parseInt(txn[3], 10);\n    return {\n      type: TransactionTypes.Normal,\n      date,\n      processedDate: date,\n      originalAmount: amount,\n      originalCurrency: SHEKEL_CURRENCY,\n      chargedAmount: amount,\n      description: txn[1],\n      status: TransactionStatuses.Pending,\n    };\n  });\n}\n\nasync function postLogin(page: Page) {\n  await Promise.race([\n    waitUntilElementFound(page, afterLoginSelector),\n    waitUntilElementFound(page, invalidPasswordSelector),\n    waitForUrl(page, CHANGE_PASSWORD_URL),\n  ]);\n}\n\nclass MizrahiScraper extends BaseScraperWithBrowser {\n  getLoginOptions(credentials: ScraperCredentials) {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector,\n      checkReadiness: async () => waitUntilElementDisappear(this.page, loginSpinnerSelector),\n      postAction: async () => postLogin(this.page),\n      possibleResults: getPossibleLoginResults(this.page),\n    };\n  }\n\n  async fetchData() {\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n    try {\n      const results: TransactionsAccount[] = [];\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        await this.page.$$eval(accountDropDownItemSelector, (els, i) => (els[i] as HTMLElement).click(), i);\n        results.push(await this.fetchAccount());\n      }\n\n      return {\n        success: true,\n        accounts: results,\n      };\n    } catch (e) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.Generic,\n        errorMessage: (e as Error).message,\n      };\n    }\n  }\n\n  private async fetchAccount() {\n    await this.navigateTo(OSH_PAGE, this.page);\n\n    const request = await this.page.waitForRequest(TRANSACTIONS_REQUEST_URL);\n    const data = createDataFromRequest(request, this.options.startDate);\n    const headers = createHeadersFromRequest(request);\n\n    const response = await fetchPostWithinPage<ScrapedTransactionsResult>(this.page,\n      TRANSACTIONS_REQUEST_URL, data, headers);\n\n    if (!response || response.header.success === false) {\n      throw new Error(`Error fetching transaction. Response message: ${response ? response.header.messages[0].text : ''}`);\n    }\n\n    const relevantRows = response.body.table.rows.filter((row) => row.RecTypeSpecified);\n    const oshTxn = convertTransactions(relevantRows);\n\n    // workaround for a bug which the bank's API returns transactions before the requested start date\n    const startMoment = getStartMoment(this.options.startDate);\n    const oshTxnAfterStartDate = oshTxn.filter((txn) => moment(txn.date).isSameOrAfter(startMoment));\n\n    await this.navigateTo(PENDING_TRANSACTIONS_PAGE, this.page);\n    const frame = await waitUntilIframeFound(this.page, (f) => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    await waitUntilElementFound(frame, pendingTrxIdentifierId);\n    const pendingTxn = await extractPendingTransactions(frame);\n\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber: response.body.fields.AccountNumber,\n      txns: allTxn,\n      balance: +response.body.fields.YitraLeloChekim,\n    };\n  }\n}\n\nexport default MizrahiScraper;\n"]}
@@ -23,8 +23,6 @@ var _elementsInteractions = require("../helpers/elements-interactions");
23
23
 
24
24
  var _transactions = require("../transactions");
25
25
 
26
- var _navigation = require("../helpers/navigation");
27
-
28
26
  var _constants = require("../constants");
29
27
 
30
28
  var _waiting = require("../helpers/waiting");
@@ -194,10 +192,10 @@ async function fetchTransactionsForAccount(page, startDate, accountNumber, scrap
194
192
  await (0, _elementsInteractions.setValue)(page, dateHiddenFieldSelector, `${currentDateIndex}`);
195
193
  debug('wait a second to workaround navigation issue in headless browser mode');
196
194
  await page.waitFor(1000);
197
- debug('click on the filter submit button');
198
- await (0, _elementsInteractions.clickButton)(page, buttonSelector);
199
- debug('wait for page navigation');
200
- await (0, _navigation.waitForNavigationAndDomLoad)(page);
195
+ debug('click on the filter submit button and wait for navigation');
196
+ await Promise.all([page.waitForNavigation({
197
+ waitUntil: 'domcontentloaded'
198
+ }), (0, _elementsInteractions.clickButton)(page, buttonSelector)]);
201
199
  debug('find the billing date');
202
200
  const billingDateLabel = await (0, _elementsInteractions.pageEval)(page, billingLabelSelector, '', element => {
203
201
  return element.innerText;
@@ -248,10 +246,10 @@ async function fetchTransactionsForAccount(page, startDate, accountNumber, scrap
248
246
  hasNextPage = await (0, _elementsInteractions.elementPresentOnPage)(page, nextPageSelector);
249
247
 
250
248
  if (hasNextPage) {
251
- debug('has another page, click on button next');
252
- await (0, _elementsInteractions.clickButton)(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]');
253
- debug('wait for page navigation');
254
- await (0, _navigation.waitForNavigationAndDomLoad)(page);
249
+ debug('has another page, click on button next and wait for page navigation');
250
+ await Promise.all([page.waitForNavigation({
251
+ waitUntil: 'domcontentloaded'
252
+ }), await (0, _elementsInteractions.clickButton)(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]')]);
255
253
  }
256
254
  } while (hasNextPage);
257
255
  }
@@ -265,15 +263,37 @@ async function fetchTransactionsForAccount(page, startDate, accountNumber, scrap
265
263
  };
266
264
  }
267
265
 
268
- async function fetchTransactions(page, startDate, scraperOptions) {
269
- var _$exec$, _$exec2;
266
+ async function getAccountNumbers(page) {
267
+ return (0, _elementsInteractions.pageEvalAll)(page, '[id$=lnkItem]', [], elements => elements.map(e => e.text)).then(res => res.map(text => {
268
+ var _$exec$, _$exec2;
269
+
270
+ return (_$exec$ = (_$exec2 = /\d+$/.exec(text.trim())) === null || _$exec2 === void 0 ? void 0 : _$exec2[0]) !== null && _$exec$ !== void 0 ? _$exec$ : '';
271
+ }));
272
+ }
273
+
274
+ async function setAccount(page, account) {
275
+ await (0, _elementsInteractions.pageEvalAll)(page, '[id$=lnkItem]', null, (elements, account) => {
276
+ for (const elem of elements) {
277
+ const a = elem;
278
+
279
+ if (a.text.includes(account)) {
280
+ a.click();
281
+ }
282
+ }
283
+ }, account);
284
+ }
270
285
 
286
+ async function fetchTransactions(page, startDate, scraperOptions) {
287
+ const accountNumbers = await getAccountNumbers(page);
271
288
  const accounts = [];
272
- const accountId = await (0, _elementsInteractions.pageEval)(page, '[id$=cboCardList_categoryList_lblCollapse]', '', item => {
273
- return item.value;
274
- }, []);
275
- const accountNumber = (_$exec$ = (_$exec2 = /\d+$/.exec(accountId.trim())) === null || _$exec2 === void 0 ? void 0 : _$exec2[0]) !== null && _$exec$ !== void 0 ? _$exec$ : '';
276
- accounts.push((await fetchTransactionsForAccount(page, startDate, accountNumber, scraperOptions)));
289
+
290
+ for (const account of accountNumbers) {
291
+ debug(`setting account: ${account}`);
292
+ await setAccount(page, account);
293
+ await page.waitFor(1000);
294
+ accounts.push((await fetchTransactionsForAccount(page, startDate, account, scraperOptions)));
295
+ }
296
+
277
297
  return accounts;
278
298
  }
279
299
 
@@ -332,4 +352,4 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
332
352
 
333
353
  var _default = VisaCalScraper;
334
354
  exports.default = _default;
335
- //# 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","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","items","el","startDateIndex","findIndex","option","accountTransactions","currentDateIndex","waitFor","billingDateLabel","element","billingDate","hasNextPage","rawTransactions","columns","getElementsByTagName","push","filter","combineInstallments","substring","fetchTransactions","accounts","accountId","trim","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 { ScaperOptions, ScaperScrapingResult, ScraperCredentials } from './base-scraper';\nimport { waitForNavigationAndDomLoad } from '../helpers/navigation';\nimport {\n  DOLLAR_CURRENCY, DOLLAR_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 {\n    const parts = amountStrCln.split(' ');\n    amount = -parseFloat(parts[0]);\n    [, currency] = parts;\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction 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: ScaperOptions): 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\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.waitFor(1000);\n    debug('click on the filter submit button');\n    await clickButton(page, buttonSelector);\n    debug('wait for page navigation');\n    await waitForNavigationAndDomLoad(page);\n    debug('find the billing date');\n    const billingDateLabel = await pageEval(page, billingLabelSelector, '', ((element) => {\n      return (element as HTMLSpanElement).innerText;\n    }));\n\n    const billingDate = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/.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          } 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');\n        await clickButton(page, '[id$=FormAreaNoBorder_FormArea_ctlGridPager_btnNext]');\n        debug('wait for page navigation');\n        await waitForNavigationAndDomLoad(page);\n      }\n    } while (hasNextPage);\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 fetchTransactions(page: Page, startDate: Moment, scraperOptions: ScaperOptions): Promise<TransactionsAccount[]> {\n  const accounts: TransactionsAccount[] = [];\n\n  const accountId = await pageEval(page, '[id$=cboCardList_categoryList_lblCollapse]', '', (item) => {\n    return (item as HTMLInputElement).value;\n  }, []);\n\n  const accountNumber = /\\d+$/.exec(accountId.trim())?.[0] ?? '';\n  accounts.push(await fetchTransactionsForAccount(page, startDate, accountNumber, scraperOptions));\n\n  return accounts;\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('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    debug('return the scraped accounts');\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
355
+ //# 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","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","items","el","startDateIndex","findIndex","option","accountTransactions","currentDateIndex","waitFor","all","waitForNavigation","waitUntil","billingDateLabel","element","billingDate","hasNextPage","rawTransactions","columns","getElementsByTagName","push","filter","combineInstallments","substring","getAccountNumbers","elements","e","text","then","res","trim","setAccount","account","elem","a","click","fetchTransactions","accountNumbers","accounts","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 { ScaperOptions, ScaperScrapingResult, ScraperCredentials } from './base-scraper';\nimport {\n  DOLLAR_CURRENCY, DOLLAR_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 {\n    const parts = amountStrCln.split(' ');\n    amount = -parseFloat(parts[0]);\n    [, currency] = parts;\n  }\n\n  return {\n    amount,\n    currency,\n  };\n}\n\nfunction 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: ScaperOptions): 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\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.waitFor(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('find the billing date');\n    const billingDateLabel = await pageEval(page, billingLabelSelector, '', ((element) => {\n      return (element as HTMLSpanElement).innerText;\n    }));\n\n    const billingDate = /\\d{1,2}[/]\\d{2}[/]\\d{2,4}/.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          } 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  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: ScaperOptions): 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.waitFor(1000);\n    accounts.push(\n      await fetchTransactionsForAccount(\n        page,\n        startDate,\n        account,\n        scraperOptions,\n      ),\n    );\n  }\n\n  return accounts;\n}\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('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    debug('return the scraped accounts');\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
@@ -56,8 +56,10 @@ describe('VisaCal legacy scraper', () => {
56
56
  expect(result).toBeDefined();
57
57
  const error = `${result.errorType || ''} ${result.errorMessage || ''}`.trim();
58
58
  expect(error).toBe('');
59
- expect(result.success).toBeTruthy();
59
+ expect(result.success).toBeTruthy(); // uncomment to test multiple accounts
60
+ // expect(result?.accounts?.length).toEqual(2)
61
+
60
62
  (0, _testsUtils.exportTransactions)(COMPANY_ID, result.accounts || []);
61
63
  });
62
64
  });
63
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zY3JhcGVycy92aXNhLWNhbC50ZXN0LnRzIl0sIm5hbWVzIjpbIkNPTVBBTllfSUQiLCJ0ZXN0c0NvbmZpZyIsImRlc2NyaWJlIiwiYmVmb3JlQWxsIiwidGVzdCIsImV4cGVjdCIsIlNDUkFQRVJTIiwidmlzYUNhbCIsInRvQmVEZWZpbmVkIiwibG9naW5GaWVsZHMiLCJ0b0NvbnRhaW4iLCJjb25maWciLCJjb21wYW55QVBJIiwiaW52YWxpZFBhc3N3b3JkIiwib3B0aW9ucyIsImNvbXBhbnlJZCIsInNjcmFwZXIiLCJWaXNhQ2FsU2NyYXBlciIsInJlc3VsdCIsInNjcmFwZSIsInVzZXJuYW1lIiwicGFzc3dvcmQiLCJzdWNjZXNzIiwidG9CZUZhbHN5IiwiZXJyb3JUeXBlIiwidG9CZSIsIkxvZ2luUmVzdWx0cyIsIkludmFsaWRQYXNzd29yZCIsImNyZWRlbnRpYWxzIiwiZXJyb3IiLCJlcnJvck1lc3NhZ2UiLCJ0cmltIiwidG9CZVRydXRoeSIsImFjY291bnRzIl0sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQTs7QUFDQTs7QUFHQTs7QUFDQTs7Ozs7Ozs7OztBQUVBLE1BQU1BLFVBQVUsR0FBRyxTQUFuQixDLENBQThCOztBQUM5QixNQUFNQyxXQUFXLEdBQUcsaUNBQXBCO0FBRUFDLFFBQVEsQ0FBQyx3QkFBRCxFQUEyQixNQUFNO0FBQ3ZDQyxFQUFBQSxTQUFTLENBQUMsTUFBTTtBQUNkLDBDQURjLENBQ1E7QUFDdkIsR0FGUSxDQUFUO0FBSUFDLEVBQUFBLElBQUksQ0FBQyxpREFBRCxFQUFvRCxNQUFNO0FBQzVEQyxJQUFBQSxNQUFNLENBQUNDLHNCQUFTQyxPQUFWLENBQU4sQ0FBeUJDLFdBQXpCO0FBQ0FILElBQUFBLE1BQU0sQ0FBQ0Msc0JBQVNDLE9BQVQsQ0FBaUJFLFdBQWxCLENBQU4sQ0FBcUNDLFNBQXJDLENBQStDLFVBQS9DO0FBQ0FMLElBQUFBLE1BQU0sQ0FBQ0Msc0JBQVNDLE9BQVQsQ0FBaUJFLFdBQWxCLENBQU4sQ0FBcUNDLFNBQXJDLENBQStDLFVBQS9DO0FBQ0QsR0FKRyxDQUFKO0FBTUEsdUNBQW9CVixVQUFwQixFQUFpQ1csTUFBRCxJQUFZQSxNQUFNLENBQUNDLFVBQVAsQ0FBa0JDLGVBQTlELEVBQStFLHVDQUEvRSxFQUF3SCxZQUFZO0FBQ2xJLFVBQU1DLE9BQU8scUJBQ1JiLFdBQVcsQ0FBQ2EsT0FESjtBQUVYQyxNQUFBQSxTQUFTLEVBQUVmO0FBRkEsTUFBYjs7QUFLQSxVQUFNZ0IsT0FBTyxHQUFHLElBQUlDLGdCQUFKLENBQW1CSCxPQUFuQixDQUFoQjtBQUVBLFVBQU1JLE1BQU0sR0FBRyxNQUFNRixPQUFPLENBQUNHLE1BQVIsQ0FBZTtBQUFFQyxNQUFBQSxRQUFRLEVBQUUsYUFBWjtBQUEyQkMsTUFBQUEsUUFBUSxFQUFFO0FBQXJDLEtBQWYsQ0FBckI7QUFFQWhCLElBQUFBLE1BQU0sQ0FBQ2EsTUFBRCxDQUFOLENBQWVWLFdBQWY7QUFDQUgsSUFBQUEsTUFBTSxDQUFDYSxNQUFNLENBQUNJLE9BQVIsQ0FBTixDQUF1QkMsU0FBdkI7QUFDQWxCLElBQUFBLE1BQU0sQ0FBQ2EsTUFBTSxDQUFDTSxTQUFSLENBQU4sQ0FBeUJDLElBQXpCLENBQThCQyxxQ0FBYUMsZUFBM0M7QUFDRCxHQWJEO0FBZUEsdUNBQW9CM0IsVUFBcEIsRUFBZ0MsNkJBQWhDLEVBQStELFlBQVk7QUFDekUsVUFBTWMsT0FBTyxxQkFDUmIsV0FBVyxDQUFDYSxPQURKO0FBRVhDLE1BQUFBLFNBQVMsRUFBRWY7QUFGQSxNQUFiOztBQUtBLFVBQU1nQixPQUFPLEdBQUcsSUFBSUMsZ0JBQUosQ0FBbUJILE9BQW5CLENBQWhCO0FBQ0EsVUFBTUksTUFBTSxHQUFHLE1BQU1GLE9BQU8sQ0FBQ0csTUFBUixDQUFlbEIsV0FBVyxDQUFDMkIsV0FBWixDQUF3QnJCLE9BQXZDLENBQXJCO0FBQ0FGLElBQUFBLE1BQU0sQ0FBQ2EsTUFBRCxDQUFOLENBQWVWLFdBQWY7QUFDQSxVQUFNcUIsS0FBSyxHQUFJLEdBQUVYLE1BQU0sQ0FBQ00sU0FBUCxJQUFvQixFQUFHLElBQUdOLE1BQU0sQ0FBQ1ksWUFBUCxJQUF1QixFQUFHLEVBQXZELENBQXlEQyxJQUF6RCxFQUFkO0FBQ0ExQixJQUFBQSxNQUFNLENBQUN3QixLQUFELENBQU4sQ0FBY0osSUFBZCxDQUFtQixFQUFuQjtBQUNBcEIsSUFBQUEsTUFBTSxDQUFDYSxNQUFNLENBQUNJLE9BQVIsQ0FBTixDQUF1QlUsVUFBdkI7QUFFQSx3Q0FBbUJoQyxVQUFuQixFQUErQmtCLE1BQU0sQ0FBQ2UsUUFBUCxJQUFtQixFQUFsRDtBQUNELEdBZEQ7QUFlRCxDQXpDTyxDQUFSIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFZpc2FDYWxTY3JhcGVyIGZyb20gJy4vdmlzYS1jYWwnO1xuaW1wb3J0IHtcbiAgbWF5YmVUZXN0Q29tcGFueUFQSSwgZXh0ZW5kQXN5bmNUaW1lb3V0LCBnZXRUZXN0c0NvbmZpZywgZXhwb3J0VHJhbnNhY3Rpb25zLFxufSBmcm9tICcuLi90ZXN0cy90ZXN0cy11dGlscyc7XG5pbXBvcnQgeyBTQ1JBUEVSUyB9IGZyb20gJy4uL2RlZmluaXRpb25zJztcbmltcG9ydCB7IExvZ2luUmVzdWx0cyB9IGZyb20gJy4vYmFzZS1zY3JhcGVyLXdpdGgtYnJvd3Nlcic7XG5cbmNvbnN0IENPTVBBTllfSUQgPSAndmlzYUNhbCc7IC8vIFRPRE8gdGhpcyBwcm9wZXJ0eSBzaG91bGQgYmUgaGFyZC1jb2RlZCBpbiB0aGUgcHJvdmlkZXJcbmNvbnN0IHRlc3RzQ29uZmlnID0gZ2V0VGVzdHNDb25maWcoKTtcblxuZGVzY3JpYmUoJ1Zpc2FDYWwgbGVnYWN5IHNjcmFwZXInLCAoKSA9PiB7XG4gIGJlZm9yZUFsbCgoKSA9PiB7XG4gICAgZXh0ZW5kQXN5bmNUaW1lb3V0KCk7IC8vIFRoZSBkZWZhdWx0IHRpbWVvdXQgaXMgNSBzZWNvbmRzIHBlciBhc3luYyB0ZXN0LCB0aGlzIGZ1bmN0aW9uIGV4dGVuZHMgdGhlIHRpbWVvdXQgdmFsdWVcbiAgfSk7XG5cbiAgdGVzdCgnc2hvdWxkIGV4cG9zZSBsb2dpbiBmaWVsZHMgaW4gc2NyYXBlcnMgY29uc3RhbnQnLCAoKSA9PiB7XG4gICAgZXhwZWN0KFNDUkFQRVJTLnZpc2FDYWwpLnRvQmVEZWZpbmVkKCk7XG4gICAgZXhwZWN0KFNDUkFQRVJTLnZpc2FDYWwubG9naW5GaWVsZHMpLnRvQ29udGFpbigndXNlcm5hbWUnKTtcbiAgICBleHBlY3QoU0NSQVBFUlMudmlzYUNhbC5sb2dpbkZpZWxkcykudG9Db250YWluKCdwYXNzd29yZCcpO1xuICB9KTtcblxuICBtYXliZVRlc3RDb21wYW55QVBJKENPTVBBTllfSUQsIChjb25maWcpID0+IGNvbmZpZy5jb21wYW55QVBJLmludmFsaWRQYXNzd29yZCkoJ3Nob3VsZCBmYWlsIG9uIGludmFsaWQgdXNlci9wYXNzd29yZFwiJywgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgICAuLi50ZXN0c0NvbmZpZy5vcHRpb25zLFxuICAgICAgY29tcGFueUlkOiBDT01QQU5ZX0lELFxuICAgIH07XG5cbiAgICBjb25zdCBzY3JhcGVyID0gbmV3IFZpc2FDYWxTY3JhcGVyKG9wdGlvbnMpO1xuXG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgc2NyYXBlci5zY3JhcGUoeyB1c2VybmFtZTogJzk3MXNkZGtzbXNsJywgcGFzc3dvcmQ6ICczZjNzc2RrU0QzZCcgfSk7XG5cbiAgICBleHBlY3QocmVzdWx0KS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChyZXN1bHQuc3VjY2VzcykudG9CZUZhbHN5KCk7XG4gICAgZXhwZWN0KHJlc3VsdC5lcnJvclR5cGUpLnRvQmUoTG9naW5SZXN1bHRzLkludmFsaWRQYXNzd29yZCk7XG4gIH0pO1xuXG4gIG1heWJlVGVzdENvbXBhbnlBUEkoQ09NUEFOWV9JRCkoJ3Nob3VsZCBzY3JhcGUgdHJhbnNhY3Rpb25zXCInLCBhc3luYyAoKSA9PiB7XG4gICAgY29uc3Qgb3B0aW9ucyA9IHtcbiAgICAgIC4uLnRlc3RzQ29uZmlnLm9wdGlvbnMsXG4gICAgICBjb21wYW55SWQ6IENPTVBBTllfSUQsXG4gICAgfTtcblxuICAgIGNvbnN0IHNjcmFwZXIgPSBuZXcgVmlzYUNhbFNjcmFwZXIob3B0aW9ucyk7XG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgc2NyYXBlci5zY3JhcGUodGVzdHNDb25maWcuY3JlZGVudGlhbHMudmlzYUNhbCk7XG4gICAgZXhwZWN0KHJlc3VsdCkudG9CZURlZmluZWQoKTtcbiAgICBjb25zdCBlcnJvciA9IGAke3Jlc3VsdC5lcnJvclR5cGUgfHwgJyd9ICR7cmVzdWx0LmVycm9yTWVzc2FnZSB8fCAnJ31gLnRyaW0oKTtcbiAgICBleHBlY3QoZXJyb3IpLnRvQmUoJycpO1xuICAgIGV4cGVjdChyZXN1bHQuc3VjY2VzcykudG9CZVRydXRoeSgpO1xuXG4gICAgZXhwb3J0VHJhbnNhY3Rpb25zKENPTVBBTllfSUQsIHJlc3VsdC5hY2NvdW50cyB8fCBbXSk7XG4gIH0pO1xufSk7XG4iXX0=
65
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zY3JhcGVycy92aXNhLWNhbC50ZXN0LnRzIl0sIm5hbWVzIjpbIkNPTVBBTllfSUQiLCJ0ZXN0c0NvbmZpZyIsImRlc2NyaWJlIiwiYmVmb3JlQWxsIiwidGVzdCIsImV4cGVjdCIsIlNDUkFQRVJTIiwidmlzYUNhbCIsInRvQmVEZWZpbmVkIiwibG9naW5GaWVsZHMiLCJ0b0NvbnRhaW4iLCJjb25maWciLCJjb21wYW55QVBJIiwiaW52YWxpZFBhc3N3b3JkIiwib3B0aW9ucyIsImNvbXBhbnlJZCIsInNjcmFwZXIiLCJWaXNhQ2FsU2NyYXBlciIsInJlc3VsdCIsInNjcmFwZSIsInVzZXJuYW1lIiwicGFzc3dvcmQiLCJzdWNjZXNzIiwidG9CZUZhbHN5IiwiZXJyb3JUeXBlIiwidG9CZSIsIkxvZ2luUmVzdWx0cyIsIkludmFsaWRQYXNzd29yZCIsImNyZWRlbnRpYWxzIiwiZXJyb3IiLCJlcnJvck1lc3NhZ2UiLCJ0cmltIiwidG9CZVRydXRoeSIsImFjY291bnRzIl0sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQTs7QUFDQTs7QUFHQTs7QUFDQTs7Ozs7Ozs7OztBQUVBLE1BQU1BLFVBQVUsR0FBRyxTQUFuQixDLENBQThCOztBQUM5QixNQUFNQyxXQUFXLEdBQUcsaUNBQXBCO0FBRUFDLFFBQVEsQ0FBQyx3QkFBRCxFQUEyQixNQUFNO0FBQ3ZDQyxFQUFBQSxTQUFTLENBQUMsTUFBTTtBQUNkLDBDQURjLENBQ1E7QUFDdkIsR0FGUSxDQUFUO0FBSUFDLEVBQUFBLElBQUksQ0FBQyxpREFBRCxFQUFvRCxNQUFNO0FBQzVEQyxJQUFBQSxNQUFNLENBQUNDLHNCQUFTQyxPQUFWLENBQU4sQ0FBeUJDLFdBQXpCO0FBQ0FILElBQUFBLE1BQU0sQ0FBQ0Msc0JBQVNDLE9BQVQsQ0FBaUJFLFdBQWxCLENBQU4sQ0FBcUNDLFNBQXJDLENBQStDLFVBQS9DO0FBQ0FMLElBQUFBLE1BQU0sQ0FBQ0Msc0JBQVNDLE9BQVQsQ0FBaUJFLFdBQWxCLENBQU4sQ0FBcUNDLFNBQXJDLENBQStDLFVBQS9DO0FBQ0QsR0FKRyxDQUFKO0FBTUEsdUNBQW9CVixVQUFwQixFQUFpQ1csTUFBRCxJQUFZQSxNQUFNLENBQUNDLFVBQVAsQ0FBa0JDLGVBQTlELEVBQStFLHVDQUEvRSxFQUF3SCxZQUFZO0FBQ2xJLFVBQU1DLE9BQU8scUJBQ1JiLFdBQVcsQ0FBQ2EsT0FESjtBQUVYQyxNQUFBQSxTQUFTLEVBQUVmO0FBRkEsTUFBYjs7QUFLQSxVQUFNZ0IsT0FBTyxHQUFHLElBQUlDLGdCQUFKLENBQW1CSCxPQUFuQixDQUFoQjtBQUVBLFVBQU1JLE1BQU0sR0FBRyxNQUFNRixPQUFPLENBQUNHLE1BQVIsQ0FBZTtBQUFFQyxNQUFBQSxRQUFRLEVBQUUsYUFBWjtBQUEyQkMsTUFBQUEsUUFBUSxFQUFFO0FBQXJDLEtBQWYsQ0FBckI7QUFFQWhCLElBQUFBLE1BQU0sQ0FBQ2EsTUFBRCxDQUFOLENBQWVWLFdBQWY7QUFDQUgsSUFBQUEsTUFBTSxDQUFDYSxNQUFNLENBQUNJLE9BQVIsQ0FBTixDQUF1QkMsU0FBdkI7QUFDQWxCLElBQUFBLE1BQU0sQ0FBQ2EsTUFBTSxDQUFDTSxTQUFSLENBQU4sQ0FBeUJDLElBQXpCLENBQThCQyxxQ0FBYUMsZUFBM0M7QUFDRCxHQWJEO0FBZUEsdUNBQW9CM0IsVUFBcEIsRUFBZ0MsNkJBQWhDLEVBQStELFlBQVk7QUFDekUsVUFBTWMsT0FBTyxxQkFDUmIsV0FBVyxDQUFDYSxPQURKO0FBRVhDLE1BQUFBLFNBQVMsRUFBRWY7QUFGQSxNQUFiOztBQUtBLFVBQU1nQixPQUFPLEdBQUcsSUFBSUMsZ0JBQUosQ0FBbUJILE9BQW5CLENBQWhCO0FBQ0EsVUFBTUksTUFBTSxHQUFHLE1BQU1GLE9BQU8sQ0FBQ0csTUFBUixDQUFlbEIsV0FBVyxDQUFDMkIsV0FBWixDQUF3QnJCLE9BQXZDLENBQXJCO0FBQ0FGLElBQUFBLE1BQU0sQ0FBQ2EsTUFBRCxDQUFOLENBQWVWLFdBQWY7QUFDQSxVQUFNcUIsS0FBSyxHQUFJLEdBQUVYLE1BQU0sQ0FBQ00sU0FBUCxJQUFvQixFQUFHLElBQUdOLE1BQU0sQ0FBQ1ksWUFBUCxJQUF1QixFQUFHLEVBQXZELENBQXlEQyxJQUF6RCxFQUFkO0FBQ0ExQixJQUFBQSxNQUFNLENBQUN3QixLQUFELENBQU4sQ0FBY0osSUFBZCxDQUFtQixFQUFuQjtBQUNBcEIsSUFBQUEsTUFBTSxDQUFDYSxNQUFNLENBQUNJLE9BQVIsQ0FBTixDQUF1QlUsVUFBdkIsR0FYeUUsQ0FZekU7QUFDQTs7QUFDQSx3Q0FBbUJoQyxVQUFuQixFQUErQmtCLE1BQU0sQ0FBQ2UsUUFBUCxJQUFtQixFQUFsRDtBQUNELEdBZkQ7QUFnQkQsQ0ExQ08sQ0FBUiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBWaXNhQ2FsU2NyYXBlciBmcm9tICcuL3Zpc2EtY2FsJztcbmltcG9ydCB7XG4gIG1heWJlVGVzdENvbXBhbnlBUEksIGV4dGVuZEFzeW5jVGltZW91dCwgZ2V0VGVzdHNDb25maWcsIGV4cG9ydFRyYW5zYWN0aW9ucyxcbn0gZnJvbSAnLi4vdGVzdHMvdGVzdHMtdXRpbHMnO1xuaW1wb3J0IHsgU0NSQVBFUlMgfSBmcm9tICcuLi9kZWZpbml0aW9ucyc7XG5pbXBvcnQgeyBMb2dpblJlc3VsdHMgfSBmcm9tICcuL2Jhc2Utc2NyYXBlci13aXRoLWJyb3dzZXInO1xuXG5jb25zdCBDT01QQU5ZX0lEID0gJ3Zpc2FDYWwnOyAvLyBUT0RPIHRoaXMgcHJvcGVydHkgc2hvdWxkIGJlIGhhcmQtY29kZWQgaW4gdGhlIHByb3ZpZGVyXG5jb25zdCB0ZXN0c0NvbmZpZyA9IGdldFRlc3RzQ29uZmlnKCk7XG5cbmRlc2NyaWJlKCdWaXNhQ2FsIGxlZ2FjeSBzY3JhcGVyJywgKCkgPT4ge1xuICBiZWZvcmVBbGwoKCkgPT4ge1xuICAgIGV4dGVuZEFzeW5jVGltZW91dCgpOyAvLyBUaGUgZGVmYXVsdCB0aW1lb3V0IGlzIDUgc2Vjb25kcyBwZXIgYXN5bmMgdGVzdCwgdGhpcyBmdW5jdGlvbiBleHRlbmRzIHRoZSB0aW1lb3V0IHZhbHVlXG4gIH0pO1xuXG4gIHRlc3QoJ3Nob3VsZCBleHBvc2UgbG9naW4gZmllbGRzIGluIHNjcmFwZXJzIGNvbnN0YW50JywgKCkgPT4ge1xuICAgIGV4cGVjdChTQ1JBUEVSUy52aXNhQ2FsKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChTQ1JBUEVSUy52aXNhQ2FsLmxvZ2luRmllbGRzKS50b0NvbnRhaW4oJ3VzZXJuYW1lJyk7XG4gICAgZXhwZWN0KFNDUkFQRVJTLnZpc2FDYWwubG9naW5GaWVsZHMpLnRvQ29udGFpbigncGFzc3dvcmQnKTtcbiAgfSk7XG5cbiAgbWF5YmVUZXN0Q29tcGFueUFQSShDT01QQU5ZX0lELCAoY29uZmlnKSA9PiBjb25maWcuY29tcGFueUFQSS5pbnZhbGlkUGFzc3dvcmQpKCdzaG91bGQgZmFpbCBvbiBpbnZhbGlkIHVzZXIvcGFzc3dvcmRcIicsIGFzeW5jICgpID0+IHtcbiAgICBjb25zdCBvcHRpb25zID0ge1xuICAgICAgLi4udGVzdHNDb25maWcub3B0aW9ucyxcbiAgICAgIGNvbXBhbnlJZDogQ09NUEFOWV9JRCxcbiAgICB9O1xuXG4gICAgY29uc3Qgc2NyYXBlciA9IG5ldyBWaXNhQ2FsU2NyYXBlcihvcHRpb25zKTtcblxuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHNjcmFwZXIuc2NyYXBlKHsgdXNlcm5hbWU6ICc5NzFzZGRrc21zbCcsIHBhc3N3b3JkOiAnM2Yzc3Nka1NEM2QnIH0pO1xuXG4gICAgZXhwZWN0KHJlc3VsdCkudG9CZURlZmluZWQoKTtcbiAgICBleHBlY3QocmVzdWx0LnN1Y2Nlc3MpLnRvQmVGYWxzeSgpO1xuICAgIGV4cGVjdChyZXN1bHQuZXJyb3JUeXBlKS50b0JlKExvZ2luUmVzdWx0cy5JbnZhbGlkUGFzc3dvcmQpO1xuICB9KTtcblxuICBtYXliZVRlc3RDb21wYW55QVBJKENPTVBBTllfSUQpKCdzaG91bGQgc2NyYXBlIHRyYW5zYWN0aW9uc1wiJywgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgICAuLi50ZXN0c0NvbmZpZy5vcHRpb25zLFxuICAgICAgY29tcGFueUlkOiBDT01QQU5ZX0lELFxuICAgIH07XG5cbiAgICBjb25zdCBzY3JhcGVyID0gbmV3IFZpc2FDYWxTY3JhcGVyKG9wdGlvbnMpO1xuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHNjcmFwZXIuc2NyYXBlKHRlc3RzQ29uZmlnLmNyZWRlbnRpYWxzLnZpc2FDYWwpO1xuICAgIGV4cGVjdChyZXN1bHQpLnRvQmVEZWZpbmVkKCk7XG4gICAgY29uc3QgZXJyb3IgPSBgJHtyZXN1bHQuZXJyb3JUeXBlIHx8ICcnfSAke3Jlc3VsdC5lcnJvck1lc3NhZ2UgfHwgJyd9YC50cmltKCk7XG4gICAgZXhwZWN0KGVycm9yKS50b0JlKCcnKTtcbiAgICBleHBlY3QocmVzdWx0LnN1Y2Nlc3MpLnRvQmVUcnV0aHkoKTtcbiAgICAvLyB1bmNvbW1lbnQgdG8gdGVzdCBtdWx0aXBsZSBhY2NvdW50c1xuICAgIC8vIGV4cGVjdChyZXN1bHQ/LmFjY291bnRzPy5sZW5ndGgpLnRvRXF1YWwoMilcbiAgICBleHBvcnRUcmFuc2FjdGlvbnMoQ09NUEFOWV9JRCwgcmVzdWx0LmFjY291bnRzIHx8IFtdKTtcbiAgfSk7XG59KTtcbiJdfQ==
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "israeli-bank-scrapers",
3
- "version": "1.5.1",
3
+ "version": "1.6.1",
4
4
  "private": false,
5
5
  "description": "Provide scrapers for all major Israeli banks and credit card companies",
6
6
  "engines": {