oxygen-cli 1.40.1 → 1.40.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,13 @@
1
1
  # CHANGELOG
2
2
 
3
+ ## v1.40.2 (2025-06-20)
4
+
5
+ #### :beetle: Bug Fix
6
+ * Various RP reporter fixes.
7
+
8
+ #### :nail_care: Polish
9
+ * Return stderr and exit code from `shell.exec`.
10
+
3
11
  ## v1.40.1 (2025-06-06)
4
12
 
5
13
  #### :beetle: Bug Fix
package/build/lib/cli.js CHANGED
@@ -147,4 +147,4 @@ Web test options:
147
147
  Mobile test options:
148
148
  -s, --server=SERVER_URL Appium server URL. Default is http://localhost:4723/wd/hub.`);
149
149
  }
150
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/lib/cli.js"],"names":["process","on","handleSigInt","error","console","argv","require","slice","v","version","log","exit","_","help","printUsage","d","delay","parseInt","targetFile","cliutil","processTargetPath","cwd","config","getConfigurations","generateTestOptions","then","options","prepareAndStartTheTest","code","e","framework","suites","Array","isArray","Error","capsArr","capabilities","exitCode","reporter","ReportAggregator","init","wsReporter","wsPort","undefined","WebSocketReporter","launcher","parallel","workers","isNaN","ParallelLauncher","Launcher","startAndWaitForClient","onLaunchStart","run","onLaunchEnd","generateReports","stop","getExitCode","trace","setTimeout"],"mappings":"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","sourcesContent":["#! /usr/bin/env node\n/*\n * Copyright (C) 2015-present CloudBeat Limited\n *\n * This program is free software: you can redistribute it and/or modify\n * it under the terms of the GNU General Public License as published by\n * the Free Software Foundation, either version 3 of the License, or\n * (at your option) any later version.\n */\nimport * as cliutil from './cli-util';\nimport Launcher from './launcher';\nimport ParallelLauncher from './parallel-launcher';\nimport ReportAggregator from '../reporter/ReportAggregator';\nimport WebSocketReporter from '../reporter/WebSocketReporter';\n\nprocess.on('SIGINT', handleSigInt);\nprocess.on('uncaughtException', error => {\n    console.error('uncaughtException', error);\n});\n\nprocess.on('unhandledRejection', error => {\n    console.error('unhandledRejection', error);\n});\n\n// parse command line arguments\nconst argv = require('minimist')(process.argv.slice(2));\n\nif (argv.v || argv.version) {\n    console.log(require('../../package.json').version);\n    process.exit(0);\n} else if (typeof(argv._[0]) === 'undefined' || argv.help) {\n    printUsage();\n    process.exit(1);\n}\nif (argv.d || argv.delay) {\n    const delay = argv.d || argv.delay;\n    if (!(parseInt(delay) > 0)) {\n        console.error(\"Invalid argument - 'delay' should be a non-negative number.\");\n        printUsage();\n        process.exit(1);\n    }\n}\n\nconst targetFile = cliutil.processTargetPath(argv._[0], argv.cwd);\nif (targetFile == null) {\n    printUsage();\n    process.exit(1);\n}\n\nconst config = cliutil.getConfigurations(targetFile, argv);\n\ncliutil.generateTestOptions(config, argv).then(\n    (options) => {\n        prepareAndStartTheTest(options).then(\n            (code) => {\n                console.log('Done!');\n                process.exit(code);\n            },\n            (e) => {\n                console.error('Test failed: ', e);\n                process.exit(1);\n            }\n        );\n    },\n    (e) => {\n        console.error('Test failed on generate state: ', e);\n        process.exit(1);\n    }\n);\n\nasync function prepareAndStartTheTest(options) {\n    if (options.framework === 'oxygen' && (!options.suites || !Array.isArray(options.suites))) {\n        throw new Error('Cannot start the test - no suites are specified.');\n    }\n    let capsArr = options.capabilities || [{}];\n    // check if capabilities object is an array or a hashtable\n    if (!(capsArr instanceof Array)) {\n        capsArr = [capsArr];\n    }\n    let exitCode = 0;\n    // start launcher\n    try {\n        const reporter = new ReportAggregator(options);\n        await reporter.init();\n        const wsReporter = options.wsPort !== undefined ?\n            new WebSocketReporter(reporter) : undefined;\n        const launcher = options.parallel && options.parallel.workers && !isNaN(options.parallel.workers) && options.parallel.workers > 1\n            ? new ParallelLauncher(options, reporter) : new Launcher(options, reporter);\n        await wsReporter?.startAndWaitForClient(options.wsPort);\n        console.log('Test started...');\n        await reporter.onLaunchStart(options);\n        await launcher.run(capsArr);\n        await reporter.onLaunchEnd();\n        // Generate file report only when no wsport argument is provided\n        if (!wsReporter) {\n            await reporter.generateReports();\n        }\n        wsReporter?.stop();\n        exitCode = reporter.getExitCode();\n    }\n    catch (e) {\n        console.error('Fatal error', e);\n        console.trace();\n        process.exit(1);\n    }\n    return exitCode;\n}\n\nfunction handleSigInt() {\n    // delay process exit to let Oxygen to properly dispose\n    setTimeout(() => process.exit(0), 2000);\n}\n\nfunction printUsage() {\n    console.log(`Usage: oxygen [OPTIONS]... FILE\n\nFILE - Path to a test script (.js) or a project configuration (.json) file.\n\nGeneral options:\n  -d, --delay=SECONDS        Delay between each command in seconds.\n      --rf={html|pdf|xml|excel|junit|json}  Reports file format. Default is html.\n      --ro=PATH              Output path for report file. If specified, the report\n                             will overwrite any previous reports.\n  -i, --iter=COUNT           Number of times to run the test. Default is 1.\n  -p, --param=FILE           Parameters file. If not specified an attempt will\n                             be made to load parameters from a file named same\n                             as the test script, located in the same directory,\n                             and having extension - xlsx, xls, csv, or txt.\n      --pm={seq|random|all}  Order in which to read the parameters - sequential,\n                             random, all. Default is seq.\n                             In 'seq' and 'random' modes test will run exact number\n                             of times specified with the -i option.\n                             In 'all' mode, all available parameters will be read\n                             sequentially. This option is mutually exclusive with\n                             -i option.\n      --dbgport=PORT         Debugger port.\n      --wsport=PORT          WebSocket events reporter port.\n      --suites               Filter out suites by name\n  -h, --help                 Display this information and exit.\n  -v, --version              Display version information and exit.\n\nWeb test options:\n  -b, --browser={chrome|ie|safari|firefox}  Browser name. Default is chrome.\n  -s, --server=SERVER_URL    Selenium hub URL. Default is http://localhost:4444/wd/hub.\n      --reopen={true|false}  Reopen browser on each iteration. Default is false.\n      \nMobile test options:\n    -s, --server=SERVER_URL  Appium server URL. Default is http://localhost:4723/wd/hub.`);\n}"]}
150
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/lib/cli.js"],"names":["process","on","handleSigInt","error","console","argv","require","slice","v","version","log","exit","_","help","printUsage","d","delay","parseInt","targetFile","cliutil","processTargetPath","cwd","config","getConfigurations","generateTestOptions","then","options","prepareAndStartTheTest","code","e","framework","suites","Array","isArray","Error","capsArr","capabilities","exitCode","reporter","ReportAggregator","init","wsReporter","wsPort","undefined","WebSocketReporter","launcher","parallel","workers","isNaN","ParallelLauncher","Launcher","startAndWaitForClient","onLaunchStart","run","onLaunchEnd","generateReports","stop","getExitCode","trace","setTimeout"],"mappings":"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","sourcesContent":["#! /usr/bin/env node\r\n/*\r\n * Copyright (C) 2015-present CloudBeat Limited\r\n *\r\n * This program is free software: you can redistribute it and/or modify\r\n * it under the terms of the GNU General Public License as published by\r\n * the Free Software Foundation, either version 3 of the License, or\r\n * (at your option) any later version.\r\n */\r\nimport * as cliutil from './cli-util';\r\nimport Launcher from './launcher';\r\nimport ParallelLauncher from './parallel-launcher';\r\nimport ReportAggregator from '../reporter/ReportAggregator';\r\nimport WebSocketReporter from '../reporter/WebSocketReporter';\r\n\r\nprocess.on('SIGINT', handleSigInt);\r\nprocess.on('uncaughtException', error => {\r\n    console.error('uncaughtException', error);\r\n});\r\n\r\nprocess.on('unhandledRejection', error => {\r\n    console.error('unhandledRejection', error);\r\n});\r\n\r\n// parse command line arguments\r\nconst argv = require('minimist')(process.argv.slice(2));\r\n\r\nif (argv.v || argv.version) {\r\n    console.log(require('../../package.json').version);\r\n    process.exit(0);\r\n} else if (typeof(argv._[0]) === 'undefined' || argv.help) {\r\n    printUsage();\r\n    process.exit(1);\r\n}\r\nif (argv.d || argv.delay) {\r\n    const delay = argv.d || argv.delay;\r\n    if (!(parseInt(delay) > 0)) {\r\n        console.error(\"Invalid argument - 'delay' should be a non-negative number.\");\r\n        printUsage();\r\n        process.exit(1);\r\n    }\r\n}\r\n\r\nconst targetFile = cliutil.processTargetPath(argv._[0], argv.cwd);\r\nif (targetFile == null) {\r\n    printUsage();\r\n    process.exit(1);\r\n}\r\n\r\nconst config = cliutil.getConfigurations(targetFile, argv);\r\n\r\ncliutil.generateTestOptions(config, argv).then(\r\n    (options) => {\r\n        prepareAndStartTheTest(options).then(\r\n            (code) => {\r\n                console.log('Done!');\r\n                process.exit(code);\r\n            },\r\n            (e) => {\r\n                console.error('Test failed: ', e);\r\n                process.exit(1);\r\n            }\r\n        );\r\n    },\r\n    (e) => {\r\n        console.error('Test failed on generate state: ', e);\r\n        process.exit(1);\r\n    }\r\n);\r\n\r\nasync function prepareAndStartTheTest(options) {\r\n    if (options.framework === 'oxygen' && (!options.suites || !Array.isArray(options.suites))) {\r\n        throw new Error('Cannot start the test - no suites are specified.');\r\n    }\r\n    let capsArr = options.capabilities || [{}];\r\n    // check if capabilities object is an array or a hashtable\r\n    if (!(capsArr instanceof Array)) {\r\n        capsArr = [capsArr];\r\n    }\r\n    let exitCode = 0;\r\n    // start launcher\r\n    try {\r\n        const reporter = new ReportAggregator(options);\r\n        await reporter.init();\r\n        const wsReporter = options.wsPort !== undefined ?\r\n            new WebSocketReporter(reporter) : undefined;\r\n        const launcher = options.parallel && options.parallel.workers && !isNaN(options.parallel.workers) && options.parallel.workers > 1\r\n            ? new ParallelLauncher(options, reporter) : new Launcher(options, reporter);\r\n        await wsReporter?.startAndWaitForClient(options.wsPort);\r\n        console.log('Test started...');\r\n        await reporter.onLaunchStart(options);\r\n        await launcher.run(capsArr);\r\n        await reporter.onLaunchEnd();\r\n        // Generate file report only when no wsport argument is provided\r\n        if (!wsReporter) {\r\n            await reporter.generateReports();\r\n        }\r\n        wsReporter?.stop();\r\n        exitCode = reporter.getExitCode();\r\n    }\r\n    catch (e) {\r\n        console.error('Fatal error', e);\r\n        console.trace();\r\n        process.exit(1);\r\n    }\r\n    return exitCode;\r\n}\r\n\r\nfunction handleSigInt() {\r\n    // delay process exit to let Oxygen to properly dispose\r\n    setTimeout(() => process.exit(0), 2000);\r\n}\r\n\r\nfunction printUsage() {\r\n    console.log(`Usage: oxygen [OPTIONS]... FILE\r\n\r\nFILE - Path to a test script (.js) or a project configuration (.json) file.\r\n\r\nGeneral options:\r\n  -d, --delay=SECONDS        Delay between each command in seconds.\r\n      --rf={html|pdf|xml|excel|junit|json}  Reports file format. Default is html.\r\n      --ro=PATH              Output path for report file. If specified, the report\r\n                             will overwrite any previous reports.\r\n  -i, --iter=COUNT           Number of times to run the test. Default is 1.\r\n  -p, --param=FILE           Parameters file. If not specified an attempt will\r\n                             be made to load parameters from a file named same\r\n                             as the test script, located in the same directory,\r\n                             and having extension - xlsx, xls, csv, or txt.\r\n      --pm={seq|random|all}  Order in which to read the parameters - sequential,\r\n                             random, all. Default is seq.\r\n                             In 'seq' and 'random' modes test will run exact number\r\n                             of times specified with the -i option.\r\n                             In 'all' mode, all available parameters will be read\r\n                             sequentially. This option is mutually exclusive with\r\n                             -i option.\r\n      --dbgport=PORT         Debugger port.\r\n      --wsport=PORT          WebSocket events reporter port.\r\n      --suites               Filter out suites by name\r\n  -h, --help                 Display this information and exit.\r\n  -v, --version              Display version information and exit.\r\n\r\nWeb test options:\r\n  -b, --browser={chrome|ie|safari|firefox}  Browser name. Default is chrome.\r\n  -s, --server=SERVER_URL    Selenium hub URL. Default is http://localhost:4444/wd/hub.\r\n      --reopen={true|false}  Reopen browser on each iteration. Default is false.\r\n      \r\nMobile test options:\r\n    -s, --server=SERVER_URL  Appium server URL. Default is http://localhost:4723/wd/hub.`);\r\n}"]}
@@ -37,8 +37,8 @@ class ShellModule extends _OxygenModule.default {
37
37
  this._alwaysInitialized = true; // pre-initialize the module
38
38
 
39
39
  this._isInitialized = true;
40
- this._lastCmd;
41
40
  this._lastStdout;
41
+ this._lastStderr;
42
42
  }
43
43
  /*
44
44
  * @summary Gets module name
@@ -54,14 +54,14 @@ class ShellModule extends _OxygenModule.default {
54
54
  * @summary Execute command
55
55
  * @description Spawn child process
56
56
  * @function exec
57
- * @param {String} command - Shell command to be executed, including arguments, if applicable.
58
- * @return {String} null | Error | stdout result
57
+ * @param {String} command - Shell command to be executed, including arguments, if applicable.
58
+ * @return {Object} Response object containing stdout, stderr, exit code, and signal (if process was terminated by a signal).
59
59
  */
60
60
 
