israeli-bank-scrapers 6.1.0 → 6.1.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.
Files changed (79) hide show
  1. package/lib/assertNever.js +5 -7
  2. package/lib/constants.js +13 -16
  3. package/lib/definitions.js +109 -113
  4. package/lib/helpers/browser.js +9 -16
  5. package/lib/helpers/dates.js +18 -19
  6. package/lib/helpers/debug.js +9 -9
  7. package/lib/helpers/elements-interactions.js +78 -84
  8. package/lib/helpers/fetch.js +82 -89
  9. package/lib/helpers/navigation.js +24 -31
  10. package/lib/helpers/storage.js +10 -12
  11. package/lib/helpers/transactions.js +33 -35
  12. package/lib/helpers/waiting.js +45 -44
  13. package/lib/index.js +15 -82
  14. package/lib/scrapers/amex.js +11 -13
  15. package/lib/scrapers/base-beinleumi-group.js +233 -252
  16. package/lib/scrapers/base-isracard-amex.js +273 -286
  17. package/lib/scrapers/base-scraper-with-browser.js +240 -274
  18. package/lib/scrapers/base-scraper.js +82 -86
  19. package/lib/scrapers/behatsdaa.js +98 -107
  20. package/lib/scrapers/beinleumi.js +11 -20
  21. package/lib/scrapers/beyahad-bishvilha.js +132 -138
  22. package/lib/scrapers/discount.js +97 -103
  23. package/lib/scrapers/errors.js +22 -25
  24. package/lib/scrapers/factory.js +66 -67
  25. package/lib/scrapers/hapoalim.js +162 -182
  26. package/lib/scrapers/interface.js +2 -5
  27. package/lib/scrapers/isracard.js +11 -13
  28. package/lib/scrapers/leumi.js +167 -176
  29. package/lib/scrapers/massad.js +11 -20
  30. package/lib/scrapers/max.js +256 -268
  31. package/lib/scrapers/mercantile.js +14 -20
  32. package/lib/scrapers/mizrahi.js +158 -159
  33. package/lib/scrapers/one-zero-queries.js +4 -7
  34. package/lib/scrapers/one-zero.js +176 -240
  35. package/lib/scrapers/otsar-hahayal.js +11 -20
  36. package/lib/scrapers/pagi.js +11 -20
  37. package/lib/scrapers/union-bank.js +172 -179
  38. package/lib/scrapers/visa-cal.js +254 -263
  39. package/lib/scrapers/yahav.js +190 -211
  40. package/lib/transactions.js +13 -16
  41. package/package.json +12 -14
  42. package/lib/scrapers/amex.test.d.ts +0 -1
  43. package/lib/scrapers/amex.test.js +0 -54
  44. package/lib/scrapers/base-scraper-with-browser.test.d.ts +0 -1
  45. package/lib/scrapers/base-scraper-with-browser.test.js +0 -58
  46. package/lib/scrapers/behatsdaa.test.d.ts +0 -1
  47. package/lib/scrapers/behatsdaa.test.js +0 -50
  48. package/lib/scrapers/beinleumi.test.d.ts +0 -1
  49. package/lib/scrapers/beinleumi.test.js +0 -52
  50. package/lib/scrapers/beyahad-bishvilha.test.d.ts +0 -1
  51. package/lib/scrapers/beyahad-bishvilha.test.js +0 -52
  52. package/lib/scrapers/discount.test.d.ts +0 -1
  53. package/lib/scrapers/discount.test.js +0 -54
  54. package/lib/scrapers/factory.test.d.ts +0 -1
  55. package/lib/scrapers/factory.test.js +0 -19
  56. package/lib/scrapers/hapoalim.test.d.ts +0 -1
  57. package/lib/scrapers/hapoalim.test.js +0 -52
  58. package/lib/scrapers/isracard.test.d.ts +0 -1
  59. package/lib/scrapers/isracard.test.js +0 -54
  60. package/lib/scrapers/leumi.test.d.ts +0 -1
  61. package/lib/scrapers/leumi.test.js +0 -52
  62. package/lib/scrapers/max.test.d.ts +0 -1
  63. package/lib/scrapers/max.test.js +0 -71
  64. package/lib/scrapers/mercantile.test.d.ts +0 -1
  65. package/lib/scrapers/mercantile.test.js +0 -50
  66. package/lib/scrapers/mizrahi.test.d.ts +0 -1
  67. package/lib/scrapers/mizrahi.test.js +0 -58
  68. package/lib/scrapers/one-zero.test.d.ts +0 -1
  69. package/lib/scrapers/one-zero.test.js +0 -56
  70. package/lib/scrapers/otsar-hahayal.test.d.ts +0 -1
  71. package/lib/scrapers/otsar-hahayal.test.js +0 -52
  72. package/lib/scrapers/pagi.test.d.ts +0 -1
  73. package/lib/scrapers/pagi.test.js +0 -52
  74. package/lib/scrapers/union-bank.test.d.ts +0 -1
  75. package/lib/scrapers/union-bank.test.js +0 -52
  76. package/lib/scrapers/visa-cal.test.d.ts +0 -1
  77. package/lib/scrapers/visa-cal.test.js +0 -54
  78. package/lib/scrapers/yahav.test.d.ts +0 -1
  79. package/lib/scrapers/yahav.test.js +0 -54
