israeli-bank-scrapers 4.1.1 → 4.2.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.
@@ -33,6 +33,8 @@ var _transactions2 = require("../transactions");
33
33
 
34
34
  var _debug = require("../helpers/debug");
35
35
 
36
+ var _constants = require("../constants");
37
+
36
38
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
37
39
 
38
40
  function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
@@ -159,6 +161,22 @@ function getInstallmentsInfo(comments) {
159
161
  };
160
162
  }
161
163
 
164
+ function getChargedCurrency(currencyId) {
165
+ switch (currencyId) {
166
+ case 376:
167
+ return _constants.SHEKEL_CURRENCY;
168
+
169
+ case 840:
170
+ return _constants.DOLLAR_CURRENCY;
171
+
172
+ case 978:
173
+ return _constants.EURO_CURRENCY;
174
+
175
+ default:
176
+ return undefined;
177
+ }
178
+ }
179
+
162
180
  function mapTransaction(rawTransaction) {
163
181
  var _rawTransaction$dealD, _rawTransaction$dealD2;
164
182
 
@@ -174,6 +192,7 @@ function mapTransaction(rawTransaction) {
174
192
  originalAmount: -rawTransaction.originalAmount,
175
193
  originalCurrency: rawTransaction.originalCurrency,
176
194
  chargedAmount: -rawTransaction.actualPaymentAmount,
195
+ chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),
177
196
  description: rawTransaction.merchantName.trim(),
178
197
  memo: rawTransaction.comments,
179
198
  category: categories.get(rawTransaction === null || rawTransaction === void 0 ? void 0 : rawTransaction.categoryId),
@@ -319,4 +338,4 @@ class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
319
338
 
320
339
  var _default = MaxScraper;
321
340
  exports.default = _default;
322
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["debug","BASE_ACTIONS_URL","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","NORMAL_TYPE_NAME","ATM_TYPE_NAME","INTERNET_SHOPPING_TYPE_NAME","INSTALLMENTS_TYPE_NAME","MONTHLY_CHARGE_TYPE_NAME","ONE_MONTH_POSTPONED_TYPE_NAME","MONTHLY_POSTPONED_TYPE_NAME","MONTHLY_PAYMENT_TYPE_NAME","FUTURE_PURCHASE_FINANCING","MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME","THIRTY_DAYS_PLUS_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME2","MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME","CREDIT_TYPE_NAME","CREDIT_OUTSIDE_THE_LIMIT","ACCUMULATING_BASKET","POSTPONED_TRANSACTION_INSTALLMENTS","REPLACEMENT_CARD","EARLY_REPAYMENT","MONTHLY_CARD_FEE","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","loadCategories","res","Array","isArray","result","length","forEach","id","name","set","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","description","merchantName","memo","category","get","categoryId","fetchTransactionsForMonth","url","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","enableTransactionsFilterByDate","clonedTxns","from","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startDate","toDate","moment","max","allMonths","i","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","checkReadiness","postAction","possibleResults","waitUntil","fetchData","results","accounts","map","success"],"mappings":";;;;;;;;;;;;;;;AAAA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;;;;;;;AAGA,MAAMA,KAAK,GAAG,qBAAS,KAAT,CAAd;AAkBA,MAAMC,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,+BAA+B,GAAG,oBAAxC;AACA,MAAMC,sCAAsC,GAAG,eAA/C;AACA,MAAMC,gBAAgB,GAAG,OAAzB;AACA,MAAMC,wBAAwB,GAAG,mBAAjC;AACA,MAAMC,mBAAmB,GAAG,UAA5B;AACA,MAAMC,kCAAkC,GAAG,oBAA3C;AACA,MAAMC,gBAAgB,GAAG,aAAzB;AACA,MAAMC,eAAe,GAAG,aAAxB;AACA,MAAMC,gBAAgB,GAAG,WAAzB;AAEA,MAAMC,wBAAwB,GAAG,oBAAjC;AACA,MAAMC,oBAAoB,GAAG,6BAA7B;AAEA,MAAMC,UAAU,GAAG,IAAIC,GAAJ,EAAnB;;AAEA,SAASC,gBAAT,CAA0BC,IAA1B,EAAsC;AACpC,SAAOC,OAAO,CAACC,IAAR,CAAa,CAClB,iCAAgBF,IAAhB,EAAsB,KAAtB,EAA6B,KAA7B,EAAoC,CAAC9B,gBAAD,EAAoB,GAAEA,gBAAiB,GAAvC,CAApC,CADkB,EAElB,iDAAsB8B,IAAtB,EAA4BL,wBAA5B,EAAsD,IAAtD,CAFkB,EAGlB,iDAAsBK,IAAtB,EAA4BJ,oBAA5B,EAAkD,IAAlD,CAHkB,CAAb,CAAP;AAKD;;AAED,SAASO,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,uBAASpC,oBAAT,EAA+B;AACpCuC,IAAAA,IAAI,EAAG,kIAAiID,IAAK;AADzG,GAA/B,CAAP;AAGD;;AASD,eAAeE,cAAf,CAA8BT,IAA9B,EAA0C;AACxCjC,EAAAA,KAAK,CAAC,oBAAD,CAAL;AACA,QAAM2C,GAAG,GAAG,MAAM,+BAAwCV,IAAxC,EAA+C,GAAE/B,oBAAqB,6BAAtE,CAAlB;;AACA,MAAIyC,GAAG,IAAIC,KAAK,CAACC,OAAN,CAAcF,GAAG,CAACG,MAAlB,CAAX,EAAsC;AAAA;;AACpC9C,IAAAA,KAAK,CAAE,GAAE2C,GAAG,CAACG,MAAJ,CAAWC,MAAO,oBAAtB,CAAL;AACE,mBAAAJ,GAAG,CAACG,MAAJ,4DAAYE,OAAZ,CAAoB,CAAC;AAAEC,MAAAA,EAAF;AAAMC,MAAAA;AAAN,KAAD,KAAkBpB,UAAU,CAACqB,GAAX,CAAeF,EAAf,EAAmBC,IAAnB,CAAtC;AACH;AACF;;AAED,SAASE,kBAAT,CAA4BC,UAA5B,EAAgD;AAC9C,QAAMC,mBAAmB,GAAGD,UAAU,CAACE,OAAX,CAAmB,IAAnB,EAAyB,GAAzB,EAA8BC,IAA9B,EAA5B;;AACA,UAAQF,mBAAR;AACE,SAAK9C,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,SAAKC,+BAAL;AACA,SAAKI,mBAAL;AACA,SAAKd,2BAAL;AACA,SAAKW,sCAAL;AACA,SAAKI,kCAAL;AACA,SAAKC,gBAAL;AACA,SAAKC,eAAL;AACA,SAAKC,gBAAL;AACE,aAAO8B,gCAAiBC,MAAxB;;AACF,SAAKhD,sBAAL;AACA,SAAKW,gBAAL;AACA,SAAKC,wBAAL;AACE,aAAOmC,gCAAiBE,YAAxB;;AACF;AACE,YAAM,IAAIC,KAAJ,CAAW,4BAA2BN,mBAAoB,EAA1D,CAAN;AAzBJ;AA2BD;;AAED,SAASO,mBAAT,CAA6BC,QAA7B,EAA+C;AAC7C,MAAI,CAACA,QAAL,EAAe;AACb,WAAOC,SAAP;AACD;;AACD,QAAMC,OAAO,GAAGF,QAAQ,CAACG,KAAT,CAAe,MAAf,CAAhB;;AACA,MAAI,CAACD,OAAD,IAAYA,OAAO,CAACjB,MAAR,GAAiB,CAAjC,EAAoC;AAClC,WAAOgB,SAAP;AACD;;AAED,SAAO;AACLG,IAAAA,MAAM,EAAEC,QAAQ,CAACH,OAAO,CAAC,CAAD,CAAR,EAAa,EAAb,CADX;AAELI,IAAAA,KAAK,EAAED,QAAQ,CAACH,OAAO,CAAC,CAAD,CAAR,EAAa,EAAb;AAFV,GAAP;AAID;;AACD,SAASK,cAAT,CAAwBC,cAAxB,EAAyE;AAAA;;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,QAAMC,YAAY,GAAGnB,mBAAmB,CAACS,cAAc,CAACR,QAAhB,CAAxC;AACA,QAAMmB,UAAU,GAAGD,YAAY,GAC5B,GAAD,yBAAGV,cAAc,CAACY,QAAlB,0DAAG,sBAAyBC,GAAI,IAAGH,YAAY,CAACd,MAAO,EAD1B,6BAE7BI,cAAc,CAACY,QAFc,2DAE7B,uBAAyBC,GAF3B;AAIA,SAAO;AACLC,IAAAA,IAAI,EAAEhC,kBAAkB,CAACkB,cAAc,CAACe,QAAhB,CADnB;AAEL7C,IAAAA,IAAI,EAAE,qBAAO8B,cAAc,CAACI,YAAtB,EAAoCC,WAApC,EAFD;AAGLF,IAAAA,aAHK;AAILa,IAAAA,cAAc,EAAE,CAAChB,cAAc,CAACgB,cAJ3B;AAKLC,IAAAA,gBAAgB,EAAEjB,cAAc,CAACiB,gBAL5B;AAMLC,IAAAA,aAAa,EAAE,CAAClB,cAAc,CAACmB,mBAN1B;AAOLC,IAAAA,WAAW,EAAEpB,cAAc,CAACqB,YAAf,CAA4BnC,IAA5B,EAPR;AAQLoC,IAAAA,IAAI,EAAEtB,cAAc,CAACR,QARhB;AASL+B,IAAAA,QAAQ,EAAE/D,UAAU,CAACgE,GAAX,CAAexB,cAAf,aAAeA,cAAf,uBAAeA,cAAc,CAAEyB,UAA/B,CATL;AAULf,IAAAA,YAVK;AAWLC,IAAAA,UAXK;AAYLL,IAAAA;AAZK,GAAP;AAcD;;AAOD,eAAeoB,yBAAf,CAAyC/D,IAAzC,EAAqDI,WAArD,EAA0E;AACxE,QAAM4D,GAAG,GAAG7D,kBAAkB,CAACC,WAAD,CAA9B;AAEA,QAAM6D,IAAI,GAAG,MAAM,+BAA8CjE,IAA9C,EAAoDgE,GAApD,CAAnB;AACA,QAAME,qBAAoD,GAAG,EAA7D;AAEA,MAAI,CAACD,IAAD,IAAS,CAACA,IAAI,CAACpD,MAAnB,EAA2B,OAAOqD,qBAAP;AAE3BD,EAAAA,IAAI,CAACpD,MAAL,CAAYsD,YAAZ,CACE;AADF,GAEGC,MAFH,CAEWC,WAAD,IAAiB,CAAC,CAACA,WAAW,CAACjB,QAFzC,EAGGrC,OAHH,CAGYsD,WAAD,IAAqC;AAC5C,QAAI,CAACH,qBAAqB,CAACG,WAAW,CAACC,eAAb,CAA1B,EAAyD;AACvDJ,MAAAA,qBAAqB,CAACG,WAAW,CAACC,eAAb,CAArB,GAAqD,EAArD;AACD;;AAED,UAAMC,iBAAiB,GAAGnC,cAAc,CAACiC,WAAD,CAAxC;AACAH,IAAAA,qBAAqB,CAACG,WAAW,CAACC,eAAb,CAArB,CAAmDE,IAAnD,CAAwDD,iBAAxD;AACD,GAVH;AAYA,SAAOL,qBAAP;AACD;;AAED,SAASO,SAAT,CAAmBC,UAAnB,EAA8D7D,MAA9D,EAAqG;AACnG,QAAM8D,aAA4C,qBAAQD,UAAR,CAAlD;;AACAE,EAAAA,MAAM,CAACC,IAAP,CAAYhE,MAAZ,EAAoBE,OAApB,CAA6B+D,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,GAAG3D,MAAM,CAACiE,aAAD,CAA3C;AACD,GALD;AAMA,SAAOH,aAAP;AACD;;AAED,SAASI,mBAAT,CAA6BC,IAA7B,EAAkDC,WAAlD,EAA8EC,mBAA9E,EAA4GC,8BAA5G,EAAqJ;AACnJ,MAAIC,UAAU,GAAGzE,KAAK,CAAC0E,IAAN,CAAWL,IAAX,CAAjB;;AACA,MAAI,CAACE,mBAAL,EAA0B;AACxBE,IAAAA,UAAU,GAAG,mCAAgBA,UAAhB,CAAb;AACD;;AACDA,EAAAA,UAAU,GAAG,0CAAuBA,UAAvB,CAAb;AACAA,EAAAA,UAAU,GAAGD,8BAA8B,GACzC,yCAAsBC,UAAtB,EAAkCH,WAAlC,EAA+CC,mBAAmB,IAAI,KAAtE,CADyC,GAEzCE,UAFF;AAGA,SAAOA,UAAP;AACD;;AAED,eAAeE,iBAAf,CAAiCtF,IAAjC,EAA6CuF,OAA7C,EAAsE;AAAA;;AACpE,QAAMC,oBAAoB,4BAAGD,OAAO,CAACC,oBAAX,yEAAmC,CAA7D;AACA,QAAMC,kBAAkB,GAAG,uBAASC,QAAT,CAAkB,CAAlB,EAAqB,OAArB,CAA3B;AACA,QAAMC,SAAS,GAAGJ,OAAO,CAACI,SAAR,IAAqBF,kBAAkB,CAACG,MAAnB,EAAvC;;AACA,QAAMX,WAAW,GAAGY,gBAAOC,GAAP,CAAWL,kBAAX,EAA+B,qBAAOE,SAAP,CAA/B,CAApB;;AACA,QAAMI,SAAS,GAAG,oBAAmBd,WAAnB,EAAgCO,oBAAhC,CAAlB;AAEA,QAAM/E,cAAc,CAACT,IAAD,CAApB;AAEA,MAAI0E,UAAyC,GAAG,EAAhD;;AACA,OAAK,IAAIsB,CAAC,GAAG,CAAb,EAAgBA,CAAC,GAAGD,SAAS,CAACjF,MAA9B,EAAsCkF,CAAC,IAAI,CAA3C,EAA8C;AAC5C,UAAMnF,MAAM,GAAG,MAAMkD,yBAAyB,CAAC/D,IAAD,EAAO+F,SAAS,CAACC,CAAD,CAAhB,CAA9C;AACAtB,IAAAA,UAAU,GAAGD,SAAS,CAACC,UAAD,EAAa7D,MAAb,CAAtB;AACD;;AAED+D,EAAAA,MAAM,CAACC,IAAP,CAAYH,UAAZ,EAAwB3D,OAAxB,CAAiC+D,aAAD,IAAmB;AAAA;;AACjD,QAAIE,IAAI,GAAGN,UAAU,CAACI,aAAD,CAArB;AACAE,IAAAA,IAAI,GAAGD,mBAAmB,CAACC,IAAD,EAAOC,WAAP,EAAoBM,OAAO,CAACL,mBAAR,IAA+B,KAAnD,kDACvBK,OAAO,CAACU,UADe,wDACvB,oBAAoBd,8BADG,yEAC+B,IAD/B,CAA1B;AAEAT,IAAAA,UAAU,CAACI,aAAD,CAAV,GAA4BE,IAA5B;AACD,GALD;AAOA,SAAON,UAAP;AACD;;AAED,SAASwB,uBAAT,CAAiClG,IAAjC,EAAmE;AACjE,QAAMmG,IAA0B,GAAG,EAAnC;AACAA,EAAAA,IAAI,CAACC,qCAAaC,OAAd,CAAJ,GAA6B,CAAChI,WAAD,CAA7B;AACA8H,EAAAA,IAAI,CAACC,qCAAaE,cAAd,CAAJ,GAAoC,CAAClI,oBAAD,CAApC;AACA+H,EAAAA,IAAI,CAACC,qCAAaG,eAAd,CAAJ,GAAqC,CAAC,YAAY;AAChD,WAAO,gDAAqBvG,IAArB,EAA2BL,wBAA3B,CAAP;AACD,GAFoC,CAArC;AAGAwG,EAAAA,IAAI,CAACC,qCAAaI,YAAd,CAAJ,GAAkC,CAAC,YAAY;AAC7C,WAAO,gDAAqBxG,IAArB,EAA2BJ,oBAA3B,CAAP;AACD,GAFiC,CAAlC;AAGA,SAAOuG,IAAP;AACD;;AAED,SAASM,iBAAT,CAA2BC,WAA3B,EAAoE;AAClE,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;;AAID,MAAMC,UAAN,SAAyBC,8CAAzB,CAA4E;AAC1EC,EAAAA,eAAe,CAACP,WAAD,EAA0C;AACvD,WAAO;AACLQ,MAAAA,QAAQ,EAAE/I,SADL;AAELgJ,MAAAA,MAAM,EAAEV,iBAAiB,CAACC,WAAD,CAFpB;AAGLU,MAAAA,oBAAoB,EAAE,4BAHjB;AAILC,MAAAA,SAAS,EAAE,YAAY;AACrB,YAAI,MAAM,gDAAqB,KAAKrH,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;AAaLsH,MAAAA,cAAc,EAAE,YAAY;AAC1B,cAAM,iDAAsB,KAAKtH,IAA3B,EAAiC,wCAAjC,EAA2E,IAA3E,CAAN;AACD,OAfI;AAgBLuH,MAAAA,UAAU,EAAE,YAAYxH,gBAAgB,CAAC,KAAKC,IAAN,CAhBnC;AAiBLwH,MAAAA,eAAe,EAAEtB,uBAAuB,CAAC,KAAKlG,IAAN,CAjBnC;AAkBLyH,MAAAA,SAAS,EAAE;AAlBN,KAAP;AAoBD;;AAED,QAAMC,SAAN,GAAkB;AAChB,UAAMC,OAAO,GAAG,MAAMrC,iBAAiB,CAAC,KAAKtF,IAAN,EAAY,KAAKuF,OAAjB,CAAvC;AACA,UAAMqC,QAAQ,GAAGhD,MAAM,CAACC,IAAP,CAAY8C,OAAZ,EAAqBE,GAArB,CAA0B/C,aAAD,IAAmB;AAC3D,aAAO;AACLA,QAAAA,aADK;AAELE,QAAAA,IAAI,EAAE2C,OAAO,CAAC7C,aAAD;AAFR,OAAP;AAID,KALgB,CAAjB;AAOA,WAAO;AACLgD,MAAAA,OAAO,EAAE,IADJ;AAELF,MAAAA;AAFK,KAAP;AAID;;AArCyE;;eAwC7Db,U","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\nimport { Page, LoadEvent } 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 { getDebug } from '../helpers/debug';\nimport { ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\ninterface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_ACTIONS_URL = 'https://online.max.co.il';\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_ACTIONS_URL}/Anonymous/Login/PasswordExpired.aspx`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nconst NORMAL_TYPE_NAME = 'רגילה';\nconst ATM_TYPE_NAME = 'חיוב עסקות מיידי';\nconst INTERNET_SHOPPING_TYPE_NAME = 'אינטרנט/חו\"ל';\nconst INSTALLMENTS_TYPE_NAME = 'תשלומים';\nconst MONTHLY_CHARGE_TYPE_NAME = 'חיוב חודשי';\nconst ONE_MONTH_POSTPONED_TYPE_NAME = 'דחוי חודש';\nconst MONTHLY_POSTPONED_TYPE_NAME = 'דחוי לחיוב החודשי';\nconst MONTHLY_PAYMENT_TYPE_NAME = 'תשלום חודשי';\nconst FUTURE_PURCHASE_FINANCING = 'מימון לרכישה עתידית';\nconst MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME = 'דחוי חודש תשלומים';\nconst THIRTY_DAYS_PLUS_TYPE_NAME = 'עסקת 30 פלוס';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME = 'דחוי חודשיים';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME2 = 'דחוי 2 ח\\' תשלומים';\nconst MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME = 'חודשי + ריבית';\nconst CREDIT_TYPE_NAME = 'קרדיט';\nconst CREDIT_OUTSIDE_THE_LIMIT = 'קרדיט-מחוץ למסגרת';\nconst ACCUMULATING_BASKET = 'סל מצטבר';\nconst POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';\nconst REPLACEMENT_CARD = 'כרטיס חליפי';\nconst EARLY_REPAYMENT = 'פרעון מוקדם';\nconst MONTHLY_CARD_FEE = 'דמי כרטיס';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n     * url explanation:\n     * userIndex: -1 for all account owners\n     * cardIndex: -1 for all cards under the account\n     * all other query params are static, beside the date which changes for request per month\n     */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result? : Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n      res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(txnTypeStr: string) {\n  const cleanedUpTxnTypeStr = txnTypeStr.replace('\\t', ' ').trim();\n  switch (cleanedUpTxnTypeStr) {\n    case ATM_TYPE_NAME:\n    case NORMAL_TYPE_NAME:\n    case MONTHLY_CHARGE_TYPE_NAME:\n    case ONE_MONTH_POSTPONED_TYPE_NAME:\n    case MONTHLY_POSTPONED_TYPE_NAME:\n    case FUTURE_PURCHASE_FINANCING:\n    case MONTHLY_PAYMENT_TYPE_NAME:\n    case MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME:\n    case THIRTY_DAYS_PLUS_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME2:\n    case ACCUMULATING_BASKET:\n    case INTERNET_SHOPPING_TYPE_NAME:\n    case MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME:\n    case POSTPONED_TRANSACTION_INSTALLMENTS:\n    case REPLACEMENT_CARD:\n    case EARLY_REPAYMENT:\n    case MONTHLY_CARD_FEE:\n      return TransactionTypes.Normal;\n    case INSTALLMENTS_TYPE_NAME:\n    case CREDIT_TYPE_NAME:\n    case CREDIT_OUTSIDE_THE_LIMIT:\n      return TransactionTypes.Installments;\n    default:\n      throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ?\n    rawTransaction.purchaseDate :\n    rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments ?\n    `${rawTransaction.dealData?.arn}_${installments.number}` :\n    rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    description: rawTransaction.merchantName.trim(),\n    memo: rawTransaction.comments,\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult{\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter((transaction) => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach((accountNumber) => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean, enableTransactionsFilterByDate: boolean) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = enableTransactionsFilterByDate ?\n    filterOldTransactions(clonedTxns, startMoment, combineInstallments || false) :\n    clonedTxns;\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach((accountNumber) => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(txns, startMoment, options.combineInstallments || false,\n      (options.outputData?.enableTransactionsFilterByDate ?? true));\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: ScraperSpecificCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = {username: string, password: string};\n\nclass MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\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      waitUntil: 'domcontentloaded' as LoadEvent,\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"]}
341
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/max.ts"],"names":["debug","BASE_ACTIONS_URL","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","NORMAL_TYPE_NAME","ATM_TYPE_NAME","INTERNET_SHOPPING_TYPE_NAME","INSTALLMENTS_TYPE_NAME","MONTHLY_CHARGE_TYPE_NAME","ONE_MONTH_POSTPONED_TYPE_NAME","MONTHLY_POSTPONED_TYPE_NAME","MONTHLY_PAYMENT_TYPE_NAME","FUTURE_PURCHASE_FINANCING","MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME","THIRTY_DAYS_PLUS_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME","TWO_MONTHS_POSTPONED_TYPE_NAME2","MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME","CREDIT_TYPE_NAME","CREDIT_OUTSIDE_THE_LIMIT","ACCUMULATING_BASKET","POSTPONED_TRANSACTION_INSTALLMENTS","REPLACEMENT_CARD","EARLY_REPAYMENT","MONTHLY_CARD_FEE","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","getTransactionsUrl","monthMoment","month","year","date","path","loadCategories","res","Array","isArray","result","length","forEach","id","name","set","getTransactionType","txnTypeStr","cleanedUpTxnTypeStr","replace","trim","TransactionTypes","Normal","Installments","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","getChargedCurrency","currencyId","SHEKEL_CURRENCY","DOLLAR_CURRENCY","EURO_CURRENCY","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","planName","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","chargedCurrency","paymentCurrency","description","merchantName","memo","category","get","categoryId","fetchTransactionsForMonth","url","data","transactionsByAccount","transactions","filter","transaction","shortCardNumber","mappedTransaction","push","addResult","allResults","clonedResults","Object","keys","accountNumber","prepareTransactions","txns","startMoment","combineInstallments","enableTransactionsFilterByDate","clonedTxns","from","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startDate","toDate","moment","max","allMonths","i","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","checkReadiness","postAction","possibleResults","waitUntil","fetchData","results","accounts","map","success"],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\nimport { Page, LoadEvent } 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 { getDebug } from '../helpers/debug';\nimport { ScraperOptions } from './interface';\nimport { SHEKEL_CURRENCY, DOLLAR_CURRENCY, EURO_CURRENCY } from '../constants';\n\nconst debug = getDebug('max');\n\ninterface ScrapedTransaction {\n  shortCardNumber: string;\n  paymentDate?: string;\n  purchaseDate: string;\n  actualPaymentAmount: string;\n  paymentCurrency: number | null;\n  originalCurrency: string;\n  originalAmount: number;\n  planName: string;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  dealData?: {\n    arn: string;\n  };\n}\n\nconst BASE_ACTIONS_URL = 'https://online.max.co.il';\nconst BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';\nconst BASE_WELCOME_URL = 'https://www.max.co.il';\n\nconst LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;\nconst PASSWORD_EXPIRED_URL = `${BASE_ACTIONS_URL}/Anonymous/Login/PasswordExpired.aspx`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nconst NORMAL_TYPE_NAME = 'רגילה';\nconst ATM_TYPE_NAME = 'חיוב עסקות מיידי';\nconst INTERNET_SHOPPING_TYPE_NAME = 'אינטרנט/חו\"ל';\nconst INSTALLMENTS_TYPE_NAME = 'תשלומים';\nconst MONTHLY_CHARGE_TYPE_NAME = 'חיוב חודשי';\nconst ONE_MONTH_POSTPONED_TYPE_NAME = 'דחוי חודש';\nconst MONTHLY_POSTPONED_TYPE_NAME = 'דחוי לחיוב החודשי';\nconst MONTHLY_PAYMENT_TYPE_NAME = 'תשלום חודשי';\nconst FUTURE_PURCHASE_FINANCING = 'מימון לרכישה עתידית';\nconst MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME = 'דחוי חודש תשלומים';\nconst THIRTY_DAYS_PLUS_TYPE_NAME = 'עסקת 30 פלוס';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME = 'דחוי חודשיים';\nconst TWO_MONTHS_POSTPONED_TYPE_NAME2 = 'דחוי 2 ח\\' תשלומים';\nconst MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME = 'חודשי + ריבית';\nconst CREDIT_TYPE_NAME = 'קרדיט';\nconst CREDIT_OUTSIDE_THE_LIMIT = 'קרדיט-מחוץ למסגרת';\nconst ACCUMULATING_BASKET = 'סל מצטבר';\nconst POSTPONED_TRANSACTION_INSTALLMENTS = 'פריסת העסקה הדחויה';\nconst REPLACEMENT_CARD = 'כרטיס חליפי';\nconst EARLY_REPAYMENT = 'פרעון מוקדם';\nconst MONTHLY_CARD_FEE = 'דמי כרטיס';\n\nconst INVALID_DETAILS_SELECTOR = '#popupWrongDetails';\nconst LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';\n\nconst categories = new Map<number, string>();\n\nfunction redirectOrDialog(page: Page) {\n  return Promise.race([\n    waitForRedirect(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),\n    waitUntilElementFound(page, INVALID_DETAILS_SELECTOR, true),\n    waitUntilElementFound(page, LOGIN_ERROR_SELECTOR, true),\n  ]);\n}\n\nfunction getTransactionsUrl(monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const date = `${year}-${month}-01`;\n\n  /**\n     * url explanation:\n     * userIndex: -1 for all account owners\n     * cardIndex: -1 for all cards under the account\n     * all other query params are static, beside the date which changes for request per month\n     */\n  return buildUrl(BASE_API_ACTIONS_URL, {\n    path: `/api/registered/transactionDetails/getTransactionsAndGraphs?filterData={\"userIndex\":-1,\"cardIndex\":-1,\"monthView\":true,\"date\":\"${date}\",\"dates\":{\"startDate\":\"0\",\"endDate\":\"0\"},\"bankAccount\":{\"bankAccountIndex\":-1,\"cards\":null}}&firstCallCardIndex=-1`,\n  });\n}\n\ninterface FetchCategoryResult {\n  result? : Array<{\n    id: number;\n    name: string;\n  }>;\n}\n\nasync function loadCategories(page: Page) {\n  debug('Loading categories');\n  const res = await fetchGetWithinPage<FetchCategoryResult>(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);\n  if (res && Array.isArray(res.result)) {\n    debug(`${res.result.length} categories loaded`);\n      res.result?.forEach(({ id, name }) => categories.set(id, name));\n  }\n}\n\nfunction getTransactionType(txnTypeStr: string) {\n  const cleanedUpTxnTypeStr = txnTypeStr.replace('\\t', ' ').trim();\n  switch (cleanedUpTxnTypeStr) {\n    case ATM_TYPE_NAME:\n    case NORMAL_TYPE_NAME:\n    case MONTHLY_CHARGE_TYPE_NAME:\n    case ONE_MONTH_POSTPONED_TYPE_NAME:\n    case MONTHLY_POSTPONED_TYPE_NAME:\n    case FUTURE_PURCHASE_FINANCING:\n    case MONTHLY_PAYMENT_TYPE_NAME:\n    case MONTHLY_POSTPONED_INSTALLMENTS_TYPE_NAME:\n    case THIRTY_DAYS_PLUS_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME:\n    case TWO_MONTHS_POSTPONED_TYPE_NAME2:\n    case ACCUMULATING_BASKET:\n    case INTERNET_SHOPPING_TYPE_NAME:\n    case MONTHLY_CHARGE_PLUS_INTEREST_TYPE_NAME:\n    case POSTPONED_TRANSACTION_INSTALLMENTS:\n    case REPLACEMENT_CARD:\n    case EARLY_REPAYMENT:\n    case MONTHLY_CARD_FEE:\n      return TransactionTypes.Normal;\n    case INSTALLMENTS_TYPE_NAME:\n    case CREDIT_TYPE_NAME:\n    case CREDIT_OUTSIDE_THE_LIMIT:\n      return TransactionTypes.Installments;\n    default:\n      throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);\n  }\n}\n\nfunction getInstallmentsInfo(comments: string) {\n  if (!comments) {\n    return undefined;\n  }\n  const matches = comments.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getChargedCurrency(currencyId: number | null) {\n  switch (currencyId) {\n    case 376:\n      return SHEKEL_CURRENCY;\n    case 840:\n      return DOLLAR_CURRENCY;\n    case 978:\n      return EURO_CURRENCY;\n    default:\n      return undefined;\n  }\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ?\n    rawTransaction.purchaseDate :\n    rawTransaction.paymentDate).toISOString();\n  const status = isPending ? TransactionStatuses.Pending : TransactionStatuses.Completed;\n\n  const installments = getInstallmentsInfo(rawTransaction.comments);\n  const identifier = installments ?\n    `${rawTransaction.dealData?.arn}_${installments.number}` :\n    rawTransaction.dealData?.arn;\n\n  return {\n    type: getTransactionType(rawTransaction.planName),\n    date: moment(rawTransaction.purchaseDate).toISOString(),\n    processedDate,\n    originalAmount: -rawTransaction.originalAmount,\n    originalCurrency: rawTransaction.originalCurrency,\n    chargedAmount: -rawTransaction.actualPaymentAmount,\n    chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),\n    description: rawTransaction.merchantName.trim(),\n    memo: rawTransaction.comments,\n    category: categories.get(rawTransaction?.categoryId),\n    installments,\n    identifier,\n    status,\n  };\n}\ninterface ScrapedTransactionsResult{\n  result?: {\n    transactions: ScrapedTransaction[];\n  };\n}\n\nasync function fetchTransactionsForMonth(page: Page, monthMoment: Moment) {\n  const url = getTransactionsUrl(monthMoment);\n\n  const data = await fetchGetWithinPage<ScrapedTransactionsResult>(page, url);\n  const transactionsByAccount: Record<string, Transaction[]> = {};\n\n  if (!data || !data.result) return transactionsByAccount;\n\n  data.result.transactions\n    // Filter out non-transactions without a plan type, e.g. summary rows\n    .filter((transaction) => !!transaction.planName)\n    .forEach((transaction: ScrapedTransaction) => {\n      if (!transactionsByAccount[transaction.shortCardNumber]) {\n        transactionsByAccount[transaction.shortCardNumber] = [];\n      }\n\n      const mappedTransaction = mapTransaction(transaction);\n      transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);\n    });\n\n  return transactionsByAccount;\n}\n\nfunction addResult(allResults: Record<string, Transaction[]>, result: Record<string, Transaction[]>) {\n  const clonedResults: Record<string, Transaction[]> = { ...allResults };\n  Object.keys(result).forEach((accountNumber) => {\n    if (!clonedResults[accountNumber]) {\n      clonedResults[accountNumber] = [];\n    }\n    clonedResults[accountNumber].push(...result[accountNumber]);\n  });\n  return clonedResults;\n}\n\nfunction prepareTransactions(txns: Transaction[], startMoment: moment.Moment, combineInstallments: boolean, enableTransactionsFilterByDate: boolean) {\n  let clonedTxns = Array.from(txns);\n  if (!combineInstallments) {\n    clonedTxns = fixInstallments(clonedTxns);\n  }\n  clonedTxns = sortTransactionsByDate(clonedTxns);\n  clonedTxns = enableTransactionsFilterByDate ?\n    filterOldTransactions(clonedTxns, startMoment, combineInstallments || false) :\n    clonedTxns;\n  return clonedTxns;\n}\n\nasync function fetchTransactions(page: Page, options: ScraperOptions) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const defaultStartMoment = moment().subtract(1, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(defaultStartMoment, moment(startDate));\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n\n  await loadCategories(page);\n\n  let allResults: Record<string, Transaction[]> = {};\n  for (let i = 0; i < allMonths.length; i += 1) {\n    const result = await fetchTransactionsForMonth(page, allMonths[i]);\n    allResults = addResult(allResults, result);\n  }\n\n  Object.keys(allResults).forEach((accountNumber) => {\n    let txns = allResults[accountNumber];\n    txns = prepareTransactions(txns, startMoment, options.combineInstallments || false,\n      (options.outputData?.enableTransactionsFilterByDate ?? true));\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: ScraperSpecificCredentials) {\n  return [\n    { selector: '#user-name', value: credentials.username },\n    { selector: '#password', value: credentials.password },\n  ];\n}\n\ntype ScraperSpecificCredentials = {username: string, password: string};\n\nclass MaxScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\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      waitUntil: 'domcontentloaded' as LoadEvent,\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"]}
@@ -190,9 +190,9 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
190
190
  await this.page.$eval(`a[href*="${OSH_PAGE}"]`, el => el.click());