61
61
 
62
62
  exec(command, options = {}) {
63
- this._lastCmd = command;
64
63
  this._lastStdout = null;
64
+ this._lastStderr = null;
65
65
 
66
66
  const spawn = require('cross-spawn');
67
67
 
@@ -78,15 +78,16 @@ class ShellModule extends _OxygenModule.default {
78
78
  }
79
79
 
80
80
  this._lastStdout = result.stdout ? result.stdout.toString() : null;
81
-
82
- if (result.stdout) {
83
- return result.stdout.toString();
84
- }
85
-
86
- return null;
81
+ this._lastStderr = result.stderr ? result.stderr.toString() : null;
82
+ return {
83
+ stdout: this._lastStdout,
84
+ stderr: this._lastStderr,
85
+ exitCode: result.status,
86
+ signal: result.signal
87
+ };
87
88
  }
88
89
  /**
89
- * @summary Assert whether the shell command output (stdout) is matching the specified pattern.
90
+ * @summary Assert that the shell command's output (stdout) matches the specified pattern.
90
91
  * @function assertOutput
91
92
  * @param {String} pattern - Pattern to assert.
92
93
  */
@@ -103,8 +104,26 @@ class ShellModule extends _OxygenModule.default {
103
104
 
104
105
  return true;
105
106
  }
107
+ /**
108
+ * @summary Assert that the shell command's error output (stderr) matches the specified pattern.
109
+ * @function assertErrorOutput
110
+ * @param {String} pattern - Pattern to assert.
111
+ */
112
+
113
+
114
+ assertErrorOutput(pattern) {
115
+ if (!this._lastStderr) {
116
+ return false;
117
+ }
118
+
119
+ if (!_utils.default.matchPattern(this._lastStderr, pattern)) {
120
+ throw new _OxygenError.default(_helper.default.errorCode.ASSERT_ERROR, `Expected shell command error output to match: "${pattern}" but got: "${this._lastStderr}"`);
121
+ }
122
+
123
+ return true;
124
+ }
106
125
 
107
126
  }
108
127
 
109
128
  exports.default = ShellModule;
110
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9veF9tb2R1bGVzL21vZHVsZS1zaGVsbC5qcyJdLCJuYW1lcyI6WyJNT0RVTEVfTkFNRSIsIlNoZWxsTW9kdWxlIiwiT3h5Z2VuTW9kdWxlIiwiY29uc3RydWN0b3IiLCJvcHRpb25zIiwiY29udGV4dCIsInJzIiwibG9nZ2VyIiwibW9kdWxlcyIsInNlcnZpY2VzIiwiX2Fsd2F5c0luaXRpYWxpemVkIiwiX2lzSW5pdGlhbGl6ZWQiLCJfbGFzdENtZCIsIl9sYXN0U3Rkb3V0IiwibmFtZSIsImV4ZWMiLCJjb21tYW5kIiwic3Bhd24iLCJyZXF1aXJlIiwiY3dkIiwic2hlbGwiLCJlbnYiLCJwcm9jZXNzIiwicmVzdWx0Iiwic3luYyIsImVycm9yIiwiT3hFcnJvciIsImVyckhlbHBlciIsImVycm9yQ29kZSIsIlNIRUxMX0VSUk9SIiwibWVzc2FnZSIsInN0ZG91dCIsInRvU3RyaW5nIiwiYXNzZXJ0T3V0cHV0IiwicGF0dGVybiIsIm1vZFV0aWxzIiwibWF0Y2hQYXR0ZXJuIiwiQVNTRVJUX0VSUk9SIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFRQTs7QUFDQTs7QUFDQTs7QUFDQTs7OztBQVhBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFNQSxNQUFNQSxXQUFXLEdBQUcsT0FBcEI7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFDZSxNQUFNQyxXQUFOLFNBQTBCQyxxQkFBMUIsQ0FBdUM7QUFDbERDLEVBQUFBLFdBQVcsQ0FBQ0MsT0FBRCxFQUFVQyxPQUFWLEVBQW1CQyxFQUFuQixFQUF1QkMsTUFBdkIsRUFBK0JDLE9BQS9CLEVBQXdDQyxRQUF4QyxFQUFrRDtBQUN6RCxVQUFNTCxPQUFOLEVBQWVDLE9BQWYsRUFBd0JDLEVBQXhCLEVBQTRCQyxNQUE1QixFQUFvQ0MsT0FBcEMsRUFBNkNDLFFBQTdDO0FBQ0EsU0FBS0Msa0JBQUwsR0FBMEIsSUFBMUIsQ0FGeUQsQ0FHekQ7O0FBQ0EsU0FBS0MsY0FBTCxHQUFzQixJQUF0QjtBQUNBLFNBQUtDLFFBQUw7QUFDQSxTQUFLQyxXQUFMO0FBQ0g7QUFFRDtBQUNKO0FBQ0E7QUFDQTtBQUNBOzs7QUFDWSxNQUFKQyxJQUFJLEdBQUc7QUFDUCxXQUFPZCxXQUFQO0FBQ0g7QUFFRDtBQUNKO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7O0FBQ0llLEVBQUFBLElBQUksQ0FBQ0MsT0FBRCxFQUFVWixPQUFPLEdBQUcsRUFBcEIsRUFBd0I7QUFDeEIsU0FBS1EsUUFBTCxHQUFnQkksT0FBaEI7QUFDQSxTQUFLSCxXQUFMLEdBQW1CLElBQW5COztBQUNBLFVBQU1JLEtBQUssR0FBR0MsT0FBTyxDQUFDLGFBQUQsQ0FBckI7O0FBQ0FkLElBQUFBLE9BQU8sR0FBRztBQUFFZSxNQUFBQSxHQUFHLEVBQUUsS0FBS2YsT0FBTCxDQUFhZSxHQUFwQjtBQUF5QkMsTUFBQUEsS0FBSyxFQUFFLElBQWhDO0FBQXNDQyxNQUFBQSxHQUFHLEVBQUVDLE9BQU8sQ0FBQ0QsR0FBbkQ7QUFBd0QsU0FBR2pCO0FBQTNELEtBQVY7QUFDQSxVQUFNbUIsTUFBTSxHQUFHTixLQUFLLENBQUNPLElBQU4sQ0FBV1IsT0FBWCxFQUFvQlosT0FBcEIsQ0FBZjs7QUFDQSxRQUFJbUIsTUFBTSxDQUFDRSxLQUFYLEVBQWtCO0FBQ2QsWUFBTSxJQUFJQyxvQkFBSixDQUFZQyxnQkFBVUMsU0FBVixDQUFvQkMsV0FBaEMsRUFBNkNOLE1BQU0sQ0FBQ0UsS0FBUCxDQUFhSyxPQUExRCxFQUFtRSxJQUFuRSxFQUF5RSxJQUF6RSxFQUErRVAsTUFBTSxDQUFDRSxLQUF0RixDQUFOO0FBQ0g7O0FBQ0QsU0FBS1osV0FBTCxHQUFtQlUsTUFBTSxDQUFDUSxNQUFQLEdBQWdCUixNQUFNLENBQUNRLE1BQVAsQ0FBY0MsUUFBZCxFQUFoQixHQUEyQyxJQUE5RDs7QUFDQSxRQUFJVCxNQUFNLENBQUNRLE1BQVgsRUFBbUI7QUFDZixhQUFPUixNQUFNLENBQUNRLE1BQVAsQ0FBY0MsUUFBZCxFQUFQO0FBQ0g7O0FBQ0QsV0FBTyxJQUFQO0FBQ0g7QUFFRDtBQUNKO0FBQ0E7QUFDQTtBQUNBOzs7QUFDSUMsRUFBQUEsWUFBWSxDQUFDQyxPQUFELEVBQVU7QUFDbEIsUUFBSSxDQUFDLEtBQUtyQixXQUFWLEVBQXVCO0FBQ25CLGFBQU8sS0FBUDtBQUNIOztBQUNELFFBQUksQ0FBQ3NCLGVBQVNDLFlBQVQsQ0FBc0IsS0FBS3ZCLFdBQTNCLEVBQXdDcUIsT0FBeEMsQ0FBTCxFQUF1RDtBQUNuRCxZQUFNLElBQUlSLG9CQUFKLENBQVlDLGdCQUFVQyxTQUFWLENBQW9CUyxZQUFoQyxFQUErQyw0Q0FBMkNILE9BQVEsZUFBYyxLQUFLckIsV0FBWSxHQUFqSSxDQUFOO0FBQ0g7O0FBQ0QsV0FBTyxJQUFQO0FBQ0g7O0FBdkRpRCIsInNvdXJjZXNDb250ZW50IjpbIi8qXHJcbiAqIENvcHlyaWdodCAoQykgMjAxNS0yMDE4IENsb3VkQmVhdCBMaW1pdGVkXHJcbiAqXHJcbiAqIFRoaXMgcHJvZ3JhbSBpcyBmcmVlIHNvZnR3YXJlOiB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IgbW9kaWZ5XHJcbiAqIGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgYXMgcHVibGlzaGVkIGJ5XHJcbiAqIHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb24sIGVpdGhlciB2ZXJzaW9uIDMgb2YgdGhlIExpY2Vuc2UsIG9yXHJcbiAqIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uXHJcbiAqL1xyXG5pbXBvcnQgT3h5Z2VuTW9kdWxlIGZyb20gJy4uL2NvcmUvT3h5Z2VuTW9kdWxlJztcclxuaW1wb3J0IE94RXJyb3IgZnJvbSAnLi4vZXJyb3JzL094eWdlbkVycm9yJztcclxuaW1wb3J0IGVyckhlbHBlciBmcm9tICcuLi9lcnJvcnMvaGVscGVyJztcclxuaW1wb3J0IG1vZFV0aWxzIGZyb20gJy4vdXRpbHMnO1xyXG5cclxuY29uc3QgTU9EVUxFX05BTUUgPSAnc2hlbGwnO1xyXG5cclxuLyoqXHJcbiAqIEBuYW1lIHNoZWxsXHJcbiAqIEBkZXNjcmlwdGlvbiBQcm92aWRlcyBtZXRob2RzIGZvciB3b3JraW5nIHdpdGggb3BlcmF0aW5nIHN5c3RlbSBzaGVsbC5cclxuICovXHJcbmV4cG9ydCBkZWZhdWx0IGNsYXNzIFNoZWxsTW9kdWxlIGV4dGVuZHMgT3h5Z2VuTW9kdWxlIHtcclxuICAgIGNvbnN0cnVjdG9yKG9wdGlvbnMsIGNvbnRleHQsIHJzLCBsb2dnZXIsIG1vZHVsZXMsIHNlcnZpY2VzKSB7XHJcbiAgICAgICAgc3VwZXIob3B0aW9ucywgY29udGV4dCwgcnMsIGxvZ2dlciwgbW9kdWxlcywgc2VydmljZXMpO1xyXG4gICAgICAgIHRoaXMuX2Fsd2F5c0luaXRpYWxpemVkID0gdHJ1ZTtcclxuICAgICAgICAvLyBwcmUtaW5pdGlhbGl6ZSB0aGUgbW9kdWxlXHJcbiAgICAgICAgdGhpcy5faXNJbml0aWFsaXplZCA9IHRydWU7XHJcbiAgICAgICAgdGhpcy5fbGFzdENtZDtcclxuICAgICAgICB0aGlzLl9sYXN0U3Rkb3V0O1xyXG4gICAgfVxyXG5cclxuICAgIC8qXHJcbiAgICAgKiBAc3VtbWFyeSBHZXRzIG1vZHVsZSBuYW1lXHJcbiAgICAgKiBAZnVuY3Rpb24gbmFtZVxyXG4gICAgICogQHJldHVybiB7U3RyaW5nfSBDb25zdGFudCB2YWx1ZSBcInNoZWxsXCIuXHJcbiAgICAgKi9cclxuICAgIGdldCBuYW1lKCkge1xyXG4gICAgICAgIHJldHVybiBNT0RVTEVfTkFNRTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIEBzdW1tYXJ5IEV4ZWN1dGUgY29tbWFuZFxyXG4gICAgICogQGRlc2NyaXB0aW9uIFNwYXduIGNoaWxkIHByb2Nlc3NcclxuICAgICAqIEBmdW5jdGlvbiBleGVjXHJcbiAgICAgKiBAcGFyYW0ge1N0cmluZ30gY29tbWFuZCAtIFNoZWxsIGNvbW1hbmQgdG8gYmUgZXhlY3V0ZWQsIGluY2x1ZGluZyBhcmd1bWVudHMsIGlmIGFwcGxpY2FibGUuIFxyXG4gICAgICogQHJldHVybiB7U3RyaW5nfSBudWxsIHwgRXJyb3IgfCBzdGRvdXQgcmVzdWx0XHJcbiAgICAgKi9cclxuICAgIGV4ZWMoY29tbWFuZCwgb3B0aW9ucyA9IHt9KSB7XHJcbiAgICAgICAgdGhpcy5fbGFzdENtZCA9IGNvbW1hbmQ7XHJcbiAgICAgICAgdGhpcy5fbGFzdFN0ZG91dCA9IG51bGw7XHJcbiAgICAgICAgY29uc3Qgc3Bhd24gPSByZXF1aXJlKCdjcm9zcy1zcGF3bicpO1xyXG4gICAgICAgIG9wdGlvbnMgPSB7IGN3ZDogdGhpcy5vcHRpb25zLmN3ZCwgc2hlbGw6IHRydWUsIGVudjogcHJvY2Vzcy5lbnYsIC4uLm9wdGlvbnMgfTtcclxuICAgICAgICBjb25zdCByZXN1bHQgPSBzcGF3bi5zeW5jKGNvbW1hbmQsIG9wdGlvbnMpO1xyXG4gICAgICAgIGlmIChyZXN1bHQuZXJyb3IpIHtcclxuICAgICAgICAgICAgdGhyb3cgbmV3IE94RXJyb3IoZXJySGVscGVyLmVycm9yQ29kZS5TSEVMTF9FUlJPUiwgcmVzdWx0LmVycm9yLm1lc3NhZ2UsIG51bGwsIHRydWUsIHJlc3VsdC5lcnJvcik7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHRoaXMuX2xhc3RTdGRvdXQgPSByZXN1bHQuc3Rkb3V0ID8gcmVzdWx0LnN0ZG91dC50b1N0cmluZygpIDogbnVsbDtcclxuICAgICAgICBpZiAocmVzdWx0LnN0ZG91dCkge1xyXG4gICAgICAgICAgICByZXR1cm4gcmVzdWx0LnN0ZG91dC50b1N0cmluZygpO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXR1cm4gbnVsbDtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIEBzdW1tYXJ5IEFzc2VydCB3aGV0aGVyIHRoZSBzaGVsbCBjb21tYW5kIG91dHB1dCAoc3Rkb3V0KSBpcyBtYXRjaGluZyB0aGUgc3BlY2lmaWVkIHBhdHRlcm4uXHJcbiAgICAgKiBAZnVuY3Rpb24gYXNzZXJ0T3V0cHV0XHJcbiAgICAgKiBAcGFyYW0ge1N0cmluZ30gcGF0dGVybiAtIFBhdHRlcm4gdG8gYXNzZXJ0LlxyXG4gICAgICovXHJcbiAgICBhc3NlcnRPdXRwdXQocGF0dGVybikge1xyXG4gICAgICAgIGlmICghdGhpcy5fbGFzdFN0ZG91dCkge1xyXG4gICAgICAgICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmICghbW9kVXRpbHMubWF0Y2hQYXR0ZXJuKHRoaXMuX2xhc3RTdGRvdXQsIHBhdHRlcm4pKSB7XHJcbiAgICAgICAgICAgIHRocm93IG5ldyBPeEVycm9yKGVyckhlbHBlci5lcnJvckNvZGUuQVNTRVJUX0VSUk9SLCBgRXhwZWN0ZWQgc2hlbGwgY29tbWFuZCBvdXRwdXQgdG8gbWF0Y2g6IFwiJHtwYXR0ZXJufVwiIGJ1dCBnb3Q6IFwiJHt0aGlzLl9sYXN0U3Rkb3V0fVwiYCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHJldHVybiB0cnVlO1xyXG4gICAgfVxyXG59Il19
129
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ox_modules/module-shell.js"],"names":["MODULE_NAME","ShellModule","OxygenModule","constructor","options","context","rs","logger","modules","services","_alwaysInitialized","_isInitialized","_lastStdout","_lastStderr","name","exec","command","spawn","require","cwd","shell","env","process","result","sync","error","OxError","errHelper","errorCode","SHELL_ERROR","message","stdout","toString","stderr","exitCode","status","signal","assertOutput","pattern","modUtils","matchPattern","ASSERT_ERROR","assertErrorOutput"],"mappings":"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","sourcesContent":["/*\r\n * Copyright (C) 2015-2018 CloudBeat Limited\r\n *\r\n * This program is free software: you can redistribute it and/or modify\r\n * it under the terms of the GNU General Public License as published by\r\n * the Free Software Foundation, either version 3 of the License, or\r\n * (at your option) any later version.\r\n */\r\nimport OxygenModule from '../core/OxygenModule';\r\nimport OxError from '../errors/OxygenError';\r\nimport errHelper from '../errors/helper';\r\nimport modUtils from './utils';\r\n\r\nconst MODULE_NAME = 'shell';\r\n\r\n/**\r\n * @name shell\r\n * @description Provides methods for working with operating system shell.\r\n */\r\nexport default class ShellModule extends OxygenModule {\r\n    constructor(options, context, rs, logger, modules, services) {\r\n        super(options, context, rs, logger, modules, services);\r\n        this._alwaysInitialized = true;\r\n        // pre-initialize the module\r\n        this._isInitialized = true;\r\n        this._lastStdout;\r\n        this._lastStderr;\r\n    }\r\n\r\n    /*\r\n     * @summary Gets module name\r\n     * @function name\r\n     * @return {String} Constant value \"shell\".\r\n     */\r\n    get name() {\r\n        return MODULE_NAME;\r\n    }\r\n\r\n    /**\r\n     * @summary Execute command\r\n     * @description Spawn child process\r\n     * @function exec\r\n     * @param {String} command - Shell command to be executed, including arguments, if applicable.\r\n     * @return {Object} Response object containing stdout, stderr, exit code, and signal (if process was terminated by a signal).\r\n     */\r\n    exec(command, options = {}) {\r\n        this._lastStdout = null;\r\n        this._lastStderr = null;\r\n\r\n        const spawn = require('cross-spawn');\r\n        options = { cwd: this.options.cwd, shell: true, env: process.env, ...options };\r\n        const result = spawn.sync(command, options);\r\n\r\n        if (result.error) {\r\n            throw new OxError(errHelper.errorCode.SHELL_ERROR, result.error.message, null, true, result.error);\r\n        }\r\n\r\n        this._lastStdout = result.stdout ? result.stdout.toString() : null;\r\n        this._lastStderr = result.stderr ? result.stderr.toString() : null;\r\n\r\n        return {\r\n            stdout: this._lastStdout,\r\n            stderr: this._lastStderr,\r\n            exitCode: result.status,\r\n            signal: result.signal\r\n        };\r\n    }\r\n\r\n    /**\r\n     * @summary Assert that the shell command's output (stdout) matches the specified pattern.\r\n     * @function assertOutput\r\n     * @param {String} pattern - Pattern to assert.\r\n     */\r\n    assertOutput(pattern) {\r\n        if (!this._lastStdout) {\r\n            return false;\r\n        }\r\n        if (!modUtils.matchPattern(this._lastStdout, pattern)) {\r\n            throw new OxError(errHelper.errorCode.ASSERT_ERROR, `Expected shell command output to match: \"${pattern}\" but got: \"${this._lastStdout}\"`);\r\n        }\r\n        return true;\r\n    }\r\n\r\n    /**\r\n     * @summary Assert that the shell command's error output (stderr) matches the specified pattern.\r\n     * @function assertErrorOutput\r\n     * @param {String} pattern - Pattern to assert.\r\n     */\r\n    assertErrorOutput(pattern) {\r\n        if (!this._lastStderr) {\r\n            return false;\r\n        }\r\n        if (!modUtils.matchPattern(this._lastStderr, pattern)) {\r\n            throw new OxError(errHelper.errorCode.ASSERT_ERROR, `Expected shell command error output to match: \"${pattern}\" but got: \"${this._lastStderr}\"`);\r\n        }\r\n        return true;\r\n    }\r\n}"]}
@@ -36,8 +36,9 @@ class ReportPortalReporter extends _ReporterBase.default {
36
36
  this.cbCaseToRpIdHash = {};
37
37
  this.cbStepToRpIdHash = {};
38
38
  this.cbSuiteResultListByRefId = {};
39
- this.currentTransactionStepId = undefined;
40
- this.currentSubStepId = undefined;
39
+ this.cbCaseToCurrentTransactionId = {};
40
+ this.cbCaseToCurrentSubStepId = {};
41
+ this.promises = [];
41
42
 
42
43
  if (!this.reporterOpts || !this.reporterOpts.apiKey || !this.reporterOpts.endpoint || !this.reporterOpts.project) {
43
44
  throw new Error('ReportPortal options are missing in oxygen.conf file.');
@@ -72,7 +73,7 @@ class ReportPortalReporter extends _ReporterBase.default {
72
73
  debug: false
73
74
  });
74
75
  this.tempLaunchId = tempId;
75
- await this.promiseWithTimeout(promise);
76
+ this.handlePromise(promise);
76
77
  } catch (e) {
77
78
  console.dir(`RP - Failed to start launch: ${e}`);
78
79
  }
@@ -85,12 +86,13 @@ class ReportPortalReporter extends _ReporterBase.default {
85
86
  const hasFailed = results.some(x => x.status === _status.default.FAILED);
86
87
 
87
88
  try {
89
+ await Promise.all(this.promises);
88
90
  const {
89
91
  promise
90
92
  } = await this.rpClient.finishLaunch(this.tempLaunchId, {
91
93
  status: hasFailed ? 'FAILED' : 'PASSED'
92
94
  });
93
- await this.promiseWithTimeout(promise);
95
+ await promise;
94
96
  } catch (e) {
95
97
  console.dir(`RP - Failed to end launch: ${e}`);
96
98
  }
@@ -131,7 +133,7 @@ class ReportPortalReporter extends _ReporterBase.default {
131
133
  promise
132
134
  } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId);
133
135
  this.cbSuiteToRpIdHash[suiteRefId] = tempId;
134
- await this.promiseWithTimeout(promise);
136
+ this.handlePromise(promise);
135
137
  } catch (e) {
136
138
  console.dir(`RP - Failed to start suite item: ${e}`);
137
139
  }
@@ -170,9 +172,9 @@ class ReportPortalReporter extends _ReporterBase.default {
170
172
  promise
171
173
  } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId, rpSuiteId);
