israeli-bank-scrapers 3.8.1 → 3.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/definitions.d.ts +10 -0
- package/lib/definitions.js +15 -2
- package/lib/helpers/waiting.d.ts +4 -1
- package/lib/helpers/waiting.js +9 -4
- package/lib/scrapers/base-isracard-amex.js +9 -9
- package/lib/scrapers/base-scraper-with-browser.d.ts +1 -0
- package/lib/scrapers/base-scraper-with-browser.js +36 -34
- package/lib/scrapers/base-scraper.d.ts +9 -16
- package/lib/scrapers/base-scraper.js +8 -19
- package/lib/scrapers/factory.test.d.ts +1 -0
- package/lib/scrapers/factory.test.js +21 -0
- package/lib/scrapers/interface.d.ts +6 -1
- package/lib/scrapers/visa-cal.js +5 -6
- package/package.json +1 -1
package/lib/definitions.d.ts
CHANGED
|
@@ -88,3 +88,13 @@ export declare const SCRAPERS: {
|
|
|
88
88
|
loginFields: string[];
|
|
89
89
|
};
|
|
90
90
|
};
|
|
91
|
+
export declare enum ScraperProgressTypes {
|
|
92
|
+
Initializing = "INITIALIZING",
|
|
93
|
+
StartScraping = "START_SCRAPING",
|
|
94
|
+
LoggingIn = "LOGGING_IN",
|
|
95
|
+
LoginSuccess = "LOGIN_SUCCESS",
|
|
96
|
+
LoginFailed = "LOGIN_FAILED",
|
|
97
|
+
ChangePassword = "CHANGE_PASSWORD",
|
|
98
|
+
EndScraping = "END_SCRAPING",
|
|
99
|
+
Terminating = "TERMINATING"
|
|
100
|
+
}
|
package/lib/definitions.js
CHANGED
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.SCRAPERS = exports.CompanyTypes = exports.PASSWORD_FIELD = void 0;
|
|
6
|
+
exports.ScraperProgressTypes = exports.SCRAPERS = exports.CompanyTypes = exports.PASSWORD_FIELD = void 0;
|
|
7
7
|
// NOTICE: avoid changing exported keys as they are part of the public api
|
|
8
8
|
const PASSWORD_FIELD = 'password';
|
|
9
9
|
exports.PASSWORD_FIELD = PASSWORD_FIELD;
|
|
@@ -103,4 +103,17 @@ const SCRAPERS = {
|
|
|
103
103
|
}
|
|
104
104
|
};
|
|
105
105
|
exports.SCRAPERS = SCRAPERS;
|
|
106
|
-
|
|
106
|
+
let ScraperProgressTypes;
|
|
107
|
+
exports.ScraperProgressTypes = ScraperProgressTypes;
|
|
108
|
+
|
|
109
|
+
(function (ScraperProgressTypes) {
|
|
110
|
+
ScraperProgressTypes["Initializing"] = "INITIALIZING";
|
|
111
|
+
ScraperProgressTypes["StartScraping"] = "START_SCRAPING";
|
|
112
|
+
ScraperProgressTypes["LoggingIn"] = "LOGGING_IN";
|
|
113
|
+
ScraperProgressTypes["LoginSuccess"] = "LOGIN_SUCCESS";
|
|
114
|
+
ScraperProgressTypes["LoginFailed"] = "LOGIN_FAILED";
|
|
115
|
+
ScraperProgressTypes["ChangePassword"] = "CHANGE_PASSWORD";
|
|
116
|
+
ScraperProgressTypes["EndScraping"] = "END_SCRAPING";
|
|
117
|
+
ScraperProgressTypes["Terminating"] = "TERMINATING";
|
|
118
|
+
})(ScraperProgressTypes || (exports.ScraperProgressTypes = ScraperProgressTypes = {}));
|
|
119
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9kZWZpbml0aW9ucy50cyJdLCJuYW1lcyI6WyJQQVNTV09SRF9GSUVMRCIsIkNvbXBhbnlUeXBlcyIsIlNDUkFQRVJTIiwiaGFwb2FsaW0iLCJuYW1lIiwibG9naW5GaWVsZHMiLCJoYXBvYWxpbUJlT25saW5lIiwibGV1bWkiLCJtaXpyYWhpIiwiZGlzY291bnQiLCJvdHNhckhhaGF5YWwiLCJsZXVtaUNhcmQiLCJtYXgiLCJ2aXNhQ2FsIiwiaXNyYWNhcmQiLCJhbWV4IiwidW5pb24iLCJiZWlubGV1bWkiLCJtYXNzYWQiLCJ5YWhhdiIsImJleWFoYWRCaXNodmlsaGEiLCJvbmVaZXJvIiwiU2NyYXBlclByb2dyZXNzVHlwZXMiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBO0FBRU8sTUFBTUEsY0FBYyxHQUFHLFVBQXZCOztJQUVLQyxZOzs7V0FBQUEsWTtBQUFBQSxFQUFBQSxZO0FBQUFBLEVBQUFBLFk7QUFBQUEsRUFBQUEsWTtBQUFBQSxFQUFBQSxZO0FBQUFBLEVBQUFBLFk7QUFBQUEsRUFBQUEsWTtBQUFBQSxFQUFBQSxZO0FBQUFBLEVBQUFBLFk7QUFBQUEsRUFBQUEsWTtBQUFBQSxFQUFBQSxZO0FBQUFBLEVBQUFBLFk7QUFBQUEsRUFBQUEsWTtBQUFBQSxFQUFBQSxZO0FBQUFBLEVBQUFBLFk7QUFBQUEsRUFBQUEsWTtBQUFBQSxFQUFBQSxZO0FBQUFBLEVBQUFBLFk7R0FBQUEsWSw0QkFBQUEsWTs7QUFvQkwsTUFBTUMsUUFBUSxHQUFHO0FBQ3RCLEdBQUNELFlBQVksQ0FBQ0UsUUFBZCxHQUF5QjtBQUN2QkMsSUFBQUEsSUFBSSxFQUFFLGVBRGlCO0FBRXZCQyxJQUFBQSxXQUFXLEVBQUUsQ0FBQyxVQUFELEVBQWFMLGNBQWI7QUFGVSxHQURIO0FBS3RCLEdBQUNDLFlBQVksQ0FBQ0ssZ0JBQWQsR0FBaUM7QUFBRTtBQUNqQ0YsSUFBQUEsSUFBSSxFQUFFLGVBRHlCO0FBRS9CQyxJQUFBQSxXQUFXLEVBQUUsQ0FBQyxVQUFELEVB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RGdCO0FBRXRCQyxJQUFBQSxXQUFXLEVBQUUsQ0FBQyxPQUFELEVBQVVMLGNBQVYsRUFBMEIsa0JBQTFCLEVBQThDLGFBQTlDLEVBQTZELGtCQUE3RDtBQUZTO0FBakVGLENBQWpCOztJQXVFS3NCLG9COzs7V0FBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7QUFBQUEsRUFBQUEsb0I7R0FBQUEsb0Isb0NBQUFBLG9CIiwic291cmNlc0NvbnRlbnQiOlsiLy8gTk9USUNFOiBhdm9pZCBjaGFuZ2luZyBleHBvcnRlZCBrZXlzIGFzIHRoZXkgYXJlIHBhcnQgb2YgdGhlIHB1YmxpYyBhcGlcblxuZXhwb3J0IGNvbnN0IFBBU1NXT1JEX0ZJRUxEID0gJ3Bhc3N3b3JkJztcblxuZXhwb3J0IGVudW0gQ29tcGFueVR5cGVzIHtcbiAgaGFwb2FsaW0gPSAnaGFwb2FsaW0nLFxuICBoYXBvYWxpbUJlT25saW5lID0gJ2hhcG9hbGltQmVPbmxpbmUnLFxuICBiZWlubGV1bWkgPSAnYmVpbmxldW1pJyxcbiAgdW5pb24gPSAndW5pb24nLFxuICBhbWV4ID0gJ2FtZXgnLFxuICBpc3JhY2FyZCA9ICdpc3JhY2FyZCcsXG4gIHZpc2FDYWwgPSAndmlzYUNhbCcsXG4gIG1heCA9ICdtYXgnLFxuICBsZXVtaUNhcmQgPSAnbGV1bWlDYXJkJyxcbiAgb3RzYXJIYWhheWFsID0gJ290c2FySGFoYXlhbCcsXG4gIGRpc2NvdW50ID0gJ2Rpc2NvdW50JyxcbiAgbWl6cmFoaSA9ICdtaXpyYWhpJyxcbiAgbGV1bWkgPSAnbGV1bWknLFxuICBtYXNzYWQgPSAnbWFzc2FkJyxcbiAgeWFoYXYgPSAneWFoYXYnLFxuICBiZXlhaGFkQmlzaHZpbGhhID0gJ2JleWFoYWRCaXNodmlsaGEnLFxuICBvbmVaZXJvID0gJ29uZVplcm8nXG59XG5cbmV4cG9ydCBjb25zdCBTQ1JBUEVSUyA9IHtcbiAgW0NvbXBhbnlUeXBlcy5oYXBvYWxpbV06IHtcbiAgICBuYW1lOiAnQmFuayBIYXBvYWxpbScsXG4gICAgbG9naW5GaWVsZHM6IFsndXNlckNvZGUnLCBQQVNTV09SRF9GSUVMRF0sXG4gIH0sXG4gIFtDb21wYW55VHlwZXMuaGFwb2FsaW1CZU9ubGluZV06IHsgLy8gVE9ETyByZW1vdmUgaW4gTWFqb3IgdmVyc2lvblxuICAgIG5hbWU6ICdCYW5rIEhhcG9hbGltJyxcbiAgICBsb2dpbkZpZWxkczogWyd1c2VyQ29kZScsIFBBU1NXT1JEX0ZJRUxEXSxcbiAgfSxcbiAgW0NvbXBhbnlUeXBlcy5sZXVtaV06IHtcbiAgICBuYW1lOiAnQmFuayBMZXVtaScsXG4gICAgbG9naW5GaWVsZHM6IFsndXNlcm5hbWUnLCBQQVNTV09SRF9GSUVMRF0sXG4gIH0sXG4gIFtDb21wYW55VHlwZXMubWl6cmFoaV06IHtcbiAgICBuYW1lOiAnTWl6cmFoaSBCYW5rJyxcbiAgICBsb2dpbkZpZWxkczogWyd1c2VybmFtZScsIFBBU1NXT1JEX0ZJRUxEXSxcbiAgfSxcbiAgW0NvbXBhbnlUeXBlcy5kaXNjb3VudF06IHtcbiAgICBuYW1lOiAnRGlzY291bnQgQmFuaycsXG4gICAgbG9naW5GaWVsZHM6IFsnaWQnLCBQQVNTV09SRF9GSUVMRCwgJ251bSddLFxuICB9LFxuICBbQ29tcGFueVR5cGVzLm90c2FySGFoYXlhbF06IHtcbiAgICBuYW1lOiAnQmFuayBPdHNhciBIYWhheWFsJyxcbiAgICBsb2dpbkZpZWxkczogWyd1c2VybmFtZScsIFBBU1NXT1JEX0ZJRUxEXSxcbiAgfSxcbiAgW0NvbXBhbnlUeXBlcy5sZXVtaUNhcmRdOiB7IC8vIFRPRE8gcmVtb3ZlIGluIE1ham9yIHZlcnNpb25cbiAgICBuYW1lOiAnTGV1bWkgQ2FyZCcsXG4gICAgbG9naW5GaWVsZHM6IFsndXNlcm5hbWUnLCBQQVNTV09SRF9GSUVMRF0sXG4gIH0sXG4gIFtDb21wYW55VHlwZXMubWF4XToge1xuICAgIG5hbWU6ICdNYXgnLFxuICAgIGxvZ2luRmllbGRzOiBbJ3VzZXJuYW1lJywgUEFTU1dPUkRfRklFTERdLFxuICB9LFxuICBbQ29tcGFueVR5cGVzLnZpc2FDYWxdOiB7XG4gICAgbmFtZTogJ1Zpc2EgQ2FsJyxcbiAgICBsb2dpbkZpZWxkczogWyd1c2VybmFtZScsIFBBU1NXT1JEX0ZJRUxEXSxcbiAgfSxcbiAgW0NvbXBhbnlUeXBlcy5pc3JhY2FyZF06IHtcbiAgICBuYW1lOiAnSXNyYWNhcmQnLFxuICAgIGxvZ2luRmllbGRzOiBbJ2lkJywgJ2NhcmQ2RGlnaXRzJywgUEFTU1dPUkRfRklFTERdLFxuICB9LFxuICBbQ29tcGFueVR5cGVzLmFtZXhdOiB7XG4gICAgbmFtZTogJ0FtZXgnLFxuICAgIGxvZ2luRmllbGRzOiBbJ2lkJywgJ2NhcmQ2RGlnaXRzJywgUEFTU1dPUkRfRklFTERdLFxuICB9LFxuICBbQ29tcGFueVR5cGVzLnVuaW9uXToge1xuICAgIG5hbWU6ICdVbmlvbicsXG4gICAgbG9naW5GaWVsZHM6IFsndXNlcm5hbWUnLCBQQVNTV09SRF9GSUVMRF0sXG4gIH0sXG4gIFtDb21wYW55VHlwZXMuYmVpbmxldW1pXToge1xuICAgIG5hbWU6ICdCZWlubGV1bWknLFxuICAgIGxvZ2luRmllbGRzOiBbJ3VzZXJuYW1lJywgUEFTU1dPUkRfRklFTERdLFxuICB9LFxuICBbQ29tcGFueVR5cGVzLm1hc3NhZF06IHtcbiAgICBuYW1lOiAnTWFzc2FkJyxcbiAgICBsb2dpbkZpZWxkczogWyd1c2VybmFtZScsIFBBU1NXT1JEX0ZJRUxEXSxcbiAgfSxcbiAgW0NvbXBhbnlUeXBlcy55YWhhdl06IHtcbiAgICBuYW1lOiAnQmFuayBZYWhhdicsXG4gICAgbG9naW5GaWVsZHM6IFsndXNlcm5hbWUnLCAnbmF0aW9uYWxJRCcsIFBBU1NXT1JEX0ZJRUxEXSxcbiAgfSxcbiAgW0NvbXBhbnlUeXBlcy5iZXlhaGFkQmlzaHZpbGhhXToge1xuICAgIG5hbWU6ICdCZXlhaGFkIEJpc2h2aWxoYScsXG4gICAgbG9naW5GaWVsZHM6IFsnaWQnLCBQQVNTV09SRF9GSUVMRF0sXG4gIH0sXG4gIFtDb21wYW55VHlwZXMub25lWmVyb106IHtcbiAgICBuYW1lOiAnT25lIFplcm8nLFxuICAgIGxvZ2luRmllbGRzOiBbJ2VtYWlsJywgUEFTU1dPUkRfRklFTEQsICdvdHBDb2RlUmV0cmlldmVyJywgJ3Bob25lTnVtYmVyJywgJ290cExvbmdUZXJtVG9rZW4nXSxcbiAgfSxcbn07XG5cbmV4cG9ydCBlbnVtIFNjcmFwZXJQcm9ncmVzc1R5cGVzIHtcbiAgSW5pdGlhbGl6aW5nID0gJ0lOSVRJQUxJWklORycsXG4gIFN0YXJ0U2NyYXBpbmcgPSAnU1RBUlRfU0NSQVBJTkcnLFxuICBMb2dnaW5nSW4gPSAnTE9HR0lOR19JTicsXG4gIExvZ2luU3VjY2VzcyA9ICdMT0dJTl9TVUNDRVNTJyxcbiAgTG9naW5GYWlsZWQgPSAnTE9HSU5fRkFJTEVEJyxcbiAgQ2hhbmdlUGFzc3dvcmQgPSAnQ0hBTkdFX1BBU1NXT1JEJyxcbiAgRW5kU2NyYXBpbmcgPSAnRU5EX1NDUkFQSU5HJyxcbiAgVGVybWluYXRpbmcgPSAnVEVSTUlOQVRJTkcnLFxufVxuIl19
|
package/lib/helpers/waiting.d.ts
CHANGED
|
@@ -1,6 +1,9 @@
|
|
|
1
1
|
export declare class TimeoutError extends Error {
|
|
2
2
|
}
|
|
3
3
|
export declare const SECOND = 1000;
|
|
4
|
-
|
|
4
|
+
/**
|
|
5
|
+
* Wait until a promise resolves with a truthy value or reject after a timeout
|
|
6
|
+
*/
|
|
7
|
+
export declare function waitUntil<T>(asyncTest: () => Promise<T>, description?: string, timeout?: number, interval?: number): Promise<T>;
|
|
5
8
|
export declare function raceTimeout(ms: number, promise: Promise<any>): Promise<any>;
|
|
6
9
|
export declare function runSerial<T>(actions: (() => Promise<T>)[]): Promise<T[]>;
|
package/lib/helpers/waiting.js
CHANGED
|
@@ -26,15 +26,20 @@ function timeoutPromise(ms, promise, description) {
|
|
|
26
26
|
reject(error);
|
|
27
27
|
}, ms);
|
|
28
28
|
});
|
|
29
|
-
return Promise.race([promise,
|
|
29
|
+
return Promise.race([promise, // casting to avoid type error- safe since this promise will always reject
|
|
30
|
+
timeout]);
|
|
30
31
|
}
|
|
32
|
+
/**
|
|
33
|
+
* Wait until a promise resolves with a truthy value or reject after a timeout
|
|
34
|
+
*/
|
|
35
|
+
|
|
31
36
|
|
|
32
37
|
function waitUntil(asyncTest, description = '', timeout = 10000, interval = 100) {
|
|
33
38
|
const promise = new Promise((resolve, reject) => {
|
|
34
39
|
function wait() {
|
|
35
40
|
asyncTest().then(value => {
|
|
36
|
-
if (value
|
|
37
|
-
resolve();
|
|
41
|
+
if (value) {
|
|
42
|
+
resolve(value);
|
|
38
43
|
} else {
|
|
39
44
|
setTimeout(wait, interval);
|
|
40
45
|
}
|
|
@@ -57,4 +62,4 @@ function raceTimeout(ms, promise) {
|
|
|
57
62
|
function runSerial(actions) {
|
|
58
63
|
return actions.reduce((m, a) => m.then(async x => [...x, await a()]), Promise.resolve(new Array()));
|
|
59
64
|
}
|
|
60
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
65
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9oZWxwZXJzL3dhaXRpbmcudHMiXSwibmFtZXMiOlsiVGltZW91dEVycm9yIiwiRXJyb3IiLCJTRUNPTkQiLCJ0aW1lb3V0UHJvbWlzZSIsIm1zIiwicHJvbWlzZSIsImRlc2NyaXB0aW9uIiwidGltZW91dCIsIlByb21pc2UiLCJfIiwicmVqZWN0IiwiaWQiLCJzZXRUaW1lb3V0IiwiY2xlYXJUaW1lb3V0IiwiZXJyb3IiLCJyYWNlIiwid2FpdFVudGlsIiwiYXN5bmNUZXN0IiwiaW50ZXJ2YWwiLCJyZXNvbHZlIiwid2FpdCIsInRoZW4iLCJ2YWx1ZSIsImNhdGNoIiwicmFjZVRpbWVvdXQiLCJlcnIiLCJydW5TZXJpYWwiLCJhY3Rpb25zIiwicmVkdWNlIiwibSIsImEiLCJ4IiwiQXJyYXkiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7O0FBQ08sTUFBTUEsWUFBTixTQUEyQkMsS0FBM0IsQ0FBaUM7OztBQUlqQyxNQUFNQyxNQUFNLEdBQUcsSUFBZjs7O0FBRVAsU0FBU0MsY0FBVCxDQUEyQkMsRUFBM0IsRUFBdUNDLE9BQXZDLEVBQTREQyxXQUE1RCxFQUE2RjtBQUMzRixRQUFNQyxPQUFPLEdBQUcsSUFBSUMsT0FBSixDQUFZLENBQUNDLENBQUQsRUFBSUMsTUFBSixLQUFlO0FBQ3pDLFVBQU1DLEVBQUUsR0FBR0MsVUFBVSxDQUFDLE1BQU07QUFDMUJDLE1BQUFBLFlBQVksQ0FBQ0YsRUFBRCxDQUFaO0FBQ0EsWUFBTUcsS0FBSyxHQUFHLElBQUlkLFlBQUosQ0FBaUJNLFdBQWpCLENBQWQ7QUFDQUksTUFBQUEsTUFBTSxDQUFDSSxLQUFELENBQU47QUFDRCxLQUpvQixFQUlsQlYsRUFKa0IsQ0FBckI7QUFLRCxHQU5lLENBQWhCO0FBUUEsU0FBT0ksT0FBTyxDQUFDTyxJQUFSLENBQWEsQ0FDbEJWLE9BRGtCLEVBRWxCO0FBQ0FFLEVBQUFBLE9BSGtCLENBQWIsQ0FBUDtBQUtEO0FBRUQ7Ozs7O0FBR08sU0FBU1MsU0FBVCxDQUFzQkMsU0FBdEIsRUFBbURYLFdBQVcsR0FBRyxFQUFqRSxFQUFxRUMsT0FBTyxHQUFHLEtBQS9FLEVBQXNGVyxRQUFRLEdBQUcsR0FBakcsRUFBc0c7QUFDM0csUUFBTWIsT0FBTyxHQUFHLElBQUlHLE9BQUosQ0FBZSxDQUFDVyxPQUFELEVBQVVULE1BQVYsS0FBcUI7QUFDbEQsYUFBU1UsSUFBVCxHQUFnQjtBQUNkSCxNQUFBQSxTQUFTLEdBQUdJLElBQVosQ0FBa0JDLEtBQUQsSUFBVztBQUMxQixZQUFJQSxLQUFKLEVBQVc7QUFDVEgsVUFBQUEsT0FBTyxDQUFDRyxLQUFELENBQVA7QUFDRCxTQUZELE1BRU87QUFDTFYsVUFBQUEsVUFBVSxDQUFDUSxJQUFELEVBQU9GLFFBQVAsQ0FBVjtBQUNEO0FBQ0YsT0FORCxFQU1HSyxLQU5ILENBTVMsTUFBTTtBQUNiYixRQUFBQSxNQUFNO0FBQ1AsT0FSRDtBQVNEOztBQUNEVSxJQUFBQSxJQUFJO0FBQ0wsR0FiZSxDQUFoQjtBQWNBLFNBQU9qQixjQUFjLENBQUNJLE9BQUQsRUFBVUYsT0FBVixFQUFtQkMsV0FBbkIsQ0FBckI7QUFDRDs7QUFFTSxTQUFTa0IsV0FBVCxDQUFxQnBCLEVBQXJCLEVBQWlDQyxPQUFqQyxFQUF3RDtBQUM3RCxTQUFPRixjQUFjLENBQUNDLEVBQUQsRUFBS0MsT0FBTCxFQUFjLFNBQWQsQ0FBZCxDQUF1Q2tCLEtBQXZDLENBQThDRSxHQUFELElBQVM7QUFDM0QsUUFBSSxFQUFFQSxHQUFHLFlBQVl6QixZQUFqQixDQUFKLEVBQW9DLE1BQU15QixHQUFOO0FBQ3JDLEdBRk0sQ0FBUDtBQUdEOztBQUVNLFNBQVNDLFNBQVQsQ0FBc0JDLE9BQXRCLEVBQW1FO0FBQ3hFLFNBQU9BLE9BQU8sQ0FBQ0MsTUFBUixDQUFlLENBQUNDLENBQUQsRUFBSUMsQ0FBSixLQUFVRCxDQUFDLENBQUNSLElBQUYsQ0FBTyxNQUFPVSxDQUFQLElBQWEsQ0FBQyxHQUFHQSxDQUFKLEVBQU8sTUFBTUQsQ0FBQyxFQUFkLENBQXBCLENBQXpCLEVBQWlFdEIsT0FBTyxDQUFDVyxPQUFSLENBQXFCLElBQUlhLEtBQUosRUFBckIsQ0FBakUsQ0FBUDtBQUNEIiwic291cmNlc0NvbnRlbnQiOlsiXG5leHBvcnQgY2xhc3MgVGltZW91dEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuXG59XG5cbmV4cG9ydCBjb25zdCBTRUNPTkQgPSAxMDAwO1xuXG5mdW5jdGlvbiB0aW1lb3V0UHJvbWlzZTxUPihtczogbnVtYmVyLCBwcm9taXNlOiBQcm9taXNlPFQ+LCBkZXNjcmlwdGlvbjogc3RyaW5nKTogUHJvbWlzZTxUPiB7XG4gIGNvbnN0IHRpbWVvdXQgPSBuZXcgUHJvbWlzZSgoXywgcmVqZWN0KSA9PiB7XG4gICAgY29uc3QgaWQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGNsZWFyVGltZW91dChpZCk7XG4gICAgICBjb25zdCBlcnJvciA9IG5ldyBUaW1lb3V0RXJyb3IoZGVzY3JpcHRpb24pO1xuICAgICAgcmVqZWN0KGVycm9yKTtcbiAgICB9LCBtcyk7XG4gIH0pO1xuXG4gIHJldHVybiBQcm9taXNlLnJhY2UoW1xuICAgIHByb21pc2UsXG4gICAgLy8gY2FzdGluZyB0byBhdm9pZCB0eXBlIGVycm9yLSBzYWZlIHNpbmNlIHRoaXMgcHJvbWlzZSB3aWxsIGFsd2F5cyByZWplY3RcbiAgICB0aW1lb3V0IGFzIFByb21pc2U8VD4sXG4gIF0pO1xufVxuXG4vKipcbiAqIFdhaXQgdW50aWwgYSBwcm9taXNlIHJlc29sdmVzIHdpdGggYSB0cnV0aHkgdmFsdWUgb3IgcmVqZWN0IGFmdGVyIGEgdGltZW91dFxuICovXG5leHBvcnQgZnVuY3Rpb24gd2FpdFVudGlsPFQ+KGFzeW5jVGVzdDogKCkgPT4gUHJvbWlzZTxUPiwgZGVzY3JpcHRpb24gPSAnJywgdGltZW91dCA9IDEwMDAwLCBpbnRlcnZhbCA9IDEwMCkge1xuICBjb25zdCBwcm9taXNlID0gbmV3IFByb21pc2U8VD4oKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgIGZ1bmN0aW9uIHdhaXQoKSB7XG4gICAgICBhc3luY1Rlc3QoKS50aGVuKCh2YWx1ZSkgPT4ge1xuICAgICAgICBpZiAodmFsdWUpIHtcbiAgICAgICAgICByZXNvbHZlKHZhbHVlKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBzZXRUaW1lb3V0KHdhaXQsIGludGVydmFsKTtcbiAgICAgICAgfVxuICAgICAgfSkuY2F0Y2goKCkgPT4ge1xuICAgICAgICByZWplY3QoKTtcbiAgICAgIH0pO1xuICAgIH1cbiAgICB3YWl0KCk7XG4gIH0pO1xuICByZXR1cm4gdGltZW91dFByb21pc2UodGltZW91dCwgcHJvbWlzZSwgZGVzY3JpcHRpb24pO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmFjZVRpbWVvdXQobXM6IG51bWJlciwgcHJvbWlzZTogUHJvbWlzZTxhbnk+KSB7XG4gIHJldHVybiB0aW1lb3V0UHJvbWlzZShtcywgcHJvbWlzZSwgJ3RpbWVvdXQnKS5jYXRjaCgoZXJyKSA9PiB7XG4gICAgaWYgKCEoZXJyIGluc3RhbmNlb2YgVGltZW91dEVycm9yKSkgdGhyb3cgZXJyO1xuICB9KTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHJ1blNlcmlhbDxUPihhY3Rpb25zOiAoKCkgPT4gUHJvbWlzZTxUPilbXSk6IFByb21pc2U8VFtdPiB7XG4gIHJldHVybiBhY3Rpb25zLnJlZHVjZSgobSwgYSkgPT4gbS50aGVuKGFzeW5jICh4KSA9PiBbLi4ueCwgYXdhaXQgYSgpXSksIFByb21pc2UucmVzb2x2ZTxUW10+KG5ldyBBcnJheTxUPigpKSk7XG59XG4iXX0=
|
|
@@ -29,14 +29,14 @@ var _transactions = require("../helpers/transactions");
|
|
|
29
29
|
|
|
30
30
|
var _transactions2 = require("../transactions");
|
|
31
31
|
|
|
32
|
-
var _baseScraper = require("./base-scraper");
|
|
33
|
-
|
|
34
32
|
var _debug = require("../helpers/debug");
|
|
35
33
|
|
|
36
34
|
var _waiting = require("../helpers/waiting");
|
|
37
35
|
|
|
38
36
|
var _errors = require("./errors");
|
|
39
37
|
|
|
38
|
+
var _definitions = require("../definitions");
|
|
39
|
+
|
|
40
40
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
41
41
|
|
|
42
42
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
|
|
@@ -305,7 +305,7 @@ class IsracardAmexBaseScraper extends _baseScraperWithBrowser.BaseScraperWithBro
|
|
|
305
305
|
});
|
|
306
306
|
debug('navigate to login page');
|
|
307
307
|
await this.navigateTo(`${this.baseUrl}/personalarea/Login`);
|
|
308
|
-
this.emitProgress(
|
|
308
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoggingIn);
|
|
309
309
|
const validateUrl = `${this.servicesUrl}?reqName=ValidateIdData`;
|
|
310
310
|
const validateRequest = {
|
|
311
311
|
id: credentials.id,
|
|
@@ -341,21 +341,21 @@ class IsracardAmexBaseScraper extends _baseScraperWithBrowser.BaseScraperWithBro
|
|
|
341
341
|
debug(`user login with status '${loginResult === null || loginResult === void 0 ? void 0 : loginResult.status}'`);
|
|
342
342
|
|
|
343
343
|
if (loginResult && loginResult.status === '1') {
|
|
344
|
-
this.emitProgress(
|
|
344
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoginSuccess);
|
|
345
345
|
return {
|
|
346
346
|
success: true
|
|
347
347
|
};
|
|
348
348
|
}
|
|
349
349
|
|
|
350
350
|
if (loginResult && loginResult.status === '3') {
|
|
351
|
-
this.emitProgress(
|
|
351
|
+
this.emitProgress(_definitions.ScraperProgressTypes.ChangePassword);
|
|
352
352
|
return {
|
|
353
353
|
success: false,
|
|
354
354
|
errorType: _errors.ScraperErrorTypes.ChangePassword
|
|
355
355
|
};
|
|
356
356
|
}
|
|
357
357
|
|
|
358
|
-
this.emitProgress(
|
|
358
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoginFailed);
|
|
359
359
|
return {
|
|
360
360
|
success: false,
|
|
361
361
|
errorType: _errors.ScraperErrorTypes.InvalidPassword
|
|
@@ -363,14 +363,14 @@ class IsracardAmexBaseScraper extends _baseScraperWithBrowser.BaseScraperWithBro
|
|
|
363
363
|
}
|
|
364
364
|
|
|
365
365
|
if (validateReturnCode === '4') {
|
|
366
|
-
this.emitProgress(
|
|
366
|
+
this.emitProgress(_definitions.ScraperProgressTypes.ChangePassword);
|
|
367
367
|
return {
|
|
368
368
|
success: false,
|
|
369
369
|
errorType: _errors.ScraperErrorTypes.ChangePassword
|
|
370
370
|
};
|
|
371
371
|
}
|
|
372
372
|
|
|
373
|
-
this.emitProgress(
|
|
373
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoginFailed);
|
|
374
374
|
return {
|
|
375
375
|
success: false,
|
|
376
376
|
errorType: _errors.ScraperErrorTypes.InvalidPassword
|
|
@@ -393,4 +393,4 @@ class IsracardAmexBaseScraper extends _baseScraperWithBrowser.BaseScraperWithBro
|
|
|
393
393
|
|
|
394
394
|
var _default = IsracardAmexBaseScraper;
|
|
395
395
|
exports.default = _default;
|
|
396
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-isracard-amex.ts"],"names":["COUNTRY_CODE","ID_TYPE","INSTALLMENTS_KEYWORD","DATE_FORMAT","debug","getAccountsUrl","servicesUrl","monthMoment","billingDate","format","queryParams","reqName","actionCode","fetchAccounts","page","dataUrl","dataResult","_","get","DashboardMonthBean","cardsCharges","map","cardCharge","index","parseInt","cardIndex","accountNumber","cardNumber","processedDate","toISOString","getTransactionsUrl","month","year","monthStr","toString","requiredDate","convertCurrency","currencyStr","SHEKEL_CURRENCY_KEYWORD","ALT_SHEKEL_CURRENCY","SHEKEL_CURRENCY","getInstallmentsInfo","txn","moreInfo","includes","undefined","matches","match","length","number","total","getTransactionType","TransactionTypes","Installments","Normal","convertTransactions","txns","filteredTxns","filter","dealSumType","voucherNumberRatz","voucherNumberRatzOutbound","isOutbound","dealSumOutbound","txnDateStr","fullPurchaseDateOutbound","fullPurchaseDate","txnMoment","currentProcessedDate","fullPaymentDate","result","type","identifier","date","originalAmount","dealSum","originalCurrency","currencyId","chargedAmount","paymentSumOutbound","paymentSum","description","fullSupplierNameOutbound","fullSupplierNameHeb","memo","installments","status","TransactionStatuses","Completed","fetchTransactions","options","startMoment","accounts","CardsTransactionsListBean","accountTxns","forEach","account","txnGroups","allTxns","txnGroup","txnIsrael","push","txnAbroad","combineInstallments","outputData","enableTransactionsFilterByDate","getTransactionExtraDetails","accountIndex","transaction","moedChiuv","CardIndex","shovarRatz","getExtraScrapTransaction","data","rawCategory","category","trim","getExtraScrapTransactions","accountWithIndex","promises","t","Promise","all","getExtraScrapAccount","accountMap","Object","keys","a","reduce","m","x","getExtraScrap","accountsWithIndex","allMonths","actions","i","fetchAllTransactions","futureMonthsToScrape","results","finalResult","additionalTransactionInformation","combinedTxns","txnsForAccount","toBeAddedTxns","success","IsracardAmexBaseScraper","BaseScraperWithBrowser","constructor","baseUrl","companyCode","login","credentials","setRequestInterception","on","request","url","abort","continue","navigateTo","emitProgress","ScraperProgressTypes","LoggingIn","validateUrl","validateRequest","id","cardSuffix","card6Digits","countryCode","idType","checkLevel","validateResult","Header","Status","ValidateIdDataBean","Error","validateReturnCode","returnCode","userName","loginUrl","KodMishtamesh","MisparZihuy","Sisma","password","loginResult","LoginSuccess","ChangePassword","errorType","ScraperErrorTypes","LoginFailed","InvalidPassword","fetchData","defaultStartMoment","subtract","startDate","toDate","moment","max"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\n\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser } from './base-scraper-with-browser';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport {\n  SHEKEL_CURRENCY_KEYWORD,\n  SHEKEL_CURRENCY,\n  ALT_SHEKEL_CURRENCY,\n} from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { fixInstallments, filterOldTransactions } from '../helpers/transactions';\nimport {\n  TransactionsAccount, Transaction, TransactionInstallments,\n  TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport {\n  ScraperProgressTypes,\n\n} from './base-scraper';\nimport { getDebug } from '../helpers/debug';\nimport { runSerial } from '../helpers/waiting';\nimport { ScraperErrorTypes } from './errors';\nimport { ScraperScrapingResult, ScraperOptions } from './interface';\n\nconst COUNTRY_CODE = '212';\nconst ID_TYPE = '1';\nconst INSTALLMENTS_KEYWORD = 'תשלום';\n\nconst DATE_FORMAT = 'DD/MM/YYYY';\n\nconst debug = getDebug('base-isracard-amex');\n\ninterface ExtendedScraperOptions extends ScraperOptions {\n  servicesUrl: string;\n  companyCode: string;\n}\n\ntype ScrapedAccountsWithIndex = Record<string, TransactionsAccount & { index: number }>;\n\ninterface ScrapedTransaction {\n  dealSumType: string;\n  voucherNumberRatzOutbound: string;\n  voucherNumberRatz: string;\n  moreInfo?: string;\n  dealSumOutbound: boolean;\n  currencyId: string;\n  dealSum: number;\n  fullPaymentDate?: string;\n  fullPurchaseDate?: string;\n  fullPurchaseDateOutbound?: string;\n  fullSupplierNameHeb: string;\n  fullSupplierNameOutbound: string;\n  paymentSum: number;\n  paymentSumOutbound: number;\n}\n\n\ninterface ScrapedAccount {\n  index: number;\n  accountNumber: string;\n  processedDate: string;\n}\n\ninterface ScrapedLoginValidation {\n  Header: {\n    Status: string;\n  };\n  ValidateIdDataBean?: {\n    userName?: string;\n    returnCode: string;\n  };\n}\n\ninterface ScrapedAccountsWithinPageResponse {\n  Header: {\n    Status: string;\n  };\n  DashboardMonthBean?: {\n    cardsCharges: {\n      cardIndex: string;\n      cardNumber: string;\n      billingDate: string;\n    }[];\n  };\n}\n\ninterface ScrapedCurrentCardTransactions {\n  txnIsrael?: ScrapedTransaction[];\n  txnAbroad?: ScrapedTransaction[];\n}\n\ninterface ScrapedTransactionData {\n  Header?: {\n    Status: string;\n  };\n  CardsTransactionsListBean?: Record<string, {\n    CurrentCardTransactions: ScrapedCurrentCardTransactions[];\n  }>;\n}\n\nfunction getAccountsUrl(servicesUrl: string, monthMoment: Moment) {\n  const billingDate = monthMoment.format('YYYY-MM-DD');\n  return buildUrl(servicesUrl, {\n    queryParams: {\n      reqName: 'DashboardMonth',\n      actionCode: '0',\n      billingDate,\n      format: 'Json',\n    },\n  });\n}\n\nasync function fetchAccounts(page: Page, servicesUrl: string, monthMoment: Moment): Promise<ScrapedAccount[]> {\n  const dataUrl = getAccountsUrl(servicesUrl, monthMoment);\n  const dataResult = await fetchGetWithinPage<ScrapedAccountsWithinPageResponse>(page, dataUrl);\n  if (dataResult && _.get(dataResult, 'Header.Status') === '1' && dataResult.DashboardMonthBean) {\n    const { cardsCharges } = dataResult.DashboardMonthBean;\n    if (cardsCharges) {\n      return cardsCharges.map((cardCharge) => {\n        return {\n          index: parseInt(cardCharge.cardIndex, 10),\n          accountNumber: cardCharge.cardNumber,\n          processedDate: moment(cardCharge.billingDate, DATE_FORMAT).toISOString(),\n        };\n      });\n    }\n  }\n  return [];\n}\n\nfunction getTransactionsUrl(servicesUrl: string, monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const monthStr = month < 10 ? `0${month}` : month.toString();\n  return buildUrl(servicesUrl, {\n    queryParams: {\n      reqName: 'CardsTransactionsList',\n      month: monthStr,\n      year: `${year}`,\n      requiredDate: 'N',\n    },\n  });\n}\n\nfunction convertCurrency(currencyStr: string) {\n  if (currencyStr === SHEKEL_CURRENCY_KEYWORD || currencyStr === ALT_SHEKEL_CURRENCY) {\n    return SHEKEL_CURRENCY;\n  }\n  return currencyStr;\n}\n\nfunction getInstallmentsInfo(txn: ScrapedTransaction): TransactionInstallments | undefined {\n  if (!txn.moreInfo || !txn.moreInfo.includes(INSTALLMENTS_KEYWORD)) {\n    return undefined;\n  }\n  const matches = txn.moreInfo.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getTransactionType(txn: ScrapedTransaction) {\n  return getInstallmentsInfo(txn) ? TransactionTypes.Installments : TransactionTypes.Normal;\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], processedDate: string): Transaction[] {\n  const filteredTxns = txns.filter((txn) => txn.dealSumType !== '1' &&\n                                            txn.voucherNumberRatz !== '000000000' &&\n                                            txn.voucherNumberRatzOutbound !== '000000000');\n\n  return filteredTxns.map((txn) => {\n    const isOutbound = txn.dealSumOutbound;\n    const txnDateStr = isOutbound ? txn.fullPurchaseDateOutbound : txn.fullPurchaseDate;\n    const txnMoment = moment(txnDateStr, DATE_FORMAT);\n\n    const currentProcessedDate = txn.fullPaymentDate ?\n      moment(txn.fullPaymentDate, DATE_FORMAT).toISOString() :\n      processedDate;\n    const result: Transaction = {\n      type: getTransactionType(txn),\n      identifier: parseInt(isOutbound ? txn.voucherNumberRatzOutbound : txn.voucherNumberRatz, 10),\n      date: txnMoment.toISOString(),\n      processedDate: currentProcessedDate,\n      originalAmount: isOutbound ? -txn.dealSumOutbound : -txn.dealSum,\n      originalCurrency: convertCurrency(txn.currencyId),\n      chargedAmount: isOutbound ? -txn.paymentSumOutbound : -txn.paymentSum,\n      description: isOutbound ? txn.fullSupplierNameOutbound : txn.fullSupplierNameHeb,\n      memo: txn.moreInfo || '',\n      installments: getInstallmentsInfo(txn) || undefined,\n      status: TransactionStatuses.Completed,\n    };\n\n    return result;\n  });\n}\n\nasync function fetchTransactions(page: Page, options: ExtendedScraperOptions, startMoment: Moment, monthMoment: Moment): Promise<ScrapedAccountsWithIndex> {\n  const accounts = await fetchAccounts(page, options.servicesUrl, monthMoment);\n  const dataUrl = getTransactionsUrl(options.servicesUrl, monthMoment);\n  const dataResult = await fetchGetWithinPage<ScrapedTransactionData>(page, dataUrl);\n  if (dataResult && _.get(dataResult, 'Header.Status') === '1' && dataResult.CardsTransactionsListBean) {\n    const accountTxns: ScrapedAccountsWithIndex = {};\n    accounts.forEach((account) => {\n      const txnGroups: ScrapedCurrentCardTransactions[] = _.get(dataResult, `CardsTransactionsListBean.Index${account.index}.CurrentCardTransactions`);\n      if (txnGroups) {\n        let allTxns: Transaction[] = [];\n        txnGroups.forEach((txnGroup) => {\n          if (txnGroup.txnIsrael) {\n            const txns = convertTransactions(txnGroup.txnIsrael, account.processedDate);\n            allTxns.push(...txns);\n          }\n          if (txnGroup.txnAbroad) {\n            const txns = convertTransactions(txnGroup.txnAbroad, account.processedDate);\n            allTxns.push(...txns);\n          }\n        });\n\n        if (!options.combineInstallments) {\n          allTxns = fixInstallments(allTxns);\n        }\n        if (options.outputData?.enableTransactionsFilterByDate ?? true) {\n          allTxns = filterOldTransactions(allTxns, startMoment, options.combineInstallments || false);\n        }\n        accountTxns[account.accountNumber] = {\n          accountNumber: account.accountNumber,\n          index: account.index,\n          txns: allTxns,\n        };\n      }\n    });\n    return accountTxns;\n  }\n\n  return {};\n}\n\nfunction getTransactionExtraDetails(servicesUrl: string, month: Moment, accountIndex: number, transaction: Transaction): string {\n  const moedChiuv = month.format('MMYYYY');\n  return buildUrl(servicesUrl, {\n    queryParams: {\n      reqName: 'PirteyIska_204',\n      CardIndex: accountIndex.toString(),\n      shovarRatz: transaction.identifier!.toString(),\n      moedChiuv,\n    },\n  });\n}\nasync function getExtraScrapTransaction(page: Page, options: ExtendedScraperOptions, month: Moment, accountIndex: number, transaction: Transaction): Promise<Transaction> {\n  const dataUrl = getTransactionExtraDetails(options.servicesUrl, month, accountIndex, transaction);\n  const data = await fetchGetWithinPage<ScrapedTransactionData>(page, dataUrl);\n  const rawCategory = _.get(data, 'PirteyIska_204Bean.sector');\n  return {\n    ...transaction,\n    category: rawCategory.trim(),\n  };\n}\n\nfunction getExtraScrapTransactions(accountWithIndex: TransactionsAccount & { index: number }, page: Page, options: ExtendedScraperOptions, month: moment.Moment): Promise<Transaction[]> {\n  const promises = accountWithIndex.txns\n    .map((t) => getExtraScrapTransaction(page, options, month, accountWithIndex.index, t));\n  return Promise.all(promises);\n}\n\nasync function getExtraScrapAccount(page: Page, options: ExtendedScraperOptions, accountMap: ScrapedAccountsWithIndex, month: moment.Moment): Promise<ScrapedAccountsWithIndex> {\n  const promises = Object.keys(accountMap)\n    .map(async (a) => ({\n      ...accountMap[a],\n      txns: await getExtraScrapTransactions(accountMap[a], page, options, month),\n    }));\n  const accounts = await Promise.all(promises);\n  return accounts.reduce((m, x) => ({ ...m, [x.accountNumber]: x }), {});\n}\n\nfunction getExtraScrap(accountsWithIndex: ScrapedAccountsWithIndex[], page: Page, options: ExtendedScraperOptions, allMonths: moment.Moment[]): Promise<ScrapedAccountsWithIndex[]> {\n  const actions = accountsWithIndex.map((a, i) => () => getExtraScrapAccount(page, options, a, allMonths[i]));\n  return runSerial(actions);\n}\n\nasync function fetchAllTransactions(page: Page, options: ExtendedScraperOptions, startMoment: Moment) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n  const results: ScrapedAccountsWithIndex[] = await Promise.all(allMonths.map(async (monthMoment) => {\n    return fetchTransactions(page, options, startMoment, monthMoment);\n  }));\n\n  const finalResult = options.additionalTransactionInformation ?\n    await getExtraScrap(results, page, options, allMonths) : results;\n\n  const combinedTxns: Record<string, Transaction[]> = {};\n\n  finalResult.forEach((result) => {\n    Object.keys(result).forEach((accountNumber) => {\n      let txnsForAccount = combinedTxns[accountNumber];\n      if (!txnsForAccount) {\n        txnsForAccount = [];\n        combinedTxns[accountNumber] = txnsForAccount;\n      }\n      const toBeAddedTxns = result[accountNumber].txns;\n      combinedTxns[accountNumber].push(...toBeAddedTxns);\n    });\n  });\n\n  const accounts = Object.keys(combinedTxns).map((accountNumber) => {\n    return {\n      accountNumber,\n      txns: combinedTxns[accountNumber],\n    };\n  });\n\n  return {\n    success: true,\n    accounts,\n  };\n}\n\ntype ScraperSpecificCredentials = {id: string, password: string, card6Digits: string};\nclass IsracardAmexBaseScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  private baseUrl: string;\n\n  private companyCode: string;\n\n  private servicesUrl: string;\n\n  constructor(options: ScraperOptions, baseUrl: string, companyCode: string) {\n    super(options);\n\n    this.baseUrl = baseUrl;\n    this.companyCode = companyCode;\n    this.servicesUrl = `${baseUrl}/services/ProxyRequestHandler.ashx`;\n  }\n\n  async login(credentials: ScraperSpecificCredentials): Promise<ScraperScrapingResult> {\n    await this.page.setRequestInterception(true);\n    this.page.on('request', (request) => {\n      if (request.url().includes('detector-dom.min.js')) {\n        debug('force abort for request do download detector-dom.min.js resource');\n        request.abort();\n      } else {\n        request.continue();\n      }\n    });\n\n    debug('navigate to login page');\n    await this.navigateTo(`${this.baseUrl}/personalarea/Login`);\n\n    this.emitProgress(ScraperProgressTypes.LoggingIn);\n\n    const validateUrl = `${this.servicesUrl}?reqName=ValidateIdData`;\n    const validateRequest = {\n      id: credentials.id,\n      cardSuffix: credentials.card6Digits,\n      countryCode: COUNTRY_CODE,\n      idType: ID_TYPE,\n      checkLevel: '1',\n      companyCode: this.companyCode,\n    };\n    const validateResult = await fetchPostWithinPage<ScrapedLoginValidation>(this.page, validateUrl, validateRequest);\n    if (!validateResult || !validateResult.Header || validateResult.Header.Status !== '1' || !validateResult.ValidateIdDataBean) {\n      throw new Error('unknown error during login');\n    }\n\n    const validateReturnCode = validateResult.ValidateIdDataBean.returnCode;\n    debug(`user validate with return code '${validateReturnCode}'`);\n    if (validateReturnCode === '1') {\n      const { userName } = validateResult.ValidateIdDataBean;\n\n      const loginUrl = `${this.servicesUrl}?reqName=performLogonI`;\n      const request = {\n        KodMishtamesh: userName,\n        MisparZihuy: credentials.id,\n        Sisma: credentials.password,\n        cardSuffix: credentials.card6Digits,\n        countryCode: COUNTRY_CODE,\n        idType: ID_TYPE,\n      };\n      const loginResult = await fetchPostWithinPage<{status: string}>(this.page, loginUrl, request);\n      debug(`user login with status '${loginResult?.status}'`);\n\n      if (loginResult && loginResult.status === '1') {\n        this.emitProgress(ScraperProgressTypes.LoginSuccess);\n        return { success: true };\n      }\n\n      if (loginResult && loginResult.status === '3') {\n        this.emitProgress(ScraperProgressTypes.ChangePassword);\n        return {\n          success: false,\n          errorType: ScraperErrorTypes.ChangePassword,\n        };\n      }\n\n      this.emitProgress(ScraperProgressTypes.LoginFailed);\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.InvalidPassword,\n      };\n    }\n\n    if (validateReturnCode === '4') {\n      this.emitProgress(ScraperProgressTypes.ChangePassword);\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.ChangePassword,\n      };\n    }\n\n    this.emitProgress(ScraperProgressTypes.LoginFailed);\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.InvalidPassword,\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    return fetchAllTransactions(this.page, {\n      ...this.options,\n      servicesUrl: this.servicesUrl,\n      companyCode: this.companyCode,\n    }, startMoment);\n  }\n}\n\nexport default IsracardAmexBaseScraper;\n"]}
|
|
396
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-isracard-amex.ts"],"names":["COUNTRY_CODE","ID_TYPE","INSTALLMENTS_KEYWORD","DATE_FORMAT","debug","getAccountsUrl","servicesUrl","monthMoment","billingDate","format","queryParams","reqName","actionCode","fetchAccounts","page","dataUrl","dataResult","_","get","DashboardMonthBean","cardsCharges","map","cardCharge","index","parseInt","cardIndex","accountNumber","cardNumber","processedDate","toISOString","getTransactionsUrl","month","year","monthStr","toString","requiredDate","convertCurrency","currencyStr","SHEKEL_CURRENCY_KEYWORD","ALT_SHEKEL_CURRENCY","SHEKEL_CURRENCY","getInstallmentsInfo","txn","moreInfo","includes","undefined","matches","match","length","number","total","getTransactionType","TransactionTypes","Installments","Normal","convertTransactions","txns","filteredTxns","filter","dealSumType","voucherNumberRatz","voucherNumberRatzOutbound","isOutbound","dealSumOutbound","txnDateStr","fullPurchaseDateOutbound","fullPurchaseDate","txnMoment","currentProcessedDate","fullPaymentDate","result","type","identifier","date","originalAmount","dealSum","originalCurrency","currencyId","chargedAmount","paymentSumOutbound","paymentSum","description","fullSupplierNameOutbound","fullSupplierNameHeb","memo","installments","status","TransactionStatuses","Completed","fetchTransactions","options","startMoment","accounts","CardsTransactionsListBean","accountTxns","forEach","account","txnGroups","allTxns","txnGroup","txnIsrael","push","txnAbroad","combineInstallments","outputData","enableTransactionsFilterByDate","getTransactionExtraDetails","accountIndex","transaction","moedChiuv","CardIndex","shovarRatz","getExtraScrapTransaction","data","rawCategory","category","trim","getExtraScrapTransactions","accountWithIndex","promises","t","Promise","all","getExtraScrapAccount","accountMap","Object","keys","a","reduce","m","x","getExtraScrap","accountsWithIndex","allMonths","actions","i","fetchAllTransactions","futureMonthsToScrape","results","finalResult","additionalTransactionInformation","combinedTxns","txnsForAccount","toBeAddedTxns","success","IsracardAmexBaseScraper","BaseScraperWithBrowser","constructor","baseUrl","companyCode","login","credentials","setRequestInterception","on","request","url","abort","continue","navigateTo","emitProgress","ScraperProgressTypes","LoggingIn","validateUrl","validateRequest","id","cardSuffix","card6Digits","countryCode","idType","checkLevel","validateResult","Header","Status","ValidateIdDataBean","Error","validateReturnCode","returnCode","userName","loginUrl","KodMishtamesh","MisparZihuy","Sisma","password","loginResult","LoginSuccess","ChangePassword","errorType","ScraperErrorTypes","LoginFailed","InvalidPassword","fetchData","defaultStartMoment","subtract","startDate","toDate","moment","max"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport buildUrl from 'build-url';\nimport moment, { Moment } from 'moment';\n\nimport { Page } from 'puppeteer';\nimport { BaseScraperWithBrowser } from './base-scraper-with-browser';\nimport { fetchGetWithinPage, fetchPostWithinPage } from '../helpers/fetch';\nimport {\n  SHEKEL_CURRENCY_KEYWORD,\n  SHEKEL_CURRENCY,\n  ALT_SHEKEL_CURRENCY,\n} from '../constants';\nimport getAllMonthMoments from '../helpers/dates';\nimport { fixInstallments, filterOldTransactions } from '../helpers/transactions';\nimport {\n  TransactionsAccount, Transaction, TransactionInstallments,\n  TransactionStatuses, TransactionTypes,\n} from '../transactions';\nimport { getDebug } from '../helpers/debug';\nimport { runSerial } from '../helpers/waiting';\nimport { ScraperErrorTypes } from './errors';\nimport { ScraperScrapingResult, ScraperOptions } from './interface';\nimport { ScraperProgressTypes } from '../definitions';\n\nconst COUNTRY_CODE = '212';\nconst ID_TYPE = '1';\nconst INSTALLMENTS_KEYWORD = 'תשלום';\n\nconst DATE_FORMAT = 'DD/MM/YYYY';\n\nconst debug = getDebug('base-isracard-amex');\n\ninterface ExtendedScraperOptions extends ScraperOptions {\n  servicesUrl: string;\n  companyCode: string;\n}\n\ntype ScrapedAccountsWithIndex = Record<string, TransactionsAccount & { index: number }>;\n\ninterface ScrapedTransaction {\n  dealSumType: string;\n  voucherNumberRatzOutbound: string;\n  voucherNumberRatz: string;\n  moreInfo?: string;\n  dealSumOutbound: boolean;\n  currencyId: string;\n  dealSum: number;\n  fullPaymentDate?: string;\n  fullPurchaseDate?: string;\n  fullPurchaseDateOutbound?: string;\n  fullSupplierNameHeb: string;\n  fullSupplierNameOutbound: string;\n  paymentSum: number;\n  paymentSumOutbound: number;\n}\n\n\ninterface ScrapedAccount {\n  index: number;\n  accountNumber: string;\n  processedDate: string;\n}\n\ninterface ScrapedLoginValidation {\n  Header: {\n    Status: string;\n  };\n  ValidateIdDataBean?: {\n    userName?: string;\n    returnCode: string;\n  };\n}\n\ninterface ScrapedAccountsWithinPageResponse {\n  Header: {\n    Status: string;\n  };\n  DashboardMonthBean?: {\n    cardsCharges: {\n      cardIndex: string;\n      cardNumber: string;\n      billingDate: string;\n    }[];\n  };\n}\n\ninterface ScrapedCurrentCardTransactions {\n  txnIsrael?: ScrapedTransaction[];\n  txnAbroad?: ScrapedTransaction[];\n}\n\ninterface ScrapedTransactionData {\n  Header?: {\n    Status: string;\n  };\n  CardsTransactionsListBean?: Record<string, {\n    CurrentCardTransactions: ScrapedCurrentCardTransactions[];\n  }>;\n}\n\nfunction getAccountsUrl(servicesUrl: string, monthMoment: Moment) {\n  const billingDate = monthMoment.format('YYYY-MM-DD');\n  return buildUrl(servicesUrl, {\n    queryParams: {\n      reqName: 'DashboardMonth',\n      actionCode: '0',\n      billingDate,\n      format: 'Json',\n    },\n  });\n}\n\nasync function fetchAccounts(page: Page, servicesUrl: string, monthMoment: Moment): Promise<ScrapedAccount[]> {\n  const dataUrl = getAccountsUrl(servicesUrl, monthMoment);\n  const dataResult = await fetchGetWithinPage<ScrapedAccountsWithinPageResponse>(page, dataUrl);\n  if (dataResult && _.get(dataResult, 'Header.Status') === '1' && dataResult.DashboardMonthBean) {\n    const { cardsCharges } = dataResult.DashboardMonthBean;\n    if (cardsCharges) {\n      return cardsCharges.map((cardCharge) => {\n        return {\n          index: parseInt(cardCharge.cardIndex, 10),\n          accountNumber: cardCharge.cardNumber,\n          processedDate: moment(cardCharge.billingDate, DATE_FORMAT).toISOString(),\n        };\n      });\n    }\n  }\n  return [];\n}\n\nfunction getTransactionsUrl(servicesUrl: string, monthMoment: Moment) {\n  const month = monthMoment.month() + 1;\n  const year = monthMoment.year();\n  const monthStr = month < 10 ? `0${month}` : month.toString();\n  return buildUrl(servicesUrl, {\n    queryParams: {\n      reqName: 'CardsTransactionsList',\n      month: monthStr,\n      year: `${year}`,\n      requiredDate: 'N',\n    },\n  });\n}\n\nfunction convertCurrency(currencyStr: string) {\n  if (currencyStr === SHEKEL_CURRENCY_KEYWORD || currencyStr === ALT_SHEKEL_CURRENCY) {\n    return SHEKEL_CURRENCY;\n  }\n  return currencyStr;\n}\n\nfunction getInstallmentsInfo(txn: ScrapedTransaction): TransactionInstallments | undefined {\n  if (!txn.moreInfo || !txn.moreInfo.includes(INSTALLMENTS_KEYWORD)) {\n    return undefined;\n  }\n  const matches = txn.moreInfo.match(/\\d+/g);\n  if (!matches || matches.length < 2) {\n    return undefined;\n  }\n\n  return {\n    number: parseInt(matches[0], 10),\n    total: parseInt(matches[1], 10),\n  };\n}\n\nfunction getTransactionType(txn: ScrapedTransaction) {\n  return getInstallmentsInfo(txn) ? TransactionTypes.Installments : TransactionTypes.Normal;\n}\n\nfunction convertTransactions(txns: ScrapedTransaction[], processedDate: string): Transaction[] {\n  const filteredTxns = txns.filter((txn) => txn.dealSumType !== '1' &&\n                                            txn.voucherNumberRatz !== '000000000' &&\n                                            txn.voucherNumberRatzOutbound !== '000000000');\n\n  return filteredTxns.map((txn) => {\n    const isOutbound = txn.dealSumOutbound;\n    const txnDateStr = isOutbound ? txn.fullPurchaseDateOutbound : txn.fullPurchaseDate;\n    const txnMoment = moment(txnDateStr, DATE_FORMAT);\n\n    const currentProcessedDate = txn.fullPaymentDate ?\n      moment(txn.fullPaymentDate, DATE_FORMAT).toISOString() :\n      processedDate;\n    const result: Transaction = {\n      type: getTransactionType(txn),\n      identifier: parseInt(isOutbound ? txn.voucherNumberRatzOutbound : txn.voucherNumberRatz, 10),\n      date: txnMoment.toISOString(),\n      processedDate: currentProcessedDate,\n      originalAmount: isOutbound ? -txn.dealSumOutbound : -txn.dealSum,\n      originalCurrency: convertCurrency(txn.currencyId),\n      chargedAmount: isOutbound ? -txn.paymentSumOutbound : -txn.paymentSum,\n      description: isOutbound ? txn.fullSupplierNameOutbound : txn.fullSupplierNameHeb,\n      memo: txn.moreInfo || '',\n      installments: getInstallmentsInfo(txn) || undefined,\n      status: TransactionStatuses.Completed,\n    };\n\n    return result;\n  });\n}\n\nasync function fetchTransactions(page: Page, options: ExtendedScraperOptions, startMoment: Moment, monthMoment: Moment): Promise<ScrapedAccountsWithIndex> {\n  const accounts = await fetchAccounts(page, options.servicesUrl, monthMoment);\n  const dataUrl = getTransactionsUrl(options.servicesUrl, monthMoment);\n  const dataResult = await fetchGetWithinPage<ScrapedTransactionData>(page, dataUrl);\n  if (dataResult && _.get(dataResult, 'Header.Status') === '1' && dataResult.CardsTransactionsListBean) {\n    const accountTxns: ScrapedAccountsWithIndex = {};\n    accounts.forEach((account) => {\n      const txnGroups: ScrapedCurrentCardTransactions[] = _.get(dataResult, `CardsTransactionsListBean.Index${account.index}.CurrentCardTransactions`);\n      if (txnGroups) {\n        let allTxns: Transaction[] = [];\n        txnGroups.forEach((txnGroup) => {\n          if (txnGroup.txnIsrael) {\n            const txns = convertTransactions(txnGroup.txnIsrael, account.processedDate);\n            allTxns.push(...txns);\n          }\n          if (txnGroup.txnAbroad) {\n            const txns = convertTransactions(txnGroup.txnAbroad, account.processedDate);\n            allTxns.push(...txns);\n          }\n        });\n\n        if (!options.combineInstallments) {\n          allTxns = fixInstallments(allTxns);\n        }\n        if (options.outputData?.enableTransactionsFilterByDate ?? true) {\n          allTxns = filterOldTransactions(allTxns, startMoment, options.combineInstallments || false);\n        }\n        accountTxns[account.accountNumber] = {\n          accountNumber: account.accountNumber,\n          index: account.index,\n          txns: allTxns,\n        };\n      }\n    });\n    return accountTxns;\n  }\n\n  return {};\n}\n\nfunction getTransactionExtraDetails(servicesUrl: string, month: Moment, accountIndex: number, transaction: Transaction): string {\n  const moedChiuv = month.format('MMYYYY');\n  return buildUrl(servicesUrl, {\n    queryParams: {\n      reqName: 'PirteyIska_204',\n      CardIndex: accountIndex.toString(),\n      shovarRatz: transaction.identifier!.toString(),\n      moedChiuv,\n    },\n  });\n}\nasync function getExtraScrapTransaction(page: Page, options: ExtendedScraperOptions, month: Moment, accountIndex: number, transaction: Transaction): Promise<Transaction> {\n  const dataUrl = getTransactionExtraDetails(options.servicesUrl, month, accountIndex, transaction);\n  const data = await fetchGetWithinPage<ScrapedTransactionData>(page, dataUrl);\n  const rawCategory = _.get(data, 'PirteyIska_204Bean.sector');\n  return {\n    ...transaction,\n    category: rawCategory.trim(),\n  };\n}\n\nfunction getExtraScrapTransactions(accountWithIndex: TransactionsAccount & { index: number }, page: Page, options: ExtendedScraperOptions, month: moment.Moment): Promise<Transaction[]> {\n  const promises = accountWithIndex.txns\n    .map((t) => getExtraScrapTransaction(page, options, month, accountWithIndex.index, t));\n  return Promise.all(promises);\n}\n\nasync function getExtraScrapAccount(page: Page, options: ExtendedScraperOptions, accountMap: ScrapedAccountsWithIndex, month: moment.Moment): Promise<ScrapedAccountsWithIndex> {\n  const promises = Object.keys(accountMap)\n    .map(async (a) => ({\n      ...accountMap[a],\n      txns: await getExtraScrapTransactions(accountMap[a], page, options, month),\n    }));\n  const accounts = await Promise.all(promises);\n  return accounts.reduce((m, x) => ({ ...m, [x.accountNumber]: x }), {});\n}\n\nfunction getExtraScrap(accountsWithIndex: ScrapedAccountsWithIndex[], page: Page, options: ExtendedScraperOptions, allMonths: moment.Moment[]): Promise<ScrapedAccountsWithIndex[]> {\n  const actions = accountsWithIndex.map((a, i) => () => getExtraScrapAccount(page, options, a, allMonths[i]));\n  return runSerial(actions);\n}\n\nasync function fetchAllTransactions(page: Page, options: ExtendedScraperOptions, startMoment: Moment) {\n  const futureMonthsToScrape = options.futureMonthsToScrape ?? 1;\n  const allMonths = getAllMonthMoments(startMoment, futureMonthsToScrape);\n  const results: ScrapedAccountsWithIndex[] = await Promise.all(allMonths.map(async (monthMoment) => {\n    return fetchTransactions(page, options, startMoment, monthMoment);\n  }));\n\n  const finalResult = options.additionalTransactionInformation ?\n    await getExtraScrap(results, page, options, allMonths) : results;\n\n  const combinedTxns: Record<string, Transaction[]> = {};\n\n  finalResult.forEach((result) => {\n    Object.keys(result).forEach((accountNumber) => {\n      let txnsForAccount = combinedTxns[accountNumber];\n      if (!txnsForAccount) {\n        txnsForAccount = [];\n        combinedTxns[accountNumber] = txnsForAccount;\n      }\n      const toBeAddedTxns = result[accountNumber].txns;\n      combinedTxns[accountNumber].push(...toBeAddedTxns);\n    });\n  });\n\n  const accounts = Object.keys(combinedTxns).map((accountNumber) => {\n    return {\n      accountNumber,\n      txns: combinedTxns[accountNumber],\n    };\n  });\n\n  return {\n    success: true,\n    accounts,\n  };\n}\n\ntype ScraperSpecificCredentials = {id: string, password: string, card6Digits: string};\nclass IsracardAmexBaseScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  private baseUrl: string;\n\n  private companyCode: string;\n\n  private servicesUrl: string;\n\n  constructor(options: ScraperOptions, baseUrl: string, companyCode: string) {\n    super(options);\n\n    this.baseUrl = baseUrl;\n    this.companyCode = companyCode;\n    this.servicesUrl = `${baseUrl}/services/ProxyRequestHandler.ashx`;\n  }\n\n  async login(credentials: ScraperSpecificCredentials): Promise<ScraperScrapingResult> {\n    await this.page.setRequestInterception(true);\n    this.page.on('request', (request) => {\n      if (request.url().includes('detector-dom.min.js')) {\n        debug('force abort for request do download detector-dom.min.js resource');\n        request.abort();\n      } else {\n        request.continue();\n      }\n    });\n\n    debug('navigate to login page');\n    await this.navigateTo(`${this.baseUrl}/personalarea/Login`);\n\n    this.emitProgress(ScraperProgressTypes.LoggingIn);\n\n    const validateUrl = `${this.servicesUrl}?reqName=ValidateIdData`;\n    const validateRequest = {\n      id: credentials.id,\n      cardSuffix: credentials.card6Digits,\n      countryCode: COUNTRY_CODE,\n      idType: ID_TYPE,\n      checkLevel: '1',\n      companyCode: this.companyCode,\n    };\n    const validateResult = await fetchPostWithinPage<ScrapedLoginValidation>(this.page, validateUrl, validateRequest);\n    if (!validateResult || !validateResult.Header || validateResult.Header.Status !== '1' || !validateResult.ValidateIdDataBean) {\n      throw new Error('unknown error during login');\n    }\n\n    const validateReturnCode = validateResult.ValidateIdDataBean.returnCode;\n    debug(`user validate with return code '${validateReturnCode}'`);\n    if (validateReturnCode === '1') {\n      const { userName } = validateResult.ValidateIdDataBean;\n\n      const loginUrl = `${this.servicesUrl}?reqName=performLogonI`;\n      const request = {\n        KodMishtamesh: userName,\n        MisparZihuy: credentials.id,\n        Sisma: credentials.password,\n        cardSuffix: credentials.card6Digits,\n        countryCode: COUNTRY_CODE,\n        idType: ID_TYPE,\n      };\n      const loginResult = await fetchPostWithinPage<{status: string}>(this.page, loginUrl, request);\n      debug(`user login with status '${loginResult?.status}'`);\n\n      if (loginResult && loginResult.status === '1') {\n        this.emitProgress(ScraperProgressTypes.LoginSuccess);\n        return { success: true };\n      }\n\n      if (loginResult && loginResult.status === '3') {\n        this.emitProgress(ScraperProgressTypes.ChangePassword);\n        return {\n          success: false,\n          errorType: ScraperErrorTypes.ChangePassword,\n        };\n      }\n\n      this.emitProgress(ScraperProgressTypes.LoginFailed);\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.InvalidPassword,\n      };\n    }\n\n    if (validateReturnCode === '4') {\n      this.emitProgress(ScraperProgressTypes.ChangePassword);\n      return {\n        success: false,\n        errorType: ScraperErrorTypes.ChangePassword,\n      };\n    }\n\n    this.emitProgress(ScraperProgressTypes.LoginFailed);\n    return {\n      success: false,\n      errorType: ScraperErrorTypes.InvalidPassword,\n    };\n  }\n\n  async fetchData() {\n    const defaultStartMoment = moment().subtract(1, 'years');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n\n    return fetchAllTransactions(this.page, {\n      ...this.options,\n      servicesUrl: this.servicesUrl,\n      companyCode: this.companyCode,\n    }, startMoment);\n  }\n}\n\nexport default IsracardAmexBaseScraper;\n"]}
|
|
@@ -49,5 +49,6 @@ declare class BaseScraperWithBrowser<TCredentials extends ScraperCredentials> ex
|
|
|
49
49
|
}[]): Promise<void>;
|
|
50
50
|
login(credentials: ScraperCredentials): Promise<ScraperScrapingResult>;
|
|
51
51
|
terminate(_success: boolean): Promise<void>;
|
|
52
|
+
private handleLoginResult;
|
|
52
53
|
}
|
|
53
54
|
export { BaseScraperWithBrowser };
|
|
@@ -21,6 +21,8 @@ var _debug = require("../helpers/debug");
|
|
|
21
21
|
|
|
22
22
|
var _errors = require("./errors");
|
|
23
23
|
|
|
24
|
+
var _definitions = require("../definitions");
|
|
25
|
+
|
|
24
26
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
25
27
|
|
|
26
28
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
|
|
@@ -86,35 +88,6 @@ async function getKeyByValue(object, value, page) {
|
|
|
86
88
|
return Promise.resolve(LoginResults.UnknownError);
|
|
87
89
|
}
|
|
88
90
|
|
|
89
|
-
function handleLoginResult(scraper, loginResult) {
|
|
90
|
-
switch (loginResult) {
|
|
91
|
-
case LoginResults.Success:
|
|
92
|
-
scraper.emitProgress(_baseScraper.ScraperProgressTypes.LoginSuccess);
|
|
93
|
-
return {
|
|
94
|
-
success: true
|
|
95
|
-
};
|
|
96
|
-
|
|
97
|
-
case LoginResults.InvalidPassword:
|
|
98
|
-
case LoginResults.UnknownError:
|
|
99
|
-
scraper.emitProgress(_baseScraper.ScraperProgressTypes.LoginFailed);
|
|
100
|
-
return {
|
|
101
|
-
success: false,
|
|
102
|
-
errorType: loginResult === LoginResults.InvalidPassword ? _errors.ScraperErrorTypes.InvalidPassword : _errors.ScraperErrorTypes.General,
|
|
103
|
-
errorMessage: `Login failed with ${loginResult} error`
|
|
104
|
-
};
|
|
105
|
-
|
|
106
|
-
case LoginResults.ChangePassword:
|
|
107
|
-
scraper.emitProgress(_baseScraper.ScraperProgressTypes.ChangePassword);
|
|
108
|
-
return {
|
|
109
|
-
success: false,
|
|
110
|
-
errorType: _errors.ScraperErrorTypes.ChangePassword
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
default:
|
|
114
|
-
throw new Error(`unexpected login result "${loginResult}"`);
|
|
115
|
-
}
|
|
116
|
-
}
|
|
117
|
-
|
|
118
91
|
function createGeneralError() {
|
|
119
92
|
return {
|
|
120
93
|
success: false,
|
|
@@ -141,7 +114,7 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
|
|
|
141
114
|
async initialize() {
|
|
142
115
|
await super.initialize();
|
|
143
116
|
debug('initialize scraper');
|
|
144
|
-
this.emitProgress(
|
|
117
|
+
this.emitProgress(_definitions.ScraperProgressTypes.Initializing);
|
|
145
118
|
let env;
|
|
146
119
|
|
|
147
120
|
if (this.options.verbose) {
|
|
@@ -291,7 +264,7 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
|
|
|
291
264
|
await loginOptions.submitButtonSelector();
|
|
292
265
|
}
|
|
293
266
|
|
|
294
|
-
this.emitProgress(
|
|
267
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoggingIn);
|
|
295
268
|
|
|
296
269
|
if (loginOptions.postAction) {
|
|
297
270
|
debug('execute \'postAction\' interceptor provided in login options');
|
|
@@ -305,12 +278,12 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
|
|
|
305
278
|
const current = await (0, _navigation.getCurrentUrl)(this.page, true);
|
|
306
279
|
const loginResult = await getKeyByValue(loginOptions.possibleResults, current, this.page);
|
|
307
280
|
debug(`handle login results ${loginResult}`);
|
|
308
|
-
return handleLoginResult(
|
|
281
|
+
return this.handleLoginResult(loginResult);
|
|
309
282
|
}
|
|
310
283
|
|
|
311
284
|
async terminate(_success) {
|
|
312
285
|
debug(`terminating browser with success = ${_success}`);
|
|
313
|
-
this.emitProgress(
|
|
286
|
+
this.emitProgress(_definitions.ScraperProgressTypes.Terminating);
|
|
314
287
|
|
|
315
288
|
if (!_success && !!this.options.storeFailureScreenShotPath) {
|
|
316
289
|
debug(`create a snapshot before terminated in ${this.options.storeFailureScreenShotPath}`);
|
|
@@ -327,7 +300,36 @@ class BaseScraperWithBrowser extends _baseScraper.BaseScraper {
|
|
|
327
300
|
await this.browser.close();
|
|
328
301
|
}
|
|
329
302
|
|
|
303
|
+
handleLoginResult(loginResult) {
|
|
304
|
+
switch (loginResult) {
|
|
305
|
+
case LoginResults.Success:
|
|
306
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoginSuccess);
|
|
307
|
+
return {
|
|
308
|
+
success: true
|
|
309
|
+
};
|
|
310
|
+
|
|
311
|
+
case LoginResults.InvalidPassword:
|
|
312
|
+
case LoginResults.UnknownError:
|
|
313
|
+
this.emitProgress(_definitions.ScraperProgressTypes.LoginFailed);
|
|
314
|
+
return {
|
|
315
|
+
success: false,
|
|
316
|
+
errorType: loginResult === LoginResults.InvalidPassword ? _errors.ScraperErrorTypes.InvalidPassword : _errors.ScraperErrorTypes.General,
|
|
317
|
+
errorMessage: `Login failed with ${loginResult} error`
|
|
318
|
+
};
|
|
319
|
+
|
|
320
|
+
case LoginResults.ChangePassword:
|
|
321
|
+
this.emitProgress(_definitions.ScraperProgressTypes.ChangePassword);
|
|
322
|
+
return {
|
|
323
|
+
success: false,
|
|
324
|
+
errorType: _errors.ScraperErrorTypes.ChangePassword
|
|
325
|
+
};
|
|
326
|
+
|
|
327
|
+
default:
|
|
328
|
+
throw new Error(`unexpected login result "${loginResult}"`);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
|
|
330
332
|
}
|
|
331
333
|
|
|
332
334
|
exports.BaseScraperWithBrowser = BaseScraperWithBrowser;
|
|
333
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper-with-browser.ts"],"names":["VIEWPORT_WIDTH","VIEWPORT_HEIGHT","OK_STATUS","debug","LoginBaseResults","Timeout","Generic","General","ScraperErrorTypes","rest","LoginResults","getKeyByValue","object","value","page","keys","Object","key","conditions","condition","result","RegExp","test","toLowerCase","Promise","resolve","UnknownError","handleLoginResult","scraper","loginResult","Success","emitProgress","ScraperProgressTypes","LoginSuccess","success","InvalidPassword","LoginFailed","errorType","errorMessage","ChangePassword","Error","createGeneralError","BaseScraperWithBrowser","BaseScraper","getViewPort","width","height","initialize","Initializing","env","options","verbose","DEBUG","process","browser","executablePath","undefined","args","timeout","headless","showBrowser","puppeteer","launch","prepareBrowser","pages","length","newPage","defaultTimeout","setDefaultTimeout","preparePage","viewport","setViewport","on","request","failure","errorText","url","navigateTo","pageToUse","response","goto","status","getLoginOptions","_credentials","companyId","fillInputs","pageOrFrame","fields","modified","input","shift","selector","login","credentials","loginOptions","userAgent","setUserAgent","loginUrl","checkReadiness","submitButtonSelector","loginFrameOrPage","preAction","LoggingIn","postAction","current","possibleResults","terminate","_success","Terminating","storeFailureScreenShotPath","screenshot","path","fullPage","close"],"mappings":"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","sourcesContent":["import puppeteer, { Browser, Frame, Page } from 'puppeteer';\n\nimport {\n  BaseScraper, ScraperProgressTypes,\n\n} from './base-scraper';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { clickButton, fillInput, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { getDebug } from '../helpers/debug';\nimport { ScraperErrorTypes } from './errors';\nimport { ScraperScrapingResult, ScraperCredentials } from './interface';\n\nconst VIEWPORT_WIDTH = 1024;\nconst VIEWPORT_HEIGHT = 768;\nconst OK_STATUS = 200;\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\nexport type LoginResults = Exclude<ScraperErrorTypes,\nScraperErrorTypes.Timeout\n| ScraperErrorTypes.Generic\n| ScraperErrorTypes.General> | 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}\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 handleLoginResult(scraper: BaseScraperWithBrowser<ScraperCredentials>, loginResult: LoginResults) {\n  switch (loginResult) {\n    case LoginResults.Success:\n      scraper.emitProgress(ScraperProgressTypes.LoginSuccess);\n      return { success: true };\n    case LoginResults.InvalidPassword:\n    case LoginResults.UnknownError:\n      scraper.emitProgress(ScraperProgressTypes.LoginFailed);\n      return {\n        success: false,\n        errorType: loginResult === LoginResults.InvalidPassword ? ScraperErrorTypes.InvalidPassword :\n          ScraperErrorTypes.General,\n        errorMessage: `Login failed with ${loginResult} error`,\n      };\n    case LoginResults.ChangePassword:\n      scraper.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\nfunction createGeneralError(): ScraperScrapingResult {\n  return {\n    success: false,\n    errorType: ScraperErrorTypes.General,\n  };\n}\n\nclass BaseScraperWithBrowser<TCredentials extends ScraperCredentials> extends BaseScraper<TCredentials> {\n  // NOTICE - it is discourage 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 browser!: Browser;\n\n  // NOTICE - it is discourage 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 {\n      width: VIEWPORT_WIDTH,\n      height: VIEWPORT_HEIGHT,\n    };\n  }\n\n  async initialize() {\n    await super.initialize();\n    debug('initialize scraper');\n    this.emitProgress(ScraperProgressTypes.Initializing);\n\n    let env: Record<string, any> | undefined;\n    if (this.options.verbose) {\n      env = { DEBUG: '*', ...process.env };\n    }\n\n    if (typeof this.options.browser !== 'undefined' && this.options.browser !== null) {\n      debug('use custom browser instance provided in options');\n      this.browser = this.options.browser;\n    } else {\n      const executablePath = this.options.executablePath || undefined;\n      const args = this.options.args || [];\n      const { timeout } = this.options;\n\n      const headless = !this.options.showBrowser;\n      debug(`launch a browser with headless mode = ${headless}`);\n      this.browser = await puppeteer.launch({\n        env,\n        headless,\n        executablePath,\n        args,\n        timeout,\n      });\n    }\n\n    if (this.options.prepareBrowser) {\n      debug('execute \\'prepareBrowser\\' interceptor provided in options');\n      await this.options.prepareBrowser(this.browser);\n    }\n\n    if (!this.browser) {\n      debug('failed to initiate a browser, exit');\n      return;\n    }\n\n    const pages = await this.browser.pages();\n    if (pages.length) {\n      debug('browser has already pages open, use the first one');\n      [this.page] = pages;\n    } else {\n      debug('create a new browser page');\n      this.page = await this.browser.newPage();\n    }\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  async navigateTo(url: string, page?: Page, timeout?: number): Promise<void> {\n    const pageToUse = page || this.page;\n\n    if (!pageToUse) {\n      return;\n    }\n\n    const options = { ...(timeout === null ? null : { timeout }) };\n    const response = await pageToUse.goto(url, options);\n\n    // note: response will be null when navigating to same url while changing the hash part. the condition below will always accept null as valid result.\n    if (response !== null && (response === undefined || response.status() !== OK_STATUS)) {\n      throw new Error(`Error while trying to navigate to url ${url}`);\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);\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 handleLoginResult(this, 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    if (!this.browser) {\n      return;\n    }\n\n    await this.browser.close();\n  }\n}\n\nexport { BaseScraperWithBrowser };\n"]}
|
|
335
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper-with-browser.ts"],"names":["VIEWPORT_WIDTH","VIEWPORT_HEIGHT","OK_STATUS","debug","LoginBaseResults","Timeout","Generic","General","ScraperErrorTypes","rest","LoginResults","getKeyByValue","object","value","page","keys","Object","key","conditions","condition","result","RegExp","test","toLowerCase","Promise","resolve","UnknownError","createGeneralError","success","errorType","BaseScraperWithBrowser","BaseScraper","getViewPort","width","height","initialize","emitProgress","ScraperProgressTypes","Initializing","env","options","verbose","DEBUG","process","browser","executablePath","undefined","args","timeout","headless","showBrowser","puppeteer","launch","prepareBrowser","pages","length","newPage","defaultTimeout","setDefaultTimeout","preparePage","viewport","setViewport","on","request","failure","errorText","url","navigateTo","pageToUse","response","goto","status","Error","getLoginOptions","_credentials","companyId","fillInputs","pageOrFrame","fields","modified","input","shift","selector","login","credentials","loginOptions","userAgent","setUserAgent","loginUrl","checkReadiness","submitButtonSelector","loginFrameOrPage","preAction","LoggingIn","postAction","current","loginResult","possibleResults","handleLoginResult","terminate","_success","Terminating","storeFailureScreenShotPath","screenshot","path","fullPage","close","Success","LoginSuccess","InvalidPassword","LoginFailed","errorMessage","ChangePassword"],"mappings":"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","sourcesContent":["import puppeteer, { Browser, Frame, Page } from 'puppeteer';\n\nimport {\n  BaseScraper,\n\n} from './base-scraper';\nimport { getCurrentUrl, waitForNavigation } from '../helpers/navigation';\nimport { clickButton, fillInput, waitUntilElementFound } from '../helpers/elements-interactions';\nimport { getDebug } from '../helpers/debug';\nimport { ScraperErrorTypes } from './errors';\nimport { ScraperScrapingResult, ScraperCredentials } from './interface';\nimport { ScraperProgressTypes } from '../definitions';\n\nconst VIEWPORT_WIDTH = 1024;\nconst VIEWPORT_HEIGHT = 768;\nconst OK_STATUS = 200;\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\nexport type LoginResults = Exclude<ScraperErrorTypes,\nScraperErrorTypes.Timeout\n| ScraperErrorTypes.Generic\n| ScraperErrorTypes.General> | 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}\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  // 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 browser!: Browser;\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 {\n      width: VIEWPORT_WIDTH,\n      height: VIEWPORT_HEIGHT,\n    };\n  }\n\n  async initialize() {\n    await super.initialize();\n    debug('initialize scraper');\n    this.emitProgress(ScraperProgressTypes.Initializing);\n\n    let env: Record<string, any> | undefined;\n    if (this.options.verbose) {\n      env = { DEBUG: '*', ...process.env };\n    }\n\n    if (typeof this.options.browser !== 'undefined' && this.options.browser !== null) {\n      debug('use custom browser instance provided in options');\n      this.browser = this.options.browser;\n    } else {\n      const executablePath = this.options.executablePath || undefined;\n      const args = this.options.args || [];\n      const { timeout } = this.options;\n\n      const headless = !this.options.showBrowser;\n      debug(`launch a browser with headless mode = ${headless}`);\n      this.browser = await puppeteer.launch({\n        env,\n        headless,\n        executablePath,\n        args,\n        timeout,\n      });\n    }\n\n    if (this.options.prepareBrowser) {\n      debug('execute \\'prepareBrowser\\' interceptor provided in options');\n      await this.options.prepareBrowser(this.browser);\n    }\n\n    if (!this.browser) {\n      debug('failed to initiate a browser, exit');\n      return;\n    }\n\n    const pages = await this.browser.pages();\n    if (pages.length) {\n      debug('browser has already pages open, use the first one');\n      [this.page] = pages;\n    } else {\n      debug('create a new browser page');\n      this.page = await this.browser.newPage();\n    }\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  async navigateTo(url: string, page?: Page, timeout?: number): Promise<void> {\n    const pageToUse = page || this.page;\n\n    if (!pageToUse) {\n      return;\n    }\n\n    const options = { ...(timeout === null ? null : { timeout }) };\n    const response = await pageToUse.goto(url, options);\n\n    // note: response will be null when navigating to same url while changing the hash part. the condition below will always accept null as valid result.\n    if (response !== null && (response === undefined || response.status() !== OK_STATUS)) {\n      throw new Error(`Error while trying to navigate to url ${url}`);\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);\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    if (!this.browser) {\n      return;\n    }\n\n    await this.browser.close();\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: loginResult === LoginResults.InvalidPassword ? 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"]}
|
|
@@ -1,14 +1,5 @@
|
|
|
1
1
|
import { Scraper, ScraperCredentials, ScraperGetLongTermTwoFactorTokenResult, ScraperLoginResult, ScraperOptions, ScraperScrapingResult, ScraperTwoFactorAuthTriggerResult } from './interface';
|
|
2
|
-
|
|
3
|
-
Initializing = "INITIALIZING",
|
|
4
|
-
StartScraping = "START_SCRAPING",
|
|
5
|
-
LoggingIn = "LOGGING_IN",
|
|
6
|
-
LoginSuccess = "LOGIN_SUCCESS",
|
|
7
|
-
LoginFailed = "LOGIN_FAILED",
|
|
8
|
-
ChangePassword = "CHANGE_PASSWORD",
|
|
9
|
-
EndScraping = "END_SCRAPING",
|
|
10
|
-
Terminating = "TERMINATING"
|
|
11
|
-
}
|
|
2
|
+
import { CompanyTypes, ScraperProgressTypes } from '../definitions';
|
|
12
3
|
export declare class BaseScraper<TCredentials extends ScraperCredentials> implements Scraper<TCredentials> {
|
|
13
4
|
options: ScraperOptions;
|
|
14
5
|
private eventEmitter;
|
|
@@ -17,10 +8,12 @@ export declare class BaseScraper<TCredentials extends ScraperCredentials> implem
|
|
|
17
8
|
scrape(credentials: TCredentials): Promise<ScraperScrapingResult>;
|
|
18
9
|
triggerTwoFactorAuth(_phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult>;
|
|
19
10
|
getLongTermTwoFactorToken(_otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult>;
|
|
20
|
-
login(_credentials: TCredentials): Promise<ScraperLoginResult>;
|
|
21
|
-
fetchData(): Promise<ScraperScrapingResult>;
|
|
22
|
-
terminate(_success: boolean): Promise<void>;
|
|
23
|
-
emitProgress(type: ScraperProgressTypes): void;
|
|
24
|
-
emit(eventName: string, payload: Record<string, any>): void;
|
|
25
|
-
onProgress(func: (
|
|
11
|
+
protected login(_credentials: TCredentials): Promise<ScraperLoginResult>;
|
|
12
|
+
protected fetchData(): Promise<ScraperScrapingResult>;
|
|
13
|
+
protected terminate(_success: boolean): Promise<void>;
|
|
14
|
+
protected emitProgress(type: ScraperProgressTypes): void;
|
|
15
|
+
protected emit(eventName: string, payload: Record<string, any>): void;
|
|
16
|
+
onProgress(func: (companyId: CompanyTypes, payload: {
|
|
17
|
+
type: ScraperProgressTypes;
|
|
18
|
+
}) => void): void;
|
|
26
19
|
}
|
|
@@ -5,7 +5,7 @@ require("core-js/modules/es.promise");
|
|
|
5
5
|
Object.defineProperty(exports, "__esModule", {
|
|
6
6
|
value: true
|
|
7
7
|
});
|
|
8
|
-
exports.BaseScraper =
|
|
8
|
+
exports.BaseScraper = void 0;
|
|
9
9
|
|
|
10
10
|
var _events = require("events");
|
|
11
11
|
|
|
@@ -15,24 +15,13 @@ var _waiting = require("../helpers/waiting");
|
|
|
15
15
|
|
|
16
16
|
var _errors = require("./errors");
|
|
17
17
|
|
|
18
|
+
var _definitions = require("../definitions");
|
|
19
|
+
|
|
18
20
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
19
21
|
|
|
20
22
|
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
21
23
|
|
|
22
24
|
const SCRAPE_PROGRESS = 'SCRAPE_PROGRESS';
|
|
23
|
-
let ScraperProgressTypes;
|
|
24
|
-
exports.ScraperProgressTypes = ScraperProgressTypes;
|
|
25
|
-
|
|
26
|
-
(function (ScraperProgressTypes) {
|
|
27
|
-
ScraperProgressTypes["Initializing"] = "INITIALIZING";
|
|
28
|
-
ScraperProgressTypes["StartScraping"] = "START_SCRAPING";
|
|
29
|
-
ScraperProgressTypes["LoggingIn"] = "LOGGING_IN";
|
|
30
|
-
ScraperProgressTypes["LoginSuccess"] = "LOGIN_SUCCESS";
|
|
31
|
-
ScraperProgressTypes["LoginFailed"] = "LOGIN_FAILED";
|
|
32
|
-
ScraperProgressTypes["ChangePassword"] = "CHANGE_PASSWORD";
|
|
33
|
-
ScraperProgressTypes["EndScraping"] = "END_SCRAPING";
|
|
34
|
-
ScraperProgressTypes["Terminating"] = "TERMINATING";
|
|
35
|
-
})(ScraperProgressTypes || (exports.ScraperProgressTypes = ScraperProgressTypes = {}));
|
|
36
25
|
|
|
37
26
|
class BaseScraper {
|
|
38
27
|
constructor(options) {
|
|
@@ -43,13 +32,13 @@ class BaseScraper {
|
|
|
43
32
|
|
|
44
33
|
|
|
45
34
|
async initialize() {
|
|
46
|
-
this.emitProgress(ScraperProgressTypes.Initializing);
|
|
35
|
+
this.emitProgress(_definitions.ScraperProgressTypes.Initializing);
|
|
47
36
|
|
|
48
37
|
_momentTimezone.default.tz.setDefault('Asia/Jerusalem');
|
|
49
38
|
}
|
|
50
39
|
|
|
51
40
|
async scrape(credentials) {
|
|
52
|
-
this.emitProgress(ScraperProgressTypes.StartScraping);
|
|
41
|
+
this.emitProgress(_definitions.ScraperProgressTypes.StartScraping);
|
|
53
42
|
await this.initialize();
|
|
54
43
|
let loginResult;
|
|
55
44
|
|
|
@@ -78,7 +67,7 @@ class BaseScraper {
|
|
|
78
67
|
scrapeResult = (0, _errors.createGenericError)(e.message);
|
|
79
68
|
}
|
|
80
69
|
|
|
81
|
-
this.emitProgress(ScraperProgressTypes.EndScraping);
|
|
70
|
+
this.emitProgress(_definitions.ScraperProgressTypes.EndScraping);
|
|
82
71
|
return scrapeResult;
|
|
83
72
|
} // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await
|
|
84
73
|
|
|
@@ -104,7 +93,7 @@ class BaseScraper {
|
|
|
104
93
|
|
|
105
94
|
|
|
106
95
|
async terminate(_success) {
|
|
107
|
-
this.emitProgress(ScraperProgressTypes.Terminating);
|
|
96
|
+
this.emitProgress(_definitions.ScraperProgressTypes.Terminating);
|
|
108
97
|
}
|
|
109
98
|
|
|
110
99
|
emitProgress(type) {
|
|
@@ -124,4 +113,4 @@ class BaseScraper {
|
|
|
124
113
|
}
|
|
125
114
|
|
|
126
115
|
exports.BaseScraper = BaseScraper;
|
|
127
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
116
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/base-scraper.ts"],"names":["SCRAPE_PROGRESS","BaseScraper","constructor","options","EventEmitter","initialize","emitProgress","ScraperProgressTypes","Initializing","moment","tz","setDefault","scrape","credentials","StartScraping","loginResult","login","e","TimeoutError","message","scrapeResult","success","fetchData","terminate","EndScraping","triggerTwoFactorAuth","_phoneNumber","Error","companyId","getLongTermTwoFactorToken","_otpCode","_credentials","_success","Terminating","type","emit","eventName","payload","eventEmitter","onProgress","func","on"],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport moment from 'moment-timezone';\nimport { TimeoutError } from '../helpers/waiting';\nimport { createGenericError, createTimeoutError } from './errors';\nimport {\n  Scraper,\n  ScraperCredentials,\n  ScraperGetLongTermTwoFactorTokenResult,\n  ScraperLoginResult,\n  ScraperOptions,\n  ScraperScrapingResult,\n  ScraperTwoFactorAuthTriggerResult,\n} from './interface';\nimport { CompanyTypes, ScraperProgressTypes } from '../definitions';\n\nconst SCRAPE_PROGRESS = 'SCRAPE_PROGRESS';\n\n\nexport class BaseScraper<TCredentials extends ScraperCredentials> implements Scraper<TCredentials> {\n  private eventEmitter = new EventEmitter();\n\n  constructor(public options: ScraperOptions) {\n  }\n\n  // eslint-disable-next-line  @typescript-eslint/require-await\n  async initialize() {\n    this.emitProgress(ScraperProgressTypes.Initializing);\n    moment.tz.setDefault('Asia/Jerusalem');\n  }\n\n  async scrape(credentials: TCredentials): Promise<ScraperScrapingResult> {\n    this.emitProgress(ScraperProgressTypes.StartScraping);\n    await this.initialize();\n\n    let loginResult;\n    try {\n      loginResult = await this.login(credentials);\n    } catch (e) {\n      loginResult = e instanceof TimeoutError ?\n        createTimeoutError(e.message) :\n        createGenericError(e.message);\n    }\n\n    let scrapeResult;\n    if (loginResult.success) {\n      try {\n        scrapeResult = await this.fetchData();\n      } catch (e) {\n        scrapeResult =\n          e instanceof TimeoutError ?\n            createTimeoutError(e.message) :\n            createGenericError(e.message);\n      }\n    } else {\n      scrapeResult = loginResult;\n    }\n\n    try {\n      const success = scrapeResult && scrapeResult.success === true;\n      await this.terminate(success);\n    } catch (e) {\n      scrapeResult = createGenericError(e.message);\n    }\n    this.emitProgress(ScraperProgressTypes.EndScraping);\n\n    return scrapeResult;\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  triggerTwoFactorAuth(_phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult> {\n    throw new Error(`triggerOtp() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  getLongTermTwoFactorToken(_otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult> {\n    throw new Error(`getPermanentOtpToken() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  protected async login(_credentials: TCredentials): Promise<ScraperLoginResult> {\n    throw new Error(`login() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line  @typescript-eslint/require-await\n  protected async fetchData(): Promise<ScraperScrapingResult> {\n    throw new Error(`fetchData() is not created in ${this.options.companyId}`);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/require-await\n  protected async terminate(_success: boolean) {\n    this.emitProgress(ScraperProgressTypes.Terminating);\n  }\n\n  protected emitProgress(type: ScraperProgressTypes) {\n    this.emit(SCRAPE_PROGRESS, { type });\n  }\n\n  protected emit(eventName: string, payload: Record<string, any>) {\n    this.eventEmitter.emit(eventName, this.options.companyId, payload);\n  }\n\n  onProgress(func: (companyId: CompanyTypes, payload: {type: ScraperProgressTypes}) => void) {\n    this.eventEmitter.on(SCRAPE_PROGRESS, func);\n  }\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _definitions = require("../definitions");
|
|
4
|
+
|
|
5
|
+
var _factory = _interopRequireDefault(require("./factory"));
|
|
6
|
+
|
|
7
|
+
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
8
|
+
|
|
9
|
+
/* eslint-disable @typescript-eslint/unbound-method */
|
|
10
|
+
describe('Factory', () => {
|
|
11
|
+
test('should return a scraper instance', () => {
|
|
12
|
+
const scraper = (0, _factory.default)({
|
|
13
|
+
companyId: _definitions.CompanyTypes.hapoalim,
|
|
14
|
+
startDate: new Date()
|
|
15
|
+
});
|
|
16
|
+
expect(scraper).toBeDefined();
|
|
17
|
+
expect(scraper.scrape).toBeInstanceOf(Function);
|
|
18
|
+
expect(scraper.onProgress).toBeInstanceOf(Function);
|
|
19
|
+
});
|
|
20
|
+
});
|
|
21
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zY3JhcGVycy9mYWN0b3J5LnRlc3QudHMiXSwibmFtZXMiOlsiZGVzY3JpYmUiLCJ0ZXN0Iiwic2NyYXBlciIsImNvbXBhbnlJZCIsIkNvbXBhbnlUeXBlcyIsImhhcG9hbGltIiwic3RhcnREYXRlIiwiRGF0ZSIsImV4cGVjdCIsInRvQmVEZWZpbmVkIiwic2NyYXBlIiwidG9CZUluc3RhbmNlT2YiLCJGdW5jdGlvbiIsIm9uUHJvZ3Jlc3MiXSwibWFwcGluZ3MiOiI7O0FBQ0E7O0FBQ0E7Ozs7QUFGQTtBQUlBQSxRQUFRLENBQUMsU0FBRCxFQUFZLE1BQU07QUFDeEJDLEVBQUFBLElBQUksQ0FBQyxrQ0FBRCxFQUFxQyxNQUFNO0FBQzdDLFVBQU1DLE9BQU8sR0FBRyxzQkFBYztBQUM1QkMsTUFBQUEsU0FBUyxFQUFFQywwQkFBYUMsUUFESTtBQUU1QkMsTUFBQUEsU0FBUyxFQUFFLElBQUlDLElBQUo7QUFGaUIsS0FBZCxDQUFoQjtBQUlBQyxJQUFBQSxNQUFNLENBQUNOLE9BQUQsQ0FBTixDQUFnQk8sV0FBaEI7QUFFQUQsSUFBQUEsTUFBTSxDQUFDTixPQUFPLENBQUNRLE1BQVQsQ0FBTixDQUF1QkMsY0FBdkIsQ0FBc0NDLFFBQXRDO0FBQ0FKLElBQUFBLE1BQU0sQ0FBQ04sT0FBTyxDQUFDVyxVQUFULENBQU4sQ0FBMkJGLGNBQTNCLENBQTBDQyxRQUExQztBQUNELEdBVEcsQ0FBSjtBQVVELENBWE8sQ0FBUiIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC91bmJvdW5kLW1ldGhvZCAqL1xuaW1wb3J0IHsgQ29tcGFueVR5cGVzIH0gZnJvbSAnLi4vZGVmaW5pdGlvbnMnO1xuaW1wb3J0IGNyZWF0ZVNjcmFwZXIgZnJvbSAnLi9mYWN0b3J5JztcblxuZGVzY3JpYmUoJ0ZhY3RvcnknLCAoKSA9PiB7XG4gIHRlc3QoJ3Nob3VsZCByZXR1cm4gYSBzY3JhcGVyIGluc3RhbmNlJywgKCkgPT4ge1xuICAgIGNvbnN0IHNjcmFwZXIgPSBjcmVhdGVTY3JhcGVyKHtcbiAgICAgIGNvbXBhbnlJZDogQ29tcGFueVR5cGVzLmhhcG9hbGltLFxuICAgICAgc3RhcnREYXRlOiBuZXcgRGF0ZSgpLFxuICAgIH0pO1xuICAgIGV4cGVjdChzY3JhcGVyKS50b0JlRGVmaW5lZCgpO1xuXG4gICAgZXhwZWN0KHNjcmFwZXIuc2NyYXBlKS50b0JlSW5zdGFuY2VPZihGdW5jdGlvbik7XG4gICAgZXhwZWN0KHNjcmFwZXIub25Qcm9ncmVzcykudG9CZUluc3RhbmNlT2YoRnVuY3Rpb24pO1xuICB9KTtcbn0pO1xuIl19
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Browser, Page } from 'puppeteer';
|
|
2
|
-
import { CompanyTypes } from '../definitions';
|
|
2
|
+
import { CompanyTypes, ScraperProgressTypes } from '../definitions';
|
|
3
3
|
import { TransactionsAccount } from '../transactions';
|
|
4
4
|
import { ErrorResult, ScraperErrorTypes } from './errors';
|
|
5
5
|
export declare type ScraperCredentials = {
|
|
@@ -130,6 +130,11 @@ export interface ScraperScrapingResult {
|
|
|
130
130
|
}
|
|
131
131
|
export interface Scraper<TCredentials extends ScraperCredentials> {
|
|
132
132
|
scrape(credentials: TCredentials): Promise<ScraperScrapingResult>;
|
|
133
|
+
onProgress(func: (companyId: CompanyTypes, payload: {
|
|
134
|
+
type: ScraperProgressTypes;
|
|
135
|
+
}) => void): void;
|
|
136
|
+
triggerTwoFactorAuth(phoneNumber: string): Promise<ScraperTwoFactorAuthTriggerResult>;
|
|
137
|
+
getLongTermTwoFactorToken(otpCode: string): Promise<ScraperGetLongTermTwoFactorTokenResult>;
|
|
133
138
|
}
|
|
134
139
|
export declare type ScraperTwoFactorAuthTriggerResult = ErrorResult | {
|
|
135
140
|
success: true;
|
package/lib/scrapers/visa-cal.js
CHANGED
|
@@ -128,7 +128,8 @@ function convertParsedDataToTransactions(parsedData) {
|
|
|
128
128
|
processedDate: transaction.debCrdDate,
|
|
129
129
|
status: _transactions2.TransactionStatuses.Completed,
|
|
130
130
|
date: installments ? date.add(installments.number - 1, 'month').toISOString() : date.toISOString(),
|
|
131
|
-
type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ? _transactions2.TransactionTypes.Normal : _transactions2.TransactionTypes.Installments
|
|
131
|
+
type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ? _transactions2.TransactionTypes.Normal : _transactions2.TransactionTypes.Installments,
|
|
132
|
+
memo: transaction.transTypeCommentDetails.toString() || undefined
|
|
132
133
|
};
|
|
133
134
|
|
|
134
135
|
if (installments) {
|
|
@@ -161,7 +162,7 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
161
162
|
}
|
|
162
163
|
|
|
163
164
|
async getCards() {
|
|
164
|
-
const initData = await (0, _storage.getFromSessionStorage)(this.page, 'init');
|
|
165
|
+
const initData = await (0, _waiting.waitUntil)(() => (0, _storage.getFromSessionStorage)(this.page, 'init'), 'get init data in session storage', 10000, 1000);
|
|
165
166
|
|
|
166
167
|
if (!initData) {
|
|
167
168
|
throw new Error('could not find \'init\' data in session storage');
|
|
@@ -241,9 +242,7 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
241
242
|
const startMoment = _moment.default.max(defaultStartMoment, (0, _moment.default)(startDate));
|
|
242
243
|
|
|
243
244
|
debug(`fetch transactions starting ${startMoment.format()}`);
|
|
244
|
-
const Authorization = await this.getAuthorizationHeader();
|
|
245
|
-
|
|
246
|
-
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
245
|
+
const Authorization = await this.getAuthorizationHeader();
|
|
247
246
|
const cards = await this.getCards();
|
|
248
247
|
const xSiteId = await this.getXSiteId();
|
|
249
248
|
const futureMonthsToScrape = (_this$options$futureM = this.options.futureMonthsToScrape) !== null && _this$options$futureM !== void 0 ? _this$options$futureM : 1;
|
|
@@ -295,4 +294,4 @@ class VisaCalScraper extends _baseScraperWithBrowser.BaseScraperWithBrowser {
|
|
|
295
294
|
|
|
296
295
|
var _default = VisaCalScraper;
|
|
297
296
|
exports.default = _default;
|
|
298
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","trnTypeCode","getLoginFrame","page","frame","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","errorMessage","item","innerText","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","createLoginFields","credentials","selector","value","username","password","convertParsedDataToTransactions","parsedData","flatMap","monthData","result","bankAccounts","accounts","debitDates","debitDate","transactions","map","transaction","installments","curPaymentNum","numOfPayments","number","total","undefined","date","trnPurchaseDate","chargedAmount","amtBeforeConvAndIndex","originalAmount","trnAmt","credit","description","merchantName","originalCurrency","trnCurrencySymbol","processedDate","debCrdDate","status","TransactionStatuses","Completed","add","toISOString","type","regular","standingOrder","TransactionTypes","Normal","Installments","VisaCalScraper","BaseScraperWithBrowser","getCards","initData","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","auth","calConnectToken","getXSiteId","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","postAction","currentUrl","endsWith","e","userAgent","isCardTransactionDetails","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","moment","max","format","Authorization","setTimeout","xSiteId","futureMonthsToScrape","all","card","finalMonthToFetchMoment","months","diff","allMonthsData","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","combineInstallments","accountNumber","JSON","stringify","success"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { Frame, Page } from 'puppeteer';\n\nimport { getDebug } from '../helpers/debug';\nimport {\n  clickButton, elementPresentOnPage, pageEval, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport {\n  Transaction,\n  TransactionStatuses,\n  TransactionTypes,\n  TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum trnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: trnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = '₪' | string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: [], debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [async (options?: { page?: Page }) => {\n      const page = options?.page;\n      if (!page) {\n        return false;\n      }\n      return hasInvalidPasswordError(page);\n    }],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    // [LoginResults.ChangePassword]: [], // TODO add when reaching this scenario\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction convertParsedDataToTransactions(parsedData: CardTransactionDetails[]): Transaction[] {\n  return parsedData\n    .flatMap((monthData) => monthData.result.bankAccounts)\n    .flatMap((accounts) => accounts.debitDates)\n    .flatMap((debitDate) => debitDate.transactions)\n    .map((transaction) => {\n      const installments = (transaction.curPaymentNum && transaction.numOfPayments &&\n      {\n        number: transaction.curPaymentNum,\n        total: transaction.numOfPayments,\n      }) ||\n        undefined;\n\n      const date = moment(transaction.trnPurchaseDate);\n\n      let chargedAmount = transaction.amtBeforeConvAndIndex * (-1);\n      let originalAmount = transaction.trnAmt * (-1);\n\n      if (transaction.trnTypeCode === trnTypeCode.credit) {\n        chargedAmount = transaction.amtBeforeConvAndIndex;\n        originalAmount = transaction.trnAmt;\n      }\n\n      const result: Transaction = {\n        chargedAmount,\n        description: transaction.merchantName,\n        originalAmount,\n        originalCurrency: transaction.trnCurrencySymbol,\n        processedDate: transaction.debCrdDate,\n        status: TransactionStatuses.Completed,\n        date: installments ?\n          date.add(installments.number - 1, 'month').toISOString() :\n          date.toISOString(),\n        type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ?\n          TransactionTypes.Normal :\n          TransactionTypes.Installments,\n      };\n\n      if (installments) {\n        result.installments = installments;\n      }\n\n      return result;\n    });\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await getFromSessionStorage<InitResponse>(this.page, 'init');\n    if (!initData) {\n      throw new Error('could not find \\'init\\' data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error('could not find \\'auth-module\\' in session storage');\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitUntilElementFound(this.page, 'button.btn-close');\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          throw e;\n        }\n      },\n      userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\n    };\n  }\n\n  isCardTransactionDetails(result: CardTransactionDetails | CardTransactionDetailsError):\n    result is CardTransactionDetails {\n    return (result as CardTransactionDetails).result !== undefined;\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const Authorization = await this.getAuthorizationHeader();\n    // Wait a little before `this.getCards` so that it would exist\n    await new Promise((resolve) => setTimeout(resolve, 1000));\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    const accounts = await Promise.all(\n      cards.map(async (card) => {\n        debug(`fetch transactions for card ${card.cardUniqueId}`);\n\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n\n        const allMonthsData: (CardTransactionDetails)[] = [];\n        for (let i = 0; i <= months; i += 1) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page, TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);\n\n          if (!this.isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData);\n\n        debug('filer out old transactions');\n        const txns = (this.options.outputData?.enableTransactionsFilterByDate ?? true) ?\n          filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false) :\n          transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
|
|
297
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scrapers/visa-cal.ts"],"names":["LOGIN_URL","TRANSACTIONS_REQUEST_ENDPOINT","InvalidPasswordMessage","debug","trnTypeCode","getLoginFrame","page","frame","frames","find","f","url","includes","Promise","resolve","Error","hasInvalidPasswordError","errorFound","errorMessage","item","innerText","getPossibleLoginResults","urls","LoginResults","Success","InvalidPassword","options","createLoginFields","credentials","selector","value","username","password","convertParsedDataToTransactions","parsedData","flatMap","monthData","result","bankAccounts","accounts","debitDates","debitDate","transactions","map","transaction","installments","curPaymentNum","numOfPayments","number","total","undefined","date","trnPurchaseDate","chargedAmount","amtBeforeConvAndIndex","originalAmount","trnAmt","credit","description","merchantName","originalCurrency","trnCurrencySymbol","processedDate","debCrdDate","status","TransactionStatuses","Completed","add","toISOString","type","regular","standingOrder","TransactionTypes","Normal","Installments","memo","transTypeCommentDetails","toString","VisaCalScraper","BaseScraperWithBrowser","getCards","initData","cards","cardUniqueId","last4Digits","getAuthorizationHeader","authModule","auth","calConnectToken","getXSiteId","getLoginOptions","loginUrl","fields","submitButtonSelector","possibleResults","checkReadiness","preAction","openLoginPopup","postAction","currentUrl","endsWith","e","userAgent","isCardTransactionDetails","fetchData","defaultStartMoment","subtract","startDate","toDate","startMoment","moment","max","format","Authorization","xSiteId","futureMonthsToScrape","all","card","finalMonthToFetchMoment","months","diff","allMonthsData","i","month","clone","year","statusCode","title","push","txns","outputData","enableTransactionsFilterByDate","combineInstallments","accountNumber","JSON","stringify","success"],"mappings":"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","sourcesContent":["import moment from 'moment';\nimport { Frame, Page } from 'puppeteer';\n\nimport { getDebug } from '../helpers/debug';\nimport {\n  clickButton, elementPresentOnPage, pageEval, waitUntilElementFound,\n} from '../helpers/elements-interactions';\nimport { fetchPostWithinPage } from '../helpers/fetch';\nimport { getCurrentUrl } from '../helpers/navigation';\nimport { getFromSessionStorage } from '../helpers/storage';\nimport { filterOldTransactions } from '../helpers/transactions';\nimport { waitUntil } from '../helpers/waiting';\nimport {\n  Transaction,\n  TransactionStatuses,\n  TransactionTypes,\n  TransactionsAccount,\n} from '../transactions';\nimport { BaseScraperWithBrowser, LoginOptions, LoginResults } from './base-scraper-with-browser';\nimport { ScraperScrapingResult } from './interface';\n\nconst LOGIN_URL = 'https://www.cal-online.co.il/';\nconst TRANSACTIONS_REQUEST_ENDPOINT = 'https://api.cal-online.co.il/Transactions/api/transactionsDetails/getCardTransactionsDetails';\n\nconst InvalidPasswordMessage = 'שם המשתמש או הסיסמה שהוזנו שגויים';\n\nconst debug = getDebug('visa-cal');\n\nenum trnTypeCode {\n  regular = '5',\n  credit = '6',\n  installments = '8',\n  standingOrder = '9',\n}\n\ninterface ScrapedTransaction {\n  amtBeforeConvAndIndex: number;\n  branchCodeDesc: string;\n  cashAccManagerName: null;\n  cashAccountManager: null;\n  cashAccountTrnAmt: number;\n  chargeExternalToCardComment: string;\n  comments: [];\n  curPaymentNum: number;\n  debCrdCurrencySymbol: CurrencySymbol;\n  debCrdDate: string;\n  debitSpreadInd: boolean;\n  discountAmount: unknown;\n  discountReason: unknown;\n  immediateComments: [];\n  isImmediateCommentInd: boolean;\n  isImmediateHHKInd: boolean;\n  isMargarita: boolean;\n  isSpreadPaymenstAbroad: boolean;\n  linkedComments: [];\n  merchantAddress: string;\n  merchantName: string;\n  merchantPhoneNo: string;\n  numOfPayments: number;\n  onGoingTransactionsComment: string;\n  refundInd: boolean;\n  roundingAmount: unknown;\n  roundingReason: unknown;\n  tokenInd: 0;\n  tokenNumberPart4: '';\n  transCardPresentInd: boolean;\n  transTypeCommentDetails: [];\n  trnAmt: number;\n  trnCurrencySymbol: CurrencySymbol;\n  trnExacWay: number;\n  trnIntId: string;\n  trnNumaretor: number;\n  trnPurchaseDate: string;\n  trnType: string;\n  trnTypeCode: trnTypeCode;\n  walletProviderCode: 0;\n  walletProviderDesc: '';\n}\ninterface InitResponse {\n  result: {\n    cards: {\n      cardUniqueId: string;\n      last4Digits: string;\n      [key: string]: unknown;\n    }[];\n  };\n}\ntype CurrencySymbol = '₪' | string;\ninterface CardTransactionDetailsError {\n  title: string;\n  statusCode: number;\n}\ninterface CardTransactionDetails extends CardTransactionDetailsError {\n  result: {\n    bankAccounts: {\n      bankAccountNum: string;\n      bankName: string;\n      choiceExternalTransactions: any;\n      currentBankAccountInd: boolean;\n      debitDates: {\n        basketAmountComment: unknown;\n        choiceHHKDebit: number;\n        date: string;\n        debitReason: unknown;\n        fixDebitAmount: number;\n        fromPurchaseDate: string;\n        isChoiceRepaiment: boolean;\n        toPurchaseDate: string;\n        totalBasketAmount: number;\n        totalDebits: {\n          currencySymbol: CurrencySymbol;\n          amount: number;\n        }[];\n        transactions: ScrapedTransaction[];\n      }[];\n      immidiateDebits: { totalDebits: [], debitDays: [] };\n    }[];\n    blockedCardInd: boolean;\n  };\n  statusCode: 1;\n  statusDescription: string;\n  statusTitle: string;\n}\n\n\nasync function getLoginFrame(page: Page) {\n  let frame: Frame | null = null;\n  debug('wait until login frame found');\n  await waitUntil(() => {\n    frame = page\n      .frames()\n      .find((f) => f.url().includes('calconnect')) || null;\n    return Promise.resolve(!!frame);\n  }, 'wait for iframe with login form', 10000, 1000);\n\n  if (!frame) {\n    debug('failed to find login frame for 10 seconds');\n    throw new Error('failed to extract login iframe');\n  }\n\n  return frame;\n}\n\nasync function hasInvalidPasswordError(page: Page) {\n  const frame = await getLoginFrame(page);\n  const errorFound = await elementPresentOnPage(frame, 'div.general-error > div');\n  const errorMessage = errorFound ? await pageEval(frame, 'div.general-error > div', '', (item) => {\n    return (item as HTMLDivElement).innerText;\n  }) : '';\n  return errorMessage === InvalidPasswordMessage;\n}\n\nfunction getPossibleLoginResults() {\n  debug('return possible login results');\n  const urls: LoginOptions['possibleResults'] = {\n    [LoginResults.Success]: [/dashboard/i],\n    [LoginResults.InvalidPassword]: [async (options?: { page?: Page }) => {\n      const page = options?.page;\n      if (!page) {\n        return false;\n      }\n      return hasInvalidPasswordError(page);\n    }],\n    // [LoginResults.AccountBlocked]: [], // TODO add when reaching this scenario\n    // [LoginResults.ChangePassword]: [], // TODO add when reaching this scenario\n  };\n  return urls;\n}\n\nfunction createLoginFields(credentials: ScraperSpecificCredentials) {\n  debug('create login fields for username and password');\n  return [\n    { selector: '[formcontrolname=\"userName\"]', value: credentials.username },\n    { selector: '[formcontrolname=\"password\"]', value: credentials.password },\n  ];\n}\n\nfunction convertParsedDataToTransactions(parsedData: CardTransactionDetails[]): Transaction[] {\n  return parsedData\n    .flatMap((monthData) => monthData.result.bankAccounts)\n    .flatMap((accounts) => accounts.debitDates)\n    .flatMap((debitDate) => debitDate.transactions)\n    .map((transaction) => {\n      const installments = (transaction.curPaymentNum && transaction.numOfPayments &&\n      {\n        number: transaction.curPaymentNum,\n        total: transaction.numOfPayments,\n      }) ||\n        undefined;\n\n      const date = moment(transaction.trnPurchaseDate);\n\n      let chargedAmount = transaction.amtBeforeConvAndIndex * (-1);\n      let originalAmount = transaction.trnAmt * (-1);\n\n      if (transaction.trnTypeCode === trnTypeCode.credit) {\n        chargedAmount = transaction.amtBeforeConvAndIndex;\n        originalAmount = transaction.trnAmt;\n      }\n\n      const result: Transaction = {\n        chargedAmount,\n        description: transaction.merchantName,\n        originalAmount,\n        originalCurrency: transaction.trnCurrencySymbol,\n        processedDate: transaction.debCrdDate,\n        status: TransactionStatuses.Completed,\n        date: installments ?\n          date.add(installments.number - 1, 'month').toISOString() :\n          date.toISOString(),\n        type: [trnTypeCode.regular, trnTypeCode.standingOrder].includes(transaction.trnTypeCode) ?\n          TransactionTypes.Normal :\n          TransactionTypes.Installments,\n        memo: transaction.transTypeCommentDetails.toString() || undefined,\n      };\n\n      if (installments) {\n        result.installments = installments;\n      }\n\n      return result;\n    });\n}\n\ntype ScraperSpecificCredentials = { username: string, password: string };\n\nclass VisaCalScraper extends BaseScraperWithBrowser<ScraperSpecificCredentials> {\n  openLoginPopup = async () => {\n    debug('open login popup, wait until login button available');\n    await waitUntilElementFound(this.page, '#ccLoginDesktopBtn', true);\n    debug('click on the login button');\n    await clickButton(this.page, '#ccLoginDesktopBtn');\n    debug('get the frame that holds the login');\n    const frame = await getLoginFrame(this.page);\n    debug('wait until the password login tab header is available');\n    await waitUntilElementFound(frame, '#regular-login');\n    debug('navigate to the password login tab');\n    await clickButton(frame, '#regular-login');\n    debug('wait until the password login tab is active');\n    await waitUntilElementFound(frame, 'regular-login');\n\n    return frame;\n  };\n\n  async getCards() {\n    const initData = await waitUntil(\n      () => getFromSessionStorage<InitResponse>(this.page, 'init'),\n      'get init data in session storage',\n      10000,\n      1000,\n    );\n    if (!initData) {\n      throw new Error('could not find \\'init\\' data in session storage');\n    }\n    return initData?.result.cards.map(({ cardUniqueId, last4Digits }) => ({ cardUniqueId, last4Digits }));\n  }\n\n  async getAuthorizationHeader() {\n    const authModule = await getFromSessionStorage<{ auth: { calConnectToken: string } }>(this.page, 'auth-module');\n    if (!authModule) {\n      throw new Error('could not find \\'auth-module\\' in session storage');\n    }\n    return `CALAuthScheme ${authModule.auth.calConnectToken}`;\n  }\n\n  async getXSiteId() {\n    /*\n      I don't know if the constant below will change in the feature.\n      If so, use the next code:\n\n      return this.page.evaluate(() => new Ut().xSiteId);\n\n      To get the classname search for 'xSiteId' in the page source\n      class Ut {\n        constructor(_e, on, yn) {\n            this.store = _e,\n            this.config = on,\n            this.eventBusService = yn,\n            this.xSiteId = \"09031987-273E-2311-906C-8AF85B17C8D9\",\n    */\n    return Promise.resolve('09031987-273E-2311-906C-8AF85B17C8D9');\n  }\n\n  getLoginOptions(credentials: ScraperSpecificCredentials): LoginOptions {\n    return {\n      loginUrl: `${LOGIN_URL}`,\n      fields: createLoginFields(credentials),\n      submitButtonSelector: 'button[type=\"submit\"]',\n      possibleResults: getPossibleLoginResults(),\n      checkReadiness: async () => waitUntilElementFound(this.page, '#ccLoginDesktopBtn'),\n      preAction: this.openLoginPopup,\n      postAction: async () => {\n        try {\n          await waitUntilElementFound(this.page, 'button.btn-close');\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('site-tutorial')) {\n            await clickButton(this.page, 'button.btn-close');\n          }\n        } catch (e) {\n          const currentUrl = await getCurrentUrl(this.page);\n          if (currentUrl.endsWith('dashboard')) return;\n          throw e;\n        }\n      },\n      userAgent: 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36',\n    };\n  }\n\n  isCardTransactionDetails(result: CardTransactionDetails | CardTransactionDetailsError):\n    result is CardTransactionDetails {\n    return (result as CardTransactionDetails).result !== undefined;\n  }\n\n  async fetchData(): Promise<ScraperScrapingResult> {\n    const defaultStartMoment = moment().subtract(1, 'years').subtract(6, 'months').add(1, 'day');\n    const startDate = this.options.startDate || defaultStartMoment.toDate();\n    const startMoment = moment.max(defaultStartMoment, moment(startDate));\n    debug(`fetch transactions starting ${startMoment.format()}`);\n\n    const Authorization = await this.getAuthorizationHeader();\n    const cards = await this.getCards();\n    const xSiteId = await this.getXSiteId();\n    const futureMonthsToScrape = this.options.futureMonthsToScrape ?? 1;\n\n    const accounts = await Promise.all(\n      cards.map(async (card) => {\n        debug(`fetch transactions for card ${card.cardUniqueId}`);\n\n        const finalMonthToFetchMoment = moment().add(futureMonthsToScrape, 'month');\n        const months = finalMonthToFetchMoment.diff(startMoment, 'months');\n\n        const allMonthsData: (CardTransactionDetails)[] = [];\n        for (let i = 0; i <= months; i += 1) {\n          const month = finalMonthToFetchMoment.clone().subtract(i, 'months');\n          const monthData = await fetchPostWithinPage<CardTransactionDetails | CardTransactionDetailsError>(\n            this.page, TRANSACTIONS_REQUEST_ENDPOINT,\n            { cardUniqueId: card.cardUniqueId, month: month.format('M'), year: month.format('YYYY') },\n            {\n              Authorization,\n              'X-Site-Id': xSiteId,\n              'Content-Type': 'application/json',\n            },\n          );\n\n          if (monthData?.statusCode !== 1) throw new Error(`failed to fetch transactions for card ${card.last4Digits}. Message: ${monthData?.title || ''}`);\n\n          if (!this.isCardTransactionDetails(monthData)) {\n            throw new Error('monthData is not of type CardTransactionDetails');\n          }\n\n          allMonthsData.push(monthData);\n        }\n\n        const transactions = convertParsedDataToTransactions(allMonthsData);\n\n        debug('filer out old transactions');\n        const txns = (this.options.outputData?.enableTransactionsFilterByDate ?? true) ?\n          filterOldTransactions(transactions, moment(startDate), this.options.combineInstallments || false) :\n          transactions;\n\n        return {\n          txns,\n          accountNumber: card.last4Digits,\n        } as TransactionsAccount;\n      }),\n    );\n\n    debug('return the scraped accounts');\n\n    debug(JSON.stringify(accounts, null, 2));\n    return {\n      success: true,\n      accounts,\n    };\n  }\n}\n\nexport default VisaCalScraper;\n"]}
|