israeli-bank-scrapers 6.1.1 → 6.1.2

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.
Files changed (79) hide show
  1. package/lib/assertNever.js +7 -5
  2. package/lib/constants.js +16 -13
  3. package/lib/definitions.js +113 -109
  4. package/lib/helpers/browser.js +13 -9
  5. package/lib/helpers/dates.js +19 -18
  6. package/lib/helpers/debug.js +9 -9
  7. package/lib/helpers/elements-interactions.js +82 -78
  8. package/lib/helpers/fetch.js +85 -82
  9. package/lib/helpers/navigation.js +28 -24
  10. package/lib/helpers/storage.js +11 -10
  11. package/lib/helpers/transactions.js +32 -33
  12. package/lib/helpers/waiting.js +42 -45
  13. package/lib/index.js +82 -15
  14. package/lib/scrapers/amex.js +13 -11
  15. package/lib/scrapers/amex.test.d.ts +1 -0
  16. package/lib/scrapers/amex.test.js +49 -0
  17. package/lib/scrapers/base-beinleumi-group.js +239 -233
  18. package/lib/scrapers/base-isracard-amex.js +273 -273
  19. package/lib/scrapers/base-scraper-with-browser.js +263 -241
  20. package/lib/scrapers/base-scraper-with-browser.test.d.ts +1 -0
  21. package/lib/scrapers/base-scraper-with-browser.test.js +53 -0
  22. package/lib/scrapers/base-scraper.js +82 -82
  23. package/lib/scrapers/behatsdaa.js +103 -98
  24. package/lib/scrapers/behatsdaa.test.d.ts +1 -0
  25. package/lib/scrapers/behatsdaa.test.js +46 -0
  26. package/lib/scrapers/beinleumi.js +13 -11
  27. package/lib/scrapers/beinleumi.test.d.ts +1 -0
  28. package/lib/scrapers/beinleumi.test.js +47 -0
  29. package/lib/scrapers/beyahad-bishvilha.js +132 -132
  30. package/lib/scrapers/beyahad-bishvilha.test.d.ts +1 -0
  31. package/lib/scrapers/beyahad-bishvilha.test.js +47 -0
  32. package/lib/scrapers/discount.js +101 -97
  33. package/lib/scrapers/discount.test.d.ts +1 -0
  34. package/lib/scrapers/discount.test.js +49 -0
  35. package/lib/scrapers/errors.js +25 -22
  36. package/lib/scrapers/factory.js +67 -66
  37. package/lib/scrapers/factory.test.d.ts +1 -0
  38. package/lib/scrapers/factory.test.js +19 -0
  39. package/lib/scrapers/hapoalim.js +175 -162
  40. package/lib/scrapers/hapoalim.test.d.ts +1 -0
  41. package/lib/scrapers/hapoalim.test.js +47 -0
  42. package/lib/scrapers/interface.js +5 -2
  43. package/lib/scrapers/isracard.js +13 -11
  44. package/lib/scrapers/isracard.test.d.ts +1 -0
  45. package/lib/scrapers/isracard.test.js +49 -0
  46. package/lib/scrapers/leumi.js +170 -167
  47. package/lib/scrapers/leumi.test.d.ts +1 -0
  48. package/lib/scrapers/leumi.test.js +47 -0
  49. package/lib/scrapers/massad.js +13 -11
  50. package/lib/scrapers/max.js +261 -261
  51. package/lib/scrapers/max.test.d.ts +1 -0
  52. package/lib/scrapers/max.test.js +65 -0
  53. package/lib/scrapers/mercantile.js +16 -14
  54. package/lib/scrapers/mercantile.test.d.ts +1 -0
  55. package/lib/scrapers/mercantile.test.js +45 -0
  56. package/lib/scrapers/mizrahi.js +154 -158
  57. package/lib/scrapers/mizrahi.test.d.ts +1 -0
  58. package/lib/scrapers/mizrahi.test.js +53 -0
  59. package/lib/scrapers/one-zero-queries.js +7 -4
  60. package/lib/scrapers/one-zero.js +221 -176
  61. package/lib/scrapers/one-zero.test.d.ts +1 -0
  62. package/lib/scrapers/one-zero.test.js +51 -0
  63. package/lib/scrapers/otsar-hahayal.js +13 -11
  64. package/lib/scrapers/otsar-hahayal.test.d.ts +1 -0
  65. package/lib/scrapers/otsar-hahayal.test.js +47 -0
  66. package/lib/scrapers/pagi.js +13 -11
  67. package/lib/scrapers/pagi.test.d.ts +1 -0
  68. package/lib/scrapers/pagi.test.js +47 -0
  69. package/lib/scrapers/union-bank.js +173 -172
  70. package/lib/scrapers/union-bank.test.d.ts +1 -0
  71. package/lib/scrapers/union-bank.test.js +47 -0
  72. package/lib/scrapers/visa-cal.js +250 -254
  73. package/lib/scrapers/visa-cal.test.d.ts +1 -0
  74. package/lib/scrapers/visa-cal.test.js +49 -0
  75. package/lib/scrapers/yahav.js +206 -190
  76. package/lib/scrapers/yahav.test.d.ts +1 -0
  77. package/lib/scrapers/yahav.test.js +49 -0
  78. package/lib/transactions.js +16 -13
  79. package/package.json +8 -3
@@ -1,299 +1,299 @@
1
1
  "use strict";
2
- var __importDefault = (this && this.__importDefault) || function (mod) {
3
- return (mod && mod.__esModule) ? mod : { "default": mod };
4
- };
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.getMemo = void 0;
7
- const build_url_1 = __importDefault(require("build-url"));
8
- const moment_1 = __importDefault(require("moment"));
9
- const constants_1 = require("../constants");
10
- const dates_1 = __importDefault(require("../helpers/dates"));
11
- const debug_1 = require("../helpers/debug");
12
- const elements_interactions_1 = require("../helpers/elements-interactions");
13
- const fetch_1 = require("../helpers/fetch");
14
- const navigation_1 = require("../helpers/navigation");
15
- const transactions_1 = require("../helpers/transactions");
16
- const transactions_2 = require("../transactions");
17
- const base_scraper_with_browser_1 = require("./base-scraper-with-browser");
18
- const debug = (0, debug_1.getDebug)('max');
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ exports.getMemo = getMemo;
8
+ var _buildUrl = _interopRequireDefault(require("build-url"));
9
+ var _moment = _interopRequireDefault(require("moment"));
10
+ var _constants = require("../constants");
11
+ var _dates = _interopRequireDefault(require("../helpers/dates"));
12
+ var _debug = require("../helpers/debug");
13
+ var _elementsInteractions = require("../helpers/elements-interactions");
14
+ var _fetch = require("../helpers/fetch");
15
+ var _navigation = require("../helpers/navigation");
16
+ var _transactions = require("../helpers/transactions");
17
+ var _transactions2 = require("../transactions");
18
+ var _baseScraperWithBrowser = require("./base-scraper-with-browser");
19
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
20
+ const debug = (0, _debug.getDebug)('max');
19
21
  const BASE_API_ACTIONS_URL = 'https://onlinelcapi.max.co.il';