172
174
  this.cbCaseToRpIdHash[caseId] = tempId;
173
- await this.promiseWithTimeout(promise);
175
+ this.handlePromise(promise);
174
176
  } catch (e) {
175
- console.dir(`RP - Failed to start test item: ${e}`);
177
+ console.dir(`RP - Failed to start case item: ${e}`);
176
178
  }
177
179
  }
178
180
 
@@ -210,13 +212,33 @@ class ReportPortalReporter extends _ReporterBase.default {
210
212
  const {
211
213
  promise
212
214
  } = this.rpClient.sendLog(rpTestId, logReq, rpFile);
213
- await this.promiseWithTimeout(promise);
215
+ this.handlePromise(promise);
214
216
  } catch (e) {
215
- console.dir(`RP - Failed to send log for finished test item: ${e}`);
217
+ console.dir(`RP - Failed to send log for finished case item: ${e}`);
216
218
  }
217
219
  }
218
220
  }
219
221
 
222
+ const transId = this.cbCaseToCurrentTransactionId[caseId];
223
+
224
+ if (transId) {
225
+ const finishTransactionReq = {
226
+ status: 'passed'
227
+ };
228
+
229
+ try {
230
+ const transRpId = this.cbStepToRpIdHash[transId];
231
+ const {
232
+ promise
233
+ } = this.rpClient.finishTestItem(transRpId, finishTransactionReq);
234
+ this.handlePromise(promise);
235
+ } catch (e) {
236
+ console.dir(`RP - Failed to finish transaction item on case end: ${e}`);
237
+ }
238
+
239
+ delete this.cbCaseToCurrentTransactionId[caseId];
240
+ }
241
+
220
242
  const finishTestItemReq = {
221
243
  status: result.status.toLowerCase()
222
244
  };
@@ -225,12 +247,10 @@ class ReportPortalReporter extends _ReporterBase.default {
225
247
  const {
226
248
  promise
227
249
  } = this.rpClient.finishTestItem(rpTestId, finishTestItemReq);
228
- await this.promiseWithTimeout(promise);
250
+ this.handlePromise(promise);
229
251
  } catch (e) {
230
- console.dir(`RP - Failed to finish test item: ${e}`);
252
+ console.dir(`RP - Failed to finish case item: ${e}`);
231
253
  }
232
-
233
- this.currentTransactionStepId = undefined;
234
254
  }
235
255
 