@@ -1,257 +1,193 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.default = void 0;
7
- require("core-js/modules/es.symbol.description.js");
8
- require("core-js/modules/es.array.flat-map.js");
9
- require("core-js/modules/es.array.iterator.js");
10
- require("core-js/modules/es.array.sort.js");
11
- require("core-js/modules/es.array.unscopables.flat-map.js");
12
- require("core-js/modules/es.promise.js");
13
- require("core-js/modules/es.regexp.exec.js");
14
- require("core-js/modules/es.string.replace.js");
15
- require("core-js/modules/es.string.trim.js");
16
- require("core-js/modules/esnext.string.match-all.js");
17
- var _moment = _interopRequireDefault(require("moment/moment"));
18
- var _debug = require("../helpers/debug");
19
- var _fetch = require("../helpers/fetch");
20
- var _transactions = require("../transactions");
21
- var _baseScraper = require("./base-scraper");
22
- var _errors = require("./errors");
23
- var _oneZeroQueries = require("./one-zero-queries");
24
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
25
- function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
26
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
27
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
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
+ const moment_1 = __importDefault(require("moment/moment"));
7
+ const debug_1 = require("../helpers/debug");
8
+ const fetch_1 = require("../helpers/fetch");
9
+ const transactions_1 = require("../transactions");
10
+ const base_scraper_1 = require("./base-scraper");
11
+ const errors_1 = require("./errors");
12
+ const one_zero_queries_1 = require("./one-zero-queries");
28
13
  const HEBREW_WORDS_REGEX = /[\u0590-\u05FF][\u0590-\u05FF"'\-_ /\\]*[\u0590-\u05FF]/g;
29
- const debug = (0, _debug.getDebug)('one-zero');
14
+ const debug = (0, debug_1.getDebug)('one-zero');
30
15
  const IDENTITY_SERVER_URL = 'https://identity.tfd-bank.com/v1/';
31
16
  const GRAPHQL_API_URL = 'https://mobile.tfd-bank.com/mobile-graph/graphql';
32
- class OneZeroScraper extends _baseScraper.BaseScraper {
33
- constructor(...args) {
34
- super(...args);
35
- _defineProperty(this, "otpContext", void 0);
36
- _defineProperty(this, "accessToken", void 0);
37
- }
38
- async triggerTwoFactorAuth(phoneNumber) {
39
- if (!phoneNumber.startsWith('+')) {
40
- return (0, _errors.createGenericError)('A full international phone number starting with + and a three digit country code is required');
41
- }
42
- debug('Fetching device token');
43
- const deviceTokenResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/devices/token`, {
44
- extClientId: 'mobile',
45
- os: 'Android'
46
- });
47
- const {
48
- resultData: {
49
- deviceToken
50
- }
51
- } = deviceTokenResponse;
52
- debug(`Sending OTP to phone number ${phoneNumber}`);
53
- const otpPrepareResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/otp/prepare`, {
54
- factorValue: phoneNumber,
55
- deviceToken,
56
- otpChannel: 'SMS_OTP'
57
- });
58
- const {
59
- resultData: {
60
- otpContext
61
- }
62
- } = otpPrepareResponse;
63
- this.otpContext = otpContext;
64
- return {
65
- success: true
66
- };
67
- }
68
- async getLongTermTwoFactorToken(otpCode) {
69
- if (!this.otpContext) {
70
- return (0, _errors.createGenericError)('triggerOtp was not called before calling getPermenantOtpToken()');
71
- }
72
- debug('Requesting OTP token');
73
- const otpVerifyResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/otp/verify`, {
74
- otpContext: this.otpContext,
75
- otpCode
76
- });
77
- const {
78
- resultData: {
79
- otpToken
80
- }
81
- } = otpVerifyResponse;
82
- return {
83
- success: true,
84
- longTermTwoFactorAuthToken: otpToken
85
- };
86
- }
87
- async resolveOtpToken(credentials) {
88
- if ('otpLongTermToken' in credentials) {
89
- if (!credentials.otpLongTermToken) {
90
- return (0, _errors.createGenericError)('Invalid otpLongTermToken');
91
- }
92
- return {
93
- success: true,
94
- longTermTwoFactorAuthToken: credentials.otpLongTermToken
95
- };
96
- }
97
- if (!credentials.otpCodeRetriever) {
98
- return {
99
- success: false,
100
- errorType: _errors.ScraperErrorTypes.TwoFactorRetrieverMissing,
101
- errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided'
102
- };
17
+ class OneZeroScraper extends base_scraper_1.BaseScraper {
18
+ otpContext;
19
+ accessToken;
20
+ async triggerTwoFactorAuth(phoneNumber) {
21
+ if (!phoneNumber.startsWith('+')) {
22
+ return (0, errors_1.createGenericError)('A full international phone number starting with + and a three digit country code is required');
23
+ }
24
+ debug('Fetching device token');
25
+ const deviceTokenResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/devices/token`, {
26
+ extClientId: 'mobile',
27
+ os: 'Android',
28
+ });
29
+ const { resultData: { deviceToken }, } = deviceTokenResponse;
30
+ debug(`Sending OTP to phone number ${phoneNumber}`);
31
+ const otpPrepareResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/otp/prepare`, {
32
+ factorValue: phoneNumber,
33
+ deviceToken,
34
+ otpChannel: 'SMS_OTP',
35
+ });
36
+ const { resultData: { otpContext }, } = otpPrepareResponse;
37
+ this.otpContext = otpContext;
38
+ return {
39
+ success: true,
40
+ };
103
41
  }
104
- if (!credentials.phoneNumber) {
105
- return (0, _errors.createGenericError)('phoneNumber is required when providing a otpCodeRetriever callback');
42
+ async getLongTermTwoFactorToken(otpCode) {
43
+ if (!this.otpContext) {
44
+ return (0, errors_1.createGenericError)('triggerOtp was not called before calling getPermenantOtpToken()');
45
+ }
46
+ debug('Requesting OTP token');
47
+ const otpVerifyResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/otp/verify`, {
48
+ otpContext: this.otpContext,
49
+ otpCode,
50
+ });
51
+ const { resultData: { otpToken }, } = otpVerifyResponse;
52
+ return { success: true, longTermTwoFactorAuthToken: otpToken };
106
53
  }
107
- debug('Triggering user supplied otpCodeRetriever callback');
108
- const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);
109
- if (!triggerResult.success) {
110
- return triggerResult;
54
+ async resolveOtpToken(credentials) {
55
+ if ('otpLongTermToken' in credentials) {
56
+ if (!credentials.otpLongTermToken) {
57
+ return (0, errors_1.createGenericError)('Invalid otpLongTermToken');
58
+ }
59
+ return { success: true, longTermTwoFactorAuthToken: credentials.otpLongTermToken };
60
+ }
61
+ if (!credentials.otpCodeRetriever) {
62
+ return {
63
+ success: false,
64
+ errorType: errors_1.ScraperErrorTypes.TwoFactorRetrieverMissing,
65
+ errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided',
66
+ };
67
+ }
68
+ if (!credentials.phoneNumber) {
69
+ return (0, errors_1.createGenericError)('phoneNumber is required when providing a otpCodeRetriever callback');
70
+ }
71
+ debug('Triggering user supplied otpCodeRetriever callback');
72
+ const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);
73
+ if (!triggerResult.success) {
74
+ return triggerResult;
75
+ }
76
+ const otpCode = await credentials.otpCodeRetriever();
77
+ const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);
78
+ if (!otpTokenResult.success) {
79
+ return otpTokenResult;
80
+ }
81
+ return { success: true, longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken };
111
82
  }