191
191
  await (0, _elementsInteractions.waitUntilElementFound)(this.page, `a[href*="${TRANSACTIONS_PAGE}"]`);
192
192
  await this.page.$eval(`a[href*="${TRANSACTIONS_PAGE}"]`, el => el.click());
193
- const accountNumberElement = (await this.page.$$('#AccountPicker b'))[0];
194
- const accountNumberHandle = await accountNumberElement.getProperty('title');
195
- const accountNumber = await accountNumberHandle.jsonValue();
193
+ const accountNumberElement = await this.page.$('#dropdownBasic b');
194
+ const accountNumberHandle = await (accountNumberElement === null || accountNumberElement === void 0 ? void 0 : accountNumberElement.getProperty('title'));
195
+ const accountNumber = await (accountNumberHandle === null || accountNumberHandle === void 0 ? void 0 : accountNumberHandle.jsonValue());
196
196
  const response = await Promise.any(TRANSACTIONS_REQUEST_URLS.map(async url => {
197
197
  const request = await this.page.waitForRequest(url);
198
198
  const data = createDataFromRequest(request, this.options.startDate);
@@ -222,4 +222,4 @@ class MizrahiScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
222
222
 
223
223
  var _default = MizrahiScraper;
224
224
  exports.default = _default;
225
- //# 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_PAGE","TRANSACTIONS_REQUEST_URLS","PENDING_TRANSACTIONS_PAGE","PENDING_TRANSACTIONS_IFRAME","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","accountDropDownItemSelector","pendingTrxIdentifierId","checkingAccountTabHebrewName","checkingAccountTabEnglishName","createLoginFields","credentials","selector","value","username","password","getPossibleLoginResults","page","LoginResults","Success","$x","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","$eval","el","click","numOfAccounts","$$","length","results","i","push","fetchAccount","success","accounts","e","errorType","ScraperErrorTypes","Generic","errorMessage","message","getPendingTransactions","frame","f","url","includes","isPending","then","catch","accountNumberElement","accountNumberHandle","getProperty","accountNumber","jsonValue","response","any","waitForRequest","options","header","Error","messages","text","relevantRows","body","rows","filter","RecTypeSpecified","oshTxn","startMoment","oshTxnAfterStartDate","isSameOrAfter","allTxn","concat","balance","Yitra"],"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 { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\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      Yitra: 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\\/OnlineApp\\/.*/;\nconst OSH_PAGE = '/osh/legacy/legacy-Osh-Main';\nconst TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';\nconst TRANSACTIONS_REQUEST_URLS = [\n  `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,\n  `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,\n];\nconst PENDING_TRANSACTIONS_PAGE = '/osh/legacy/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 = '#dropdownBasic';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#AccountPicker .item';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\nconst checkingAccountTabHebrewName = 'עובר ושב';\nconst checkingAccountTabEnglishName = 'Checking Account';\n\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\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, async () => !!(await page.$x(`//a//span[contains(., \"${checkingAccountTabHebrewName}\") or contains(., \"${checkingAccountTabEnglishName}\")]`))],\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\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass MizrahiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\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.page.$eval('#dropdownBasic, .item', (el) => (el as HTMLElement).click());\n\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n\n    try {\n      const results: TransactionsAccount[] = [];\n\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        if (i > 0) {\n          await this.page.$eval('#dropdownBasic, .item', (el) => (el as HTMLElement).click());\n        }\n\n        await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, (el) => (el as HTMLElement).click());\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.message,\n      };\n    }\n  }\n\n  private async getPendingTransactions(): Promise<Transaction[]> {\n    await this.page.$eval(`a[href*=\"${PENDING_TRANSACTIONS_PAGE}\"]`, (el) => (el as HTMLElement).click());\n    const frame = await waitUntilIframeFound(this.page, (f) => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    const isPending = await waitUntilElementFound(frame, pendingTrxIdentifierId).then(() => true).catch(() => false);\n    if (!isPending) {\n      return [];\n    }\n\n    const pendingTxn = await extractPendingTransactions(frame);\n    return pendingTxn;\n  }\n\n  private async fetchAccount() {\n    await this.page.$eval(`a[href*=\"${OSH_PAGE}\"]`, (el) => (el as HTMLElement).click());\n    await waitUntilElementFound(this.page, `a[href*=\"${TRANSACTIONS_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${TRANSACTIONS_PAGE}\"]`, (el) => (el as HTMLElement).click());\n\n    const accountNumberElement = (await this.page.$$('#AccountPicker b'))[0];\n    const accountNumberHandle = await accountNumberElement.getProperty('title');\n    const accountNumber = ((await accountNumberHandle.jsonValue()) as string);\n\n    const response = await Promise.any(TRANSACTIONS_REQUEST_URLS.map(async (url) => {\n      const request = await this.page.waitForRequest(url);\n      const data = createDataFromRequest(request, this.options.startDate);\n      const headers = createHeadersFromRequest(request);\n\n      return fetchPostWithinPage<ScrapedTransactionsResult>(this.page, url, data, headers);\n    }));\n\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    const pendingTxn = await this.getPendingTransactions();\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber,\n      txns: allTxn,\n      balance: +response.body.fields?.Yitra,\n    };\n  }\n}\n\nexport default MizrahiScraper;\n"]}
225
+ //# 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_PAGE","TRANSACTIONS_REQUEST_URLS","PENDING_TRANSACTIONS_PAGE","PENDING_TRANSACTIONS_IFRAME","CHANGE_PASSWORD_URL","DATE_FORMAT","MAX_ROWS_PER_REQUEST","usernameSelector","passwordSelector","submitButtonSelector","invalidPasswordSelector","afterLoginSelector","loginSpinnerSelector","accountDropDownItemSelector","pendingTrxIdentifierId","checkingAccountTabHebrewName","checkingAccountTabEnglishName","createLoginFields","credentials","selector","value","username","password","getPossibleLoginResults","page","LoginResults","Success","$x","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","$eval","el","click","numOfAccounts","$$","length","results","i","push","fetchAccount","success","accounts","e","errorType","ScraperErrorTypes","Generic","errorMessage","message","getPendingTransactions","frame","f","url","includes","isPending","then","catch","accountNumberElement","accountNumberHandle","getProperty","accountNumber","jsonValue","response","any","waitForRequest","options","header","Error","messages","text","relevantRows","body","rows","filter","RecTypeSpecified","oshTxn","startMoment","oshTxnAfterStartDate","isSameOrAfter","allTxn","concat","balance","Yitra"],"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 { BaseScraperWithBrowser, LoginResults, PossibleLoginResults } from './base-scraper-with-browser';\nimport { ScraperErrorTypes } from './errors';\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      Yitra: 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\\/OnlineApp\\/.*/;\nconst OSH_PAGE = '/osh/legacy/legacy-Osh-Main';\nconst TRANSACTIONS_PAGE = '/osh/legacy/root-main-osh-p428New';\nconst TRANSACTIONS_REQUEST_URLS = [\n  `${BASE_APP_URL}/OnlinePilot/api/SkyOSH/get428Index`,\n  `${BASE_APP_URL}/Online/api/SkyOSH/get428Index`,\n];\nconst PENDING_TRANSACTIONS_PAGE = '/osh/legacy/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 = '#dropdownBasic';\nconst loginSpinnerSelector = 'div.ngx-overlay.loading-foreground';\nconst accountDropDownItemSelector = '#AccountPicker .item';\nconst pendingTrxIdentifierId = '#ctl00_ContentPlaceHolder2_panel1';\nconst checkingAccountTabHebrewName = 'עובר ושב';\nconst checkingAccountTabEnglishName = 'Checking Account';\n\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\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, async () => !!(await page.$x(`//a//span[contains(., \"${checkingAccountTabHebrewName}\") or contains(., \"${checkingAccountTabEnglishName}\")]`))],\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\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass MizrahiScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  getLoginOptions(credentials: ScraperSpecificCredentials) {\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.page.$eval('#dropdownBasic, .item', (el) => (el as HTMLElement).click());\n\n    const numOfAccounts = (await this.page.$$(accountDropDownItemSelector)).length;\n\n    try {\n      const results: TransactionsAccount[] = [];\n\n      for (let i = 0; i < numOfAccounts; i += 1) {\n        if (i > 0) {\n          await this.page.$eval('#dropdownBasic, .item', (el) => (el as HTMLElement).click());\n        }\n\n        await this.page.$eval(`${accountDropDownItemSelector}:nth-child(${i + 1})`, (el) => (el as HTMLElement).click());\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.message,\n      };\n    }\n  }\n\n  private async getPendingTransactions(): Promise<Transaction[]> {\n    await this.page.$eval(`a[href*=\"${PENDING_TRANSACTIONS_PAGE}\"]`, (el) => (el as HTMLElement).click());\n    const frame = await waitUntilIframeFound(this.page, (f) => f.url().includes(PENDING_TRANSACTIONS_IFRAME));\n    const isPending = await waitUntilElementFound(frame, pendingTrxIdentifierId).then(() => true).catch(() => false);\n    if (!isPending) {\n      return [];\n    }\n\n    const pendingTxn = await extractPendingTransactions(frame);\n    return pendingTxn;\n  }\n\n  private async fetchAccount() {\n    await this.page.$eval(`a[href*=\"${OSH_PAGE}\"]`, (el) => (el as HTMLElement).click());\n    await waitUntilElementFound(this.page, `a[href*=\"${TRANSACTIONS_PAGE}\"]`);\n    await this.page.$eval(`a[href*=\"${TRANSACTIONS_PAGE}\"]`, (el) => (el as HTMLElement).click());\n\n    const accountNumberElement = await this.page.$('#dropdownBasic b');\n    const accountNumberHandle = await accountNumberElement?.getProperty('title');\n    const accountNumber = ((await accountNumberHandle?.jsonValue()) as string);\n\n    const response = await Promise.any(TRANSACTIONS_REQUEST_URLS.map(async (url) => {\n      const request = await this.page.waitForRequest(url);\n      const data = createDataFromRequest(request, this.options.startDate);\n      const headers = createHeadersFromRequest(request);\n\n      return fetchPostWithinPage<ScrapedTransactionsResult>(this.page, url, data, headers);\n    }));\n\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    const pendingTxn = await this.getPendingTransactions();\n    const allTxn = oshTxnAfterStartDate.concat(pendingTxn);\n\n    return {\n      accountNumber,\n      txns: allTxn,\n      balance: +response.body.fields?.Yitra,\n    };\n  }\n}\n\nexport default MizrahiScraper;\n"]}
@@ -106,20 +106,17 @@ function createLoginFields(credentials) {
106
106
  }
107
107
 
108
108
  function convertParsedDataToTransactions(parsedData) {
109
- return parsedData.flatMap(monthData => monthData.result.bankAccounts).flatMap(accounts => accounts.debitDates).flatMap(debitDate => debitDate.transactions).map(transaction => {
109
+ const bankAccounts = parsedData.flatMap(monthData => monthData.result.bankAccounts);
110
+ const regularDebitDays = bankAccounts.flatMap(accounts => accounts.debitDates);
111
+ const immediateDebitDays = bankAccounts.flatMap(accounts => accounts.immidiateDebits.debitDays);
112
+ return [...regularDebitDays, ...immediateDebitDays].flatMap(debitDate => debitDate.transactions).map(transaction => {
110
113
  const installments = transaction.curPaymentNum && transaction.numOfPayments && {
111
114
  number: transaction.curPaymentNum,
112
115
  total: transaction.numOfPayments
113
116
  } || undefined;
114
117
  const date = (0, _moment.default)(transaction.trnPurchaseDate);
115
- let chargedAmount = transaction.amtBeforeConvAndIndex * -1;
116
- let originalAmount = transaction.trnAmt * -1;
117
-
118
- if (transaction.trnTypeCode === trnTypeCode.credit) {
119
- chargedAmount = transaction.amtBeforeConvAndIndex;
120
- originalAmount = transaction.trnAmt;
121
- }
122
-
118
+ const chargedAmount = transaction.amtBeforeConvAndIndex * -1;
119
+ const originalAmount = transaction.trnAmt * -1;
123
120
  const result = {
124
121
  identifier: transaction.trnIntId,
125
122
  type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ? _transactions2.TransactionTypes.Normal : _transactions2.TransactionTypes.Installments,
@@ -297,4 +294,4 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
297
294
 
298
295
  var _default = VisaCalScraper;
299
296
  exports.default = _default;
300
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","trnTypeCode","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","convertParsedDataToTransactions","parsedData","flatMap","monthData","result","bankAccounts","accounts","debitDates","debitDate","transactions","map","transaction","installments","curPaymentNum","numOfPayments","number","total","undefined","date","trnPurchaseDate","chargedAmount","amtBeforeConvAndIndex","originalAmount","trnAmt","credit","identifier","trnIntId","type","regular","standingOrder","TransactionTypes","Normal","Installments","status","TransactionStatuses","Completed","add","toISOString","processedDate","Date","debCrdDate","originalCurrency","trnCurrencySymbol","chargedCurrency","debCrdCurrencySymbol","description","merchantName","memo","transTypeCommentDetails","toString","category","branchCodeDesc","VisaCalScraper","BaseScraperWithBrowser","getCards","initData","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","auth","calConnectToken","getXSiteId","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","postAction","currentUrl","endsWith","e","userAgent","isCardTransactionDetails","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","moment","max","format","Authorization","xSiteId","futureMonthsToScrape","all","card","finalMonthToFetchMoment","months","diff","allMonthsData","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","combineInstallments","accountNumber","JSON","stringify","success"],"mappings":";;;;;;;;;;;;;;;AAAA;;AAGA;;AACA;;AAGA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAMA;;;;;;AAGA,MAAMA,SAAS,GAAG,+BAAlB;AACA,MAAMC,6BAA6B,GAAG,8FAAtC;AAEA,MAAMC,sBAAsB,GAAG,mCAA/B;AAEA,MAAMC,KAAK,GAAG,qBAAS,UAAT,CAAd;IAEKC,W;;WAAAA,W;AAAAA,EAAAA,W;AAAAA,EAAAA,W;AAAAA,EAAAA,W;AAAAA,EAAAA,W;GAAAA,W,KAAAA,W;;AAiGL,eAAeC,aAAf,CAA6BC,IAA7B,EAAyC;AACvC,MAAIC,KAAmB,GAAG,IAA1B;AACAJ,EAAAA,KAAK,CAAC,8BAAD,CAAL;AACA,QAAM,wBAAU,MAAM;AACpBI,IAAAA,KAAK,GAAGD,IAAI,CACTE,MADK,GAELC,IAFK,CAECC,CAAD,IAAOA,CAAC,CAACC,GAAF,GAAQC,QAAR,CAAiB,YAAjB,CAFP,KAE0C,IAFlD;AAGA,WAAOC,OAAO,CAACC,OAAR,CAAgB,CAAC,CAACP,KAAlB,CAAP;AACD,GALK,EAKH,iCALG,EAKgC,KALhC,EAKuC,IALvC,CAAN;;AAOA,MAAI,CAACA,KAAL,EAAY;AACVJ,IAAAA,KAAK,CAAC,2CAAD,CAAL;AACA,UAAM,IAAIY,KAAJ,CAAU,gCAAV,CAAN;AACD;;AAED,SAAOR,KAAP;AACD;;AAED,eAAeS,uBAAf,CAAuCV,IAAvC,EAAmD;AACjD,QAAMC,KAAK,GAAG,MAAMF,aAAa,CAACC,IAAD,CAAjC;AACA,QAAMW,UAAU,GAAG,MAAM,gDAAqBV,KAArB,EAA4B,yBAA5B,CAAzB;AACA,QAAMW,YAAY,GAAGD,UAAU,GAAG,MAAM,oCAASV,KAAT,EAAgB,yBAAhB,EAA2C,EAA3C,EAAgDY,IAAD,IAAU;AAC/F,WAAQA,IAAD,CAAyBC,SAAhC;AACD,GAFuC,CAAT,GAE1B,EAFL;AAGA,SAAOF,YAAY,KAAKhB,sBAAxB;AACD;;AAED,SAASmB,uBAAT,GAAmC;AACjClB,EAAAA,KAAK,CAAC,+BAAD,CAAL;AACA,QAAMmB,IAAqC,GAAG;AAC5C,KAACC,qCAAaC,OAAd,GAAwB,CAAC,YAAD,CADoB;AAE5C,KAACD,qCAAaE,eAAd,GAAgC,CAAC,MAAOC,OAAP,IAAqC;AACpE,YAAMpB,IAAI,GAAGoB,OAAH,aAAGA,OAAH,uBAAGA,OAAO,CAAEpB,IAAtB;;AACA,UAAI,CAACA,IAAL,EAAW;AACT,eAAO,KAAP;AACD;;AACD,aAAOU,uBAAuB,CAACV,IAAD,CAA9B;AACD,KAN+B,CAFY,CAS5C;AACA;;AAV4C,GAA9C;AAYA,SAAOgB,IAAP;AACD;;AAED,SAASK,iBAAT,CAA2BC,WAA3B,EAAoE;AAClEzB,EAAAA,KAAK,CAAC,+CAAD,CAAL;AACA,SAAO,CACL;AAAE0B,IAAAA,QAAQ,EAAE,8BAAZ;AAA4CC,IAAAA,KAAK,EAAEF,WAAW,CAACG;AAA/D,GADK,EAEL;AAAEF,IAAAA,QAAQ,EAAE,8BAAZ;AAA4CC,IAAAA,KAAK,EAAEF,WAAW,CAACI;AAA/D,GAFK,CAAP;AAID;;AAED,SAASC,+BAAT,CAAyCC,UAAzC,EAA8F;AAC5F,SAAOA,UAAU,CACdC,OADI,CACKC,SAAD,IAAeA,SAAS,CAACC,MAAV,CAAiBC,YADpC,EAEJH,OAFI,CAEKI,QAAD,IAAcA,QAAQ,CAACC,UAF3B,EAGJL,OAHI,CAGKM,SAAD,IAAeA,SAAS,CAACC,YAH7B,EAIJC,GAJI,CAICC,WAAD,IAAiB;AACpB,UAAMC,YAAY,GAAID,WAAW,CAACE,aAAZ,IAA6BF,WAAW,CAACG,aAAzC,IACtB;AACEC,MAAAA,MAAM,EAAEJ,WAAW,CAACE,aADtB;AAEEG,MAAAA,KAAK,EAAEL,WAAW,CAACG;AAFrB,KADqB,IAKnBG,SALF;AAOA,UAAMC,IAAI,GAAG,qBAAOP,WAAW,CAACQ,eAAnB,CAAb;AAEA,QAAIC,aAAa,GAAGT,WAAW,CAACU,qBAAZ,GAAqC,CAAC,CAA1D;AACA,QAAIC,cAAc,GAAGX,WAAW,CAACY,MAAZ,GAAsB,CAAC,CAA5C;;AAEA,QAAIZ,WAAW,CAACxC,WAAZ,KAA4BA,WAAW,CAACqD,MAA5C,EAAoD;AAClDJ,MAAAA,aAAa,GAAGT,WAAW,CAACU,qBAA5B;AACAC,MAAAA,cAAc,GAAGX,WAAW,CAACY,MAA7B;AACD;;AAED,UAAMnB,MAAmB,GAAG;AAC1BqB,MAAAA,UAAU,EAAEd,WAAW,CAACe,QADE;AAE1BC,MAAAA,IAAI,EAAE,CAACxD,WAAW,CAACyD,OAAb,EAAsBzD,WAAW,CAAC0D,aAAlC,EAAiDlD,QAAjD,CAA0DgC,WAAW,CAACxC,WAAtE,IACJ2D,gCAAiBC,MADb,GAEJD,gCAAiBE,YAJO;AAK1BC,MAAAA,MAAM,EAAEC,mCAAoBC,SALF;AAM1BjB,MAAAA,IAAI,EAAEN,YAAY,GAChBM,IAAI,CAACkB,GAAL,CAASxB,YAAY,CAACG,MAAb,GAAsB,CAA/B,EAAkC,OAAlC,EAA2CsB,WAA3C,EADgB,GAEhBnB,IAAI,CAACmB,WAAL,EARwB;AAS1BC,MAAAA,aAAa,EAAE,IAAIC,IAAJ,CAAS5B,WAAW,CAAC6B,UAArB,EAAiCH,WAAjC,EATW;AAU1Bf,MAAAA,cAV0B;AAW1BmB,MAAAA,gBAAgB,EAAE9B,WAAW,CAAC+B,iBAXJ;AAY1BtB,MAAAA,aAZ0B;AAa1BuB,MAAAA,eAAe,EAAEhC,WAAW,CAACiC,oBAbH;AAc1BC,MAAAA,WAAW,EAAElC,WAAW,CAACmC,YAdC;AAe1BC,MAAAA,IAAI,EAAEpC,WAAW,CAACqC,uBAAZ,CAAoCC,QAApC,EAfoB;AAgB1BC,MAAAA,QAAQ,EAAEvC,WAAW,CAACwC;AAhBI,KAA5B;;AAmBA,QAAIvC,YAAJ,EAAkB;AAChBR,MAAAA,MAAM,CAACQ,YAAP,GAAsBA,YAAtB;AACD;;AAED,WAAOR,MAAP;AACD,GA9CI,CAAP;AA+CD;;AAID,MAAMgD,cAAN,SAA6BC,8CAA7B,CAAgF;AAAA;AAAA;;AAAA,4CAC7D,YAAY;AAC3BnF,MAAAA,KAAK,CAAC,qDAAD,CAAL;AACA,YAAM,iDAAsB,KAAKG,IAA3B,EAAiC,oBAAjC,EAAuD,IAAvD,CAAN;AACAH,MAAAA,KAAK,CAAC,2BAAD,CAAL;AACA,YAAM,uCAAY,KAAKG,IAAjB,EAAuB,oBAAvB,CAAN;AACAH,MAAAA,KAAK,CAAC,oCAAD,CAAL;AACA,YAAMI,KAAK,GAAG,MAAMF,aAAa,CAAC,KAAKC,IAAN,CAAjC;AACAH,MAAAA,KAAK,CAAC,uDAAD,CAAL;AACA,YAAM,iDAAsBI,KAAtB,EAA6B,gBAA7B,CAAN;AACAJ,MAAAA,KAAK,CAAC,oCAAD,CAAL;AACA,YAAM,uCAAYI,KAAZ,EAAmB,gBAAnB,CAAN;AACAJ,MAAAA,KAAK,CAAC,6CAAD,CAAL;AACA,YAAM,iDAAsBI,KAAtB,EAA6B,eAA7B,CAAN;AAEA,aAAOA,KAAP;AACD,KAhB6E;AAAA;;AAkB9E,QAAMgF,QAAN,GAAiB;AACf,UAAMC,QAAQ,GAAG,MAAM,wBACrB,MAAM,oCAAoC,KAAKlF,IAAzC,EAA+C,MAA/C,CADe,EAErB,kCAFqB,EAGrB,KAHqB,EAIrB,IAJqB,CAAvB;;AAMA,QAAI,CAACkF,QAAL,EAAe;AACb,YAAM,IAAIzE,KAAJ,CAAU,iDAAV,CAAN;AACD;;AACD,WAAOyE,QAAP,aAAOA,QAAP,uBAAOA,QAAQ,CAAEnD,MAAV,CAAiBoD,KAAjB,CAAuB9C,GAAvB,CAA2B,CAAC;AAAE+C,MAAAA,YAAF;AAAgBC,MAAAA;AAAhB,KAAD,MAAoC;AAAED,MAAAA,YAAF;AAAgBC,MAAAA;AAAhB,KAApC,CAA3B,CAAP;AACD;;AAED,QAAMC,sBAAN,GAA+B;AAC7B,UAAMC,UAAU,GAAG,MAAM,oCAA6D,KAAKvF,IAAlE,EAAwE,aAAxE,CAAzB;;AACA,QAAI,CAACuF,UAAL,EAAiB;AACf,YAAM,IAAI9E,KAAJ,CAAU,mDAAV,CAAN;AACD;;AACD,WAAQ,iBAAgB8E,UAAU,CAACC,IAAX,CAAgBC,eAAgB,EAAxD;AACD;;AAED,QAAMC,UAAN,GAAmB;AACjB;;;;;;;;;;;;AAcA,WAAOnF,OAAO,CAACC,OAAR,CAAgB,sCAAhB,CAAP;AACD;;AAEDmF,EAAAA,eAAe,CAACrE,WAAD,EAAwD;AACrE,WAAO;AACLsE,MAAAA,QAAQ,EAAG,GAAElG,SAAU,EADlB;AAELmG,MAAAA,MAAM,EAAExE,iBAAiB,CAACC,WAAD,CAFpB;AAGLwE,MAAAA,oBAAoB,EAAE,uBAHjB;AAILC,MAAAA,eAAe,EAAEhF,uBAAuB,EAJnC;AAKLiF,MAAAA,cAAc,EAAE,YAAY,iDAAsB,KAAKhG,IAA3B,EAAiC,oBAAjC,CALvB;AAMLiG,MAAAA,SAAS,EAAE,KAAKC,cANX;AAOLC,MAAAA,UAAU,EAAE,YAAY;AACtB,YAAI;AACF,gBAAM,iDAAsB,KAAKnG,IAA3B,EAAiC,kBAAjC,CAAN;AACA,gBAAMoG,UAAU,GAAG,MAAM,+BAAc,KAAKpG,IAAnB,CAAzB;;AACA,cAAIoG,UAAU,CAACC,QAAX,CAAoB,eAApB,CAAJ,EAA0C;AACxC,kBAAM,uCAAY,KAAKrG,IAAjB,EAAuB,kBAAvB,CAAN;AACD;AACF,SAND,CAME,OAAOsG,CAAP,EAAU;AACV,gBAAMF,UAAU,GAAG,MAAM,+BAAc,KAAKpG,IAAnB,CAAzB;AACA,cAAIoG,UAAU,CAACC,QAAX,CAAoB,WAApB,CAAJ,EAAsC;AACtC,gBAAMC,CAAN;AACD;AACF,OAnBI;AAoBLC,MAAAA,SAAS,EAAE;AApBN,KAAP;AAsBD;;AAEDC,EAAAA,wBAAwB,CAACzE,MAAD,EACW;AACjC,WAAQA,MAAD,CAAmCA,MAAnC,KAA8Ca,SAArD;AACD;;AAED,QAAM6D,SAAN,GAAkD;AAAA;;AAChD,UAAMC,kBAAkB,GAAG,uBAASC,QAAT,CAAkB,CAAlB,EAAqB,OAArB,EAA8BA,QAA9B,CAAuC,CAAvC,EAA0C,QAA1C,EAAoD5C,GAApD,CAAwD,CAAxD,EAA2D,KAA3D,CAA3B;AACA,UAAM6C,SAAS,GAAG,KAAKxF,OAAL,CAAawF,SAAb,IAA0BF,kBAAkB,CAACG,MAAnB,EAA5C;;AACA,UAAMC,WAAW,GAAGC,gBAAOC,GAAP,CAAWN,kBAAX,EAA+B,qBAAOE,SAAP,CAA/B,CAApB;;AACA/G,IAAAA,KAAK,CAAE,+BAA8BiH,WAAW,CAACG,MAAZ,EAAqB,EAArD,CAAL;AAEA,UAAMC,aAAa,GAAG,MAAM,KAAK5B,sBAAL,EAA5B;AACA,UAAMH,KAAK,GAAG,MAAM,KAAKF,QAAL,EAApB;AACA,UAAMkC,OAAO,GAAG,MAAM,KAAKzB,UAAL,EAAtB;AACA,UAAM0B,oBAAoB,4BAAG,KAAKhG,OAAL,CAAagG,oBAAhB,yEAAwC,CAAlE;AAEA,UAAMnF,QAAQ,GAAG,MAAM1B,OAAO,CAAC8G,GAAR,CACrBlC,KAAK,CAAC9C,GAAN,CAAU,MAAOiF,IAAP,IAAgB;AAAA;;AACxBzH,MAAAA,KAAK,CAAE,+BAA8ByH,IAAI,CAAClC,YAAa,EAAlD,CAAL;AAEA,YAAMmC,uBAAuB,GAAG,uBAASxD,GAAT,CAAaqD,oBAAb,EAAmC,OAAnC,CAAhC;AACA,YAAMI,MAAM,GAAGD,uBAAuB,CAACE,IAAxB,CAA6BX,WAA7B,EAA0C,QAA1C,CAAf;AAEA,YAAMY,aAAyC,GAAG,EAAlD;;AACA,WAAK,IAAIC,CAAC,GAAG,CAAb,EAAgBA,CAAC,IAAIH,MAArB,EAA6BG,CAAC,IAAI,CAAlC,EAAqC;AACnC,cAAMC,KAAK,GAAGL,uBAAuB,CAACM,KAAxB,GAAgClB,QAAhC,CAAyCgB,CAAzC,EAA4C,QAA5C,CAAd;AACA,cAAM7F,SAAS,GAAG,MAAM,gCACtB,KAAK9B,IADiB,EACXL,6BADW,EAEtB;AAAEyF,UAAAA,YAAY,EAAEkC,IAAI,CAAClC,YAArB;AAAmCwC,UAAAA,KAAK,EAAEA,KAAK,CAACX,MAAN,CAAa,GAAb,CAA1C;AAA6Da,UAAAA,IAAI,EAAEF,KAAK,CAACX,MAAN,CAAa,MAAb;AAAnE,SAFsB,EAGtB;AACEC,UAAAA,aADF;AAEE,uBAAaC,OAFf;AAGE,0BAAgB;AAHlB,SAHsB,CAAxB;AAUA,YAAI,CAAArF,SAAS,SAAT,IAAAA,SAAS,WAAT,YAAAA,SAAS,CAAEiG,UAAX,MAA0B,CAA9B,EAAiC,MAAM,IAAItH,KAAJ,CAAW,yCAAwC6G,IAAI,CAACjC,WAAY,cAAa,CAAAvD,SAAS,SAAT,IAAAA,SAAS,WAAT,YAAAA,SAAS,CAAEkG,KAAX,KAAoB,EAAG,EAAxG,CAAN;;AAEjC,YAAI,CAAC,KAAKxB,wBAAL,CAA8B1E,SAA9B,CAAL,EAA+C;AAC7C,gBAAM,IAAIrB,KAAJ,CAAU,iDAAV,CAAN;AACD;;AAEDiH,QAAAA,aAAa,CAACO,IAAd,CAAmBnG,SAAnB;AACD;;AAED,YAAMM,YAAY,GAAGT,+BAA+B,CAAC+F,aAAD,CAApD;AAEA7H,MAAAA,KAAK,CAAC,4BAAD,CAAL;AACA,YAAMqI,IAAI,GAAG,oDAAC,KAAK9G,OAAL,CAAa+G,UAAd,2DAAC,uBAAyBC,8BAA1B,yEAA4D,IAA5D,IACX,yCAAsBhG,YAAtB,EAAoC,qBAAOwE,SAAP,CAApC,EAAuD,KAAKxF,OAAL,CAAaiH,mBAAb,IAAoC,KAA3F,CADW,GAEXjG,YAFF;AAIA,aAAO;AACL8F,QAAAA,IADK;AAELI,QAAAA,aAAa,EAAEhB,IAAI,CAACjC;AAFf,OAAP;AAID,KAvCD,CADqB,CAAvB;AA2CAxF,IAAAA,KAAK,CAAC,6BAAD,CAAL;AAEAA,IAAAA,KAAK,CAAC0I,IAAI,CAACC,SAAL,CAAevG,QAAf,EAAyB,IAAzB,EAA+B,CAA/B,CAAD,CAAL;AACA,WAAO;AACLwG,MAAAA,OAAO,EAAE,IADJ;AAELxG,MAAAA;AAFK,KAAP;AAID;;AApJ6E;;eAuJjE8C,c","sourcesContent":["import moment from 'moment';\nimport { Frame, Page } from 'puppeteer';\n\nimport { getDebug } from '../helpers/debug';\nimport {\n  clickButton, elementPresentOnPage, pageEval, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport {\n  Transaction,\n  TransactionStatuses,\n  TransactionTypes,\n  TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum trnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: trnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = '₪' | string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: [], debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/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: ScraperSpecificCredentials) {\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\nfunction convertParsedDataToTransactions(parsedData: CardTransactionDetails[]): Transaction[] {\n  return parsedData\n    .flatMap((monthData) => monthData.result.bankAccounts)\n    .flatMap((accounts) => accounts.debitDates)\n    .flatMap((debitDate) => debitDate.transactions)\n    .map((transaction) => {\n      const installments = (transaction.curPaymentNum && transaction.numOfPayments &&\n      {\n        number: transaction.curPaymentNum,\n        total: transaction.numOfPayments,\n      }) ||\n        undefined;\n\n      const date = moment(transaction.trnPurchaseDate);\n\n      let chargedAmount = transaction.amtBeforeConvAndIndex * (-1);\n      let originalAmount = transaction.trnAmt * (-1);\n\n      if (transaction.trnTypeCode === trnTypeCode.credit) {\n        chargedAmount = transaction.amtBeforeConvAndIndex;\n        originalAmount = transaction.trnAmt;\n      }\n\n      const result: Transaction = {\n        identifier: transaction.trnIntId,\n        type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ?\n          TransactionTypes.Normal :\n          TransactionTypes.Installments,\n        status: TransactionStatuses.Completed,\n        date: installments ?\n          date.add(installments.number - 1, 'month').toISOString() :\n          date.toISOString(),\n        processedDate: new Date(transaction.debCrdDate).toISOString(),\n        originalAmount,\n        originalCurrency: transaction.trnCurrencySymbol,\n        chargedAmount,\n        chargedCurrency: transaction.debCrdCurrencySymbol,\n        description: transaction.merchantName,\n        memo: transaction.transTypeCommentDetails.toString(),\n        category: transaction.branchCodeDesc,\n      };\n\n      if (installments) {\n        result.installments = installments;\n      }\n\n      return result;\n    });\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\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  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error('could not find \\'init\\' data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error('could not find \\'auth-module\\' in session storage');\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\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      postAction: async () => {\n        try {\n          await waitUntilElementFound(this.page, 'button.btn-close');\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          throw e;\n        }\n      },\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  isCardTransactionDetails(result: CardTransactionDetails | CardTransactionDetailsError):\n    result is CardTransactionDetails {\n    return (result as CardTransactionDetails).result !== undefined;\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').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    const Authorization = await this.getAuthorizationHeader();\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    const accounts = await Promise.all(\n      cards.map(async (card) => {\n        debug(`fetch transactions for card ${card.cardUniqueId}`);\n\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n\n        const allMonthsData: (CardTransactionDetails)[] = [];\n        for (let i = 0; i <= months; i += 1) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page, TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);\n\n          if (!this.isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData);\n\n        debug('filer out old transactions');\n        const txns = (this.options.outputData?.enableTransactionsFilterByDate ?? true) ?\n          filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false) :\n          transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
297
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","trnTypeCode","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","convertParsedDataToTransactions","parsedData","bankAccounts","flatMap","monthData","result","regularDebitDays","accounts","debitDates","immediateDebitDays","immidiateDebits","debitDays","debitDate","transactions","map","transaction","installments","curPaymentNum","numOfPayments","number","total","undefined","date","trnPurchaseDate","chargedAmount","amtBeforeConvAndIndex","originalAmount","trnAmt","identifier","trnIntId","type","regular","standingOrder","TransactionTypes","Normal","Installments","status","TransactionStatuses","Completed","add","toISOString","processedDate","Date","debCrdDate","originalCurrency","trnCurrencySymbol","chargedCurrency","debCrdCurrencySymbol","description","merchantName","memo","transTypeCommentDetails","toString","category","branchCodeDesc","VisaCalScraper","BaseScraperWithBrowser","getCards","initData","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","auth","calConnectToken","getXSiteId","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","postAction","currentUrl","endsWith","e","userAgent","isCardTransactionDetails","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","moment","max","format","Authorization","xSiteId","futureMonthsToScrape","all","card","finalMonthToFetchMoment","months","diff","allMonthsData","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","combineInstallments","accountNumber","JSON","stringify","success"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { Frame, Page } from 'puppeteer';\n\nimport { getDebug } from '../helpers/debug';\nimport {\n  clickButton, elementPresentOnPage, pageEval, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport {\n  Transaction,\n  TransactionStatuses,\n  TransactionTypes,\n  TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum trnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: trnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = '₪' | string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: [], debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/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: ScraperSpecificCredentials) {\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\nfunction convertParsedDataToTransactions(parsedData: CardTransactionDetails[]): Transaction[] {\n  const bankAccounts = parsedData\n    .flatMap((monthData) => monthData.result.bankAccounts);\n\n  const regularDebitDays = bankAccounts\n    .flatMap((accounts) => accounts.debitDates);\n  const immediateDebitDays = bankAccounts\n    .flatMap((accounts) => accounts.immidiateDebits.debitDays);\n\n  return [...regularDebitDays, ...immediateDebitDays]\n    .flatMap((debitDate) => debitDate.transactions)\n    .map((transaction) => {\n      const installments = (transaction.curPaymentNum && transaction.numOfPayments &&\n      {\n        number: transaction.curPaymentNum,\n        total: transaction.numOfPayments,\n      }) ||\n        undefined;\n\n      const date = moment(transaction.trnPurchaseDate);\n\n      const chargedAmount = transaction.amtBeforeConvAndIndex * (-1);\n      const originalAmount = transaction.trnAmt * (-1);\n\n      const result: Transaction = {\n        identifier: transaction.trnIntId,\n        type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ?\n          TransactionTypes.Normal :\n          TransactionTypes.Installments,\n        status: TransactionStatuses.Completed,\n        date: installments ?\n          date.add(installments.number - 1, 'month').toISOString() :\n          date.toISOString(),\n        processedDate: new Date(transaction.debCrdDate).toISOString(),\n        originalAmount,\n        originalCurrency: transaction.trnCurrencySymbol,\n        chargedAmount,\n        chargedCurrency: transaction.debCrdCurrencySymbol,\n        description: transaction.merchantName,\n        memo: transaction.transTypeCommentDetails.toString(),\n        category: transaction.branchCodeDesc,\n      };\n\n      if (installments) {\n        result.installments = installments;\n      }\n\n      return result;\n    });\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\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  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error('could not find \\'init\\' data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error('could not find \\'auth-module\\' in session storage');\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\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      postAction: async () => {\n        try {\n          await waitUntilElementFound(this.page, 'button.btn-close');\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          throw e;\n        }\n      },\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  isCardTransactionDetails(result: CardTransactionDetails | CardTransactionDetailsError):\n    result is CardTransactionDetails {\n    return (result as CardTransactionDetails).result !== undefined;\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').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    const Authorization = await this.getAuthorizationHeader();\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    const accounts = await Promise.all(\n      cards.map(async (card) => {\n        debug(`fetch transactions for card ${card.cardUniqueId}`);\n\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n\n        const allMonthsData: (CardTransactionDetails)[] = [];\n        for (let i = 0; i <= months; i += 1) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page, TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);\n\n          if (!this.isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData);\n\n        debug('filer out old transactions');\n        const txns = (this.options.outputData?.enableTransactionsFilterByDate ?? true) ?\n          filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false) :\n          transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "israeli-bank-scrapers",
3
- "version": "4.1.1",
3
+ "version": "4.2.1",
4
4
  "private": false,
5
5
  "description": "Provide scrapers for all major Israeli banks and credit card companies",
6
6
  "engines": {