236
256
  async onStepStart({
@@ -249,14 +269,31 @@ class ReportPortalReporter extends _ReporterBase.default {
249
269
 
250
270
  const rpCaseId = this.cbCaseToRpIdHash[caseId];
251
271
  let rpParentId;
272
+ const transId = this.cbCaseToCurrentTransactionId[caseId];
252
273
 
253
274
  if (step.name === 'transaction') {
254
- this.currentTransactionStepId = step.id;
255
- this.currentSubStepId = undefined;
275
+ if (transId) {
276
+ const finishTransactionReq = {
277
+ status: 'passed'
278
+ };
279
+
280
+ try {
281
+ const transRpId = this.cbStepToRpIdHash[transId];
282
+ const {
283
+ promise
284
+ } = this.rpClient.finishTestItem(transRpId, finishTransactionReq);
285
+ this.handlePromise(promise);
286
+ } catch (e) {
287
+ console.dir(`RP - Failed to finish transaction item on case end: ${e}`);
288
+ }
289
+ }
290
+
291
+ this.cbCaseToCurrentTransactionId[caseId] = step.id;
292
+ delete this.cbCaseToCurrentSubStepId[caseId];
256
293
  rpParentId = rpCaseId;
257
294
  } else {
258
- this.currentSubStepId = step.id;
259
- rpParentId = this.currentTransactionStepId ? this.cbStepToRpIdHash[this.currentTransactionStepId] : rpCaseId;
295
+ this.cbCaseToCurrentSubStepId[caseId] = step.id;
296
+ rpParentId = transId ? this.cbStepToRpIdHash[transId] : rpCaseId;
260
297
  }
261
298
 
262
299
  if (!rpParentId || !rpCaseId) {
@@ -279,7 +316,7 @@ class ReportPortalReporter extends _ReporterBase.default {
279
316
  promise
280
317
  } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId, rpParentId);
281
318
  this.cbStepToRpIdHash[step.id] = tempId;
282
- await this.promiseWithTimeout(promise);
319
+ this.handlePromise(promise);
283
320
  } catch (e) {
284
321
  console.dir(`RP - Failed to start step item: ${e}`);
285
322
  }
@@ -287,13 +324,14 @@ class ReportPortalReporter extends _ReporterBase.default {
287
324
 
288
325
  async onStepEnd({
289
326
  rid,
327
+ caseId,
290
328
  step: result
291
329
  }) {
292
330
  if (!this.reportSteps) {
293
331
  return;
294
332
  }
295
333
 
296
- if (result.name && result.name.startsWith('log.')) {
334
+ if (result.name && (result.name.startsWith('log.') || result.name.startsWith('web.transaction') || result.name.startsWith('mob.transaction'))) {
297
335
  return;
298
336
  }
299
337
 
@@ -303,10 +341,7 @@ class ReportPortalReporter extends _ReporterBase.default {
303
341
  return;
304
342
  }
305
343
 
306
- if (result.name !== 'transaction') {
307
- this.currentSubStepId = undefined;
308
- }
309
-
344
+ delete this.cbCaseToCurrentSubStepId[caseId];
310
345
  const status = result.status.toLowerCase();
311
346
 
312
347
  if (status === 'failed' && result.failure) {
@@ -329,9 +364,9 @@ class ReportPortalReporter extends _ReporterBase.default {
329
364
  const {
330
365
  promise
331
366
  } = this.rpClient.sendLog(rpStepId, logReq, rpFile);
332
- await this.promiseWithTimeout(promise);
367
+ this.handlePromise(promise);
333
368
  } catch (e) {
334
- console.dir(`RP - Failed to send log for finished test item: ${e}`);
369
+ console.dir(`RP - Failed to send log for finished step item: ${e}`);
335
370
  }
336
371
  }
337
372
  }
@@ -344,7 +379,7 @@ class ReportPortalReporter extends _ReporterBase.default {
344
379
  const {
345
380
  promise
346
381
  } = this.rpClient.finishTestItem(rpStepId, finishTestItemReq);
347
- await this.promiseWithTimeout(promise);
382
+ this.handlePromise(promise);
348
383
  } catch (e) {
349
384
  console.dir(`RP - Failed to finish step item: ${e}`);
350
385
  }
@@ -384,7 +419,7 @@ class ReportPortalReporter extends _ReporterBase.default {
384
419
  const {
385
420
  promise
386
421
  } = this.rpClient.sendLog(rpParentId || this.tempLaunchId, logReq);
387
- await this.promiseWithTimeout(promise);
422
+ this.handlePromise(promise);
388
423
  } catch (e) {
389
424
  console.dir(`RP - Failed to create log item: ${e}`);
390
425
  }
@@ -401,10 +436,10 @@ class ReportPortalReporter extends _ReporterBase.default {
401
436
  const time = step.time;
402
437
  let stepId = undefined;
403
438
 
404
- if (this.currentSubStepId) {
405
- stepId = this.currentSubStepId;
406
- } else if (this.currentTransactionStepId) {
407
- stepId = this.currentTransactionStepId;
439
+ if (this.cbCaseToCurrentSubStepId[caseId]) {
440
+ stepId = this.cbCaseToCurrentSubStepId[caseId];
441
+ } else if (this.cbCaseToCurrentTransactionId[caseId]) {
442
+ stepId = this.cbCaseToCurrentTransactionId[caseId];
408
443
  }
409
444
 
410
445
  await this.sendRpLog({
@@ -446,7 +481,7 @@ class ReportPortalReporter extends _ReporterBase.default {
446
481
  const {
447
482
  promise
448
483
  } = this.rpClient.sendLog(rpParentId || this.tempLaunchId, logReq);
449
- await this.promiseWithTimeout(promise);
484
+ this.handlePromise(promise);
450
485
  } catch (e) {
451
486
  console.dir(`RP - Failed to create log item: ${e}`);
452
487
  }
@@ -467,7 +502,7 @@ class ReportPortalReporter extends _ReporterBase.default {
467
502
  const {
468
503
  promise
469
504
  } = this.rpClient.finishTestItem(rpSuiteId, finishTestItemReq);
470
- await this.promiseWithTimeout(promise);
505
+ this.handlePromise(promise);
471
506
  } catch (e) {
472
507
  console.dir(`RP - Failed to finish suite item: ${e}`);
473
508
  }
@@ -518,18 +553,14 @@ class ReportPortalReporter extends _ReporterBase.default {
518
553
  }
519
554
  }
520
555
 
521
- promiseWithTimeout(promise, timeout = 30 * 1000) {
522
- return new Promise((resolve, reject) => {
523
- if (!promise || !promise.then) {
524
- reject(new Error(`Promise await timeout of ${timeout} ms`));
525
- }
526
-
527
- promise.then(resolve, reject);
528
- setTimeout(reject, timeout);
556
+ handlePromise(promise) {
557
+ promise.catch(err => {
558
+ console.log('RP error: ', err);
529
559
  });
560
+ this.promises.push(promise);
530
561
  }
531
562
 
532
563
  }
533
564
 
534
565
  exports.default = ReportPortalReporter;
535
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ox_reporters/reporter-rp.js"],"names":["RPClient","require","LAUNCH_MODES","DEFAULT","DEBUG","TEST_ITEM_TYPES","SUITE","TEST","STEP","MAX_NAME_LENGTH","ReportPortalReporter","ReporterBase","constructor","options","reporterOpts","aggregator","rp","reportSteps","reportLogs","cbSuiteToRpIdHash","cbCaseToRpIdHash","cbStepToRpIdHash","cbSuiteResultListByRefId","currentTransactionStepId","undefined","currentSubStepId","apiKey","endpoint","project","Error","rpClient","launch","name","init","checkConnect","e","message","generate","results","onLaunchStart","tempId","promise","startLaunch","mode","debug","tempLaunchId","promiseWithTimeout","console","dir","onLaunchEnd","_reportEndOfStartedSuites","hasFailed","some","x","status","Status","FAILED","finishLaunch","onRunnerStart","rid","opts","caps","onRunnerEnd","result","onSuiteStart","suiteId","suite","suiteDef","suiteRefId","refId","startTestItemReq","type","startTestItem","onSuiteEnd","push","onCaseStart","caseId","caseDef","codeRef","path","rpSuiteId","onCaseEnd","rpTestId","toLowerCase","failure","rpFile","screenshot","content","logMsg","_stringify","logReq","level","sendLog","finishTestItemReq","finishTestItem","onStepStart","step","module","reportLogStep","rpCaseId","rpParentId","id","stepName","_getStepName","location","testCaseId","hasStats","onStepEnd","startsWith","rpStepId","file","onLog","stepId","msg","time","src","rpLevel","_getRpLevel","args","length","sendRpLog","Object","keys","hasFailedSuite","oxLevel","signature","substring","String","JSON","stringify","error","timeout","Promise","resolve","reject","then","setTimeout"],"mappings":"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","sourcesContent":["/*\r\n * Copyright (C) 2015-present CloudBeat Limited\r\n *\r\n * This program is free software: you can redistribute it and/or modify\r\n * it under the terms of the GNU General Public License as published by\r\n * the Free Software Foundation, either version 3 of the License, or\r\n * (at your option) any later version.\r\n */\r\n\r\n/*\r\n * Oxygen Reporter for ReportPortal\r\n */\r\nimport ReporterBase from '../reporter/ReporterBase';\r\nimport Status from '../model/status';\r\nconst RPClient = require('@oxygenhq/rp-client-javascript');\r\nconst LAUNCH_MODES = {\r\n    DEFAULT: 'DEFAULT',\r\n    DEBUG: 'DEBUG',\r\n};\r\nconst TEST_ITEM_TYPES = {\r\n    SUITE: 'SUITE',\r\n    TEST: 'TEST',\r\n    STEP: 'STEP',\r\n};\r\nconst MAX_NAME_LENGTH = 128;\r\n\r\nexport default class ReportPortalReporter extends ReporterBase {\r\n    constructor(options, reporterOpts, aggregator) {\r\n        super(options);\r\n        this.reporterOpts = reporterOpts || options.rp;\r\n        this.reportSteps = reporterOpts.reportSteps || true;\r\n        this.reportLogs = reporterOpts.reportLogs || false;\r\n        this.cbSuiteToRpIdHash = {};\r\n        this.cbCaseToRpIdHash = {};\r\n        this.cbStepToRpIdHash = {};\r\n        this.cbSuiteResultListByRefId = {};\r\n        this.currentTransactionStepId = undefined;\r\n        this.currentSubStepId = undefined;\r\n        if (\r\n            !this.reporterOpts\r\n            || !this.reporterOpts.apiKey\r\n            || !this.reporterOpts.endpoint\r\n            || !this.reporterOpts.project\r\n        ) {\r\n            throw new Error('ReportPortal options are missing in oxygen.conf file.');\r\n        }\r\n        this.aggregator = aggregator;\r\n        this.rpClient = new RPClient({\r\n            launch: this.reporterOpts.launch || this.options.name,\r\n            ...this.reporterOpts\r\n        });\r\n    }\r\n\r\n    async init() {\r\n        try {\r\n            await this.rpClient.checkConnect();\r\n        }\r\n        catch (e) {\r\n            throw new Error(`RP - Error connecting to the server: ${e.message}`);\r\n        }\r\n    }\r\n    async generate(results) {\r\n    }\r\n    // Events\r\n    async onLaunchStart({ options }) {\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startLaunch({\r\n                mode: this.reporterOpts.mode || LAUNCH_MODES.DEFAULT,\r\n                debug: false,\r\n            });\r\n            this.tempLaunchId = tempId;\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start launch: ${e}`);\r\n        }\r\n    }\r\n    async onLaunchEnd({ results }) {\r\n        // report the end of all started suites\r\n        await this._reportEndOfStartedSuites();\r\n        // Calculate launch status\r\n        const hasFailed = results.some(x => x.status === Status.FAILED);\r\n\r\n        try {\r\n            const { promise } = await this.rpClient.finishLaunch(this.tempLaunchId, {\r\n                status: hasFailed ? 'FAILED' : 'PASSED'\r\n            });\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to end launch: ${e}`);\r\n        }\r\n    }\r\n    async onRunnerStart({ rid, opts, caps }) {\r\n    }\r\n    async onRunnerEnd({ rid, result }) {\r\n    }\r\n    async onSuiteStart({ rid, suiteId, suite: suiteDef }) {\r\n        // in parallel test execution, onSuiteStart will be called multiple times for the same suite \r\n        // make sure we call \"startTestItem\" the same suite only once\r\n        const suiteRefId = suiteDef.refId;\r\n        if (this.cbSuiteToRpIdHash[suiteRefId]) {\r\n            return;\r\n        }\r\n        // assign empty object as a form of multi-thread lock,\r\n        // so the parallel call to onSuiteStart from the next thread\r\n        // will not try to proceed with the code below\r\n        this.cbSuiteToRpIdHash[suiteRefId] = {};\r\n        this.cbSuiteResultListByRefId[suiteRefId] = [];\r\n        const startTestItemReq = {\r\n            name: suiteDef.name || this.options.name,\r\n            type: TEST_ITEM_TYPES.SUITE,\r\n        };\r\n\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId);\r\n            this.cbSuiteToRpIdHash[suiteRefId] = tempId;\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start suite item: ${e}`);\r\n        }\r\n    }\r\n    async onSuiteEnd({ rid, suiteId, result }) {\r\n        const suiteRefId = result.refId;\r\n        this.cbSuiteResultListByRefId[suiteRefId] && this.cbSuiteResultListByRefId[suiteRefId].push(result);\r\n    }\r\n    async onCaseStart({ rid, suiteId, suiteRefId, caseId, 'case': caseDef }) {\r\n        const startTestItemReq = {\r\n            name: caseDef.name,\r\n            type: TEST_ITEM_TYPES.TEST,\r\n            codeRef: caseDef.path,\r\n        };\r\n        const rpSuiteId = this.cbSuiteToRpIdHash[suiteRefId];\r\n        if (!rpSuiteId) {\r\n            return;\r\n        }\r\n\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId, rpSuiteId);\r\n            this.cbCaseToRpIdHash[caseId] = tempId;\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start test item: ${e}`);\r\n        }\r\n    }\r\n    async onCaseEnd({ rid, suiteId, suiteRefId, caseId, result }) {\r\n        const rpTestId = this.cbCaseToRpIdHash[caseId];\r\n        if (!rpTestId) {\r\n            return;\r\n        }\r\n        const status = result.status.toLowerCase();\r\n        if (status === 'failed' && result.failure) {\r\n            const rpFile = result.screenshot ?\r\n            {\r\n                name: 'screenshot',\r\n                type: 'image/png',\r\n                content: result.screenshot,\r\n            } : undefined;\r\n\r\n            const logMsg = this._stringify(result.failure.message);\r\n            if (logMsg) {\r\n                const logReq = {\r\n                    message: logMsg,\r\n                    level: 'error'\r\n                };\r\n\r\n                try {\r\n                    const { promise } = this.rpClient.sendLog(rpTestId, logReq, rpFile);\r\n                    await this.promiseWithTimeout(promise);\r\n                } catch (e) {\r\n                    console.dir(`RP - Failed to send log for finished test item: ${e}`);\r\n                }\r\n            }\r\n        }\r\n        const finishTestItemReq = {\r\n            status: result.status.toLowerCase(),\r\n        };\r\n\r\n        try {\r\n            const { promise } = this.rpClient.finishTestItem(rpTestId, finishTestItemReq);\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to finish test item: ${e}`);\r\n        }\r\n\r\n        this.currentTransactionStepId = undefined;\r\n    }\r\n    async onStepStart({ rid, caseId, step }) {\r\n        if (!this.reportSteps) {\r\n            return;\r\n        }\r\n        if (step.module && step.module === 'log') {\r\n            await this.reportLogStep(caseId, step);\r\n            return;\r\n        }\r\n        const rpCaseId = this.cbCaseToRpIdHash[caseId];\r\n        let rpParentId;\r\n\r\n        if (step.name === 'transaction') {\r\n            this.currentTransactionStepId = step.id;\r\n            this.currentSubStepId = undefined;\r\n            rpParentId = rpCaseId;\r\n        }\r\n        else {\r\n            this.currentSubStepId = step.id;\r\n            rpParentId = this.currentTransactionStepId ? this.cbStepToRpIdHash[this.currentTransactionStepId] : rpCaseId;\r\n        }\r\n\r\n        if (!rpParentId || !rpCaseId) {\r\n            return;\r\n        }\r\n        const stepName = this._getStepName(step);\r\n        const startTestItemReq = {\r\n            name: stepName,\r\n            type: TEST_ITEM_TYPES.STEP,\r\n            // \"codeRef\" + \"parameters\" used by RP lib to generate testCaseId if testCaseId is not defined\r\n            // see https://github.com/reportportal/client-javascript/blob/486ef70c638a9a23267290c62df1faf900f7df6a/lib/report-portal-client.js#L497\r\n            // this is disabled for now. not sure if actually needed...\r\n            //parameters: this._getRpArgs(step.args),\r\n            codeRef: step.location,\r\n            testCaseId: rpCaseId,\r\n            hasStats: false,\r\n        };\r\n\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId, rpParentId);\r\n            this.cbStepToRpIdHash[step.id] = tempId;\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start step item: ${e}`);\r\n        }\r\n    }\r\n    async onStepEnd({ rid, step: result }) {\r\n        if (!this.reportSteps) {\r\n            return;\r\n        }\r\n        if (result.name && result.name.startsWith('log.')) {\r\n            return;\r\n        }\r\n        const rpStepId = this.cbStepToRpIdHash[result.id];\r\n        if (!rpStepId) {\r\n            return;\r\n        }\r\n        if (result.name !== 'transaction') {\r\n            this.currentSubStepId = undefined;\r\n        }\r\n        const status = result.status.toLowerCase();\r\n        if (status === 'failed' && result.failure) {\r\n            const rpFile = result.screenshot ?\r\n            {\r\n                name: 'screenshot',\r\n                type: 'image/png',\r\n                content: result.screenshot,\r\n            } : undefined;\r\n\r\n            const logMsg = this._stringify(result.failure.message);\r\n            if (logMsg) {\r\n                const logReq = {\r\n                    message: logMsg,\r\n                    level: 'error',\r\n                    file: rpFile,\r\n                };\r\n\r\n                try {\r\n                    const { promise } = this.rpClient.sendLog(rpStepId, logReq, rpFile);\r\n                    await this.promiseWithTimeout(promise);\r\n                } catch (e) {\r\n                    console.dir(`RP - Failed to send log for finished test item: ${e}`);\r\n                }\r\n            }\r\n        }\r\n        const finishTestItemReq = {\r\n            status: result.status.toLowerCase(),\r\n        };\r\n\r\n        try {\r\n            const { promise } = this.rpClient.finishTestItem(rpStepId, finishTestItemReq);\r\n            await this.promiseWithTimeout(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to finish step item: ${e}`);\r\n        }\r\n    }\r\n    async onLog({ suiteId, caseId, stepId, level, msg, time, src }) {\r\n        if (!this.reportLogs) {\r\n            return;\r\n        }\r\n        // Oxygen might start generating logs before onRunnerStart event, ignore them\r\n        if (!this.tempLaunchId) {\r\n            return;\r\n        }\r\n        const rpParentId = stepId ?\r\n            this.cbStepToRpIdHash[stepId]\r\n            : caseId ? this.cbCaseToRpIdHash[caseId]\r\n            : suiteId ? this.cbSuiteToRpIdHash[suiteId]\r\n            : undefined;\r\n        const rpLevel = this._getRpLevel(level);\r\n\r\n        const logMsg = this._stringify(msg);\r\n        if (logMsg) {\r\n            const logReq = {\r\n                message: logMsg,\r\n                level: rpLevel,\r\n                time: time,\r\n            };\r\n\r\n            try {\r\n                const { promise } = this.rpClient.sendLog(rpParentId || this.tempLaunchId, logReq);\r\n                await this.promiseWithTimeout(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to create log item: ${e}`);\r\n            }\r\n        }\r\n    }\r\n    async reportLogStep(caseId, step) {\r\n        if (!step.args || !step.args.length) {\r\n            return;\r\n        }\r\n        const level = step.name;\r\n        const msg = step.args[0];\r\n        const time = step.time;\r\n        let stepId = undefined;\r\n        if (this.currentSubStepId) {\r\n            stepId = this.currentSubStepId;\r\n        }\r\n        else if (this.currentTransactionStepId) {\r\n            stepId = this.currentTransactionStepId;\r\n        }\r\n        await this.sendRpLog({ suiteId: undefined, caseId, stepId, level, msg, time });\r\n    }\r\n    async sendRpLog({ suiteId, caseId, stepId, level, msg, time }) {\r\n        // Oxygen might start generating logs before onRunnerStart event, ignore them\r\n        if (!this.tempLaunchId) {\r\n            return;\r\n        }\r\n        const rpParentId = stepId ?\r\n            this.cbStepToRpIdHash[stepId]\r\n            : caseId ? this.cbCaseToRpIdHash[caseId]\r\n            : suiteId ? this.cbSuiteToRpIdHash[suiteId]\r\n            : undefined;\r\n        const rpLevel = this._getRpLevel(level);\r\n\r\n        const logMsg = this._stringify(msg);\r\n        if (logMsg) {\r\n            const logReq = {\r\n                message: logMsg,\r\n                level: rpLevel,\r\n                time: time,\r\n            };\r\n\r\n            try {\r\n                const { promise } = this.rpClient.sendLog(rpParentId || this.tempLaunchId, logReq);\r\n                await this.promiseWithTimeout(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to create log item: ${e}`);\r\n            }\r\n        }\r\n    }\r\n    async _reportEndOfStartedSuites() {\r\n        for (const suiteRefId of Object.keys(this.cbSuiteToRpIdHash)) {\r\n            const rpSuiteId = this.cbSuiteToRpIdHash[suiteRefId];\r\n            const results = this.cbSuiteResultListByRefId[suiteRefId];\r\n            delete this.cbSuiteToRpIdHash[suiteRefId];\r\n            const hasFailedSuite = results.some(x => x.status === Status.FAILED);\r\n            const finishTestItemReq = {\r\n                status: hasFailedSuite ? 'failed' : 'passed',\r\n            };\r\n\r\n            try {\r\n                const { promise } = this.rpClient.finishTestItem(rpSuiteId, finishTestItemReq);\r\n                await this.promiseWithTimeout(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to finish suite item: ${e}`);\r\n            }\r\n        }\r\n    }\r\n    _getRpLevel(oxLevel) {\r\n        if (oxLevel) {\r\n            return oxLevel.toLowerCase();\r\n        }\r\n        return 'info';\r\n    }\r\n    /*_getRpArgs(cbArgs) {\r\n        if (!cbArgs || !cbArgs.length) {\r\n            return undefined;\r\n        }\r\n        const rpArgs = [];\r\n        for (let i=0; i < cbArgs.length; i++) {\r\n            const arg = cbArgs[i];\r\n            let argStr = typeof arg === 'string' || arg instanceof String ? arg : JSON.stringify(arg);\r\n            rpArgs.push({\r\n                key: `arg${i}`,\r\n                value: argStr\r\n            });\r\n        }\r\n        return rpArgs;\r\n    }*/\r\n    _getStepName(step) {\r\n        let name;\r\n\r\n        if (step.name === 'transaction' && step.args.length > 0) {  // FIXME: steps need to be nested under transactions\r\n            name = step.args[0];\r\n        }\r\n        else if (step.signature) {\r\n            name = step.signature;\r\n        }\r\n        else if (step.module) {                         // FIXME: is this ever reached? \"step.signature\" seems to always exist\r\n            name = `${step.module}.${step.name}`;\r\n        }\r\n        else {\r\n            name = step.name;\r\n        }\r\n\r\n        // maximum allowed name length in RP is 1024 bytes\r\n        // we truncate it even lower so it will display nicely\r\n        if (name.length > MAX_NAME_LENGTH) {\r\n            name = name.substring(0, MAX_NAME_LENGTH-3) + '...';\r\n        }\r\n        return name;\r\n    }\r\n\r\n    // returns null if msg cannot be stringified, or strigifies it if it's not a string already\r\n    _stringify(msg) {\r\n        // we test for undefined and null explicitly on purpose because other falsy values (e.g. 0) are ok\r\n        if (typeof msg === 'undefined' || msg === null) {\r\n            return null;\r\n        }\r\n\r\n        if (typeof msg === 'string' || msg instanceof String) {\r\n            return msg;\r\n        }\r\n\r\n        try {\r\n            return JSON.stringify(msg);\r\n        } catch (e) {\r\n            console.error('RP - error sending log: msg object cannot be serialized');\r\n        }\r\n    }\r\n\r\n    promiseWithTimeout(promise, timeout = 30 * 1000) {\r\n        return new Promise((resolve, reject) => {\r\n            if (!promise || !promise.then) {\r\n                reject(new Error(`Promise await timeout of ${timeout} ms`));\r\n            }\r\n            promise.then(resolve, reject);\r\n            setTimeout(reject, timeout);\r\n        });\r\n    }\r\n}\r\n\r\n"]}
566
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/ox_reporters/reporter-rp.js"],"names":["RPClient","require","LAUNCH_MODES","DEFAULT","DEBUG","TEST_ITEM_TYPES","SUITE","TEST","STEP","MAX_NAME_LENGTH","ReportPortalReporter","ReporterBase","constructor","options","reporterOpts","aggregator","rp","reportSteps","reportLogs","cbSuiteToRpIdHash","cbCaseToRpIdHash","cbStepToRpIdHash","cbSuiteResultListByRefId","cbCaseToCurrentTransactionId","cbCaseToCurrentSubStepId","promises","apiKey","endpoint","project","Error","rpClient","launch","name","init","checkConnect","e","message","generate","results","onLaunchStart","tempId","promise","startLaunch","mode","debug","tempLaunchId","handlePromise","console","dir","onLaunchEnd","_reportEndOfStartedSuites","hasFailed","some","x","status","Status","FAILED","Promise","all","finishLaunch","onRunnerStart","rid","opts","caps","onRunnerEnd","result","onSuiteStart","suiteId","suite","suiteDef","suiteRefId","refId","startTestItemReq","type","startTestItem","onSuiteEnd","push","onCaseStart","caseId","caseDef","codeRef","path","rpSuiteId","onCaseEnd","rpTestId","toLowerCase","failure","rpFile","screenshot","content","undefined","logMsg","_stringify","logReq","level","sendLog","transId","finishTransactionReq","transRpId","finishTestItem","finishTestItemReq","onStepStart","step","module","reportLogStep","rpCaseId","rpParentId","id","stepName","_getStepName","location","testCaseId","hasStats","onStepEnd","startsWith","rpStepId","file","onLog","stepId","msg","time","src","rpLevel","_getRpLevel","args","length","sendRpLog","Object","keys","hasFailedSuite","oxLevel","signature","substring","String","JSON","stringify","error","catch","err","log"],"mappings":";;;;;;;;;AAYA;;AACA;;;;AACA,MAAMA,QAAQ,GAAGC,OAAO,CAAC,gCAAD,CAAxB;;AACA,MAAMC,YAAY,GAAG;AACjBC,EAAAA,OAAO,EAAE,SADQ;AAEjBC,EAAAA,KAAK,EAAE;AAFU,CAArB;AAIA,MAAMC,eAAe,GAAG;AACpBC,EAAAA,KAAK,EAAE,OADa;AAEpBC,EAAAA,IAAI,EAAE,MAFc;AAGpBC,EAAAA,IAAI,EAAE;AAHc,CAAxB;AAKA,MAAMC,eAAe,GAAG,GAAxB;;AAEe,MAAMC,oBAAN,SAAmCC,qBAAnC,CAAgD;AAC3DC,EAAAA,WAAW,CAACC,OAAD,EAAUC,YAAV,EAAwBC,UAAxB,EAAoC;AAC3C,UAAMF,OAAN;AACA,SAAKC,YAAL,GAAoBA,YAAY,IAAID,OAAO,CAACG,EAA5C;AACA,SAAKC,WAAL,GAAmBH,YAAY,CAACG,WAAb,IAA4B,IAA/C;AACA,SAAKC,UAAL,GAAkBJ,YAAY,CAACI,UAAb,IAA2B,KAA7C;AACA,SAAKC,iBAAL,GAAyB,EAAzB;AACA,SAAKC,gBAAL,GAAwB,EAAxB;AACA,SAAKC,gBAAL,GAAwB,EAAxB;AACA,SAAKC,wBAAL,GAAgC,EAAhC;AACA,SAAKC,4BAAL,GAAoC,EAApC;AACA,SAAKC,wBAAL,GAAgC,EAAhC;AACA,SAAKC,QAAL,GAAgB,EAAhB;;AAEA,QACI,CAAC,KAAKX,YAAN,IACG,CAAC,KAAKA,YAAL,CAAkBY,MADtB,IAEG,CAAC,KAAKZ,YAAL,CAAkBa,QAFtB,IAGG,CAAC,KAAKb,YAAL,CAAkBc,OAJ1B,EAKE;AACE,YAAM,IAAIC,KAAJ,CAAU,uDAAV,CAAN;AACH;;AACD,SAAKd,UAAL,GAAkBA,UAAlB;AACA,SAAKe,QAAL,GAAgB,IAAI9B,QAAJ,CAAa;AACzB+B,MAAAA,MAAM,EAAE,KAAKjB,YAAL,CAAkBiB,MAAlB,IAA4B,KAAKlB,OAAL,CAAamB,IADxB;AAEzB,SAAG,KAAKlB;AAFiB,KAAb,CAAhB;AAIH;;AAES,QAAJmB,IAAI,GAAG;AACT,QAAI;AACA,YAAM,KAAKH,QAAL,CAAcI,YAAd,EAAN;AACH,KAFD,CAGA,OAAOC,CAAP,EAAU;AACN,YAAM,IAAIN,KAAJ,CAAW,wCAAuCM,CAAC,CAACC,OAAQ,EAA5D,CAAN;AACH;AACJ;;AACa,QAARC,QAAQ,CAACC,OAAD,EAAU,CACvB;;AAEkB,QAAbC,aAAa,CAAC;AAAE1B,IAAAA;AAAF,GAAD,EAAc;AAC7B,QAAI;AACA,YAAM;AAAE2B,QAAAA,MAAF;AAAUC,QAAAA;AAAV,UAAsB,KAAKX,QAAL,CAAcY,WAAd,CAA0B;AAClDC,QAAAA,IAAI,EAAE,KAAK7B,YAAL,CAAkB6B,IAAlB,IAA0BzC,YAAY,CAACC,OADK;AAElDyC,QAAAA,KAAK,EAAE;AAF2C,OAA1B,CAA5B;AAIA,WAAKC,YAAL,GAAoBL,MAApB;AAEA,WAAKM,aAAL,CAAmBL,OAAnB;AACH,KARD,CASA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,gCAA+Bb,CAAE,EAA9C;AACH;AACJ;;AACgB,QAAXc,WAAW,CAAC;AAAEX,IAAAA;AAAF,GAAD,EAAc;AAE3B,UAAM,KAAKY,yBAAL,EAAN;AAEA,UAAMC,SAAS,GAAGb,OAAO,CAACc,IAAR,CAAaC,CAAC,IAAIA,CAAC,CAACC,MAAF,KAAaC,gBAAOC,MAAtC,CAAlB;;AAEA,QAAI;AACA,YAAMC,OAAO,CAACC,GAAR,CAAY,KAAKjC,QAAjB,CAAN;AAEA,YAAM;AAAEgB,QAAAA;AAAF,UAAc,MAAM,KAAKX,QAAL,CAAc6B,YAAd,CAA2B,KAAKd,YAAhC,EAA8C;AACpES,QAAAA,MAAM,EAAEH,SAAS,GAAG,QAAH,GAAc;AADqC,OAA9C,CAA1B;AAGA,YAAMV,OAAN;AACH,KAPD,CAQA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,8BAA6Bb,CAAE,EAA5C;AACH;AACJ;;AACkB,QAAbyB,aAAa,CAAC;AAAEC,IAAAA,GAAF;AAAOC,IAAAA,IAAP;AAAaC,IAAAA;AAAb,GAAD,EAAsB,CACxC;;AACgB,QAAXC,WAAW,CAAC;AAAEH,IAAAA,GAAF;AAAOI,IAAAA;AAAP,GAAD,EAAkB,CAClC;;AACiB,QAAZC,YAAY,CAAC;AAAEL,IAAAA,GAAF;AAAOM,IAAAA,OAAP;AAAgBC,IAAAA,KAAK,EAAEC;AAAvB,GAAD,EAAoC;AAGlD,UAAMC,UAAU,GAAGD,QAAQ,CAACE,KAA5B;;AACA,QAAI,KAAKpD,iBAAL,CAAuBmD,UAAvB,CAAJ,EAAwC;AACpC;AACH;;AAID,SAAKnD,iBAAL,CAAuBmD,UAAvB,IAAqC,EAArC;AACA,SAAKhD,wBAAL,CAA8BgD,UAA9B,IAA4C,EAA5C;AACA,UAAME,gBAAgB,GAAG;AACrBxC,MAAAA,IAAI,EAAEqC,QAAQ,CAACrC,IAAT,IAAiB,KAAKnB,OAAL,CAAamB,IADf;AAErByC,MAAAA,IAAI,EAAEpE,eAAe,CAACC;AAFD,KAAzB;;AAKA,QAAI;AACA,YAAM;AAAEkC,QAAAA,MAAF;AAAUC,QAAAA;AAAV,UAAsB,KAAKX,QAAL,CAAc4C,aAAd,CAA4BF,gBAA5B,EAA8C,KAAK3B,YAAnD,CAA5B;AACA,WAAK1B,iBAAL,CAAuBmD,UAAvB,IAAqC9B,MAArC;AACA,WAAKM,aAAL,CAAmBL,OAAnB;AACH,KAJD,CAKA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,oCAAmCb,CAAE,EAAlD;AACH;AACJ;;AACe,QAAVwC,UAAU,CAAC;AAAEd,IAAAA,GAAF;AAAOM,IAAAA,OAAP;AAAgBF,IAAAA;AAAhB,GAAD,EAA2B;AACvC,UAAMK,UAAU,GAAGL,MAAM,CAACM,KAA1B;AACA,SAAKjD,wBAAL,CAA8BgD,UAA9B,KAA6C,KAAKhD,wBAAL,CAA8BgD,UAA9B,EAA0CM,IAA1C,CAA+CX,MAA/C,CAA7C;AACH;;AACgB,QAAXY,WAAW,CAAC;AAAEhB,IAAAA,GAAF;AAAOM,IAAAA,OAAP;AAAgBG,IAAAA,UAAhB;AAA4BQ,IAAAA,MAA5B;AAAoC,YAAQC;AAA5C,GAAD,EAAwD;AACrE,UAAMP,gBAAgB,GAAG;AACrBxC,MAAAA,IAAI,EAAE+C,OAAO,CAAC/C,IADO;AAErByC,MAAAA,IAAI,EAAEpE,eAAe,CAACE,IAFD;AAGrByE,MAAAA,OAAO,EAAED,OAAO,CAACE;AAHI,KAAzB;AAKA,UAAMC,SAAS,GAAG,KAAK/D,iBAAL,CAAuBmD,UAAvB,CAAlB;;AACA,QAAI,CAACY,SAAL,EAAgB;AACZ;AACH;;AAED,QAAI;AACA,YAAM;AAAE1C,QAAAA,MAAF;AAAUC,QAAAA;AAAV,UAAsB,KAAKX,QAAL,CAAc4C,aAAd,CAA4BF,gBAA5B,EAA8C,KAAK3B,YAAnD,EAAiEqC,SAAjE,CAA5B;AACA,WAAK9D,gBAAL,CAAsB0D,MAAtB,IAAgCtC,MAAhC;AACA,WAAKM,aAAL,CAAmBL,OAAnB;AACH,KAJD,CAKA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,mCAAkCb,CAAE,EAAjD;AACH;AACJ;;AACc,QAATgD,SAAS,CAAC;AAAEtB,IAAAA,GAAF;AAAOM,IAAAA,OAAP;AAAgBG,IAAAA,UAAhB;AAA4BQ,IAAAA,MAA5B;AAAoCb,IAAAA;AAApC,GAAD,EAA+C;AAC1D,UAAMmB,QAAQ,GAAG,KAAKhE,gBAAL,CAAsB0D,MAAtB,CAAjB;;AACA,QAAI,CAACM,QAAL,EAAe;AACX;AACH;;AACD,UAAM9B,MAAM,GAAGW,MAAM,CAACX,MAAP,CAAc+B,WAAd,EAAf;;AACA,QAAI/B,MAAM,KAAK,QAAX,IAAuBW,MAAM,CAACqB,OAAlC,EAA2C;AACvC,YAAMC,MAAM,GAAGtB,MAAM,CAACuB,UAAP,GACf;AACIxD,QAAAA,IAAI,EAAE,YADV;AAEIyC,QAAAA,IAAI,EAAE,WAFV;AAGIgB,QAAAA,OAAO,EAAExB,MAAM,CAACuB;AAHpB,OADe,GAKXE,SALJ;;AAOA,YAAMC,MAAM,GAAG,KAAKC,UAAL,CAAgB3B,MAAM,CAACqB,OAAP,CAAelD,OAA/B,CAAf;;AACA,UAAIuD,MAAJ,EAAY;AACR,cAAME,MAAM,GAAG;AACXzD,UAAAA,OAAO,EAAEuD,MADE;AAEXG,UAAAA,KAAK,EAAE;AAFI,SAAf;;AAKA,YAAI;AACA,gBAAM;AAAErD,YAAAA;AAAF,cAAc,KAAKX,QAAL,CAAciE,OAAd,CAAsBX,QAAtB,EAAgCS,MAAhC,EAAwCN,MAAxC,CAApB;AACA,eAAKzC,aAAL,CAAmBL,OAAnB;AACH,SAHD,CAGE,OAAON,CAAP,EAAU;AACRY,UAAAA,OAAO,CAACC,GAAR,CAAa,mDAAkDb,CAAE,EAAjE;AACH;AACJ;AACJ;;AAED,UAAM6D,OAAO,GAAG,KAAKzE,4BAAL,CAAkCuD,MAAlC,CAAhB;;AAEA,QAAIkB,OAAJ,EAAa;AAET,YAAMC,oBAAoB,GAAG;AACzB3C,QAAAA,MAAM,EAAE;AADiB,OAA7B;;AAIA,UAAI;AACA,cAAM4C,SAAS,GAAG,KAAK7E,gBAAL,CAAsB2E,OAAtB,CAAlB;AAEA,cAAM;AAAEvD,UAAAA;AAAF,YAAc,KAAKX,QAAL,CAAcqE,cAAd,CAA6BD,SAA7B,EAAwCD,oBAAxC,CAApB;AACA,aAAKnD,aAAL,CAAmBL,OAAnB;AACH,OALD,CAMA,OAAON,CAAP,EAAU;AACNY,QAAAA,OAAO,CAACC,GAAR,CAAa,uDAAsDb,CAAE,EAArE;AACH;;AAED,aAAO,KAAKZ,4BAAL,CAAkCuD,MAAlC,CAAP;AACH;;AAED,UAAMsB,iBAAiB,GAAG;AACtB9C,MAAAA,MAAM,EAAEW,MAAM,CAACX,MAAP,CAAc+B,WAAd;AADc,KAA1B;;AAIA,QAAI;AACA,YAAM;AAAE5C,QAAAA;AAAF,UAAc,KAAKX,QAAL,CAAcqE,cAAd,CAA6Bf,QAA7B,EAAuCgB,iBAAvC,CAApB;AACA,WAAKtD,aAAL,CAAmBL,OAAnB;AACH,KAHD,CAIA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,oCAAmCb,CAAE,EAAlD;AACH;AACJ;;AAEgB,QAAXkE,WAAW,CAAC;AAAExC,IAAAA,GAAF;AAAOiB,IAAAA,MAAP;AAAewB,IAAAA;AAAf,GAAD,EAAwB;AACrC,QAAI,CAAC,KAAKrF,WAAV,EAAuB;AACnB;AACH;;AACD,QAAIqF,IAAI,CAACC,MAAL,IAAeD,IAAI,CAACC,MAAL,KAAgB,KAAnC,EAA0C;AACtC,YAAM,KAAKC,aAAL,CAAmB1B,MAAnB,EAA2BwB,IAA3B,CAAN;AACA;AACH;;AACD,UAAMG,QAAQ,GAAG,KAAKrF,gBAAL,CAAsB0D,MAAtB,CAAjB;AACA,QAAI4B,UAAJ;AAEA,UAAMV,OAAO,GAAG,KAAKzE,4BAAL,CAAkCuD,MAAlC,CAAhB;;AAEA,QAAIwB,IAAI,CAACtE,IAAL,KAAc,aAAlB,EAAiC;AAC7B,UAAIgE,OAAJ,EAAa;AAET,cAAMC,oBAAoB,GAAG;AACzB3C,UAAAA,MAAM,EAAE;AADiB,SAA7B;;AAIA,YAAI;AACA,gBAAM4C,SAAS,GAAG,KAAK7E,gBAAL,CAAsB2E,OAAtB,CAAlB;AACA,gBAAM;AAAEvD,YAAAA;AAAF,cAAc,KAAKX,QAAL,CAAcqE,cAAd,CAA6BD,SAA7B,EAAwCD,oBAAxC,CAApB;AACA,eAAKnD,aAAL,CAAmBL,OAAnB;AACH,SAJD,CAKA,OAAON,CAAP,EAAU;AACNY,UAAAA,OAAO,CAACC,GAAR,CAAa,uDAAsDb,CAAE,EAArE;AACH;AACJ;;AAED,WAAKZ,4BAAL,CAAkCuD,MAAlC,IAA4CwB,IAAI,CAACK,EAAjD;AACA,aAAO,KAAKnF,wBAAL,CAA8BsD,MAA9B,CAAP;AACA4B,MAAAA,UAAU,GAAGD,QAAb;AACH,KApBD,MAqBK;AACD,WAAKjF,wBAAL,CAA8BsD,MAA9B,IAAwCwB,IAAI,CAACK,EAA7C;AACAD,MAAAA,UAAU,GAAGV,OAAO,GAAG,KAAK3E,gBAAL,CAAsB2E,OAAtB,CAAH,GAAoCS,QAAxD;AACH;;AAED,QAAI,CAACC,UAAD,IAAe,CAACD,QAApB,EAA8B;AAC1B;AACH;;AACD,UAAMG,QAAQ,GAAG,KAAKC,YAAL,CAAkBP,IAAlB,CAAjB;;AACA,UAAM9B,gBAAgB,GAAG;AACrBxC,MAAAA,IAAI,EAAE4E,QADe;AAErBnC,MAAAA,IAAI,EAAEpE,eAAe,CAACG,IAFD;AAOrBwE,MAAAA,OAAO,EAAEsB,IAAI,CAACQ,QAPO;AAQrBC,MAAAA,UAAU,EAAEN,QARS;AASrBO,MAAAA,QAAQ,EAAE;AATW,KAAzB;;AAYA,QAAI;AACA,YAAM;AAAExE,QAAAA,MAAF;AAAUC,QAAAA;AAAV,UAAsB,KAAKX,QAAL,CAAc4C,aAAd,CAA4BF,gBAA5B,EAA8C,KAAK3B,YAAnD,EAAiE6D,UAAjE,CAA5B;AACA,WAAKrF,gBAAL,CAAsBiF,IAAI,CAACK,EAA3B,IAAiCnE,MAAjC;AACA,WAAKM,aAAL,CAAmBL,OAAnB;AACH,KAJD,CAKA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,mCAAkCb,CAAE,EAAjD;AACH;AACJ;;AACc,QAAT8E,SAAS,CAAC;AAAEpD,IAAAA,GAAF;AAAOiB,IAAAA,MAAP;AAAewB,IAAAA,IAAI,EAAErC;AAArB,GAAD,EAAgC;AAC3C,QAAI,CAAC,KAAKhD,WAAV,EAAuB;AACnB;AACH;;AACD,QAAIgD,MAAM,CAACjC,IAAP,KAAgBiC,MAAM,CAACjC,IAAP,CAAYkF,UAAZ,CAAuB,MAAvB,KACAjD,MAAM,CAACjC,IAAP,CAAYkF,UAAZ,CAAuB,iBAAvB,CADA,IAEAjD,MAAM,CAACjC,IAAP,CAAYkF,UAAZ,CAAuB,iBAAvB,CAFhB,CAAJ,EAEgE;AAC5D;AACH;;AACD,UAAMC,QAAQ,GAAG,KAAK9F,gBAAL,CAAsB4C,MAAM,CAAC0C,EAA7B,CAAjB;;AACA,QAAI,CAACQ,QAAL,EAAe;AACX;AACH;;AAED,WAAO,KAAK3F,wBAAL,CAA8BsD,MAA9B,CAAP;AAEA,UAAMxB,MAAM,GAAGW,MAAM,CAACX,MAAP,CAAc+B,WAAd,EAAf;;AACA,QAAI/B,MAAM,KAAK,QAAX,IAAuBW,MAAM,CAACqB,OAAlC,EAA2C;AACvC,YAAMC,MAAM,GAAGtB,MAAM,CAACuB,UAAP,GACf;AACIxD,QAAAA,IAAI,EAAE,YADV;AAEIyC,QAAAA,IAAI,EAAE,WAFV;AAGIgB,QAAAA,OAAO,EAAExB,MAAM,CAACuB;AAHpB,OADe,GAKXE,SALJ;;AAOA,YAAMC,MAAM,GAAG,KAAKC,UAAL,CAAgB3B,MAAM,CAACqB,OAAP,CAAelD,OAA/B,CAAf;;AACA,UAAIuD,MAAJ,EAAY;AACR,cAAME,MAAM,GAAG;AACXzD,UAAAA,OAAO,EAAEuD,MADE;AAEXG,UAAAA,KAAK,EAAE,OAFI;AAGXsB,UAAAA,IAAI,EAAE7B;AAHK,SAAf;;AAMA,YAAI;AACA,gBAAM;AAAE9C,YAAAA;AAAF,cAAc,KAAKX,QAAL,CAAciE,OAAd,CAAsBoB,QAAtB,EAAgCtB,MAAhC,EAAwCN,MAAxC,CAApB;AACA,eAAKzC,aAAL,CAAmBL,OAAnB;AACH,SAHD,CAGE,OAAON,CAAP,EAAU;AACRY,UAAAA,OAAO,CAACC,GAAR,CAAa,mDAAkDb,CAAE,EAAjE;AACH;AACJ;AACJ;;AACD,UAAMiE,iBAAiB,GAAG;AACtB9C,MAAAA,MAAM,EAAEW,MAAM,CAACX,MAAP,CAAc+B,WAAd;AADc,KAA1B;;AAIA,QAAI;AACA,YAAM;AAAE5C,QAAAA;AAAF,UAAc,KAAKX,QAAL,CAAcqE,cAAd,CAA6BgB,QAA7B,EAAuCf,iBAAvC,CAApB;AACA,WAAKtD,aAAL,CAAmBL,OAAnB;AACH,KAHD,CAIA,OAAON,CAAP,EAAU;AACNY,MAAAA,OAAO,CAACC,GAAR,CAAa,oCAAmCb,CAAE,EAAlD;AACH;AACJ;;AACU,QAALkF,KAAK,CAAC;AAAElD,IAAAA,OAAF;AAAWW,IAAAA,MAAX;AAAmBwC,IAAAA,MAAnB;AAA2BxB,IAAAA,KAA3B;AAAkCyB,IAAAA,GAAlC;AAAuCC,IAAAA,IAAvC;AAA6CC,IAAAA;AAA7C,GAAD,EAAqD;AAC5D,QAAI,CAAC,KAAKvG,UAAV,EAAsB;AAClB;AACH;;AAED,QAAI,CAAC,KAAK2B,YAAV,EAAwB;AACpB;AACH;;AACD,UAAM6D,UAAU,GAAGY,MAAM,GACrB,KAAKjG,gBAAL,CAAsBiG,MAAtB,CADqB,GAEnBxC,MAAM,GAAG,KAAK1D,gBAAL,CAAsB0D,MAAtB,CAAH,GACNX,OAAO,GAAG,KAAKhD,iBAAL,CAAuBgD,OAAvB,CAAH,GACPuB,SAJN;;AAKA,UAAMgC,OAAO,GAAG,KAAKC,WAAL,CAAiB7B,KAAjB,CAAhB;;AAEA,UAAMH,MAAM,GAAG,KAAKC,UAAL,CAAgB2B,GAAhB,CAAf;;AACA,QAAI5B,MAAJ,EAAY;AACR,YAAME,MAAM,GAAG;AACXzD,QAAAA,OAAO,EAAEuD,MADE;AAEXG,QAAAA,KAAK,EAAE4B,OAFI;AAGXF,QAAAA,IAAI,EAAEA;AAHK,OAAf;;AAMA,UAAI;AACA,cAAM;AAAE/E,UAAAA;AAAF,YAAc,KAAKX,QAAL,CAAciE,OAAd,CAAsBW,UAAU,IAAI,KAAK7D,YAAzC,EAAuDgD,MAAvD,CAApB;AACA,aAAK/C,aAAL,CAAmBL,OAAnB;AACH,OAHD,CAIA,OAAON,CAAP,EAAU;AACNY,QAAAA,OAAO,CAACC,GAAR,CAAa,mCAAkCb,CAAE,EAAjD;AACH;AACJ;AACJ;;AACkB,QAAbqE,aAAa,CAAC1B,MAAD,EAASwB,IAAT,EAAe;AAC9B,QAAI,CAACA,IAAI,CAACsB,IAAN,IAAc,CAACtB,IAAI,CAACsB,IAAL,CAAUC,MAA7B,EAAqC;AACjC;AACH;;AACD,UAAM/B,KAAK,GAAGQ,IAAI,CAACtE,IAAnB;AACA,UAAMuF,GAAG,GAAGjB,IAAI,CAACsB,IAAL,CAAU,CAAV,CAAZ;AACA,UAAMJ,IAAI,GAAGlB,IAAI,CAACkB,IAAlB;AACA,QAAIF,MAAM,GAAG5B,SAAb;;AACA,QAAI,KAAKlE,wBAAL,CAA8BsD,MAA9B,CAAJ,EAA2C;AACvCwC,MAAAA,MAAM,GAAG,KAAK9F,wBAAL,CAA8BsD,MAA9B,CAAT;AACH,KAFD,MAEO,IAAI,KAAKvD,4BAAL,CAAkCuD,MAAlC,CAAJ,EAA+C;AAClDwC,MAAAA,MAAM,GAAG,KAAK/F,4BAAL,CAAkCuD,MAAlC,CAAT;AACH;;AACD,UAAM,KAAKgD,SAAL,CAAe;AAAE3D,MAAAA,OAAO,EAAEuB,SAAX;AAAsBZ,MAAAA,MAAtB;AAA8BwC,MAAAA,MAA9B;AAAsCxB,MAAAA,KAAtC;AAA6CyB,MAAAA,GAA7C;AAAkDC,MAAAA;AAAlD,KAAf,CAAN;AACH;;AACc,QAATM,SAAS,CAAC;AAAE3D,IAAAA,OAAF;AAAWW,IAAAA,MAAX;AAAmBwC,IAAAA,MAAnB;AAA2BxB,IAAAA,KAA3B;AAAkCyB,IAAAA,GAAlC;AAAuCC,IAAAA;AAAvC,GAAD,EAAgD;AAE3D,QAAI,CAAC,KAAK3E,YAAV,EAAwB;AACpB;AACH;;AACD,UAAM6D,UAAU,GAAGY,MAAM,GACrB,KAAKjG,gBAAL,CAAsBiG,MAAtB,CADqB,GAEnBxC,MAAM,GAAG,KAAK1D,gBAAL,CAAsB0D,MAAtB,CAAH,GACNX,OAAO,GAAG,KAAKhD,iBAAL,CAAuBgD,OAAvB,CAAH,GACPuB,SAJN;;AAKA,UAAMgC,OAAO,GAAG,KAAKC,WAAL,CAAiB7B,KAAjB,CAAhB;;AAEA,UAAMH,MAAM,GAAG,KAAKC,UAAL,CAAgB2B,GAAhB,CAAf;;AACA,QAAI5B,MAAJ,EAAY;AACR,YAAME,MAAM,GAAG;AACXzD,QAAAA,OAAO,EAAEuD,MADE;AAEXG,QAAAA,KAAK,EAAE4B,OAFI;AAGXF,QAAAA,IAAI,EAAEA;AAHK,OAAf;;AAMA,UAAI;AACA,cAAM;AAAE/E,UAAAA;AAAF,YAAc,KAAKX,QAAL,CAAciE,OAAd,CAAsBW,UAAU,IAAI,KAAK7D,YAAzC,EAAuDgD,MAAvD,CAApB;AACA,aAAK/C,aAAL,CAAmBL,OAAnB;AACH,OAHD,CAIA,OAAON,CAAP,EAAU;AACNY,QAAAA,OAAO,CAACC,GAAR,CAAa,mCAAkCb,CAAE,EAAjD;AACH;AACJ;AACJ;;AAC8B,QAAzBe,yBAAyB,GAAG;AAC9B,SAAK,MAAMoB,UAAX,IAAyByD,MAAM,CAACC,IAAP,CAAY,KAAK7G,iBAAjB,CAAzB,EAA8D;AAC1D,YAAM+D,SAAS,GAAG,KAAK/D,iBAAL,CAAuBmD,UAAvB,CAAlB;AACA,YAAMhC,OAAO,GAAG,KAAKhB,wBAAL,CAA8BgD,UAA9B,CAAhB;AACA,aAAO,KAAKnD,iBAAL,CAAuBmD,UAAvB,CAAP;AACA,YAAM2D,cAAc,GAAG3F,OAAO,CAACc,IAAR,CAAaC,CAAC,IAAIA,CAAC,CAACC,MAAF,KAAaC,gBAAOC,MAAtC,CAAvB;AACA,YAAM4C,iBAAiB,GAAG;AACtB9C,QAAAA,MAAM,EAAE2E,cAAc,GAAG,QAAH,GAAc;AADd,OAA1B;;AAIA,UAAI;AACA,cAAM;AAAExF,UAAAA;AAAF,YAAc,KAAKX,QAAL,CAAcqE,cAAd,CAA6BjB,SAA7B,EAAwCkB,iBAAxC,CAApB;AACA,aAAKtD,aAAL,CAAmBL,OAAnB;AACH,OAHD,CAIA,OAAON,CAAP,EAAU;AACNY,QAAAA,OAAO,CAACC,GAAR,CAAa,qCAAoCb,CAAE,EAAnD;AACH;AACJ;AACJ;;AACDwF,EAAAA,WAAW,CAACO,OAAD,EAAU;AACjB,QAAIA,OAAJ,EAAa;AACT,aAAOA,OAAO,CAAC7C,WAAR,EAAP;AACH;;AACD,WAAO,MAAP;AACH;;AAgBDwB,EAAAA,YAAY,CAACP,IAAD,EAAO;AACf,QAAItE,IAAJ;;AAEA,QAAIsE,IAAI,CAACtE,IAAL,KAAc,aAAd,IAA+BsE,IAAI,CAACsB,IAAL,CAAUC,MAAV,GAAmB,CAAtD,EAAyD;AACrD7F,MAAAA,IAAI,GAAGsE,IAAI,CAACsB,IAAL,CAAU,CAAV,CAAP;AACH,KAFD,MAGK,IAAItB,IAAI,CAAC6B,SAAT,EAAoB;AACrBnG,MAAAA,IAAI,GAAGsE,IAAI,CAAC6B,SAAZ;AACH,KAFI,MAGA,IAAI7B,IAAI,CAACC,MAAT,EAAiB;AAClBvE,MAAAA,IAAI,GAAI,GAAEsE,IAAI,CAACC,MAAO,IAAGD,IAAI,CAACtE,IAAK,EAAnC;AACH,KAFI,MAGA;AACDA,MAAAA,IAAI,GAAGsE,IAAI,CAACtE,IAAZ;AACH;;AAID,QAAIA,IAAI,CAAC6F,MAAL,GAAcpH,eAAlB,EAAmC;AAC/BuB,MAAAA,IAAI,GAAGA,IAAI,CAACoG,SAAL,CAAe,CAAf,EAAkB3H,eAAe,GAAC,CAAlC,IAAuC,KAA9C;AACH;;AACD,WAAOuB,IAAP;AACH;;AAGD4D,EAAAA,UAAU,CAAC2B,GAAD,EAAM;AAEZ,QAAI,OAAOA,GAAP,KAAe,WAAf,IAA8BA,GAAG,KAAK,IAA1C,EAAgD;AAC5C,aAAO,IAAP;AACH;;AAED,QAAI,OAAOA,GAAP,KAAe,QAAf,IAA2BA,GAAG,YAAYc,MAA9C,EAAsD;AAClD,aAAOd,GAAP;AACH;;AAED,QAAI;AACA,aAAOe,IAAI,CAACC,SAAL,CAAehB,GAAf,CAAP;AACH,KAFD,CAEE,OAAOpF,CAAP,EAAU;AACRY,MAAAA,OAAO,CAACyF,KAAR,CAAc,yDAAd;AACH;AACJ;;AAED1F,EAAAA,aAAa,CAACL,OAAD,EAAU;AACnBA,IAAAA,OAAO,CAACgG,KAAR,CAAeC,GAAD,IAAS;AACnB3F,MAAAA,OAAO,CAAC4F,GAAR,CAAY,YAAZ,EAA0BD,GAA1B;AACH,KAFD;AAGA,SAAKjH,QAAL,CAAcmD,IAAd,CAAmBnC,OAAnB;AACH;;AAtd0D","sourcesContent":["/*\r\n * Copyright (C) 2015-present CloudBeat Limited\r\n *\r\n * This program is free software: you can redistribute it and/or modify\r\n * it under the terms of the GNU General Public License as published by\r\n * the Free Software Foundation, either version 3 of the License, or\r\n * (at your option) any later version.\r\n */\r\n\r\n/*\r\n * Oxygen Reporter for ReportPortal\r\n */\r\nimport ReporterBase from '../reporter/ReporterBase';\r\nimport Status from '../model/status';\r\nconst RPClient = require('@oxygenhq/rp-client-javascript');\r\nconst LAUNCH_MODES = {\r\n    DEFAULT: 'DEFAULT',\r\n    DEBUG: 'DEBUG',\r\n};\r\nconst TEST_ITEM_TYPES = {\r\n    SUITE: 'SUITE',\r\n    TEST: 'TEST',\r\n    STEP: 'STEP',\r\n};\r\nconst MAX_NAME_LENGTH = 128;\r\n\r\nexport default class ReportPortalReporter extends ReporterBase {\r\n    constructor(options, reporterOpts, aggregator) {\r\n        super(options);\r\n        this.reporterOpts = reporterOpts || options.rp;\r\n        this.reportSteps = reporterOpts.reportSteps || true;\r\n        this.reportLogs = reporterOpts.reportLogs || false;\r\n        this.cbSuiteToRpIdHash = {};\r\n        this.cbCaseToRpIdHash = {};\r\n        this.cbStepToRpIdHash = {};\r\n        this.cbSuiteResultListByRefId = {};\r\n        this.cbCaseToCurrentTransactionId = {};\r\n        this.cbCaseToCurrentSubStepId = {};\r\n        this.promises = [];\r\n\r\n        if (\r\n            !this.reporterOpts\r\n            || !this.reporterOpts.apiKey\r\n            || !this.reporterOpts.endpoint\r\n            || !this.reporterOpts.project\r\n        ) {\r\n            throw new Error('ReportPortal options are missing in oxygen.conf file.');\r\n        }\r\n        this.aggregator = aggregator;\r\n        this.rpClient = new RPClient({\r\n            launch: this.reporterOpts.launch || this.options.name,\r\n            ...this.reporterOpts\r\n        });\r\n    }\r\n\r\n    async init() {\r\n        try {\r\n            await this.rpClient.checkConnect();\r\n        }\r\n        catch (e) {\r\n            throw new Error(`RP - Error connecting to the server: ${e.message}`);\r\n        }\r\n    }\r\n    async generate(results) {\r\n    }\r\n    // Events\r\n    async onLaunchStart({ options }) {\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startLaunch({\r\n                mode: this.reporterOpts.mode || LAUNCH_MODES.DEFAULT,\r\n                debug: false,\r\n            });\r\n            this.tempLaunchId = tempId;\r\n\r\n            this.handlePromise(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start launch: ${e}`);\r\n        }\r\n    }\r\n    async onLaunchEnd({ results }) {\r\n        // report the end of all started suites\r\n        await this._reportEndOfStartedSuites();\r\n        // Calculate launch status\r\n        const hasFailed = results.some(x => x.status === Status.FAILED);\r\n\r\n        try {\r\n            await Promise.all(this.promises);\r\n\r\n            const { promise } = await this.rpClient.finishLaunch(this.tempLaunchId, {\r\n                status: hasFailed ? 'FAILED' : 'PASSED'\r\n            });\r\n            await promise;\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to end launch: ${e}`);\r\n        }\r\n    }\r\n    async onRunnerStart({ rid, opts, caps }) {\r\n    }\r\n    async onRunnerEnd({ rid, result }) {\r\n    }\r\n    async onSuiteStart({ rid, suiteId, suite: suiteDef }) {\r\n        // in parallel test execution, onSuiteStart will be called multiple times for the same suite \r\n        // make sure we call \"startTestItem\" the same suite only once\r\n        const suiteRefId = suiteDef.refId;\r\n        if (this.cbSuiteToRpIdHash[suiteRefId]) {\r\n            return;\r\n        }\r\n        // assign empty object as a form of multi-thread lock,\r\n        // so the parallel call to onSuiteStart from the next thread\r\n        // will not try to proceed with the code below\r\n        this.cbSuiteToRpIdHash[suiteRefId] = {};\r\n        this.cbSuiteResultListByRefId[suiteRefId] = [];\r\n        const startTestItemReq = {\r\n            name: suiteDef.name || this.options.name,\r\n            type: TEST_ITEM_TYPES.SUITE,\r\n        };\r\n\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId);\r\n            this.cbSuiteToRpIdHash[suiteRefId] = tempId;\r\n            this.handlePromise(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start suite item: ${e}`);\r\n        }\r\n    }\r\n    async onSuiteEnd({ rid, suiteId, result }) {\r\n        const suiteRefId = result.refId;\r\n        this.cbSuiteResultListByRefId[suiteRefId] && this.cbSuiteResultListByRefId[suiteRefId].push(result);\r\n    }\r\n    async onCaseStart({ rid, suiteId, suiteRefId, caseId, 'case': caseDef }) {\r\n        const startTestItemReq = {\r\n            name: caseDef.name,\r\n            type: TEST_ITEM_TYPES.TEST,\r\n            codeRef: caseDef.path,\r\n        };\r\n        const rpSuiteId = this.cbSuiteToRpIdHash[suiteRefId];\r\n        if (!rpSuiteId) {\r\n            return;\r\n        }\r\n\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId, rpSuiteId);\r\n            this.cbCaseToRpIdHash[caseId] = tempId;\r\n            this.handlePromise(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start case item: ${e}`);\r\n        }\r\n    }\r\n    async onCaseEnd({ rid, suiteId, suiteRefId, caseId, result }) {\r\n        const rpTestId = this.cbCaseToRpIdHash[caseId];\r\n        if (!rpTestId) {\r\n            return;\r\n        }\r\n        const status = result.status.toLowerCase();\r\n        if (status === 'failed' && result.failure) {\r\n            const rpFile = result.screenshot ?\r\n            {\r\n                name: 'screenshot',\r\n                type: 'image/png',\r\n                content: result.screenshot,\r\n            } : undefined;\r\n\r\n            const logMsg = this._stringify(result.failure.message);\r\n            if (logMsg) {\r\n                const logReq = {\r\n                    message: logMsg,\r\n                    level: 'error'\r\n                };\r\n\r\n                try {\r\n                    const { promise } = this.rpClient.sendLog(rpTestId, logReq, rpFile);\r\n                    this.handlePromise(promise);\r\n                } catch (e) {\r\n                    console.dir(`RP - Failed to send log for finished case item: ${e}`);\r\n                }\r\n            }\r\n        }\r\n\r\n        const transId = this.cbCaseToCurrentTransactionId[caseId];\r\n\r\n        if (transId) {\r\n            // FIXME: should calculate proper status\r\n            const finishTransactionReq = {\r\n                status: 'passed' //result.status.toLowerCase(),\r\n            };\r\n\r\n            try {\r\n                const transRpId = this.cbStepToRpIdHash[transId];\r\n\r\n                const { promise } = this.rpClient.finishTestItem(transRpId, finishTransactionReq);\r\n                this.handlePromise(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to finish transaction item on case end: ${e}`);\r\n            }\r\n\r\n            delete this.cbCaseToCurrentTransactionId[caseId];\r\n        }\r\n\r\n        const finishTestItemReq = {\r\n            status: result.status.toLowerCase(),\r\n        };\r\n\r\n        try {\r\n            const { promise } = this.rpClient.finishTestItem(rpTestId, finishTestItemReq);\r\n            this.handlePromise(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to finish case item: ${e}`);\r\n        }\r\n    }\r\n\r\n    async onStepStart({ rid, caseId, step }) {\r\n        if (!this.reportSteps) {\r\n            return;\r\n        }\r\n        if (step.module && step.module === 'log') {\r\n            await this.reportLogStep(caseId, step);\r\n            return;\r\n        }\r\n        const rpCaseId = this.cbCaseToRpIdHash[caseId];\r\n        let rpParentId;\r\n\r\n        const transId = this.cbCaseToCurrentTransactionId[caseId];\r\n\r\n        if (step.name === 'transaction') {\r\n            if (transId) {\r\n                // FIXME: should calculate proper status\r\n                const finishTransactionReq = {\r\n                    status: 'passed' //result.status.toLowerCase(),\r\n                };\r\n\r\n                try {\r\n                    const transRpId = this.cbStepToRpIdHash[transId];\r\n                    const { promise } = this.rpClient.finishTestItem(transRpId, finishTransactionReq);\r\n                    this.handlePromise(promise);\r\n                }\r\n                catch (e) {\r\n                    console.dir(`RP - Failed to finish transaction item on case end: ${e}`);\r\n                }\r\n            }\r\n\r\n            this.cbCaseToCurrentTransactionId[caseId] = step.id;\r\n            delete this.cbCaseToCurrentSubStepId[caseId];\r\n            rpParentId = rpCaseId;\r\n        }\r\n        else {\r\n            this.cbCaseToCurrentSubStepId[caseId] = step.id;\r\n            rpParentId = transId ? this.cbStepToRpIdHash[transId] : rpCaseId;\r\n        }\r\n\r\n        if (!rpParentId || !rpCaseId) {\r\n            return;\r\n        }\r\n        const stepName = this._getStepName(step);\r\n        const startTestItemReq = {\r\n            name: stepName,\r\n            type: TEST_ITEM_TYPES.STEP,\r\n            // \"codeRef\" + \"parameters\" used by RP lib to generate testCaseId if testCaseId is not defined\r\n            // see https://github.com/reportportal/client-javascript/blob/486ef70c638a9a23267290c62df1faf900f7df6a/lib/report-portal-client.js#L497\r\n            // this is disabled for now. not sure if actually needed...\r\n            //parameters: this._getRpArgs(step.args),\r\n            codeRef: step.location,\r\n            testCaseId: rpCaseId,\r\n            hasStats: false,\r\n        };\r\n\r\n        try {\r\n            const { tempId, promise } = this.rpClient.startTestItem(startTestItemReq, this.tempLaunchId, rpParentId);\r\n            this.cbStepToRpIdHash[step.id] = tempId;\r\n            this.handlePromise(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to start step item: ${e}`);\r\n        }\r\n    }\r\n    async onStepEnd({ rid, caseId, step: result }) {\r\n        if (!this.reportSteps) {\r\n            return;\r\n        }\r\n        if (result.name && (result.name.startsWith('log.') ||\r\n                            result.name.startsWith('web.transaction') ||\r\n                            result.name.startsWith('mob.transaction'))) {\r\n            return;\r\n        }\r\n        const rpStepId = this.cbStepToRpIdHash[result.id];\r\n        if (!rpStepId) {\r\n            return;\r\n        }\r\n\r\n        delete this.cbCaseToCurrentSubStepId[caseId];\r\n\r\n        const status = result.status.toLowerCase();\r\n        if (status === 'failed' && result.failure) {\r\n            const rpFile = result.screenshot ?\r\n            {\r\n                name: 'screenshot',\r\n                type: 'image/png',\r\n                content: result.screenshot,\r\n            } : undefined;\r\n\r\n            const logMsg = this._stringify(result.failure.message);\r\n            if (logMsg) {\r\n                const logReq = {\r\n                    message: logMsg,\r\n                    level: 'error',\r\n                    file: rpFile,\r\n                };\r\n\r\n                try {\r\n                    const { promise } = this.rpClient.sendLog(rpStepId, logReq, rpFile);\r\n                    this.handlePromise(promise);\r\n                } catch (e) {\r\n                    console.dir(`RP - Failed to send log for finished step item: ${e}`);\r\n                }\r\n            }\r\n        }\r\n        const finishTestItemReq = {\r\n            status: result.status.toLowerCase(),\r\n        };\r\n\r\n        try {\r\n            const { promise } = this.rpClient.finishTestItem(rpStepId, finishTestItemReq);\r\n            this.handlePromise(promise);\r\n        }\r\n        catch (e) {\r\n            console.dir(`RP - Failed to finish step item: ${e}`);\r\n        }\r\n    }\r\n    async onLog({ suiteId, caseId, stepId, level, msg, time, src }) {\r\n        if (!this.reportLogs) {\r\n            return;\r\n        }\r\n        // Oxygen might start generating logs before onRunnerStart event, ignore them\r\n        if (!this.tempLaunchId) {\r\n            return;\r\n        }\r\n        const rpParentId = stepId ?\r\n            this.cbStepToRpIdHash[stepId]\r\n            : caseId ? this.cbCaseToRpIdHash[caseId]\r\n            : suiteId ? this.cbSuiteToRpIdHash[suiteId]\r\n            : undefined;\r\n        const rpLevel = this._getRpLevel(level);\r\n\r\n        const logMsg = this._stringify(msg);\r\n        if (logMsg) {\r\n            const logReq = {\r\n                message: logMsg,\r\n                level: rpLevel,\r\n                time: time,\r\n            };\r\n\r\n            try {\r\n                const { promise } = this.rpClient.sendLog(rpParentId || this.tempLaunchId, logReq);\r\n                this.handlePromise(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to create log item: ${e}`);\r\n            }\r\n        }\r\n    }\r\n    async reportLogStep(caseId, step) {\r\n        if (!step.args || !step.args.length) {\r\n            return;\r\n        }\r\n        const level = step.name;\r\n        const msg = step.args[0];\r\n        const time = step.time;\r\n        let stepId = undefined;\r\n        if (this.cbCaseToCurrentSubStepId[caseId]) {\r\n            stepId = this.cbCaseToCurrentSubStepId[caseId];\r\n        } else if (this.cbCaseToCurrentTransactionId[caseId]) {\r\n            stepId = this.cbCaseToCurrentTransactionId[caseId];\r\n        }\r\n        await this.sendRpLog({ suiteId: undefined, caseId, stepId, level, msg, time });\r\n    }\r\n    async sendRpLog({ suiteId, caseId, stepId, level, msg, time }) {\r\n        // Oxygen might start generating logs before onRunnerStart event, ignore them\r\n        if (!this.tempLaunchId) {\r\n            return;\r\n        }\r\n        const rpParentId = stepId ?\r\n            this.cbStepToRpIdHash[stepId]\r\n            : caseId ? this.cbCaseToRpIdHash[caseId]\r\n            : suiteId ? this.cbSuiteToRpIdHash[suiteId]\r\n            : undefined;\r\n        const rpLevel = this._getRpLevel(level);\r\n\r\n        const logMsg = this._stringify(msg);\r\n        if (logMsg) {\r\n            const logReq = {\r\n                message: logMsg,\r\n                level: rpLevel,\r\n                time: time,\r\n            };\r\n\r\n            try {\r\n                const { promise } = this.rpClient.sendLog(rpParentId || this.tempLaunchId, logReq);\r\n                this.handlePromise(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to create log item: ${e}`);\r\n            }\r\n        }\r\n    }\r\n    async _reportEndOfStartedSuites() {\r\n        for (const suiteRefId of Object.keys(this.cbSuiteToRpIdHash)) {\r\n            const rpSuiteId = this.cbSuiteToRpIdHash[suiteRefId];\r\n            const results = this.cbSuiteResultListByRefId[suiteRefId];\r\n            delete this.cbSuiteToRpIdHash[suiteRefId];\r\n            const hasFailedSuite = results.some(x => x.status === Status.FAILED);\r\n            const finishTestItemReq = {\r\n                status: hasFailedSuite ? 'failed' : 'passed',\r\n            };\r\n\r\n            try {\r\n                const { promise } = this.rpClient.finishTestItem(rpSuiteId, finishTestItemReq);\r\n                this.handlePromise(promise);\r\n            }\r\n            catch (e) {\r\n                console.dir(`RP - Failed to finish suite item: ${e}`);\r\n            }\r\n        }\r\n    }\r\n    _getRpLevel(oxLevel) {\r\n        if (oxLevel) {\r\n            return oxLevel.toLowerCase();\r\n        }\r\n        return 'info';\r\n    }\r\n    /*_getRpArgs(cbArgs) {\r\n        if (!cbArgs || !cbArgs.length) {\r\n            return undefined;\r\n        }\r\n        const rpArgs = [];\r\n        for (let i=0; i < cbArgs.length; i++) {\r\n            const arg = cbArgs[i];\r\n            let argStr = typeof arg === 'string' || arg instanceof String ? arg : JSON.stringify(arg);\r\n            rpArgs.push({\r\n                key: `arg${i}`,\r\n                value: argStr\r\n            });\r\n        }\r\n        return rpArgs;\r\n    }*/\r\n    _getStepName(step) {\r\n        let name;\r\n\r\n        if (step.name === 'transaction' && step.args.length > 0) {  // FIXME: steps need to be nested under transactions\r\n            name = step.args[0];\r\n        }\r\n        else if (step.signature) {\r\n            name = step.signature;\r\n        }\r\n        else if (step.module) {                         // FIXME: is this ever reached? \"step.signature\" seems to always exist\r\n            name = `${step.module}.${step.name}`;\r\n        }\r\n        else {\r\n            name = step.name;\r\n        }\r\n\r\n        // maximum allowed name length in RP is 1024 bytes\r\n        // we truncate it even lower so it will display nicely\r\n        if (name.length > MAX_NAME_LENGTH) {\r\n            name = name.substring(0, MAX_NAME_LENGTH-3) + '...';\r\n        }\r\n        return name;\r\n    }\r\n\r\n    // returns null if msg cannot be stringified, or strigifies it if it's not a string already\r\n    _stringify(msg) {\r\n        // we test for undefined and null explicitly on purpose because other falsy values (e.g. 0) are ok\r\n        if (typeof msg === 'undefined' || msg === null) {\r\n            return null;\r\n        }\r\n\r\n        if (typeof msg === 'string' || msg instanceof String) {\r\n            return msg;\r\n        }\r\n\r\n        try {\r\n            return JSON.stringify(msg);\r\n        } catch (e) {\r\n            console.error('RP - error sending log: msg object cannot be serialized');\r\n        }\r\n    }\r\n\r\n    handlePromise(promise) {\r\n        promise.catch((err) => {\r\n            console.log('RP error: ', err);\r\n        });\r\n        this.promises.push(promise);\r\n    }\r\n}\r\n\r\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "oxygen-cli",
3
- "version": "1.40.1",
3
+ "version": "1.40.2",
4
4
  "description": "Automation framework for testing modern applications",
5
5
  "main": "./build/index.js",
6
6
  "bin": {