112
- const otpCode = await credentials.otpCodeRetriever();
113
- const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);
114
- if (!otpTokenResult.success) {
115
- return otpTokenResult;
83
+ async login(credentials) {
84
+ const otpTokenResult = await this.resolveOtpToken(credentials);
85
+ if (!otpTokenResult.success) {
86
+ return otpTokenResult;
87
+ }
88
+ debug('Requesting id token');
89
+ const getIdTokenResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/getIdToken`, {
90
+ otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,
91
+ email: credentials.email,
92
+ pass: credentials.password,
93
+ pinCode: '',
94
+ });
95
+ const { resultData: { idToken }, } = getIdTokenResponse;
96
+ debug('Requesting session token');
97
+ const getSessionTokenResponse = await (0, fetch_1.fetchPost)(`${IDENTITY_SERVER_URL}/sessions/token`, {
98
+ idToken,
99
+ pass: credentials.password,
100
+ });
101
+ const { resultData: { accessToken }, } = getSessionTokenResponse;
102
+ this.accessToken = accessToken;
103
+ return {
104
+ success: true,
105
+ persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken,
106
+ };
116
107
  }
117
- return {
118
- success: true,
119
- longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken
120
- };
121
- }
122
- async login(credentials) {
123
- const otpTokenResult = await this.resolveOtpToken(credentials);
124
- if (!otpTokenResult.success) {
125
- return otpTokenResult;
108
+ async fetchPortfolioMovements(portfolio, startDate) {
109
+ // TODO: Find out if we need the other accounts, there seems to always be one
110
+ const account = portfolio.accounts[0];
111
+ let cursor = null;
112
+ const movements = [];
113
+ while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {
114
+ debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);
115
+ const { movements: { movements: newMovements, pagination }, } = await (0, fetch_1.fetchGraphql)(GRAPHQL_API_URL, one_zero_queries_1.GET_MOVEMENTS, {
116
+ portfolioId: portfolio.portfolioId,
117
+ accountId: account.accountId,
118
+ language: 'HEBREW',
119
+ pagination: {
120
+ cursor,
121
+ limit: 50,
122
+ },
123
+ }, { authorization: `Bearer ${this.accessToken}` });
124
+ movements.unshift(...newMovements);
125
+ cursor = pagination.cursor;
126
+ if (!pagination.hasMore) {
127
+ break;
128
+ }
129
+ }
130
+ movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());
131
+ const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);
132
+ return {
133
+ accountNumber: portfolio.portfolioNum,
134
+ balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),
135
+ txns: matchingMovements.map((movement) => {
136
+ const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);
137
+ const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;
138
+ return {
139
+ identifier: movement.movementId,
140
+ date: movement.valueDate,
141
+ chargedAmount: +movement.movementAmount * modifier,
142
+ chargedCurrency: movement.movementCurrency,
143
+ originalAmount: +movement.movementAmount * modifier,
144
+ originalCurrency: movement.movementCurrency,
145
+ description: this.sanitizeHebrew(movement.description),
146
+ processedDate: movement.movementTimestamp,
147
+ status: transactions_1.TransactionStatuses.Completed,
148
+ type: hasInstallments ? transactions_1.TransactionTypes.Installments : transactions_1.TransactionTypes.Normal,
149
+ };
150
+ }),
151
+ };
126
152
  }
127
- debug('Requesting id token');
128
- const getIdTokenResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/getIdToken`, {
129
- otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,
130
- email: credentials.email,
131
- pass: credentials.password,
132
- pinCode: ''
133
- });
134
- const {
135
- resultData: {
136
- idToken
137
- }
138
- } = getIdTokenResponse;
139
- debug('Requesting session token');
140
- const getSessionTokenResponse = await (0, _fetch.fetchPost)(`${IDENTITY_SERVER_URL}/sessions/token`, {
141
- idToken,
142
- pass: credentials.password
143
- });
144
- const {
145
- resultData: {
146
- accessToken
147
- }
148
- } = getSessionTokenResponse;
149
- this.accessToken = accessToken;
150
- return {
151
- success: true,
152
- persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken
153
- };
154
- }
155
- async fetchPortfolioMovements(portfolio, startDate) {
156
- // TODO: Find out if we need the other accounts, there seems to always be one
157
- const account = portfolio.accounts[0];
158
- let cursor = null;
159
- const movements = [];
160
- while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {
161
- debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);
162
- const {
163
- movements: {
164
- movements: newMovements,
165
- pagination
153
+ /**
154
+ * one zero hebrew strings are reversed with a unicode control character that forces display in LTR order
155
+ * We need to remove the unicode control character, and then reverse hebrew substrings inside the string
156
+ */
157
+ sanitizeHebrew(text) {
158
+ if (!text.includes('\u202d')) {
159
+ return text.trim();
166
160
  }
167
- } = await (0, _fetch.fetchGraphql)(GRAPHQL_API_URL, _oneZeroQueries.GET_MOVEMENTS, {
168
- portfolioId: portfolio.portfolioId,
169
- accountId: account.accountId,
170
- language: 'HEBREW',
171
- pagination: {
172
- cursor,
173
- limit: 50
161
+ const plainString = text.replace(/\u202d/gi, '').trim();
162
+ const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];
163
+ const rangesToReverse = hebrewSubStringsRanges.map(str => ({ start: str.index, end: str.index + str[0].length }));
164
+ const out = [];
165
+ let index = 0;
166
+ for (const { start, end } of rangesToReverse) {
167
+ out.push(...plainString.substring(index, start));
168
+ index += start - index;
169
+ const reversed = [...plainString.substring(start, end)].reverse();
170
+ out.push(...reversed);
171
+ index += end - start;
174
172
  }
175
- }, {
176
- authorization: `Bearer ${this.accessToken}`
177
- });
178
- movements.unshift(...newMovements);
179
- cursor = pagination.cursor;
180
- if (!pagination.hasMore) {
181
- break;
182
- }
173
+ out.push(...plainString.substring(index, plainString.length));
174
+ return out.join('');
183
175
  }
184
- movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());
185
- const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);
186
- return {
187
- accountNumber: portfolio.portfolioNum,
188
- balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),
189
- txns: matchingMovements.map(movement => {
190
- var _movement$transaction;
191
- const hasInstallments = (_movement$transaction = movement.transaction) === null || _movement$transaction === void 0 || (_movement$transaction = _movement$transaction.enrichment) === null || _movement$transaction === void 0 || (_movement$transaction = _movement$transaction.recurrences) === null || _movement$transaction === void 0 ? void 0 : _movement$transaction.some(x => x.isRecurrent);
192
- const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;
176
+ async fetchData() {
177
+ if (!this.accessToken) {
178
+ return (0, errors_1.createGenericError)('login() was not called');
179
+ }
180
+ const defaultStartMoment = (0, moment_1.default)().subtract(1, 'years').add(1, 'day');
181
+ const startDate = this.options.startDate || defaultStartMoment.toDate();
182
+ const startMoment = moment_1.default.max(defaultStartMoment, (0, moment_1.default)(startDate));
183
+ debug('Fetching account list');
184
+ const result = await (0, fetch_1.fetchGraphql)(GRAPHQL_API_URL, one_zero_queries_1.GET_CUSTOMER, {}, { authorization: `Bearer ${this.accessToken}` });
185
+ const portfolios = result.customer.flatMap(customer => customer.portfolios || []);
193
186
  return {
194
- identifier: movement.movementId,
195
- date: movement.valueDate,
196
- chargedAmount: +movement.movementAmount * modifier,
197
- chargedCurrency: movement.movementCurrency,
198
- originalAmount: +movement.movementAmount * modifier,
199
- originalCurrency: movement.movementCurrency,
200
- description: this.sanitizeHebrew(movement.description),
201
- processedDate: movement.movementTimestamp,
202
- status: _transactions.TransactionStatuses.Completed,
203
- type: hasInstallments ? _transactions.TransactionTypes.Installments : _transactions.TransactionTypes.Normal
187
+ success: true,
188
+ accounts: await Promise.all(portfolios.map(portfolio => this.fetchPortfolioMovements(portfolio, startMoment.toDate()))),
204
189
  };
205
- })
206
- };
207
- }
208
-
209
- /**
210
- * one zero hebrew strings are reversed with a unicode control character that forces display in LTR order
211
- * We need to remove the unicode control character, and then reverse hebrew substrings inside the string
212
- */
213
- sanitizeHebrew(text) {
214
- if (!text.includes('\u202d')) {
215
- return text.trim();
216
- }
217
- const plainString = text.replace(/\u202d/gi, '').trim();
218
- const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];
219
- const rangesToReverse = hebrewSubStringsRanges.map(str => ({
220
- start: str.index,
221
- end: str.index + str[0].length
222
- }));
223
- const out = [];
224
- let index = 0;
225
- for (const {
226
- start,
227
- end
228
- } of rangesToReverse) {
229
- out.push(...plainString.substring(index, start));
230
- index += start - index;
231
- const reversed = [...plainString.substring(start, end)].reverse();
232
- out.push(...reversed);
233
- index += end - start;
234
- }
235
- out.push(...plainString.substring(index, plainString.length));
236
- return out.join('');
237
- }
238
- async fetchData() {
239
- if (!this.accessToken) {
240
- return (0, _errors.createGenericError)('login() was not called');
241
190
  }
242
- const defaultStartMoment = (0, _moment.default)().subtract(1, 'years').add(1, 'day');
243
- const startDate = this.options.startDate || defaultStartMoment.toDate();
244
- const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
245
- debug('Fetching account list');
246
- const result = await (0, _fetch.fetchGraphql)(GRAPHQL_API_URL, _oneZeroQueries.GET_CUSTOMER, {}, {
247
- authorization: `Bearer ${this.accessToken}`
248
- });
249
- const portfolios = result.customer.flatMap(customer => customer.portfolios || []);
250
- return {
251
- success: true,
252
- accounts: await Promise.all(portfolios.map(portfolio => this.fetchPortfolioMovements(portfolio, startMoment.toDate())))
253
- };
254
- }
255
191
  }
