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,296 +1,262 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.LoginResults = exports.BaseScraperWithBrowser = void 0;
7
- require("core-js/modules/es.array.iterator.js");
8
- require("core-js/modules/es.promise.js");
9
- require("core-js/modules/es.regexp.constructor.js");
10
- require("core-js/modules/es.regexp.exec.js");
11
- var _puppeteer = _interopRequireDefault(require("puppeteer"));
12
- var _definitions = require("../definitions");
13
- var _debug = require("../helpers/debug");
14
- var _elementsInteractions = require("../helpers/elements-interactions");
15
- var _navigation = require("../helpers/navigation");
16
- var _baseScraper = require("./base-scraper");
17
- var _errors = require("./errors");
18
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
19
- function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
20
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
21
- 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; }
22
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
23
- 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); }
24
- function _objectWithoutProperties(e, t) { if (null == e) return {}; var o, r, i = _objectWithoutPropertiesLoose(e, t); if (Object.getOwnPropertySymbols) { var n = Object.getOwnPropertySymbols(e); for (r = 0; r < n.length; r++) o = n[r], t.indexOf(o) >= 0 || {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]); } return i; }
25
- function _objectWithoutPropertiesLoose(r, e) { if (null == r) return {}; var t = {}; for (var n in r) if ({}.hasOwnProperty.call(r, n)) { if (e.indexOf(n) >= 0) continue; t[n] = r[n]; } return t; }
26
- const debug = (0, _debug.getDebug)('base-scraper-with-browser');
27
- var LoginBaseResults = /*#__PURE__*/function (LoginBaseResults) {
28
- LoginBaseResults["Success"] = "SUCCESS";
29
- LoginBaseResults["UnknownError"] = "UNKNOWN_ERROR";
30
- return LoginBaseResults;
31
- }(LoginBaseResults || {});
32
- const {
33
- Timeout,
34
- Generic,
35
- General
36
- } = _errors.ScraperErrorTypes,
37
- rest = _objectWithoutProperties(_errors.ScraperErrorTypes, ["Timeout", "Generic", "General"]);
38
- const LoginResults = exports.LoginResults = _objectSpread(_objectSpread({}, rest), LoginBaseResults);
39
-
40
- // eslint-disable-next-line @typescript-eslint/no-redeclare
41
-
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.BaseScraperWithBrowser = exports.LoginResults = void 0;
7
+ const puppeteer_1 = __importDefault(require("puppeteer"));
8
+ const definitions_1 = require("../definitions");
9
+ const debug_1 = require("../helpers/debug");
10
+ const elements_interactions_1 = require("../helpers/elements-interactions");
11
+ const navigation_1 = require("../helpers/navigation");
12
+ const base_scraper_1 = require("./base-scraper");
13
+ const errors_1 = require("./errors");
14
+ const debug = (0, debug_1.getDebug)('base-scraper-with-browser');
15
+ var LoginBaseResults;
16
+ (function (LoginBaseResults) {
17
+ LoginBaseResults["Success"] = "SUCCESS";
18
+ LoginBaseResults["UnknownError"] = "UNKNOWN_ERROR";
19
+ })(LoginBaseResults || (LoginBaseResults = {}));
20
+ const { Timeout, Generic, General, ...rest } = errors_1.ScraperErrorTypes;
21
+ exports.LoginResults = {
22
+ ...rest,
23
+ ...LoginBaseResults,
24
+ };
42
25
  async function getKeyByValue(object, value, page) {
43
- const keys = Object.keys(object);
44
- for (const key of keys) {
45
- // @ts-ignore
46
- const conditions = object[key];
47
- for (const condition of conditions) {
48
- let result = false;
49
- if (condition instanceof RegExp) {
50
- result = condition.test(value);
51
- } else if (typeof condition === 'function') {
52
- result = await condition({
53
- page,
54
- value
55
- });
56
- } else {
57
- result = value.toLowerCase() === condition.toLowerCase();
58
- }
59
- if (result) {
26
+ const keys = Object.keys(object);
27
+ for (const key of keys) {
60
28
  // @ts-ignore
61
- return Promise.resolve(key);
62
- }
29
+ const conditions = object[key];
30
+ for (const condition of conditions) {
31
+ let result = false;
32
+ if (condition instanceof RegExp) {
33
+ result = condition.test(value);
34
+ }
35
+ else if (typeof condition === 'function') {
36
+ result = await condition({ page, value });
37
+ }
38
+ else {
39
+ result = value.toLowerCase() === condition.toLowerCase();
40
+ }
41
+ if (result) {
42
+ // @ts-ignore
43
+ return Promise.resolve(key);
44
+ }
45
+ }
63
46
  }
64
- }
65
- return Promise.resolve(LoginResults.UnknownError);
47
+ return Promise.resolve(exports.LoginResults.UnknownError);
66
48
  }
67
49
  function createGeneralError() {
68
- return {
69
- success: false,
70
- errorType: _errors.ScraperErrorTypes.General
71
- };
50
+ return {
51
+ success: false,
52
+ errorType: errors_1.ScraperErrorTypes.General,
53
+ };
72
54
  }
73
- class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
74
- constructor(...args) {
75
- super(...args);
76
- _defineProperty(this, "cleanups", []);
77
- _defineProperty(this, "defaultViewportSize", {
78
- width: 1024,
79
- height: 768
80
- });
55
+ class BaseScraperWithBrowser extends base_scraper_1.BaseScraper {
56
+ cleanups = [];
57
+ defaultViewportSize = {
58
+ width: 1024,
59
+ height: 768,
60
+ };
81
61
  // NOTICE - it is discouraged to use bang (!) in general. It is used here because
82
62
  // all the classes that inherit from this base assume is it mandatory.
83
- _defineProperty(this, "page", void 0);
84
- }
85
- getViewPort() {
86
- var _this$options$viewpor;
87
- return (_this$options$viewpor = this.options.viewportSize) !== null && _this$options$viewpor !== void 0 ? _this$options$viewpor : this.defaultViewportSize;
88
- }
89
- async initialize() {
90
- await super.initialize();
91
- debug('initialize scraper');
92
- this.emitProgress(_definitions.ScraperProgressTypes.Initializing);
93
- const page = await this.initializePage();
94
- await page.setCacheEnabled(false); // Clear cache and avoid 300's response status
95
-
96
- if (!page) {
97
- debug('failed to initiate a browser page, exit');
98
- return;
99
- }
100
- this.page = page;
101
- this.cleanups.push(() => page.close());
102
- if (this.options.defaultTimeout) {
103
- this.page.setDefaultTimeout(this.options.defaultTimeout);
63
+ page;
64
+ getViewPort() {
65
+ return this.options.viewportSize ?? this.defaultViewportSize;
104
66
  }
105
- if (this.options.preparePage) {
106
- debug("execute 'preparePage' interceptor provided in options");
107
- await this.options.preparePage(this.page);
108
- }
109
- const viewport = this.getViewPort();
110
- debug(`set viewport to width ${viewport.width}, height ${viewport.height}`);
111
- await this.page.setViewport({
112
- width: viewport.width,
113
- height: viewport.height
114
- });
115
- this.page.on('requestfailed', request => {
116
- var _request$failure;
117
- debug('Request failed: %s %s', (_request$failure = request.failure()) === null || _request$failure === void 0 ? void 0 : _request$failure.errorText, request.url());
118
- });
119
- }
120
- async initializePage() {
121
- debug('initialize browser page');
122
- if ('browserContext' in this.options) {
123
- debug('Using the browser context provided in options');
124
- return this.options.browserContext.newPage();
67
+ async initialize() {
68
+ await super.initialize();
69
+ debug('initialize scraper');
70
+ this.emitProgress(definitions_1.ScraperProgressTypes.Initializing);
71
+ const page = await this.initializePage();
72
+ await page.setCacheEnabled(false); // Clear cache and avoid 300's response status
73
+ if (!page) {
74
+ debug('failed to initiate a browser page, exit');
75
+ return;
76
+ }
77
+ this.page = page;
78
+ this.cleanups.push(() => page.close());
79
+ if (this.options.defaultTimeout) {
80
+ this.page.setDefaultTimeout(this.options.defaultTimeout);
81
+ }
82
+ if (this.options.preparePage) {
83
+ debug("execute 'preparePage' interceptor provided in options");
84
+ await this.options.preparePage(this.page);
85
+ }
86
+ const viewport = this.getViewPort();
87
+ debug(`set viewport to width ${viewport.width}, height ${viewport.height}`);
88
+ await this.page.setViewport({
89
+ width: viewport.width,
90
+ height: viewport.height,
91
+ });
92
+ this.page.on('requestfailed', request => {
93
+ debug('Request failed: %s %s', request.failure()?.errorText, request.url());
94
+ });
125
95
  }
126
- if ('browser' in this.options) {
127
- debug('Using the browser instance provided in options');
128
- const {
129
- browser
130
- } = this.options;
131
-
132
- /**
133
- * For backward compatibility, we will close the browser even if we didn't create it
134
- */
135
- if (!this.options.skipCloseBrowser) {
96
+ async initializePage() {
97
+ debug('initialize browser page');
98
+ if ('browserContext' in this.options) {
99
+ debug('Using the browser context provided in options');
100
+ return this.options.browserContext.newPage();
101
+ }
102
+ if ('browser' in this.options) {
103
+ debug('Using the browser instance provided in options');
104
+ const { browser } = this.options;
105
+ /**
106
+ * For backward compatibility, we will close the browser even if we didn't create it
107
+ */
108
+ if (!this.options.skipCloseBrowser) {
109
+ this.cleanups.push(async () => {
110
+ debug('closing the browser');
111
+ await browser.close();
112
+ });
113
+ }
114
+ return browser.newPage();
115
+ }
116
+ const { timeout, args, executablePath, showBrowser } = this.options;
117
+ const headless = !showBrowser;
118
+ debug(`launch a browser with headless mode = ${headless}`);
119
+ const browser = await puppeteer_1.default.launch({
120
+ env: this.options.verbose ? { DEBUG: '*', ...process.env } : undefined,
121
+ headless,
122
+ executablePath,
123
+ args,
124
+ timeout,
125
+ });
136
126
  this.cleanups.push(async () => {
137
- debug('closing the browser');
138
- await browser.close();
127
+ debug('closing the browser');
128
+ await browser.close();
139
129
  });
140
- }
141
- return browser.newPage();
142
- }
143
- const {
144
- timeout,
145
- args,
146
- executablePath,
147
- showBrowser
148
- } = this.options;
149
- const headless = !showBrowser;
150
- debug(`launch a browser with headless mode = ${headless}`);
151
- const browser = await _puppeteer.default.launch({
152
- env: this.options.verbose ? _objectSpread({
153
- DEBUG: '*'
154
- }, process.env) : undefined,
155
- headless,
156
- executablePath,
157
- args,
158
- timeout
159
- });
160
- this.cleanups.push(async () => {
161
- debug('closing the browser');
162
- await browser.close();
163
- });
164
- if (this.options.prepareBrowser) {
165
- debug("execute 'prepareBrowser' interceptor provided in options");
166
- await this.options.prepareBrowser(browser);
167
- }
168
- debug('create a new browser page');
169
- return browser.newPage();
170
- }
171
- async navigateTo(url, waitUntil = 'load', retries = (_this$options$navigat => (_this$options$navigat = this.options.navigationRetryCount) !== null && _this$options$navigat !== void 0 ? _this$options$navigat : 0)()) {
172
- var _this$page;
173
- const response = await ((_this$page = this.page) === null || _this$page === void 0 ? void 0 : _this$page.goto(url, {
174
- waitUntil
175
- }));
176
- if (response === null) {
177
- // note: response will be null when navigating to same url while changing the hash part.
178
- // the condition below will always accept null as valid result.
179
- return;
180
- }
181
- if (!response) {
182
- throw new Error(`Error while trying to navigate to url ${url}, response is undefined`);
183
- }
184
- if (!response.ok()) {
185
- const status = response.status();
186
- if (retries > 0) {
187
- debug(`Failed to navigate to url ${url}, status code: ${status}, retrying ${retries} more times`);
188
- await this.navigateTo(url, waitUntil, retries - 1);
189
- } else {
190
- throw new Error(`Failed to navigate to url ${url}, status code: ${status}`);
191
- }
192
- }
193
- }
194
-
195
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
196
- getLoginOptions(_credentials) {
197
- throw new Error(`getLoginOptions() is not created in ${this.options.companyId}`);
198
- }
199
- async fillInputs(pageOrFrame, fields) {
200
- const modified = [...fields];
201
- const input = modified.shift();
202
- if (!input) {
203
- return;
204
- }
205
- await (0, _elementsInteractions.fillInput)(pageOrFrame, input.selector, input.value);
206
- if (modified.length) {
207
- await this.fillInputs(pageOrFrame, modified);
208
- }
209
- }
210
- async login(credentials) {
211
- if (!credentials || !this.page) {
212
- return createGeneralError();
213
- }
214
- debug('execute login process');
215
- const loginOptions = this.getLoginOptions(credentials);
216
- if (loginOptions.userAgent) {
217
- debug('set custom user agent provided in options');
218
- await this.page.setUserAgent(loginOptions.userAgent);
130
+ if (this.options.prepareBrowser) {
131
+ debug("execute 'prepareBrowser' interceptor provided in options");
132
+ await this.options.prepareBrowser(browser);
133
+ }
134
+ debug('create a new browser page');
135
+ return browser.newPage();
219
136
  }
220
- debug('navigate to login url');
221
- await this.navigateTo(loginOptions.loginUrl, loginOptions.waitUntil);
222
- if (loginOptions.checkReadiness) {
223
- debug("execute 'checkReadiness' interceptor provided in login options");
224
- await loginOptions.checkReadiness();
225
- } else if (typeof loginOptions.submitButtonSelector === 'string') {
226
- debug('wait until submit button is available');
227
- await (0, _elementsInteractions.waitUntilElementFound)(this.page, loginOptions.submitButtonSelector);
137
+ async navigateTo(url, waitUntil = 'load', retries = this.options.navigationRetryCount ?? 0) {
138
+ const response = await this.page?.goto(url, { waitUntil });
139
+ if (response === null) {
140
+ // note: response will be null when navigating to same url while changing the hash part.
141
+ // the condition below will always accept null as valid result.
142
+ return;
143
+ }
144
+ if (!response) {
145
+ throw new Error(`Error while trying to navigate to url ${url}, response is undefined`);
146
+ }
147
+ if (!response.ok()) {
148
+ const status = response.status();
149
+ if (retries > 0) {
150
+ debug(`Failed to navigate to url ${url}, status code: ${status}, retrying ${retries} more times`);
151
+ await this.navigateTo(url, waitUntil, retries - 1);
152
+ }
153
+ else {
154
+ throw new Error(`Failed to navigate to url ${url}, status code: ${status}`);
155
+ }
156
+ }
228
157
  }
229
- let loginFrameOrPage = this.page;
230
- if (loginOptions.preAction) {
231
- debug("execute 'preAction' interceptor provided in login options");
232
- loginFrameOrPage = (await loginOptions.preAction()) || this.page;
158
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
159
+ getLoginOptions(_credentials) {
160
+ throw new Error(`getLoginOptions() is not created in ${this.options.companyId}`);
233
161
  }
234
- debug('fill login components input with relevant values');
235
- await this.fillInputs(loginFrameOrPage, loginOptions.fields);
236
- debug('click on login submit button');
237
- if (typeof loginOptions.submitButtonSelector === 'string') {
238
- await (0, _elementsInteractions.clickButton)(loginFrameOrPage, loginOptions.submitButtonSelector);
239
- } else {
240
- await loginOptions.submitButtonSelector();
162
+ async fillInputs(pageOrFrame, fields) {
163
+ const modified = [...fields];
164
+ const input = modified.shift();
165
+ if (!input) {
166
+ return;
167
+ }
168
+ await (0, elements_interactions_1.fillInput)(pageOrFrame, input.selector, input.value);
169
+ if (modified.length) {
170
+ await this.fillInputs(pageOrFrame, modified);
171
+ }
241
172
  }
242
- this.emitProgress(_definitions.ScraperProgressTypes.LoggingIn);
243
- if (loginOptions.postAction) {
244
- debug("execute 'postAction' interceptor provided in login options");
245
- await loginOptions.postAction();
246
- } else {
247
- debug('wait for page navigation');
248
- await (0, _navigation.waitForNavigation)(this.page);
173
+ async login(credentials) {
174
+ if (!credentials || !this.page) {
175
+ return createGeneralError();
176
+ }
177
+ debug('execute login process');
178
+ const loginOptions = this.getLoginOptions(credentials);
179
+ if (loginOptions.userAgent) {
180
+ debug('set custom user agent provided in options');
181
+ await this.page.setUserAgent(loginOptions.userAgent);
182
+ }
183
+ debug('navigate to login url');
184
+ await this.navigateTo(loginOptions.loginUrl, loginOptions.waitUntil);
185
+ if (loginOptions.checkReadiness) {
186
+ debug("execute 'checkReadiness' interceptor provided in login options");
187
+ await loginOptions.checkReadiness();
188
+ }
189
+ else if (typeof loginOptions.submitButtonSelector === 'string') {
190
+ debug('wait until submit button is available');
191
+ await (0, elements_interactions_1.waitUntilElementFound)(this.page, loginOptions.submitButtonSelector);
192
+ }
193
+ let loginFrameOrPage = this.page;
194
+ if (loginOptions.preAction) {
195
+ debug("execute 'preAction' interceptor provided in login options");
196
+ loginFrameOrPage = (await loginOptions.preAction()) || this.page;
197
+ }
198
+ debug('fill login components input with relevant values');
199
+ await this.fillInputs(loginFrameOrPage, loginOptions.fields);
200
+ debug('click on login submit button');
201
+ if (typeof loginOptions.submitButtonSelector === 'string') {
202
+ await (0, elements_interactions_1.clickButton)(loginFrameOrPage, loginOptions.submitButtonSelector);
203
+ }
204
+ else {
205
+ await loginOptions.submitButtonSelector();
206
+ }
207
+ this.emitProgress(definitions_1.ScraperProgressTypes.LoggingIn);
208
+ if (loginOptions.postAction) {
209
+ debug("execute 'postAction' interceptor provided in login options");
210
+ await loginOptions.postAction();
211
+ }
212
+ else {
213
+ debug('wait for page navigation');
214
+ await (0, navigation_1.waitForNavigation)(this.page);
215
+ }
216
+ debug('check login result');
217
+ const current = await (0, navigation_1.getCurrentUrl)(this.page, true);
218
+ const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);
219
+ debug(`handle login results ${loginResult}`);
220
+ return this.handleLoginResult(loginResult);
249
221
  }
250
- debug('check login result');
251
- const current = await (0, _navigation.getCurrentUrl)(this.page, true);
252
- const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);
253
- debug(`handle login results ${loginResult}`);
254
- return this.handleLoginResult(loginResult);
255
- }
256
- async terminate(_success) {
257
- debug(`terminating browser with success = ${_success}`);
258
- this.emitProgress(_definitions.ScraperProgressTypes.Terminating);
259
- if (!_success && !!this.options.storeFailureScreenShotPath) {
260
- debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);
261
- await this.page.screenshot({
262
- path: this.options.storeFailureScreenShotPath,
263
- fullPage: true
264
- });
222
+ async terminate(_success) {
223
+ debug(`terminating browser with success = ${_success}`);
224
+ this.emitProgress(definitions_1.ScraperProgressTypes.Terminating);
225
+ if (!_success && !!this.options.storeFailureScreenShotPath) {
226
+ debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);
227
+ await this.page.screenshot({
228
+ path: this.options.storeFailureScreenShotPath,
229
+ fullPage: true,
230
+ });
231
+ }
232
+ await Promise.all(this.cleanups.reverse().map(cleanup => cleanup()));
233
+ this.cleanups = [];
265
234
  }
266
- await Promise.all(this.cleanups.reverse().map(cleanup => cleanup()));
267
- this.cleanups = [];
268
- }
269
- handleLoginResult(loginResult) {
270
- switch (loginResult) {
271
- case LoginResults.Success:
272
- this.emitProgress(_definitions.ScraperProgressTypes.LoginSuccess);
273
- return {
274
- success: true
275
- };
276
- case LoginResults.InvalidPassword:
277
- case LoginResults.UnknownError:
278
- this.emitProgress(_definitions.ScraperProgressTypes.LoginFailed);
279
- return {
280
- success: false,
281
- errorType: loginResult === LoginResults.InvalidPassword ? _errors.ScraperErrorTypes.InvalidPassword : _errors.ScraperErrorTypes.General,
282
- errorMessage: `Login failed with ${loginResult} error`
283
- };
284
- case LoginResults.ChangePassword:
285
- this.emitProgress(_definitions.ScraperProgressTypes.ChangePassword);
286
- return {
287
- success: false,
288
- errorType: _errors.ScraperErrorTypes.ChangePassword
289
- };
290
- default:
291
- throw new Error(`unexpected login result "${loginResult}"`);
235
+ handleLoginResult(loginResult) {
236
+ switch (loginResult) {
237
+ case exports.LoginResults.Success:
238
+ this.emitProgress(definitions_1.ScraperProgressTypes.LoginSuccess);
239
+ return { success: true };
240
+ case exports.LoginResults.InvalidPassword:
241
+ case exports.LoginResults.UnknownError:
242
+ this.emitProgress(definitions_1.ScraperProgressTypes.LoginFailed);
243
+ return {
244
+ success: false,
245
+ errorType: loginResult === exports.LoginResults.InvalidPassword
246
+ ? errors_1.ScraperErrorTypes.InvalidPassword
247
+ : errors_1.ScraperErrorTypes.General,
248
+ errorMessage: `Login failed with ${loginResult} error`,
249
+ };
250
+ case exports.LoginResults.ChangePassword:
251
+ this.emitProgress(definitions_1.ScraperProgressTypes.ChangePassword);
252
+ return {
253
+ success: false,
254
+ errorType: errors_1.ScraperErrorTypes.ChangePassword,
255
+ };
256
+ default:
257
+ throw new Error(`unexpected login result "${loginResult}"`);
258
+ }
292
259
  }
293
- }
294
260
  }
295
261
  exports.BaseScraperWithBrowser = BaseScraperWithBrowser;
296
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_puppeteer","_interopRequireDefault","require","_definitions","_debug","_elementsInteractions","_navigation","_baseScraper","_errors","e","__esModule","default","ownKeys","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty","getOwnPropertyDescriptors","defineProperties","defineProperty","_toPropertyKey","value","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","_objectWithoutProperties","_objectWithoutPropertiesLoose","n","indexOf","propertyIsEnumerable","hasOwnProperty","debug","getDebug","LoginBaseResults","Timeout","Generic","General","ScraperErrorTypes","rest","LoginResults","exports","getKeyByValue","object","page","key","conditions","condition","result","RegExp","test","toLowerCase","Promise","resolve","UnknownError","createGeneralError","success","errorType","BaseScraperWithBrowser","BaseScraper","constructor","args","width","height","getViewPort","_this$options$viewpor","options","viewportSize","defaultViewportSize","initialize","emitProgress","ScraperProgressTypes","Initializing","initializePage","setCacheEnabled","cleanups","close","defaultTimeout","setDefaultTimeout","preparePage","viewport","setViewport","on","request","_request$failure","failure","errorText","url","browserContext","newPage","browser","skipCloseBrowser","timeout","executablePath","showBrowser","headless","puppeteer","launch","env","verbose","DEBUG","process","undefined","prepareBrowser","navigateTo","waitUntil","retries","_this$options$navigat","navigationRetryCount","_this$page","response","goto","Error","ok","status","getLoginOptions","_credentials","companyId","fillInputs","pageOrFrame","fields","modified","input","shift","fillInput","selector","login","credentials","loginOptions","userAgent","setUserAgent","loginUrl","checkReadiness","submitButtonSelector","waitUntilElementFound","loginFrameOrPage","preAction","clickButton","LoggingIn","postAction","waitForNavigation","current","getCurrentUrl","loginResult","possibleResults","handleLoginResult","terminate","_success","Terminating","storeFailureScreenShotPath","screenshot","path","fullPage","all","reverse","map","cleanup","Success","LoginSuccess","InvalidPassword","LoginFailed","errorMessage","ChangePassword"],"sources":["../../src/scrapers/base-scraper-with-browser.ts"],"sourcesContent":["import puppeteer, { type Frame, type Page, type PuppeteerLifeCycleEvent } from 'puppeteer';\nimport { ScraperProgressTypes } from '../definitions';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, fillInput, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { BaseScraper } from './base-scraper';\nimport { ScraperErrorTypes } from './errors';\nimport { type ScraperCredentials, type ScraperScrapingResult } from './interface';\n\nconst debug = getDebug('base-scraper-with-browser');\n\nenum LoginBaseResults {\n  Success = 'SUCCESS',\n  UnknownError = 'UNKNOWN_ERROR',\n}\n\nconst {\n  Timeout, Generic, General, ...rest\n} = ScraperErrorTypes;\nexport const LoginResults = {\n  ...rest,\n  ...LoginBaseResults,\n};\n\n// eslint-disable-next-line @typescript-eslint/no-redeclare\nexport type LoginResults =\n  | Exclude<ScraperErrorTypes, ScraperErrorTypes.Timeout | ScraperErrorTypes.Generic | ScraperErrorTypes.General>\n  | LoginBaseResults;\n\nexport type PossibleLoginResults = {\n  [key in LoginResults]?: (string | RegExp | ((options?: { page?: Page }) => Promise<boolean>))[];\n};\n\nexport interface LoginOptions {\n  loginUrl: string;\n  checkReadiness?: () => Promise<void>;\n  fields: { selector: string, value: string }[];\n  submitButtonSelector: string | (() => Promise<void>);\n  preAction?: () => Promise<Frame | void>;\n  postAction?: () => Promise<void>;\n  possibleResults: PossibleLoginResults;\n  userAgent?: string;\n  waitUntil?: PuppeteerLifeCycleEvent;\n}\n\nasync function getKeyByValue(object: PossibleLoginResults, value: string, page: Page): Promise<LoginResults> {\n  const keys = Object.keys(object);\n  for (const key of keys) {\n    // @ts-ignore\n    const conditions = object[key];\n\n    for (const condition of conditions) {\n      let result = false;\n\n      if (condition instanceof RegExp) {\n        result = condition.test(value);\n      } else if (typeof condition === 'function') {\n        result = await condition({ page, value });\n      } else {\n        result = value.toLowerCase() === condition.toLowerCase();\n      }\n\n      if (result) {\n        // @ts-ignore\n        return Promise.resolve(key);\n      }\n    }\n  }\n\n  return Promise.resolve(LoginResults.UnknownError);\n}\n\nfunction createGeneralError(): ScraperScrapingResult {\n  return {\n    success: false,\n    errorType: ScraperErrorTypes.General,\n  };\n}\n\nclass BaseScraperWithBrowser<TCredentials extends ScraperCredentials> extends BaseScraper<TCredentials> {\n  private cleanups: Array<() => Promise<void>> = [];\n\n  private defaultViewportSize = {\n    width: 1024,\n    height: 768,\n  };\n\n  // NOTICE - it is discouraged to use bang (!) in general. It is used here because\n  // all the classes that inherit from this base assume is it mandatory.\n  protected page!: Page;\n\n  protected getViewPort() {\n    return this.options.viewportSize ?? this.defaultViewportSize;\n  }\n\n  async initialize() {\n    await super.initialize();\n    debug('initialize scraper');\n    this.emitProgress(ScraperProgressTypes.Initializing);\n\n    const page = await this.initializePage();\n    await page.setCacheEnabled(false); // Clear cache and avoid 300's response status\n\n    if (!page) {\n      debug('failed to initiate a browser page, exit');\n      return;\n    }\n\n    this.page = page;\n\n    this.cleanups.push(() => page.close());\n\n    if (this.options.defaultTimeout) {\n      this.page.setDefaultTimeout(this.options.defaultTimeout);\n    }\n\n    if (this.options.preparePage) {\n      debug(\"execute 'preparePage' interceptor provided in options\");\n      await this.options.preparePage(this.page);\n    }\n\n    const viewport = this.getViewPort();\n    debug(`set viewport to width ${viewport.width}, height ${viewport.height}`);\n    await this.page.setViewport({\n      width: viewport.width,\n      height: viewport.height,\n    });\n\n    this.page.on('requestfailed', (request) => {\n      debug('Request failed: %s %s', request.failure()?.errorText, request.url());\n    });\n  }\n\n  private async initializePage() {\n    debug('initialize browser page');\n    if ('browserContext' in this.options) {\n      debug('Using the browser context provided in options');\n      return this.options.browserContext.newPage();\n    }\n\n    if ('browser' in this.options) {\n      debug('Using the browser instance provided in options');\n      const { browser } = this.options;\n\n      /**\n       * For backward compatibility, we will close the browser even if we didn't create it\n       */\n      if (!this.options.skipCloseBrowser) {\n        this.cleanups.push(async () => {\n          debug('closing the browser');\n          await browser.close();\n        });\n      }\n\n      return browser.newPage();\n    }\n\n    const { timeout, args, executablePath, showBrowser } = this.options;\n\n    const headless = !showBrowser;\n    debug(`launch a browser with headless mode = ${headless}`);\n\n    const browser = await puppeteer.launch({\n      env: this.options.verbose ? { DEBUG: '*', ...process.env } : undefined,\n      headless,\n      executablePath,\n      args,\n      timeout,\n    });\n\n    this.cleanups.push(async () => {\n      debug('closing the browser');\n      await browser.close();\n    });\n\n    if (this.options.prepareBrowser) {\n      debug(\"execute 'prepareBrowser' interceptor provided in options\");\n      await this.options.prepareBrowser(browser);\n    }\n\n    debug('create a new browser page');\n    return browser.newPage();\n  }\n\n  async navigateTo(\n    url: string,\n    waitUntil: PuppeteerLifeCycleEvent | undefined = 'load',\n    retries = this.options.navigationRetryCount ?? 0,\n  ): Promise<void> {\n    const response = await this.page?.goto(url, { waitUntil });\n    if (response === null) {\n      // note: response will be null when navigating to same url while changing the hash part.\n      // the condition below will always accept null as valid result.\n      return;\n    }\n\n    if (!response) {\n      throw new Error(\n        `Error while trying to navigate to url ${url}, response is undefined`,\n      );\n    }\n\n    if (!response.ok()) {\n      const status = response.status();\n      if (retries > 0) {\n        debug(`Failed to navigate to url ${url}, status code: ${status}, retrying ${retries} more times`);\n        await this.navigateTo(url, waitUntil, retries - 1);\n      } else {\n        throw new Error( `Failed to navigate to url ${url}, status code: ${status}`);\n      }\n    }\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  getLoginOptions(_credentials: ScraperCredentials): LoginOptions {\n    throw new Error(`getLoginOptions() is not created in ${this.options.companyId}`);\n  }\n\n  async fillInputs(pageOrFrame: Page | Frame, fields: { selector: string, value: string }[]): Promise<void> {\n    const modified = [...fields];\n    const input = modified.shift();\n\n    if (!input) {\n      return;\n    }\n    await fillInput(pageOrFrame, input.selector, input.value);\n    if (modified.length) {\n      await this.fillInputs(pageOrFrame, modified);\n    }\n  }\n\n  async login(credentials: ScraperCredentials): Promise<ScraperScrapingResult> {\n    if (!credentials || !this.page) {\n      return createGeneralError();\n    }\n\n    debug('execute login process');\n    const loginOptions = this.getLoginOptions(credentials);\n\n    if (loginOptions.userAgent) {\n      debug('set custom user agent provided in options');\n      await this.page.setUserAgent(loginOptions.userAgent);\n    }\n\n    debug('navigate to login url');\n    await this.navigateTo(loginOptions.loginUrl, loginOptions.waitUntil);\n    if (loginOptions.checkReadiness) {\n      debug(\"execute 'checkReadiness' interceptor provided in login options\");\n      await loginOptions.checkReadiness();\n    } else if (typeof loginOptions.submitButtonSelector === 'string') {\n      debug('wait until submit button is available');\n      await waitUntilElementFound(this.page, loginOptions.submitButtonSelector);\n    }\n\n    let loginFrameOrPage: Page | Frame | null = this.page;\n    if (loginOptions.preAction) {\n      debug(\"execute 'preAction' interceptor provided in login options\");\n      loginFrameOrPage = (await loginOptions.preAction()) || this.page;\n    }\n\n    debug('fill login components input with relevant values');\n    await this.fillInputs(loginFrameOrPage, loginOptions.fields);\n    debug('click on login submit button');\n    if (typeof loginOptions.submitButtonSelector === 'string') {\n      await clickButton(loginFrameOrPage, loginOptions.submitButtonSelector);\n    } else {\n      await loginOptions.submitButtonSelector();\n    }\n    this.emitProgress(ScraperProgressTypes.LoggingIn);\n\n    if (loginOptions.postAction) {\n      debug(\"execute 'postAction' interceptor provided in login options\");\n      await loginOptions.postAction();\n    } else {\n      debug('wait for page navigation');\n      await waitForNavigation(this.page);\n    }\n\n    debug('check login result');\n    const current = await getCurrentUrl(this.page, true);\n    const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);\n    debug(`handle login results ${loginResult}`);\n    return this.handleLoginResult(loginResult);\n  }\n\n  async terminate(_success: boolean) {\n    debug(`terminating browser with success = ${_success}`);\n    this.emitProgress(ScraperProgressTypes.Terminating);\n\n    if (!_success && !!this.options.storeFailureScreenShotPath) {\n      debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);\n      await this.page.screenshot({\n        path: this.options.storeFailureScreenShotPath,\n        fullPage: true,\n      });\n    }\n\n    await Promise.all(this.cleanups.reverse().map((cleanup) => cleanup()));\n    this.cleanups = [];\n  }\n\n  private handleLoginResult(loginResult: LoginResults) {\n    switch (loginResult) {\n      case LoginResults.Success:\n        this.emitProgress(ScraperProgressTypes.LoginSuccess);\n        return { success: true };\n      case LoginResults.InvalidPassword:\n      case LoginResults.UnknownError:\n        this.emitProgress(ScraperProgressTypes.LoginFailed);\n        return {\n          success: false,\n          errorType:\n            loginResult === LoginResults.InvalidPassword ?\n              ScraperErrorTypes.InvalidPassword :\n              ScraperErrorTypes.General,\n          errorMessage: `Login failed with ${loginResult} error`,\n        };\n      case LoginResults.ChangePassword:\n        this.emitProgress(ScraperProgressTypes.ChangePassword);\n        return {\n          success: false,\n          errorType: ScraperErrorTypes.ChangePassword,\n        };\n      default:\n        throw new Error(`unexpected login result \"${loginResult}\"`);\n    }\n  }\n}\n\nexport { BaseScraperWithBrowser };\n"],"mappings":"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","ignoreList":[]}
262
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-scraper-with-browser.js","sourceRoot":"","sources":["../../src/scrapers/base-scraper-with-browser.ts"],"names":[],"mappings":"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","sourcesContent":["import puppeteer, { type Frame, type Page, type PuppeteerLifeCycleEvent } from 'puppeteer';\nimport { ScraperProgressTypes } from '../definitions';\nimport { getDebug } from '../helpers/debug';\nimport { clickButton, fillInput, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { BaseScraper } from './base-scraper';\nimport { ScraperErrorTypes } from './errors';\nimport { type ScraperCredentials, type ScraperScrapingResult } from './interface';\n\nconst debug = getDebug('base-scraper-with-browser');\n\nenum LoginBaseResults {\n  Success = 'SUCCESS',\n  UnknownError = 'UNKNOWN_ERROR',\n}\n\nconst { Timeout, Generic, General, ...rest } = ScraperErrorTypes;\nexport const LoginResults = {\n  ...rest,\n  ...LoginBaseResults,\n};\n\n// eslint-disable-next-line @typescript-eslint/no-redeclare\nexport type LoginResults =\n  | Exclude<ScraperErrorTypes, ScraperErrorTypes.Timeout | ScraperErrorTypes.Generic | ScraperErrorTypes.General>\n  | LoginBaseResults;\n\nexport type PossibleLoginResults = {\n  [key in LoginResults]?: (string | RegExp | ((options?: { page?: Page }) => Promise<boolean>))[];\n};\n\nexport interface LoginOptions {\n  loginUrl: string;\n  checkReadiness?: () => Promise<void>;\n  fields: { selector: string; value: string }[];\n  submitButtonSelector: string | (() => Promise<void>);\n  preAction?: () => Promise<Frame | void>;\n  postAction?: () => Promise<void>;\n  possibleResults: PossibleLoginResults;\n  userAgent?: string;\n  waitUntil?: PuppeteerLifeCycleEvent;\n}\n\nasync function getKeyByValue(object: PossibleLoginResults, value: string, page: Page): Promise<LoginResults> {\n  const keys = Object.keys(object);\n  for (const key of keys) {\n    // @ts-ignore\n    const conditions = object[key];\n\n    for (const condition of conditions) {\n      let result = false;\n\n      if (condition instanceof RegExp) {\n        result = condition.test(value);\n      } else if (typeof condition === 'function') {\n        result = await condition({ page, value });\n      } else {\n        result = value.toLowerCase() === condition.toLowerCase();\n      }\n\n      if (result) {\n        // @ts-ignore\n        return Promise.resolve(key);\n      }\n    }\n  }\n\n  return Promise.resolve(LoginResults.UnknownError);\n}\n\nfunction createGeneralError(): ScraperScrapingResult {\n  return {\n    success: false,\n    errorType: ScraperErrorTypes.General,\n  };\n}\n\nclass BaseScraperWithBrowser<TCredentials extends ScraperCredentials> extends BaseScraper<TCredentials> {\n  private cleanups: Array<() => Promise<void>> = [];\n\n  private defaultViewportSize = {\n    width: 1024,\n    height: 768,\n  };\n\n  // NOTICE - it is discouraged to use bang (!) in general. It is used here because\n  // all the classes that inherit from this base assume is it mandatory.\n  protected page!: Page;\n\n  protected getViewPort() {\n    return this.options.viewportSize ?? this.defaultViewportSize;\n  }\n\n  async initialize() {\n    await super.initialize();\n    debug('initialize scraper');\n    this.emitProgress(ScraperProgressTypes.Initializing);\n\n    const page = await this.initializePage();\n    await page.setCacheEnabled(false); // Clear cache and avoid 300's response status\n\n    if (!page) {\n      debug('failed to initiate a browser page, exit');\n      return;\n    }\n\n    this.page = page;\n\n    this.cleanups.push(() => page.close());\n\n    if (this.options.defaultTimeout) {\n      this.page.setDefaultTimeout(this.options.defaultTimeout);\n    }\n\n    if (this.options.preparePage) {\n      debug(\"execute 'preparePage' interceptor provided in options\");\n      await this.options.preparePage(this.page);\n    }\n\n    const viewport = this.getViewPort();\n    debug(`set viewport to width ${viewport.width}, height ${viewport.height}`);\n    await this.page.setViewport({\n      width: viewport.width,\n      height: viewport.height,\n    });\n\n    this.page.on('requestfailed', request => {\n      debug('Request failed: %s %s', request.failure()?.errorText, request.url());\n    });\n  }\n\n  private async initializePage() {\n    debug('initialize browser page');\n    if ('browserContext' in this.options) {\n      debug('Using the browser context provided in options');\n      return this.options.browserContext.newPage();\n    }\n\n    if ('browser' in this.options) {\n      debug('Using the browser instance provided in options');\n      const { browser } = this.options;\n\n      /**\n       * For backward compatibility, we will close the browser even if we didn't create it\n       */\n      if (!this.options.skipCloseBrowser) {\n        this.cleanups.push(async () => {\n          debug('closing the browser');\n          await browser.close();\n        });\n      }\n\n      return browser.newPage();\n    }\n\n    const { timeout, args, executablePath, showBrowser } = this.options;\n\n    const headless = !showBrowser;\n    debug(`launch a browser with headless mode = ${headless}`);\n\n    const browser = await puppeteer.launch({\n      env: this.options.verbose ? { DEBUG: '*', ...process.env } : undefined,\n      headless,\n      executablePath,\n      args,\n      timeout,\n    });\n\n    this.cleanups.push(async () => {\n      debug('closing the browser');\n      await browser.close();\n    });\n\n    if (this.options.prepareBrowser) {\n      debug(\"execute 'prepareBrowser' interceptor provided in options\");\n      await this.options.prepareBrowser(browser);\n    }\n\n    debug('create a new browser page');\n    return browser.newPage();\n  }\n\n  async navigateTo(\n    url: string,\n    waitUntil: PuppeteerLifeCycleEvent | undefined = 'load',\n    retries = this.options.navigationRetryCount ?? 0,\n  ): Promise<void> {\n    const response = await this.page?.goto(url, { waitUntil });\n    if (response === null) {\n      // note: response will be null when navigating to same url while changing the hash part.\n      // the condition below will always accept null as valid result.\n      return;\n    }\n\n    if (!response) {\n      throw new Error(`Error while trying to navigate to url ${url}, response is undefined`);\n    }\n\n    if (!response.ok()) {\n      const status = response.status();\n      if (retries > 0) {\n        debug(`Failed to navigate to url ${url}, status code: ${status}, retrying ${retries} more times`);\n        await this.navigateTo(url, waitUntil, retries - 1);\n      } else {\n        throw new Error(`Failed to navigate to url ${url}, status code: ${status}`);\n      }\n    }\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  getLoginOptions(_credentials: ScraperCredentials): LoginOptions {\n    throw new Error(`getLoginOptions() is not created in ${this.options.companyId}`);\n  }\n\n  async fillInputs(pageOrFrame: Page | Frame, fields: { selector: string; value: string }[]): Promise<void> {\n    const modified = [...fields];\n    const input = modified.shift();\n\n    if (!input) {\n      return;\n    }\n    await fillInput(pageOrFrame, input.selector, input.value);\n    if (modified.length) {\n      await this.fillInputs(pageOrFrame, modified);\n    }\n  }\n\n  async login(credentials: ScraperCredentials): Promise<ScraperScrapingResult> {\n    if (!credentials || !this.page) {\n      return createGeneralError();\n    }\n\n    debug('execute login process');\n    const loginOptions = this.getLoginOptions(credentials);\n\n    if (loginOptions.userAgent) {\n      debug('set custom user agent provided in options');\n      await this.page.setUserAgent(loginOptions.userAgent);\n    }\n\n    debug('navigate to login url');\n    await this.navigateTo(loginOptions.loginUrl, loginOptions.waitUntil);\n    if (loginOptions.checkReadiness) {\n      debug(\"execute 'checkReadiness' interceptor provided in login options\");\n      await loginOptions.checkReadiness();\n    } else if (typeof loginOptions.submitButtonSelector === 'string') {\n      debug('wait until submit button is available');\n      await waitUntilElementFound(this.page, loginOptions.submitButtonSelector);\n    }\n\n    let loginFrameOrPage: Page | Frame | null = this.page;\n    if (loginOptions.preAction) {\n      debug(\"execute 'preAction' interceptor provided in login options\");\n      loginFrameOrPage = (await loginOptions.preAction()) || this.page;\n    }\n\n    debug('fill login components input with relevant values');\n    await this.fillInputs(loginFrameOrPage, loginOptions.fields);\n    debug('click on login submit button');\n    if (typeof loginOptions.submitButtonSelector === 'string') {\n      await clickButton(loginFrameOrPage, loginOptions.submitButtonSelector);\n    } else {\n      await loginOptions.submitButtonSelector();\n    }\n    this.emitProgress(ScraperProgressTypes.LoggingIn);\n\n    if (loginOptions.postAction) {\n      debug(\"execute 'postAction' interceptor provided in login options\");\n      await loginOptions.postAction();\n    } else {\n      debug('wait for page navigation');\n      await waitForNavigation(this.page);\n    }\n\n    debug('check login result');\n    const current = await getCurrentUrl(this.page, true);\n    const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);\n    debug(`handle login results ${loginResult}`);\n    return this.handleLoginResult(loginResult);\n  }\n\n  async terminate(_success: boolean) {\n    debug(`terminating browser with success = ${_success}`);\n    this.emitProgress(ScraperProgressTypes.Terminating);\n\n    if (!_success && !!this.options.storeFailureScreenShotPath) {\n      debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);\n      await this.page.screenshot({\n        path: this.options.storeFailureScreenShotPath,\n        fullPage: true,\n      });\n    }\n\n    await Promise.all(this.cleanups.reverse().map(cleanup => cleanup()));\n    this.cleanups = [];\n  }\n\n  private handleLoginResult(loginResult: LoginResults) {\n    switch (loginResult) {\n      case LoginResults.Success:\n        this.emitProgress(ScraperProgressTypes.LoginSuccess);\n        return { success: true };\n      case LoginResults.InvalidPassword:\n      case LoginResults.UnknownError:\n        this.emitProgress(ScraperProgressTypes.LoginFailed);\n        return {\n          success: false,\n          errorType:\n            loginResult === LoginResults.InvalidPassword\n              ? ScraperErrorTypes.InvalidPassword\n              : ScraperErrorTypes.General,\n          errorMessage: `Login failed with ${loginResult} error`,\n        };\n      case LoginResults.ChangePassword:\n        this.emitProgress(ScraperProgressTypes.ChangePassword);\n        return {\n          success: false,\n          errorType: ScraperErrorTypes.ChangePassword,\n        };\n      default:\n        throw new Error(`unexpected login result \"${loginResult}\"`);\n    }\n  }\n}\n\nexport { BaseScraperWithBrowser };\n"]}