20
22
  const BASE_WELCOME_URL = 'https://www.max.co.il';
21
23
  const LOGIN_URL = `${BASE_WELCOME_URL}/homepage/welcome`;
22
24
  const PASSWORD_EXPIRED_URL = `${BASE_WELCOME_URL}/renew-password`;
23
25
  const SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;
24
- var MaxPlanName;
25
- (function (MaxPlanName) {
26
- MaxPlanName["Normal"] = "\u05E8\u05D2\u05D9\u05DC\u05D4";
27
- MaxPlanName["ImmediateCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05E2\u05E1\u05E7\u05D5\u05EA \u05DE\u05D9\u05D9\u05D3\u05D9";
28
- MaxPlanName["InternetShopping"] = "\u05D0\u05D9\u05E0\u05D8\u05E8\u05E0\u05D8/\u05D7\u05D5\"\u05DC";
29
- MaxPlanName["Installments"] = "\u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
30
- MaxPlanName["MonthlyCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05D7\u05D5\u05D3\u05E9\u05D9";
31
- MaxPlanName["OneMonthPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9";
32
- MaxPlanName["MonthlyPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05DC\u05D7\u05D9\u05D5\u05D1 \u05D4\u05D7\u05D5\u05D3\u05E9\u05D9";
33
- MaxPlanName["MonthlyPayment"] = "\u05EA\u05E9\u05DC\u05D5\u05DD \u05D7\u05D5\u05D3\u05E9\u05D9";
34
- MaxPlanName["FuturePurchaseFinancing"] = "\u05DE\u05D9\u05DE\u05D5\u05DF \u05DC\u05E8\u05DB\u05D9\u05E9\u05D4 \u05E2\u05EA\u05D9\u05D3\u05D9\u05EA";
35
- MaxPlanName["MonthlyPostponedInstallments"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9 \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
36
- MaxPlanName["ThirtyDaysPlus"] = "\u05E2\u05E1\u05E7\u05EA 30 \u05E4\u05DC\u05D5\u05E1";
37
- MaxPlanName["TwoMonthsPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9\u05D9\u05D9\u05DD";
38
- MaxPlanName["TwoMonthsPostponed2"] = "\u05D3\u05D7\u05D5\u05D9 2 \u05D7' \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
39
- MaxPlanName["MonthlyChargePlusInterest"] = "\u05D7\u05D5\u05D3\u05E9\u05D9 + \u05E8\u05D9\u05D1\u05D9\u05EA";
40
- MaxPlanName["Credit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8";
41
- MaxPlanName["CreditOutsideTheLimit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8-\u05DE\u05D7\u05D5\u05E5 \u05DC\u05DE\u05E1\u05D2\u05E8\u05EA";
42
- MaxPlanName["AccumulatingBasket"] = "\u05E1\u05DC \u05DE\u05E6\u05D8\u05D1\u05E8";
43
- MaxPlanName["PostponedTransactionInstallments"] = "\u05E4\u05E8\u05D9\u05E1\u05EA \u05D4\u05E2\u05E1\u05E7\u05D4 \u05D4\u05D3\u05D7\u05D5\u05D9\u05D4";
44
- MaxPlanName["ReplacementCard"] = "\u05DB\u05E8\u05D8\u05D9\u05E1 \u05D7\u05DC\u05D9\u05E4\u05D9";
45
- MaxPlanName["EarlyRepayment"] = "\u05E4\u05E8\u05E2\u05D5\u05DF \u05DE\u05D5\u05E7\u05D3\u05DD";
46
- MaxPlanName["MonthlyCardFee"] = "\u05D3\u05DE\u05D9 \u05DB\u05E8\u05D8\u05D9\u05E1";
47
- MaxPlanName["CurrencyPocket"] = "\u05D7\u05D9\u05D5\u05D1 \u05D0\u05E8\u05E0\u05E7 \u05DE\u05D8\u05D7";
48
- })(MaxPlanName || (MaxPlanName = {}));
26
+ var MaxPlanName = /*#__PURE__*/function (MaxPlanName) {
27
+ MaxPlanName["Normal"] = "\u05E8\u05D2\u05D9\u05DC\u05D4";
28
+ MaxPlanName["ImmediateCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05E2\u05E1\u05E7\u05D5\u05EA \u05DE\u05D9\u05D9\u05D3\u05D9";
29
+ MaxPlanName["InternetShopping"] = "\u05D0\u05D9\u05E0\u05D8\u05E8\u05E0\u05D8/\u05D7\u05D5\"\u05DC";
30
+ MaxPlanName["Installments"] = "\u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
31
+ MaxPlanName["MonthlyCharge"] = "\u05D7\u05D9\u05D5\u05D1 \u05D7\u05D5\u05D3\u05E9\u05D9";
32
+ MaxPlanName["OneMonthPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9";
33
+ MaxPlanName["MonthlyPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05DC\u05D7\u05D9\u05D5\u05D1 \u05D4\u05D7\u05D5\u05D3\u05E9\u05D9";
34
+ MaxPlanName["MonthlyPayment"] = "\u05EA\u05E9\u05DC\u05D5\u05DD \u05D7\u05D5\u05D3\u05E9\u05D9";
35
+ MaxPlanName["FuturePurchaseFinancing"] = "\u05DE\u05D9\u05DE\u05D5\u05DF \u05DC\u05E8\u05DB\u05D9\u05E9\u05D4 \u05E2\u05EA\u05D9\u05D3\u05D9\u05EA";
36
+ MaxPlanName["MonthlyPostponedInstallments"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9 \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
37
+ MaxPlanName["ThirtyDaysPlus"] = "\u05E2\u05E1\u05E7\u05EA 30 \u05E4\u05DC\u05D5\u05E1";
38
+ MaxPlanName["TwoMonthsPostponed"] = "\u05D3\u05D7\u05D5\u05D9 \u05D7\u05D5\u05D3\u05E9\u05D9\u05D9\u05DD";
39
+ MaxPlanName["TwoMonthsPostponed2"] = "\u05D3\u05D7\u05D5\u05D9 2 \u05D7' \u05EA\u05E9\u05DC\u05D5\u05DE\u05D9\u05DD";
40
+ MaxPlanName["MonthlyChargePlusInterest"] = "\u05D7\u05D5\u05D3\u05E9\u05D9 + \u05E8\u05D9\u05D1\u05D9\u05EA";
41
+ MaxPlanName["Credit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8";
42
+ MaxPlanName["CreditOutsideTheLimit"] = "\u05E7\u05E8\u05D3\u05D9\u05D8-\u05DE\u05D7\u05D5\u05E5 \u05DC\u05DE\u05E1\u05D2\u05E8\u05EA";
43
+ MaxPlanName["AccumulatingBasket"] = "\u05E1\u05DC \u05DE\u05E6\u05D8\u05D1\u05E8";
44
+ MaxPlanName["PostponedTransactionInstallments"] = "\u05E4\u05E8\u05D9\u05E1\u05EA \u05D4\u05E2\u05E1\u05E7\u05D4 \u05D4\u05D3\u05D7\u05D5\u05D9\u05D4";
45
+ MaxPlanName["ReplacementCard"] = "\u05DB\u05E8\u05D8\u05D9\u05E1 \u05D7\u05DC\u05D9\u05E4\u05D9";
46
+ MaxPlanName["EarlyRepayment"] = "\u05E4\u05E8\u05E2\u05D5\u05DF \u05DE\u05D5\u05E7\u05D3\u05DD";
47
+ MaxPlanName["MonthlyCardFee"] = "\u05D3\u05DE\u05D9 \u05DB\u05E8\u05D8\u05D9\u05E1";
48
+ MaxPlanName["CurrencyPocket"] = "\u05D7\u05D9\u05D5\u05D1 \u05D0\u05E8\u05E0\u05E7 \u05DE\u05D8\u05D7";
49
+ return MaxPlanName;
50
+ }(MaxPlanName || {});
49
51
  const INVALID_DETAILS_SELECTOR = '#popupWrongDetails';
50
52
  const LOGIN_ERROR_SELECTOR = '#popupCardHoldersLoginError';
51
53
  const categories = new Map();
52
54
  function redirectOrDialog(page) {
53
- return Promise.race([
54
- (0, navigation_1.waitForRedirect)(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]),
55
- (0, elements_interactions_1.waitUntilElementFound)(page, INVALID_DETAILS_SELECTOR, true),
56
- (0, elements_interactions_1.waitUntilElementFound)(page, LOGIN_ERROR_SELECTOR, true),
57
- ]);
55
+ return Promise.race([(0, _navigation.waitForRedirect)(page, 20000, false, [BASE_WELCOME_URL, `${BASE_WELCOME_URL}/`]), (0, _elementsInteractions.waitUntilElementFound)(page, INVALID_DETAILS_SELECTOR, true), (0, _elementsInteractions.waitUntilElementFound)(page, LOGIN_ERROR_SELECTOR, true)]);
58
56
  }
59
57
  function getTransactionsUrl(monthMoment) {
60
- const month = monthMoment.month() + 1;
61
- const year = monthMoment.year();
62
- const date = `${year}-${month}-01`;
63
- /**
64
- * url explanation:
65
- * userIndex: -1 for all account owners
66
- * cardIndex: -1 for all cards under the account
67
- * all other query params are static, beside the date which changes for request per month
68
- */
69
- return (0, build_url_1.default)(BASE_API_ACTIONS_URL, {
70
- 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`,
71
- });
58
+ const month = monthMoment.month() + 1;
59
+ const year = monthMoment.year();
60
+ const date = `${year}-${month}-01`;
61
+
62
+ /**
63
+ * url explanation:
64
+ * userIndex: -1 for all account owners
65
+ * cardIndex: -1 for all cards under the account
66
+ * all other query params are static, beside the date which changes for request per month
67
+ */
68
+ return (0, _buildUrl.default)(BASE_API_ACTIONS_URL, {
69
+ 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`
70
+ });
72
71
  }
73
72
  async function loadCategories(page) {
74
- debug('Loading categories');
75
- const res = await (0, fetch_1.fetchGetWithinPage)(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);
76
- if (res && Array.isArray(res.result)) {
77
- debug(`${res.result.length} categories loaded`);
78
- res.result?.forEach(({ id, name }) => categories.set(id, name));
79
- }
73
+ debug('Loading categories');
74
+ const res = await (0, _fetch.fetchGetWithinPage)(page, `${BASE_API_ACTIONS_URL}/api/contents/getCategories`);
75
+ if (res && Array.isArray(res.result)) {
76
+ debug(`${res.result.length} categories loaded`);
77
+ res.result?.forEach(({
78
+ id,
79
+ name
80
+ }) => categories.set(id, name));
81
+ }
80
82
  }
81
83
  function getTransactionType(planName, planTypeId) {
82
- const cleanedUpTxnTypeStr = planName.replace('\t', ' ').trim();
83
- switch (cleanedUpTxnTypeStr) {
84
- case MaxPlanName.ImmediateCharge:
85
- case MaxPlanName.Normal:
86
- case MaxPlanName.MonthlyCharge:
87
- case MaxPlanName.OneMonthPostponed:
88
- case MaxPlanName.MonthlyPostponed:
89
- case MaxPlanName.FuturePurchaseFinancing:
90
- case MaxPlanName.MonthlyPayment:
91
- case MaxPlanName.MonthlyPostponedInstallments:
92
- case MaxPlanName.ThirtyDaysPlus:
93
- case MaxPlanName.TwoMonthsPostponed:
94
- case MaxPlanName.TwoMonthsPostponed2:
95
- case MaxPlanName.AccumulatingBasket:
96
- case MaxPlanName.InternetShopping:
97
- case MaxPlanName.MonthlyChargePlusInterest:
98
- case MaxPlanName.PostponedTransactionInstallments:
99
- case MaxPlanName.ReplacementCard:
100
- case MaxPlanName.EarlyRepayment:
101
- case MaxPlanName.MonthlyCardFee:
102
- case MaxPlanName.CurrencyPocket:
103
- return transactions_2.TransactionTypes.Normal;
104
- case MaxPlanName.Installments:
105
- case MaxPlanName.Credit:
106
- case MaxPlanName.CreditOutsideTheLimit:
107
- return transactions_2.TransactionTypes.Installments;
84
+ const cleanedUpTxnTypeStr = planName.replace('\t', ' ').trim();
85
+ switch (cleanedUpTxnTypeStr) {
86
+ case MaxPlanName.ImmediateCharge:
87
+ case MaxPlanName.Normal:
88
+ case MaxPlanName.MonthlyCharge:
89
+ case MaxPlanName.OneMonthPostponed:
90
+ case MaxPlanName.MonthlyPostponed:
91
+ case MaxPlanName.FuturePurchaseFinancing:
92
+ case MaxPlanName.MonthlyPayment:
93
+ case MaxPlanName.MonthlyPostponedInstallments:
94
+ case MaxPlanName.ThirtyDaysPlus:
95
+ case MaxPlanName.TwoMonthsPostponed:
96
+ case MaxPlanName.TwoMonthsPostponed2:
97
+ case MaxPlanName.AccumulatingBasket:
98
+ case MaxPlanName.InternetShopping:
99
+ case MaxPlanName.MonthlyChargePlusInterest:
100
+ case MaxPlanName.PostponedTransactionInstallments:
101
+ case MaxPlanName.ReplacementCard:
102
+ case MaxPlanName.EarlyRepayment:
103
+ case MaxPlanName.MonthlyCardFee:
104
+ case MaxPlanName.CurrencyPocket:
105
+ return _transactions2.TransactionTypes.Normal;
106
+ case MaxPlanName.Installments:
107
+ case MaxPlanName.Credit:
108
+ case MaxPlanName.CreditOutsideTheLimit:
109
+ return _transactions2.TransactionTypes.Installments;
110
+ default:
111
+ switch (planTypeId) {
112
+ case 2:
113
+ case 3:
114
+ return _transactions2.TransactionTypes.Installments;
115
+ case 5:
116
+ return _transactions2.TransactionTypes.Normal;
108
117
  default:
109
- switch (planTypeId) {
110
- case 2:
111
- case 3:
112
- return transactions_2.TransactionTypes.Installments;
113
- case 5:
114
- return transactions_2.TransactionTypes.Normal;
115
- default:
116
- throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);
117
- }
118
- }
118
+ throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr}`);
119
+ }
120
+ }
119
121
  }
120
122
  function getInstallmentsInfo(comments) {
121
- if (!comments) {
122
- return undefined;
123
- }
124
- const matches = comments.match(/\d+/g);
125
- if (!matches || matches.length < 2) {
126
- return undefined;
127
- }
128
- return {
129
- number: parseInt(matches[0], 10),
130
- total: parseInt(matches[1], 10),
131
- };
123
+ if (!comments) {
124
+ return undefined;
125
+ }
126
+ const matches = comments.match(/\d+/g);
127
+ if (!matches || matches.length < 2) {
128
+ return undefined;
129
+ }
130
+ return {
131
+ number: parseInt(matches[0], 10),
132
+ total: parseInt(matches[1], 10)
133
+ };
132
134
  }
133
135
  function getChargedCurrency(currencyId) {
134
- switch (currencyId) {
135
- case 376:
136
- return constants_1.SHEKEL_CURRENCY;
137
- case 840:
138
- return constants_1.DOLLAR_CURRENCY;
139
- case 978:
140
- return constants_1.EURO_CURRENCY;
141
- default:
142
- return undefined;
143
- }
136
+ switch (currencyId) {
137
+ case 376:
138
+ return _constants.SHEKEL_CURRENCY;
139
+ case 840:
140
+ return _constants.DOLLAR_CURRENCY;
141
+ case 978:
142
+ return _constants.EURO_CURRENCY;
143
+ default:
144
+ return undefined;
145
+ }
144
146
  }
145
- function getMemo({ comments, fundsTransferReceiverOrTransfer, fundsTransferComment, }) {
146
- if (fundsTransferReceiverOrTransfer) {
147
- const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;
148
- return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;
149
- }
150
- return comments;
147
+ function getMemo({
148
+ comments,
149
+ fundsTransferReceiverOrTransfer,
150
+ fundsTransferComment
151
+ }) {
152
+ if (fundsTransferReceiverOrTransfer) {
153
+ const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;
154
+ return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;
155
+ }
156
+ return comments;
151
157
  }
152
- exports.getMemo = getMemo;
153
158
  function mapTransaction(rawTransaction) {
154
- const isPending = rawTransaction.paymentDate === null;
155
- const processedDate = (0, moment_1.default)(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();
156
- const status = isPending ? transactions_2.TransactionStatuses.Pending : transactions_2.TransactionStatuses.Completed;
157
- const installments = getInstallmentsInfo(rawTransaction.comments);
158
- const identifier = installments
159
- ? `${rawTransaction.dealData?.arn}_${installments.number}`
160
- : rawTransaction.dealData?.arn;
161
- return {
162
- type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),
163
- date: (0, moment_1.default)(rawTransaction.purchaseDate).toISOString(),
164
- processedDate,
165
- originalAmount: -rawTransaction.originalAmount,
166
- originalCurrency: rawTransaction.originalCurrency,
167
- chargedAmount: -rawTransaction.actualPaymentAmount,
168
- chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),
169
- description: rawTransaction.merchantName.trim(),
170
- memo: getMemo(rawTransaction),
171
- category: categories.get(rawTransaction?.categoryId),
172
- installments,
173
- identifier,
174
- status,
175
- };
159
+ const isPending = rawTransaction.paymentDate === null;
160
+ const processedDate = (0, _moment.default)(isPending ? rawTransaction.purchaseDate : rawTransaction.paymentDate).toISOString();
161
+ const status = isPending ? _transactions2.TransactionStatuses.Pending : _transactions2.TransactionStatuses.Completed;
162
+ const installments = getInstallmentsInfo(rawTransaction.comments);
163
+ const identifier = installments ? `${rawTransaction.dealData?.arn}_${installments.number}` : rawTransaction.dealData?.arn;
164
+ return {
165
+ type: getTransactionType(rawTransaction.planName, rawTransaction.planTypeId),
166
+ date: (0, _moment.default)(rawTransaction.purchaseDate).toISOString(),
167
+ processedDate,
168
+ originalAmount: -rawTransaction.originalAmount,
169
+ originalCurrency: rawTransaction.originalCurrency,
170
+ chargedAmount: -rawTransaction.actualPaymentAmount,
171
+ chargedCurrency: getChargedCurrency(rawTransaction.paymentCurrency),
172
+ description: rawTransaction.merchantName.trim(),
173
+ memo: getMemo(rawTransaction),
174
+ category: categories.get(rawTransaction?.categoryId),
175
+ installments,
176
+ identifier,
177
+ status
178
+ };
176
179
  }
177
180
  async function fetchTransactionsForMonth(page, monthMoment) {
178
- const url = getTransactionsUrl(monthMoment);
179
- const data = await (0, fetch_1.fetchGetWithinPage)(page, url);
180
- const transactionsByAccount = {};
181
- if (!data || !data.result)
182
- return transactionsByAccount;
183
- data.result.transactions
184
- // Filter out non-transactions without a plan type, e.g. summary rows
185
- .filter(transaction => !!transaction.planName)
186
- .forEach((transaction) => {
187
- if (!transactionsByAccount[transaction.shortCardNumber]) {
188
- transactionsByAccount[transaction.shortCardNumber] = [];
189
- }
190
- const mappedTransaction = mapTransaction(transaction);
191
- transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);
192
- });
193
- return transactionsByAccount;
181
+ const url = getTransactionsUrl(monthMoment);
182
+ const data = await (0, _fetch.fetchGetWithinPage)(page, url);
183
+ const transactionsByAccount = {};
184
+ if (!data || !data.result) return transactionsByAccount;
185
+ data.result.transactions
186
+ // Filter out non-transactions without a plan type, e.g. summary rows
187
+ .filter(transaction => !!transaction.planName).forEach(transaction => {
188
+ if (!transactionsByAccount[transaction.shortCardNumber]) {
189
+ transactionsByAccount[transaction.shortCardNumber] = [];
190
+ }
191
+ const mappedTransaction = mapTransaction(transaction);
192
+ transactionsByAccount[transaction.shortCardNumber].push(mappedTransaction);
193
+ });
194
+ return transactionsByAccount;
194
195
  }
195
196
  function addResult(allResults, result) {
196
- const clonedResults = { ...allResults };
197
- Object.keys(result).forEach(accountNumber => {
198
- if (!clonedResults[accountNumber]) {
199
- clonedResults[accountNumber] = [];
200
- }
201
- clonedResults[accountNumber].push(...result[accountNumber]);
202
- });
203
- return clonedResults;
197
+ const clonedResults = {
198
+ ...allResults
199
+ };
200
+ Object.keys(result).forEach(accountNumber => {
201
+ if (!clonedResults[accountNumber]) {
202
+ clonedResults[accountNumber] = [];
203
+ }
204
+ clonedResults[accountNumber].push(...result[accountNumber]);
205
+ });
206
+ return clonedResults;
204
207
  }
205
208
  function prepareTransactions(txns, startMoment, combineInstallments, enableTransactionsFilterByDate) {
206
- let clonedTxns = Array.from(txns);
207
- if (!combineInstallments) {
208
- clonedTxns = (0, transactions_1.fixInstallments)(clonedTxns);
209
- }
210
- clonedTxns = (0, transactions_1.sortTransactionsByDate)(clonedTxns);
211
- clonedTxns = enableTransactionsFilterByDate
212
- ? (0, transactions_1.filterOldTransactions)(clonedTxns, startMoment, combineInstallments || false)
213
- : clonedTxns;
214
- return clonedTxns;
209
+ let clonedTxns = Array.from(txns);
210
+ if (!combineInstallments) {
211
+ clonedTxns = (0, _transactions.fixInstallments)(clonedTxns);
212
+ }
213
+ clonedTxns = (0, _transactions.sortTransactionsByDate)(clonedTxns);
214
+ clonedTxns = enableTransactionsFilterByDate ? (0, _transactions.filterOldTransactions)(clonedTxns, startMoment, combineInstallments || false) : clonedTxns;
215
+ return clonedTxns;
215
216
  }
216
217
  async function fetchTransactions(page, options) {
217
- const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;
218
- const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years');
219
- const startMomentLimit = (0, moment_1.default)().subtract(4, 'years');
220
- const startDate = options.startDate || defaultStartMoment.toDate();
221
- const startMoment = moment_1.default.max(startMomentLimit, (0, moment_1.default)(startDate));
222
- const allMonths = (0, dates_1.default)(startMoment, futureMonthsToScrape);
223
- await loadCategories(page);
224
- let allResults = {};
225
- for (let i = 0; i < allMonths.length; i += 1) {
226
- const result = await fetchTransactionsForMonth(page, allMonths[i]);
227
- allResults = addResult(allResults, result);
228
- }
229
- Object.keys(allResults).forEach(accountNumber => {
230
- let txns = allResults[accountNumber];
231
- txns = prepareTransactions(txns, startMoment, options.combineInstallments || false, options.outputData?.enableTransactionsFilterByDate ?? true);
232
- allResults[accountNumber] = txns;
233
- });
234
- return allResults;
218
+ const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;
219
+ const defaultStartMoment = (0, _moment.default)().subtract(1, 'years');
220
+ const startMomentLimit = (0, _moment.default)().subtract(4, 'years');
221
+ const startDate = options.startDate || defaultStartMoment.toDate();
222
+ const startMoment = _moment.default.max(startMomentLimit, (0, _moment.default)(startDate));
223
+ const allMonths = (0, _dates.default)(startMoment, futureMonthsToScrape);
224
+ await loadCategories(page);
225
+ let allResults = {};
226
+ for (let i = 0; i < allMonths.length; i += 1) {
227
+ const result = await fetchTransactionsForMonth(page, allMonths[i]);
228
+ allResults = addResult(allResults, result);
229
+ }
230
+ Object.keys(allResults).forEach(accountNumber => {
231
+ let txns = allResults[accountNumber];
232
+ txns = prepareTransactions(txns, startMoment, options.combineInstallments || false, options.outputData?.enableTransactionsFilterByDate ?? true);
233
+ allResults[accountNumber] = txns;
234
+ });
235
+ return allResults;
235
236
  }
236
237
  function getPossibleLoginResults(page) {
237
- const urls = {};
238
- urls[base_scraper_with_browser_1.LoginResults.Success] = [SUCCESS_URL];
239
- urls[base_scraper_with_browser_1.LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];
240
- urls[base_scraper_with_browser_1.LoginResults.InvalidPassword] = [
241
- async () => {
242
- return (0, elements_interactions_1.elementPresentOnPage)(page, INVALID_DETAILS_SELECTOR);
243
- },
244
- ];
245
- urls[base_scraper_with_browser_1.LoginResults.UnknownError] = [
246
- async () => {
247
- return (0, elements_interactions_1.elementPresentOnPage)(page, LOGIN_ERROR_SELECTOR);
248
- },
249
- ];
250
- return urls;
238
+ const urls = {};
239
+ urls[_baseScraperWithBrowser.LoginResults.Success] = [SUCCESS_URL];
240
+ urls[_baseScraperWithBrowser.LoginResults.ChangePassword] = [PASSWORD_EXPIRED_URL];
241
+ urls[_baseScraperWithBrowser.LoginResults.InvalidPassword] = [async () => {
242
+ return (0, _elementsInteractions.elementPresentOnPage)(page, INVALID_DETAILS_SELECTOR);
243
+ }];
244
+ urls[_baseScraperWithBrowser.LoginResults.UnknownError] = [async () => {
245
+ return (0, _elementsInteractions.elementPresentOnPage)(page, LOGIN_ERROR_SELECTOR);
246
+ }];
247
+ return urls;
251
248
  }
252
249
  function createLoginFields(credentials) {
253
- return [
254
- { selector: '#user-name', value: credentials.username },
255
- { selector: '#password', value: credentials.password },
256
- ];
250
+ return [{
251
+ selector: '#user-name',
252
+ value: credentials.username
253
+ }, {
254
+ selector: '#password',
255
+ value: credentials.password
256
+ }];
257
257
  }
258
- class MaxScraper extends base_scraper_with_browser_1.BaseScraperWithBrowser {
259
- getLoginOptions(credentials) {
260
- return {
261
- loginUrl: LOGIN_URL,
262
- fields: createLoginFields(credentials),
263
- submitButtonSelector: 'app-user-login-form .general-button.send-me-code',
264
- preAction: async () => {
265
- if (await (0, elements_interactions_1.elementPresentOnPage)(this.page, '#closePopup')) {
266
- await (0, elements_interactions_1.clickButton)(this.page, '#closePopup');
267
- }
268
- await (0, elements_interactions_1.clickButton)(this.page, '.personal-area > a.go-to-personal-area');
269
- if (await (0, elements_interactions_1.elementPresentOnPage)(this.page, '.login-link#private')) {
270
- await (0, elements_interactions_1.clickButton)(this.page, '.login-link#private');
271
- }
272
- await (0, elements_interactions_1.waitUntilElementFound)(this.page, '#login-password-link', true);
273
- await (0, elements_interactions_1.clickButton)(this.page, '#login-password-link');
274
- await (0, elements_interactions_1.waitUntilElementFound)(this.page, '#login-password.tab-pane.active app-user-login-form', true);
275
- },
276
- checkReadiness: async () => {
277
- await (0, elements_interactions_1.waitUntilElementFound)(this.page, '.personal-area > a.go-to-personal-area', true);
278
- },
279
- postAction: async () => redirectOrDialog(this.page),
280
- possibleResults: getPossibleLoginResults(this.page),
281
- waitUntil: 'domcontentloaded',
282
- };
283
- }
284
- async fetchData() {
285
- const results = await fetchTransactions(this.page, this.options);
286
- const accounts = Object.keys(results).map(accountNumber => {
287
- return {
288
- accountNumber,
289
- txns: results[accountNumber],
290
- };
291
- });
292
- return {
293
- success: true,
294
- accounts,
295
- };
296
- }
258
+ class MaxScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
259
+ getLoginOptions(credentials) {
260
+ return {
261
+ loginUrl: LOGIN_URL,
262
+ fields: createLoginFields(credentials),
263
+ submitButtonSelector: 'app-user-login-form .general-button.send-me-code',
264
+ preAction: async () => {
265
+ if (await (0, _elementsInteractions.elementPresentOnPage)(this.page, '#closePopup')) {
266
+ await (0, _elementsInteractions.clickButton)(this.page, '#closePopup');
267
+ }
268
+ await (0, _elementsInteractions.clickButton)(this.page, '.personal-area > a.go-to-personal-area');
269
+ if (await (0, _elementsInteractions.elementPresentOnPage)(this.page, '.login-link#private')) {
270
+ await (0, _elementsInteractions.clickButton)(this.page, '.login-link#private');
271
+ }
272
+ await (0, _elementsInteractions.waitUntilElementFound)(this.page, '#login-password-link', true);
273
+ await (0, _elementsInteractions.clickButton)(this.page, '#login-password-link');
274
+ await (0, _elementsInteractions.waitUntilElementFound)(this.page, '#login-password.tab-pane.active app-user-login-form', true);
275
+ },
276
+ checkReadiness: async () => {
277
+ await (0, _elementsInteractions.waitUntilElementFound)(this.page, '.personal-area > a.go-to-personal-area', true);
278
+ },
279
+ postAction: async () => redirectOrDialog(this.page),
280
+ possibleResults: getPossibleLoginResults(this.page),
281
+ waitUntil: 'domcontentloaded'
282
+ };
283
+ }
284
+ async fetchData() {
285
+ const results = await fetchTransactions(this.page, this.options);
286
+ const accounts = Object.keys(results).map(accountNumber => {
287
+ return {
288
+ accountNumber,
289
+ txns: results[accountNumber]
290
+ };
291
+ });
292
+ return {
293
+ success: true,
294
+ accounts
295
+ };
296
+ }
297
297
  }
298
- exports.default = MaxScraper;
299
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"max.js","sourceRoot":"","sources":["../../src/scrapers/max.ts"],"names":[],"mappings":"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","sourcesContent":["import buildUrl from 'build-url';\nimport moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { DOLLAR_CURRENCY, EURO_CURRENCY, SHEKEL_CURRENCY } from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { filterOldTransactions, fixInstallments, sortTransactionsByDate } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport {\n  BaseScraperWithBrowser,\n  LoginResults,\n  type LoginOptions,\n  type PossibleLoginResults,\n} from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\nexport interface 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  planTypeId: number;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  fundsTransferComment?: string;\n  fundsTransferReceiverOrTransfer?: string;\n  dealData?: {\n    arn: string;\n  };\n}\n\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_WELCOME_URL}/renew-password`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nenum MaxPlanName {\n  Normal = 'רגילה',\n  ImmediateCharge = 'חיוב עסקות מיידי',\n  InternetShopping = 'אינטרנט/חו\"ל',\n  Installments = 'תשלומים',\n  MonthlyCharge = 'חיוב חודשי',\n  OneMonthPostponed = 'דחוי חודש',\n  MonthlyPostponed = 'דחוי לחיוב החודשי',\n  MonthlyPayment = 'תשלום חודשי',\n  FuturePurchaseFinancing = 'מימון לרכישה עתידית',\n  MonthlyPostponedInstallments = 'דחוי חודש תשלומים',\n  ThirtyDaysPlus = 'עסקת 30 פלוס',\n  TwoMonthsPostponed = 'דחוי חודשיים',\n  TwoMonthsPostponed2 = \"דחוי 2 ח' תשלומים\",\n  MonthlyChargePlusInterest = 'חודשי + ריבית',\n  Credit = 'קרדיט',\n  CreditOutsideTheLimit = 'קרדיט-מחוץ למסגרת',\n  AccumulatingBasket = 'סל מצטבר',\n  PostponedTransactionInstallments = 'פריסת העסקה הדחויה',\n  ReplacementCard = 'כרטיס חליפי',\n  EarlyRepayment = 'פרעון מוקדם',\n  MonthlyCardFee = 'דמי כרטיס',\n  CurrencyPocket = 'חיוב ארנק מטח',\n}\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(planName: string, planTypeId: number) {\n  const cleanedUpTxnTypeStr = planName.replace('\\t', ' ').trim() as MaxPlanName;\n  switch (cleanedUpTxnTypeStr) {\n    case MaxPlanName.ImmediateCharge:\n    case MaxPlanName.Normal:\n    case MaxPlanName.MonthlyCharge:\n    case MaxPlanName.OneMonthPostponed:\n    case MaxPlanName.MonthlyPostponed:\n    case MaxPlanName.FuturePurchaseFinancing:\n    case MaxPlanName.MonthlyPayment:\n    case MaxPlanName.MonthlyPostponedInstallments:\n    case MaxPlanName.ThirtyDaysPlus:\n    case MaxPlanName.TwoMonthsPostponed:\n    case MaxPlanName.TwoMonthsPostponed2:\n    case MaxPlanName.AccumulatingBasket:\n    case MaxPlanName.InternetShopping:\n    case MaxPlanName.MonthlyChargePlusInterest:\n    case MaxPlanName.PostponedTransactionInstallments:\n    case MaxPlanName.ReplacementCard:\n    case MaxPlanName.EarlyRepayment:\n    case MaxPlanName.MonthlyCardFee:\n    case MaxPlanName.CurrencyPocket:\n      return TransactionTypes.Normal;\n    case MaxPlanName.Installments:\n    case MaxPlanName.Credit:\n    case MaxPlanName.CreditOutsideTheLimit:\n      return TransactionTypes.Installments;\n    default:\n      switch (planTypeId) {\n        case 2:\n        case 3:\n          return TransactionTypes.Installments;\n        case 5:\n          return TransactionTypes.Normal;\n        default:\n          throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr as string}`);\n      }\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\nexport function getMemo({\n  comments,\n  fundsTransferReceiverOrTransfer,\n  fundsTransferComment,\n}: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>) {\n  if (fundsTransferReceiverOrTransfer) {\n    const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;\n    return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;\n  }\n\n  return comments;\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ? rawTransaction.purchaseDate : 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, rawTransaction.planTypeId),\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: getMemo(rawTransaction),\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(\n  txns: Transaction[],\n  startMoment: moment.Moment,\n  combineInstallments: boolean,\n  enableTransactionsFilterByDate: boolean,\n) {\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 startMomentLimit = moment().subtract(4, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(startMomentLimit, 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(\n      txns,\n      startMoment,\n      options.combineInstallments || false,\n      options.outputData?.enableTransactionsFilterByDate ?? true,\n    );\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] = [\n    async () => {\n      return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n    },\n  ];\n  urls[LoginResults.UnknownError] = [\n    async () => {\n      return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n    },\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): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'app-user-login-form .general-button.send-me-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        if (await elementPresentOnPage(this.page, '.login-link#private')) {\n          await clickButton(this.page, '.login-link#private');\n        }\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',\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"]}
298
+ var _default = exports.default = MaxScraper;
299
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_buildUrl","_interopRequireDefault","require","_moment","_constants","_dates","_debug","_elementsInteractions","_fetch","_navigation","_transactions","_transactions2","_baseScraperWithBrowser","e","__esModule","default","debug","getDebug","BASE_API_ACTIONS_URL","BASE_WELCOME_URL","LOGIN_URL","PASSWORD_EXPIRED_URL","SUCCESS_URL","MaxPlanName","INVALID_DETAILS_SELECTOR","LOGIN_ERROR_SELECTOR","categories","Map","redirectOrDialog","page","Promise","race","waitForRedirect","waitUntilElementFound","getTransactionsUrl","monthMoment","month","year","date","buildUrl","path","loadCategories","res","fetchGetWithinPage","Array","isArray","result","length","forEach","id","name","set","getTransactionType","planName","planTypeId","cleanedUpTxnTypeStr","replace","trim","ImmediateCharge","Normal","MonthlyCharge","OneMonthPostponed","MonthlyPostponed","FuturePurchaseFinancing","MonthlyPayment","MonthlyPostponedInstallments","ThirtyDaysPlus","TwoMonthsPostponed","TwoMonthsPostponed2","AccumulatingBasket","InternetShopping","MonthlyChargePlusInterest","PostponedTransactionInstallments","ReplacementCard","EarlyRepayment","MonthlyCardFee","CurrencyPocket","TransactionTypes","Installments","Credit","CreditOutsideTheLimit","Error","getInstallmentsInfo","comments","undefined","matches","match","number","parseInt","total","getChargedCurrency","currencyId","SHEKEL_CURRENCY","DOLLAR_CURRENCY","EURO_CURRENCY","getMemo","fundsTransferReceiverOrTransfer","fundsTransferComment","memo","mapTransaction","rawTransaction","isPending","paymentDate","processedDate","moment","purchaseDate","toISOString","status","TransactionStatuses","Pending","Completed","installments","identifier","dealData","arn","type","originalAmount","originalCurrency","chargedAmount","actualPaymentAmount","chargedCurrency","paymentCurrency","description","merchantName","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","fixInstallments","sortTransactionsByDate","filterOldTransactions","fetchTransactions","options","futureMonthsToScrape","defaultStartMoment","subtract","startMomentLimit","startDate","toDate","max","allMonths","getAllMonthMoments","i","outputData","getPossibleLoginResults","urls","LoginResults","Success","ChangePassword","InvalidPassword","elementPresentOnPage","UnknownError","createLoginFields","credentials","selector","value","username","password","MaxScraper","BaseScraperWithBrowser","getLoginOptions","loginUrl","fields","submitButtonSelector","preAction","clickButton","checkReadiness","postAction","possibleResults","waitUntil","fetchData","results","accounts","map","success","_default","exports"],"sources":["../../src/scrapers/max.ts"],"sourcesContent":["import buildUrl from 'build-url';\nimport moment, { type Moment } from 'moment';\nimport { type Page } from 'puppeteer';\nimport { DOLLAR_CURRENCY, EURO_CURRENCY, SHEKEL_CURRENCY } from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, elementPresentOnPage, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { fetchGetWithinPage } from '../helpers/fetch';\nimport { waitForRedirect } from '../helpers/navigation';\nimport { filterOldTransactions, fixInstallments, sortTransactionsByDate } from '../helpers/transactions';\nimport { TransactionStatuses, TransactionTypes, type Transaction } from '../transactions';\nimport {\n  BaseScraperWithBrowser,\n  LoginResults,\n  type LoginOptions,\n  type PossibleLoginResults,\n} from './base-scraper-with-browser';\nimport { type ScraperOptions } from './interface';\n\nconst debug = getDebug('max');\n\nexport interface 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  planTypeId: number;\n  comments: string;\n  merchantName: string;\n  categoryId: number;\n  fundsTransferComment?: string;\n  fundsTransferReceiverOrTransfer?: string;\n  dealData?: {\n    arn: string;\n  };\n}\n\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_WELCOME_URL}/renew-password`;\nconst SUCCESS_URL = `${BASE_WELCOME_URL}/homepage/personal`;\n\nenum MaxPlanName {\n  Normal = 'רגילה',\n  ImmediateCharge = 'חיוב עסקות מיידי',\n  InternetShopping = 'אינטרנט/חו\"ל',\n  Installments = 'תשלומים',\n  MonthlyCharge = 'חיוב חודשי',\n  OneMonthPostponed = 'דחוי חודש',\n  MonthlyPostponed = 'דחוי לחיוב החודשי',\n  MonthlyPayment = 'תשלום חודשי',\n  FuturePurchaseFinancing = 'מימון לרכישה עתידית',\n  MonthlyPostponedInstallments = 'דחוי חודש תשלומים',\n  ThirtyDaysPlus = 'עסקת 30 פלוס',\n  TwoMonthsPostponed = 'דחוי חודשיים',\n  TwoMonthsPostponed2 = \"דחוי 2 ח' תשלומים\",\n  MonthlyChargePlusInterest = 'חודשי + ריבית',\n  Credit = 'קרדיט',\n  CreditOutsideTheLimit = 'קרדיט-מחוץ למסגרת',\n  AccumulatingBasket = 'סל מצטבר',\n  PostponedTransactionInstallments = 'פריסת העסקה הדחויה',\n  ReplacementCard = 'כרטיס חליפי',\n  EarlyRepayment = 'פרעון מוקדם',\n  MonthlyCardFee = 'דמי כרטיס',\n  CurrencyPocket = 'חיוב ארנק מטח',\n}\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(planName: string, planTypeId: number) {\n  const cleanedUpTxnTypeStr = planName.replace('\\t', ' ').trim() as MaxPlanName;\n  switch (cleanedUpTxnTypeStr) {\n    case MaxPlanName.ImmediateCharge:\n    case MaxPlanName.Normal:\n    case MaxPlanName.MonthlyCharge:\n    case MaxPlanName.OneMonthPostponed:\n    case MaxPlanName.MonthlyPostponed:\n    case MaxPlanName.FuturePurchaseFinancing:\n    case MaxPlanName.MonthlyPayment:\n    case MaxPlanName.MonthlyPostponedInstallments:\n    case MaxPlanName.ThirtyDaysPlus:\n    case MaxPlanName.TwoMonthsPostponed:\n    case MaxPlanName.TwoMonthsPostponed2:\n    case MaxPlanName.AccumulatingBasket:\n    case MaxPlanName.InternetShopping:\n    case MaxPlanName.MonthlyChargePlusInterest:\n    case MaxPlanName.PostponedTransactionInstallments:\n    case MaxPlanName.ReplacementCard:\n    case MaxPlanName.EarlyRepayment:\n    case MaxPlanName.MonthlyCardFee:\n    case MaxPlanName.CurrencyPocket:\n      return TransactionTypes.Normal;\n    case MaxPlanName.Installments:\n    case MaxPlanName.Credit:\n    case MaxPlanName.CreditOutsideTheLimit:\n      return TransactionTypes.Installments;\n    default:\n      switch (planTypeId) {\n        case 2:\n        case 3:\n          return TransactionTypes.Installments;\n        case 5:\n          return TransactionTypes.Normal;\n        default:\n          throw new Error(`Unknown transaction type ${cleanedUpTxnTypeStr as string}`);\n      }\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\nexport function getMemo({\n  comments,\n  fundsTransferReceiverOrTransfer,\n  fundsTransferComment,\n}: Pick<ScrapedTransaction, 'comments' | 'fundsTransferReceiverOrTransfer' | 'fundsTransferComment'>) {\n  if (fundsTransferReceiverOrTransfer) {\n    const memo = comments ? `${comments} ${fundsTransferReceiverOrTransfer}` : fundsTransferReceiverOrTransfer;\n    return fundsTransferComment ? `${memo}: ${fundsTransferComment}` : memo;\n  }\n\n  return comments;\n}\n\nfunction mapTransaction(rawTransaction: ScrapedTransaction): Transaction {\n  const isPending = rawTransaction.paymentDate === null;\n  const processedDate = moment(isPending ? rawTransaction.purchaseDate : 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, rawTransaction.planTypeId),\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: getMemo(rawTransaction),\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(\n  txns: Transaction[],\n  startMoment: moment.Moment,\n  combineInstallments: boolean,\n  enableTransactionsFilterByDate: boolean,\n) {\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 startMomentLimit = moment().subtract(4, 'years');\n  const startDate = options.startDate || defaultStartMoment.toDate();\n  const startMoment = moment.max(startMomentLimit, 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(\n      txns,\n      startMoment,\n      options.combineInstallments || false,\n      options.outputData?.enableTransactionsFilterByDate ?? true,\n    );\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] = [\n    async () => {\n      return elementPresentOnPage(page, INVALID_DETAILS_SELECTOR);\n    },\n  ];\n  urls[LoginResults.UnknownError] = [\n    async () => {\n      return elementPresentOnPage(page, LOGIN_ERROR_SELECTOR);\n    },\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): LoginOptions {\n    return {\n      loginUrl: LOGIN_URL,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'app-user-login-form .general-button.send-me-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        if (await elementPresentOnPage(this.page, '.login-link#private')) {\n          await clickButton(this.page, '.login-link#private');\n        }\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',\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"],"mappings":"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","ignoreList":[]}