256
192
  exports.default = OneZeroScraper;
257
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_moment","_interopRequireDefault","require","_debug","_fetch","_transactions","_baseScraper","_errors","_oneZeroQueries","e","__esModule","default","_defineProperty","r","t","_toPropertyKey","Object","defineProperty","value","enumerable","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","HEBREW_WORDS_REGEX","debug","getDebug","IDENTITY_SERVER_URL","GRAPHQL_API_URL","OneZeroScraper","BaseScraper","constructor","args","triggerTwoFactorAuth","phoneNumber","startsWith","createGenericError","deviceTokenResponse","fetchPost","extClientId","os","resultData","deviceToken","otpPrepareResponse","factorValue","otpChannel","otpContext","success","getLongTermTwoFactorToken","otpCode","otpVerifyResponse","otpToken","longTermTwoFactorAuthToken","resolveOtpToken","credentials","otpLongTermToken","otpCodeRetriever","errorType","ScraperErrorTypes","TwoFactorRetrieverMissing","errorMessage","triggerResult","otpTokenResult","login","getIdTokenResponse","otpSmsToken","email","pass","password","pinCode","idToken","getSessionTokenResponse","accessToken","persistentOtpToken","fetchPortfolioMovements","portfolio","startDate","account","accounts","cursor","movements","length","Date","movementTimestamp","portfolioNum","newMovements","pagination","fetchGraphql","GET_MOVEMENTS","portfolioId","accountId","language","limit","authorization","unshift","hasMore","sort","x","y","valueOf","matchingMovements","filter","movement","accountNumber","balance","parseFloat","runningBalance","txns","map","_movement$transaction","hasInstallments","transaction","enrichment","recurrences","some","isRecurrent","modifier","creditDebit","identifier","movementId","date","valueDate","chargedAmount","movementAmount","chargedCurrency","movementCurrency","originalAmount","originalCurrency","description","sanitizeHebrew","processedDate","status","TransactionStatuses","Completed","type","TransactionTypes","Installments","Normal","text","includes","trim","plainString","replace","hebrewSubStringsRanges","matchAll","rangesToReverse","str","start","index","end","out","push","substring","reversed","reverse","join","fetchData","defaultStartMoment","moment","subtract","add","options","toDate","startMoment","max","result","GET_CUSTOMER","portfolios","customer","flatMap","Promise","all","exports"],"sources":["../../src/scrapers/one-zero.ts"],"sourcesContent":["import moment from 'moment/moment';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGraphql, fetchPost } from '../helpers/fetch';\nimport {\n  type Transaction as ScrapingTransaction,\n  TransactionStatuses,\n  TransactionTypes,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraper } from './base-scraper';\nimport { ScraperErrorTypes, createGenericError } from './errors';\nimport {\n  type ScraperGetLongTermTwoFactorTokenResult, type ScraperLoginResult,\n  type ScraperScrapingResult, type ScraperTwoFactorAuthTriggerResult,\n} from './interface';\nimport { GET_CUSTOMER, GET_MOVEMENTS } from './one-zero-queries';\n\nconst HEBREW_WORDS_REGEX = /[\\u0590-\\u05FF][\\u0590-\\u05FF\"'\\-_ /\\\\]*[\\u0590-\\u05FF]/g;\n\nconst debug = getDebug('one-zero');\n\ntype Account = {\n  accountId: string;\n};\n\ntype Portfolio = {\n  accounts: Array<Account>;\n  portfolioId: string;\n  portfolioNum: string;\n};\n\ntype Customer = {\n  customerId: string;\n  portfolios?: Array<Portfolio> | null;\n};\n\nexport type Category = {\n  categoryId: number;\n  dataSource: string;\n  subCategoryId?: number | null;\n};\n\nexport type Recurrence = {\n  dataSource: string;\n  isRecurrent: boolean;\n};\n\ntype TransactionEnrichment = {\n  categories?: Category[] | null;\n  recurrences?: Recurrence[] | null;\n};\n\ntype Transaction = {\n  enrichment?: TransactionEnrichment | null;\n  // TODO: Get installments information here\n  // transactionDetails: TransactionDetails;\n};\n\ntype Movement = {\n  accountId: string;\n  bankCurrencyAmount: string;\n  bookingDate: string;\n  conversionRate: string;\n  creditDebit: string;\n  description: string;\n  isReversed: boolean;\n  movementAmount: string;\n  movementCurrency: string;\n  movementId: string;\n  movementReversedId?: string | null;\n  movementTimestamp: string;\n  movementType: string;\n  portfolioId: string;\n  runningBalance: string;\n  transaction?: Transaction | null;\n  valueDate: string;\n};\n\ntype QueryPagination = { hasMore: boolean, cursor: string };\n\nconst IDENTITY_SERVER_URL = 'https://identity.tfd-bank.com/v1/';\n\nconst GRAPHQL_API_URL = 'https://mobile.tfd-bank.com/mobile-graph/graphql';\n\ntype ScraperSpecificCredentials = { email: string, password: string } & ({\n  otpCodeRetriever: () => Promise<string>;\n  phoneNumber: string;\n} | {\n  otpLongTermToken: string;\n});\n\nexport default class OneZeroScraper extends BaseScraper<ScraperSpecificCredentials> {\n  private otpContext?: string;\n\n  private accessToken?: string;\n\n  async triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult> {\n    if (!phoneNumber.startsWith('+')) {\n      return createGenericError('A full international phone number starting with + and a three digit country code is required');\n    }\n\n    debug('Fetching device token');\n    const deviceTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/devices/token`, {\n      extClientId: 'mobile',\n      os: 'Android',\n    });\n\n    const { resultData: { deviceToken } } = deviceTokenResponse;\n\n    debug(`Sending OTP to phone number ${phoneNumber}`);\n\n    const otpPrepareResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/prepare`, {\n      factorValue: phoneNumber,\n      deviceToken,\n      otpChannel: 'SMS_OTP',\n    });\n\n    const { resultData: { otpContext } } = otpPrepareResponse;\n\n    this.otpContext = otpContext;\n\n    return {\n      success: true,\n    };\n  }\n\n  public async getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if (!this.otpContext) {\n      return createGenericError('triggerOtp was not called before calling getPermenantOtpToken()');\n    }\n\n    debug('Requesting OTP token');\n    const otpVerifyResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/verify`, {\n      otpContext: this.otpContext,\n      otpCode,\n    });\n\n    const { resultData: { otpToken } } = otpVerifyResponse;\n    return { success: true, longTermTwoFactorAuthToken: otpToken };\n  }\n\n  private async resolveOtpToken(\n    credentials: ScraperSpecificCredentials,\n  ): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if ('otpLongTermToken' in credentials) {\n      if (!credentials.otpLongTermToken) {\n        return createGenericError('Invalid otpLongTermToken');\n      }\n      return { success: true, longTermTwoFactorAuthToken: credentials.otpLongTermToken };\n    }\n\n    if (!credentials.otpCodeRetriever) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.TwoFactorRetrieverMissing,\n        errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided',\n      };\n    }\n\n    if (!credentials.phoneNumber) {\n      return createGenericError('phoneNumber is required when providing a otpCodeRetriever callback');\n    }\n\n    debug('Triggering user supplied otpCodeRetriever callback');\n    const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);\n\n    if (!triggerResult.success) {\n      return triggerResult;\n    }\n\n    const otpCode = await credentials.otpCodeRetriever();\n\n    const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    return { success: true, longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken };\n  }\n\n  async login(credentials: ScraperSpecificCredentials):\n  Promise<ScraperLoginResult> {\n    const otpTokenResult = await this.resolveOtpToken(credentials);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    debug('Requesting id token');\n    const getIdTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/getIdToken`, {\n      otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,\n      email: credentials.email,\n      pass: credentials.password,\n      pinCode: '',\n    });\n\n    const { resultData: { idToken } } = getIdTokenResponse;\n\n    debug('Requesting session token');\n\n    const getSessionTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/sessions/token`, {\n      idToken,\n      pass: credentials.password,\n    });\n\n    const { resultData: { accessToken } } = getSessionTokenResponse;\n\n    this.accessToken = accessToken;\n\n    return {\n      success: true,\n      persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken,\n    };\n  }\n\n  private async fetchPortfolioMovements(portfolio: Portfolio, startDate: Date): Promise<TransactionsAccount> {\n    // TODO: Find out if we need the other accounts, there seems to always be one\n    const account = portfolio.accounts[0];\n    let cursor = null;\n    const movements = [];\n\n    while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {\n      debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);\n      const { movements: { movements: newMovements, pagination } }:\n      { movements: { movements: Movement[], pagination: QueryPagination } } =\n          await fetchGraphql(\n            GRAPHQL_API_URL,\n            GET_MOVEMENTS, {\n              portfolioId: portfolio.portfolioId,\n              accountId: account.accountId,\n              language: 'HEBREW',\n              pagination: {\n                cursor,\n                limit: 50,\n              },\n            },\n            { authorization: `Bearer ${this.accessToken}` },\n          );\n\n      movements.unshift(...newMovements);\n      cursor = pagination.cursor;\n      if (!pagination.hasMore) {\n        break;\n      }\n    }\n\n    movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());\n\n    const matchingMovements = movements.filter((movement) => new Date(movement.movementTimestamp) >= startDate);\n    return {\n      accountNumber: portfolio.portfolioNum,\n      balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),\n      txns: matchingMovements.map((movement): ScrapingTransaction => {\n        const hasInstallments = movement.transaction?.enrichment?.recurrences?.some((x) => x.isRecurrent);\n        const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;\n        return ({\n          identifier: movement.movementId,\n          date: movement.valueDate,\n          chargedAmount: (+movement.movementAmount) * modifier,\n          chargedCurrency: movement.movementCurrency,\n          originalAmount: (+movement.movementAmount) * modifier,\n          originalCurrency: movement.movementCurrency,\n          description: this.sanitizeHebrew(movement.description),\n          processedDate: movement.movementTimestamp,\n          status: TransactionStatuses.Completed,\n          type: hasInstallments ? TransactionTypes.Installments : TransactionTypes.Normal,\n        });\n      }),\n    };\n  }\n\n  /**\n   * one zero hebrew strings are reversed with a unicode control character that forces display in LTR order\n   * We need to remove the unicode control character, and then reverse hebrew substrings inside the string\n   */\n  private sanitizeHebrew(text: string) {\n    if (!text.includes('\\u202d')) {\n      return text.trim();\n    }\n\n    const plainString = text.replace(/\\u202d/gi, '').trim();\n    const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];\n    const rangesToReverse = hebrewSubStringsRanges.map((str) => (\n      { start: str.index!, end: str.index! + str[0].length }));\n    const out = [];\n    let index = 0;\n\n    for (const { start, end } of rangesToReverse) {\n      out.push(...plainString.substring(index, start));\n      index += (start - index);\n      const reversed = [...plainString.substring(start, end)].reverse();\n      out.push(...reversed);\n      index += (end - start);\n    }\n\n    out.push(...plainString.substring(index, plainString.length));\n\n    return out.join('');\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    if (!this.accessToken) {\n      return createGenericError('login() was not called');\n    }\n\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    debug('Fetching account list');\n    const result = await fetchGraphql<{ customer: Customer[] }>(GRAPHQL_API_URL, GET_CUSTOMER, {}, { authorization: `Bearer ${this.accessToken}` });\n    const portfolios = result.customer.flatMap((customer) => (customer.portfolios || []));\n\n    return {\n      success: true,\n      accounts: await Promise.all(portfolios.map(\n        (portfolio) => this.fetchPortfolioMovements(portfolio, startMoment.toDate()),\n      )),\n    };\n  }\n}\n"],"mappings":"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","ignoreList":[]}
193
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"one-zero.js","sourceRoot":"","sources":["../../src/scrapers/one-zero.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from 'moment/moment';\nimport { getDebug } from '../helpers/debug';\nimport { fetchGraphql, fetchPost } from '../helpers/fetch';\nimport {\n  type Transaction as ScrapingTransaction,\n  TransactionStatuses,\n  TransactionTypes,\n  type TransactionsAccount,\n} from '../transactions';\nimport { BaseScraper } from './base-scraper';\nimport { ScraperErrorTypes, createGenericError } from './errors';\nimport {\n  type ScraperGetLongTermTwoFactorTokenResult,\n  type ScraperLoginResult,\n  type ScraperScrapingResult,\n  type ScraperTwoFactorAuthTriggerResult,\n} from './interface';\nimport { GET_CUSTOMER, GET_MOVEMENTS } from './one-zero-queries';\n\nconst HEBREW_WORDS_REGEX = /[\\u0590-\\u05FF][\\u0590-\\u05FF\"'\\-_ /\\\\]*[\\u0590-\\u05FF]/g;\n\nconst debug = getDebug('one-zero');\n\ntype Account = {\n  accountId: string;\n};\n\ntype Portfolio = {\n  accounts: Array<Account>;\n  portfolioId: string;\n  portfolioNum: string;\n};\n\ntype Customer = {\n  customerId: string;\n  portfolios?: Array<Portfolio> | null;\n};\n\nexport type Category = {\n  categoryId: number;\n  dataSource: string;\n  subCategoryId?: number | null;\n};\n\nexport type Recurrence = {\n  dataSource: string;\n  isRecurrent: boolean;\n};\n\ntype TransactionEnrichment = {\n  categories?: Category[] | null;\n  recurrences?: Recurrence[] | null;\n};\n\ntype Transaction = {\n  enrichment?: TransactionEnrichment | null;\n  // TODO: Get installments information here\n  // transactionDetails: TransactionDetails;\n};\n\ntype Movement = {\n  accountId: string;\n  bankCurrencyAmount: string;\n  bookingDate: string;\n  conversionRate: string;\n  creditDebit: string;\n  description: string;\n  isReversed: boolean;\n  movementAmount: string;\n  movementCurrency: string;\n  movementId: string;\n  movementReversedId?: string | null;\n  movementTimestamp: string;\n  movementType: string;\n  portfolioId: string;\n  runningBalance: string;\n  transaction?: Transaction | null;\n  valueDate: string;\n};\n\ntype QueryPagination = { hasMore: boolean; cursor: string };\n\nconst IDENTITY_SERVER_URL = 'https://identity.tfd-bank.com/v1/';\n\nconst GRAPHQL_API_URL = 'https://mobile.tfd-bank.com/mobile-graph/graphql';\n\ntype ScraperSpecificCredentials = { email: string; password: string } & (\n  | {\n      otpCodeRetriever: () => Promise<string>;\n      phoneNumber: string;\n    }\n  | {\n      otpLongTermToken: string;\n    }\n);\n\nexport default class OneZeroScraper extends BaseScraper<ScraperSpecificCredentials> {\n  private otpContext?: string;\n\n  private accessToken?: string;\n\n  async triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult> {\n    if (!phoneNumber.startsWith('+')) {\n      return createGenericError(\n        'A full international phone number starting with + and a three digit country code is required',\n      );\n    }\n\n    debug('Fetching device token');\n    const deviceTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/devices/token`, {\n      extClientId: 'mobile',\n      os: 'Android',\n    });\n\n    const {\n      resultData: { deviceToken },\n    } = deviceTokenResponse;\n\n    debug(`Sending OTP to phone number ${phoneNumber}`);\n\n    const otpPrepareResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/prepare`, {\n      factorValue: phoneNumber,\n      deviceToken,\n      otpChannel: 'SMS_OTP',\n    });\n\n    const {\n      resultData: { otpContext },\n    } = otpPrepareResponse;\n\n    this.otpContext = otpContext;\n\n    return {\n      success: true,\n    };\n  }\n\n  public async getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if (!this.otpContext) {\n      return createGenericError('triggerOtp was not called before calling getPermenantOtpToken()');\n    }\n\n    debug('Requesting OTP token');\n    const otpVerifyResponse = await fetchPost(`${IDENTITY_SERVER_URL}/otp/verify`, {\n      otpContext: this.otpContext,\n      otpCode,\n    });\n\n    const {\n      resultData: { otpToken },\n    } = otpVerifyResponse;\n    return { success: true, longTermTwoFactorAuthToken: otpToken };\n  }\n\n  private async resolveOtpToken(\n    credentials: ScraperSpecificCredentials,\n  ): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    if ('otpLongTermToken' in credentials) {\n      if (!credentials.otpLongTermToken) {\n        return createGenericError('Invalid otpLongTermToken');\n      }\n      return { success: true, longTermTwoFactorAuthToken: credentials.otpLongTermToken };\n    }\n\n    if (!credentials.otpCodeRetriever) {\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.TwoFactorRetrieverMissing,\n        errorMessage: 'otpCodeRetriever is required when otpPermanentToken is not provided',\n      };\n    }\n\n    if (!credentials.phoneNumber) {\n      return createGenericError('phoneNumber is required when providing a otpCodeRetriever callback');\n    }\n\n    debug('Triggering user supplied otpCodeRetriever callback');\n    const triggerResult = await this.triggerTwoFactorAuth(credentials.phoneNumber);\n\n    if (!triggerResult.success) {\n      return triggerResult;\n    }\n\n    const otpCode = await credentials.otpCodeRetriever();\n\n    const otpTokenResult = await this.getLongTermTwoFactorToken(otpCode);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    return { success: true, longTermTwoFactorAuthToken: otpTokenResult.longTermTwoFactorAuthToken };\n  }\n\n  async login(credentials: ScraperSpecificCredentials): Promise<ScraperLoginResult> {\n    const otpTokenResult = await this.resolveOtpToken(credentials);\n    if (!otpTokenResult.success) {\n      return otpTokenResult;\n    }\n\n    debug('Requesting id token');\n    const getIdTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/getIdToken`, {\n      otpSmsToken: otpTokenResult.longTermTwoFactorAuthToken,\n      email: credentials.email,\n      pass: credentials.password,\n      pinCode: '',\n    });\n\n    const {\n      resultData: { idToken },\n    } = getIdTokenResponse;\n\n    debug('Requesting session token');\n\n    const getSessionTokenResponse = await fetchPost(`${IDENTITY_SERVER_URL}/sessions/token`, {\n      idToken,\n      pass: credentials.password,\n    });\n\n    const {\n      resultData: { accessToken },\n    } = getSessionTokenResponse;\n\n    this.accessToken = accessToken;\n\n    return {\n      success: true,\n      persistentOtpToken: otpTokenResult.longTermTwoFactorAuthToken,\n    };\n  }\n\n  private async fetchPortfolioMovements(portfolio: Portfolio, startDate: Date): Promise<TransactionsAccount> {\n    // TODO: Find out if we need the other accounts, there seems to always be one\n    const account = portfolio.accounts[0];\n    let cursor = null;\n    const movements = [];\n\n    while (!movements.length || new Date(movements[0].movementTimestamp) >= startDate) {\n      debug(`Fetching transactions for account ${portfolio.portfolioNum}...`);\n      const {\n        movements: { movements: newMovements, pagination },\n      }: { movements: { movements: Movement[]; pagination: QueryPagination } } = await fetchGraphql(\n        GRAPHQL_API_URL,\n        GET_MOVEMENTS,\n        {\n          portfolioId: portfolio.portfolioId,\n          accountId: account.accountId,\n          language: 'HEBREW',\n          pagination: {\n            cursor,\n            limit: 50,\n          },\n        },\n        { authorization: `Bearer ${this.accessToken}` },\n      );\n\n      movements.unshift(...newMovements);\n      cursor = pagination.cursor;\n      if (!pagination.hasMore) {\n        break;\n      }\n    }\n\n    movements.sort((x, y) => new Date(x.movementTimestamp).valueOf() - new Date(y.movementTimestamp).valueOf());\n\n    const matchingMovements = movements.filter(movement => new Date(movement.movementTimestamp) >= startDate);\n    return {\n      accountNumber: portfolio.portfolioNum,\n      balance: !movements.length ? 0 : parseFloat(movements[movements.length - 1].runningBalance),\n      txns: matchingMovements.map((movement): ScrapingTransaction => {\n        const hasInstallments = movement.transaction?.enrichment?.recurrences?.some(x => x.isRecurrent);\n        const modifier = movement.creditDebit === 'DEBIT' ? -1 : 1;\n        return {\n          identifier: movement.movementId,\n          date: movement.valueDate,\n          chargedAmount: +movement.movementAmount * modifier,\n          chargedCurrency: movement.movementCurrency,\n          originalAmount: +movement.movementAmount * modifier,\n          originalCurrency: movement.movementCurrency,\n          description: this.sanitizeHebrew(movement.description),\n          processedDate: movement.movementTimestamp,\n          status: TransactionStatuses.Completed,\n          type: hasInstallments ? TransactionTypes.Installments : TransactionTypes.Normal,\n        };\n      }),\n    };\n  }\n\n  /**\n   * one zero hebrew strings are reversed with a unicode control character that forces display in LTR order\n   * We need to remove the unicode control character, and then reverse hebrew substrings inside the string\n   */\n  private sanitizeHebrew(text: string) {\n    if (!text.includes('\\u202d')) {\n      return text.trim();\n    }\n\n    const plainString = text.replace(/\\u202d/gi, '').trim();\n    const hebrewSubStringsRanges = [...plainString.matchAll(HEBREW_WORDS_REGEX)];\n    const rangesToReverse = hebrewSubStringsRanges.map(str => ({ start: str.index!, end: str.index! + str[0].length }));\n    const out = [];\n    let index = 0;\n\n    for (const { start, end } of rangesToReverse) {\n      out.push(...plainString.substring(index, start));\n      index += start - index;\n      const reversed = [...plainString.substring(start, end)].reverse();\n      out.push(...reversed);\n      index += end - start;\n    }\n\n    out.push(...plainString.substring(index, plainString.length));\n\n    return out.join('');\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    if (!this.accessToken) {\n      return createGenericError('login() was not called');\n    }\n\n    const defaultStartMoment = moment().subtract(1, 'years').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    debug('Fetching account list');\n    const result = await fetchGraphql<{ customer: Customer[] }>(\n      GRAPHQL_API_URL,\n      GET_CUSTOMER,\n      {},\n      { authorization: `Bearer ${this.accessToken}` },\n    );\n    const portfolios = result.customer.flatMap(customer => customer.portfolios || []);\n\n    return {\n      success: true,\n      accounts: await Promise.all(\n        portfolios.map(portfolio => this.fetchPortfolioMovements(portfolio, startMoment.toDate())),\n      ),\n    };\n  }\n}\n"]}
@@ -1,22 +1,13 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.default = void 0;
7
- require("core-js/modules/es.array.iterator.js");
8
- var _baseBeinleumiGroup = _interopRequireDefault(require("./base-beinleumi-group"));
9
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
10
- function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
11
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
12
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
13
- class OtsarHahayalScraper extends _baseBeinleumiGroup.default {
14
- constructor(...args) {
15
- super(...args);
16
- _defineProperty(this, "BASE_URL", 'https://online.bankotsar.co.il');
17
- _defineProperty(this, "LOGIN_URL", `${this.BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=OTSARPRTAL&site=Private&KODSAFA=HE`);
18
- _defineProperty(this, "TRANSACTIONS_URL", `${this.BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`);
19
- }
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
+ const base_beinleumi_group_1 = __importDefault(require("./base-beinleumi-group"));
7
+ class OtsarHahayalScraper extends base_beinleumi_group_1.default {
8
+ BASE_URL = 'https://online.bankotsar.co.il';
9
+ LOGIN_URL = `${this.BASE_URL}/MatafLoginService/MatafLoginServlet?bankId=OTSARPRTAL&site=Private&KODSAFA=HE`;
10
+ TRANSACTIONS_URL = `${this.BASE_URL}/wps/myportal/FibiMenu/Online/OnAccountMngment/OnBalanceTrans/PrivateAccountFlow`;
20
11
  }
21
- var _default = exports.default = OtsarHahayalScraper;
22
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfYmFzZUJlaW5sZXVtaUdyb3VwIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsInJlcXVpcmUiLCJlIiwiX19lc01vZHVsZSIsImRlZmF1bHQiLCJfZGVmaW5lUHJvcGVydHkiLCJyIiwidCIsIl90b1Byb3BlcnR5S2V5IiwiT2JqZWN0IiwiZGVmaW5lUHJvcGVydHkiLCJ2YWx1ZSIsImVudW1lcmFibGUiLCJjb25maWd1cmFibGUiLCJ3cml0YWJsZSIsImkiLCJfdG9QcmltaXRpdmUiLCJTeW1ib2wiLCJ0b1ByaW1pdGl2ZSIsImNhbGwiLCJUeXBlRXJyb3IiLCJTdHJpbmciLCJOdW1iZXIiLCJPdHNhckhhaGF5YWxTY3JhcGVyIiwiQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciIsImNvbnN0cnVjdG9yIiwiYXJncyIsIkJBU0VfVVJMIiwiX2RlZmF1bHQiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmFwZXJzL290c2FyLWhhaGF5YWwudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEJlaW5sZXVtaUdyb3VwQmFzZVNjcmFwZXIgZnJvbSAnLi9iYXNlLWJlaW5sZXVtaS1ncm91cCc7XG5cbmNsYXNzIE90c2FySGFoYXlhbFNjcmFwZXIgZXh0ZW5kcyBCZWlubGV1bWlHcm91cEJhc2VTY3JhcGVyIHtcbiAgQkFTRV9VUkwgPSAnaHR0cHM6Ly9vbmxpbmUuYmFua290c2FyLmNvLmlsJztcblxuICBMT0dJTl9VUkwgPSBgJHt0aGlzLkJBU0VfVVJMfS9NYXRhZkxvZ2luU2VydmljZS9NYXRhZkxvZ2luU2VydmxldD9iYW5rSWQ9T1RTQVJQUlRBTCZzaXRlPVByaXZhdGUmS09EU0FGQT1IRWA7XG5cbiAgVFJBTlNBQ1RJT05TX1VSTCA9IGAke3RoaXMuQkFTRV9VUkx9L3dwcy9teXBvcnRhbC9GaWJpTWVudS9PbmxpbmUvT25BY2NvdW50TW5nbWVudC9PbkJhbGFuY2VUcmFucy9Qcml2YXRlQWNjb3VudEZsb3dgO1xufVxuXG5leHBvcnQgZGVmYXVsdCBPdHNhckhhaGF5YWxTY3JhcGVyOyJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBLElBQUFBLG1CQUFBLEdBQUFDLHNCQUFBLENBQUFDLE9BQUE7QUFBK0QsU0FBQUQsdUJBQUFFLENBQUEsV0FBQUEsQ0FBQSxJQUFBQSxDQUFBLENBQUFDLFVBQUEsR0FBQUQsQ0FBQSxLQUFBRSxPQUFBLEVBQUFGLENBQUE7QUFBQSxTQUFBRyxnQkFBQUgsQ0FBQSxFQUFBSSxDQUFBLEVBQUFDLENBQUEsWUFBQUQsQ0FBQSxHQUFBRSxjQUFBLENBQUFGLENBQUEsTUFBQUosQ0FBQSxHQUFBTyxNQUFBLENBQUFDLGNBQUEsQ0FBQVIsQ0FBQSxFQUFBSSxDQUFBLElBQUFLLEtBQUEsRUFBQUosQ0FBQSxFQUFBSyxVQUFBLE1BQUFDLFlBQUEsTUFBQUMsUUFBQSxVQUFBWixDQUFBLENBQUFJLENBQUEsSUFBQUMsQ0FBQSxFQUFBTCxDQUFBO0FBQUEsU0FBQU0sZUFBQUQsQ0FBQSxRQUFBUSxDQUFBLEdBQUFDLFlBQUEsQ0FBQVQsQ0FBQSx1Q0FBQVEsQ0FBQSxHQUFBQSxDQUFBLEdBQUFBLENBQUE7QUFBQSxTQUFBQyxhQUFBVCxDQUFBLEVBQUFELENBQUEsMkJBQUFDLENBQUEsS0FBQUEsQ0FBQSxTQUFBQSxDQUFBLE1BQUFMLENBQUEsR0FBQUssQ0FBQSxDQUFBVSxNQUFBLENBQUFDLFdBQUEsa0JBQUFoQixDQUFBLFFBQUFhLENBQUEsR0FBQWIsQ0FBQSxDQUFBaUIsSUFBQSxDQUFBWixDQUFBLEVBQUFELENBQUEsdUNBQUFTLENBQUEsU0FBQUEsQ0FBQSxZQUFBSyxTQUFBLHlFQUFBZCxDQUFBLEdBQUFlLE1BQUEsR0FBQUMsTUFBQSxFQUFBZixDQUFBO0FBRS9ELE1BQU1nQixtQkFBbUIsU0FBU0MsMkJBQXlCLENBQUM7RUFBQUMsWUFBQSxHQUFBQyxJQUFBO0lBQUEsU0FBQUEsSUFBQTtJQUFBckIsZUFBQSxtQkFDL0MsZ0NBQWdDO0lBQUFBLGVBQUEsb0JBRS9CLEdBQUcsSUFBSSxDQUFDc0IsUUFBUSxnRkFBZ0Y7SUFBQXRCLGVBQUEsMkJBRXpGLEdBQUcsSUFBSSxDQUFDc0IsUUFBUSxrRkFBa0Y7RUFBQTtBQUN2SDtBQUFDLElBQUFDLFFBQUEsR0FBQUMsT0FBQSxDQUFBekIsT0FBQSxHQUVjbUIsbUJBQW1CIiwiaWdub3JlTGlzdCI6W119
12
+ exports.default = OtsarHahayalScraper;
13
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3RzYXItaGFoYXlhbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zY3JhcGVycy9vdHNhci1oYWhheWFsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsa0ZBQStEO0FBRS9ELE1BQU0sbUJBQW9CLFNBQVEsOEJBQXlCO0lBQ3pELFFBQVEsR0FBRyxnQ0FBZ0MsQ0FBQztJQUU1QyxTQUFTLEdBQUcsR0FBRyxJQUFJLENBQUMsUUFBUSxnRkFBZ0YsQ0FBQztJQUU3RyxnQkFBZ0IsR0FBRyxHQUFHLElBQUksQ0FBQyxRQUFRLGtGQUFrRixDQUFDO0NBQ3ZIO0FBRUQsa0JBQWUsbUJBQW1CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQmVpbmxldW1pR3JvdXBCYXNlU2NyYXBlciBmcm9tICcuL2Jhc2UtYmVpbmxldW1pLWdyb3VwJztcblxuY2xhc3MgT3RzYXJIYWhheWFsU2NyYXBlciBleHRlbmRzIEJlaW5sZXVtaUdyb3VwQmFzZVNjcmFwZXIge1xuICBCQVNFX1VSTCA9ICdodHRwczovL29ubGluZS5iYW5rb3RzYXIuY28uaWwnO1xuXG4gIExPR0lOX1VSTCA9IGAke3RoaXMuQkFTRV9VUkx9L01hdGFmTG9naW5TZXJ2aWNlL01hdGFmTG9naW5TZXJ2bGV0P2JhbmtJZD1PVFNBUlBSVEFMJnNpdGU9UHJpdmF0ZSZLT0RTQUZBPUhFYDtcblxuICBUUkFOU0FDVElPTlNfVVJMID0gYCR7dGhpcy5CQVNFX1VSTH0vd3BzL215cG9ydGFsL0ZpYmlNZW51L09ubGluZS9PbkFjY291bnRNbmdtZW50L09uQmFsYW5jZVRyYW5zL1ByaXZhdGVBY2NvdW50Rmxvd2A7XG59XG5cbmV4cG9ydCBkZWZhdWx0IE90c2FySGFoYXlhbFNjcmFwZXI7XG4iXX0=