@zohodesk/testinglibrary 0.0.58-n20-experimental → 0.0.60-n20-experimental
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/README.md +0 -8
- package/build/common/data-generator/steps/DataGenerator.spec.js +1 -1
- package/build/common/data-generator/steps/DataGeneratorStepsHelper.js +4 -19
- package/build/core/dataGenerator/DataGenerator.js +25 -104
- package/build/core/dataGenerator/DataGeneratorHelper.js +4 -52
- package/build/core/playwright/builtInFixtures/cacheLayer.js +2 -197
- package/build/core/playwright/constants/reporterConstants.js +1 -0
- package/build/core/playwright/helpers/auth/getUsers.js +2 -2
- package/build/core/playwright/readConfigFile.js +1 -3
- package/build/core/playwright/reporter/PlaywrightReporter.js +44 -0
- package/build/core/playwright/reporter/UnitReporter.js +27 -0
- package/build/core/playwright/validateFeature.js +0 -11
- package/build/lib/cli.js +30 -7
- package/build/utils/commonUtils.js +9 -0
- package/build/utils/logger.js +1 -3
- package/changelog.md +0 -27
- package/npm-shrinkwrap.json +7843 -3408
- package/package.json +15 -11
- package/.claude/worktrees/thirsty-yalow/.babelrc +0 -24
- package/.claude/worktrees/thirsty-yalow/.eslintrc.js +0 -31
- package/.claude/worktrees/thirsty-yalow/.gitlab-ci.yml +0 -208
- package/.claude/worktrees/thirsty-yalow/.prettierrc +0 -6
- package/.claude/worktrees/thirsty-yalow/README.md +0 -234
- package/.claude/worktrees/thirsty-yalow/bin/cli.js +0 -3
- package/.claude/worktrees/thirsty-yalow/bin/postinstall.js +0 -1
- package/.claude/worktrees/thirsty-yalow/changelog.md +0 -167
- package/.claude/worktrees/thirsty-yalow/jest.config.js +0 -82
- package/.claude/worktrees/thirsty-yalow/package.json +0 -62
- package/.claude/worktrees/thirsty-yalow/playwright.config.js +0 -62
- package/AUTO_CLEANUP_PLAN.md +0 -171
- package/build/core/dataGenerator/validateGenerators.js +0 -82
- package/build/core/playwright/report-generator.js +0 -42
- package/build/utils/datePlaceholders.js +0 -170
- package/build/utils/timeFormat.js +0 -41
- package/unit_reports/unit-report.html +0 -277
package/README.md
CHANGED
|
@@ -17,14 +17,6 @@
|
|
|
17
17
|
|
|
18
18
|
- npm run report
|
|
19
19
|
|
|
20
|
-
### v0.0.44-n20-experimental - 23-03-2026
|
|
21
|
-
|
|
22
|
-
#### Enhancement
|
|
23
|
-
- DataGenerator now walks up the directory tree to discover data-generators folders
|
|
24
|
-
- Supports multiple JSON files in data-generators directory (no longer hardcoded to generators.json)
|
|
25
|
-
- Removed hardcoded folder name dependencies for flexible generator file placement
|
|
26
|
-
- Added duplicate generator name validation to `ZDTestingFramework validate` command
|
|
27
|
-
|
|
28
20
|
### v4.1.1/v3.3.0 - 28-01-2026
|
|
29
21
|
|
|
30
22
|
#### Features
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { test } from '@zohodesk/testinglibrary';
|
|
2
2
|
import DataGenerator from '@zohodesk/testinglibrary/DataGenerator';
|
|
3
|
-
import {getUserForSelectedEditionAndProfile
|
|
3
|
+
import {getUserForSelectedEditionAndProfile} from '@zohodesk/testinglibrary/helpers'
|
|
4
4
|
|
|
5
5
|
const dataGenerator = new DataGenerator();
|
|
6
6
|
|
|
@@ -10,27 +10,12 @@ export async function generateAndCacheTestData(executionContext, type, identifie
|
|
|
10
10
|
const scenarioName = testInfo.title.split('/').pop() || 'Unknown Scenario';
|
|
11
11
|
|
|
12
12
|
if (profile) {
|
|
13
|
-
// Explicit profile requested — resolve that profile's credentials
|
|
14
13
|
const { edition, orgName: portal, beta } = executionContext.actorInfo;
|
|
15
14
|
actorInfo = await getUserForSelectedEditionAndProfile(edition, profile, beta, portal);
|
|
16
15
|
} else {
|
|
17
|
-
// Default — use current actor, fall back to org-level data-generator if profile has none
|
|
18
16
|
actorInfo = executionContext.actorInfo;
|
|
19
|
-
if (!actorInfo['data-generator']) {
|
|
20
|
-
const { edition, orgName: portal, beta } = actorInfo;
|
|
21
|
-
const actorsData = getListOfActors(beta);
|
|
22
|
-
const portalData = actorsData.editions[edition]?.find(e => e.orgName === portal);
|
|
23
|
-
if (portalData?.['data-generator']) {
|
|
24
|
-
actorInfo = { ...actorInfo, 'data-generator': portalData['data-generator'] };
|
|
25
|
-
}
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
const { response, generators } = await dataGenerator.generate(testInfo, actorInfo, type, identifier, scenarioName, dataTable ? dataTable.hashes() : [], cacheLayer);
|
|
30
|
-
if (cacheLayer._trackForCleanup) {
|
|
31
|
-
cacheLayer._trackForCleanup(entityName, response.data, generators, actorInfo, response.logs, identifier);
|
|
32
|
-
} else {
|
|
33
|
-
cacheLayer.set(entityName, response.data);
|
|
34
|
-
cacheLayer.set(`${entityName}_logs`, response.logs);
|
|
35
17
|
}
|
|
18
|
+
|
|
19
|
+
const generatedData = await dataGenerator.generate(testInfo, actorInfo, type, identifier, scenarioName, dataTable ? dataTable.hashes() : []);
|
|
20
|
+
await cacheLayer.set(entityName, generatedData.data);
|
|
36
21
|
}
|
|
@@ -8,140 +8,61 @@ exports.default = void 0;
|
|
|
8
8
|
var _path = _interopRequireDefault(require("path"));
|
|
9
9
|
var _fs = _interopRequireDefault(require("fs"));
|
|
10
10
|
var _logger = require("../../utils/logger");
|
|
11
|
-
var _timeFormat = require("../../utils/timeFormat");
|
|
12
11
|
var _DataGeneratorHelper = require("./DataGeneratorHelper");
|
|
13
|
-
var
|
|
12
|
+
var _helpers = require("@zohodesk/testinglibrary/helpers");
|
|
14
13
|
var _DataGeneratorError = require("./DataGeneratorError");
|
|
15
|
-
var _readConfigFile = require("../playwright/readConfigFile");
|
|
16
|
-
var _configConstants = _interopRequireDefault(require("../playwright/constants/configConstants"));
|
|
17
|
-
var _ConfigurationHelper = require("../playwright/configuration/ConfigurationHelper");
|
|
18
14
|
function _classPrivateMethodInitSpec(e, a) { _checkPrivateRedeclaration(e, a), a.add(e); }
|
|
19
|
-
function _classPrivateFieldInitSpec(e, t, a) { _checkPrivateRedeclaration(e, t), t.set(e, a); }
|
|
20
15
|
function _checkPrivateRedeclaration(e, t) { if (t.has(e)) throw new TypeError("Cannot initialize the same private elements twice on an object"); }
|
|
21
|
-
function _classPrivateFieldSet(s, a, r) { return s.set(_assertClassBrand(s, a), r), r; }
|
|
22
|
-
function _classPrivateFieldGet(s, a) { return s.get(_assertClassBrand(s, a)); }
|
|
23
16
|
function _assertClassBrand(e, t, n) { if ("function" == typeof e ? e === t : e.has(t)) return arguments.length < 3 ? t : n; throw new TypeError("Private element is not present on this object"); }
|
|
24
|
-
var _generatorIndex = /*#__PURE__*/new WeakMap();
|
|
25
17
|
var _DataGenerator_brand = /*#__PURE__*/new WeakSet();
|
|
26
18
|
class DataGenerator {
|
|
27
19
|
constructor() {
|
|
28
20
|
_classPrivateMethodInitSpec(this, _DataGenerator_brand);
|
|
29
|
-
_classPrivateFieldInitSpec(this, _generatorIndex, null);
|
|
30
21
|
}
|
|
31
|
-
async generate(testInfo, actorInfo, generatorType, generatorName, scenarioName, dataTable
|
|
32
|
-
const startMs = Date.now();
|
|
33
|
-
const startLabel = (0, _timeFormat.formatTimestamp)(startMs);
|
|
34
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Data generation started | generator="${generatorName}" | scenario="${scenarioName}" | startTime=${startLabel}`);
|
|
22
|
+
async generate(testInfo, actorInfo, generatorType, generatorName, scenarioName, dataTable) {
|
|
35
23
|
try {
|
|
36
24
|
let generators;
|
|
37
25
|
if (generatorType === 'API') {
|
|
38
26
|
generators = await _assertClassBrand(_DataGenerator_brand, this, _generateAPIGenerator).call(this, generatorName);
|
|
39
27
|
} else {
|
|
40
|
-
generators = await _assertClassBrand(_DataGenerator_brand, this, _getGenerator).call(this, generatorName);
|
|
28
|
+
generators = await _assertClassBrand(_DataGenerator_brand, this, _getGenerator).call(this, testInfo, generatorName);
|
|
41
29
|
}
|
|
42
|
-
|
|
43
|
-
// Clone to prevent cross-test mutation of the cached generator index.
|
|
44
|
-
const generatorsCopy = generators.map(g => ({
|
|
45
|
-
...g,
|
|
46
|
-
params: g.params ? {
|
|
47
|
-
...g.params
|
|
48
|
-
} : undefined
|
|
49
|
-
}));
|
|
50
|
-
(0, _datePlaceholders.resolveDatePlaceholdersInGenerators)(generatorsCopy);
|
|
51
|
-
const dateResolvedTable = (0, _datePlaceholders.resolveDatePlaceholders)(dataTable);
|
|
52
|
-
const resolvedDataTable = await (0, _DataGeneratorHelper.resolveCacheReferences)(dateResolvedTable, cacheLayer);
|
|
53
|
-
const processedGenerators = await (0, _DataGeneratorHelper.processGenerator)(generatorsCopy, resolvedDataTable);
|
|
30
|
+
const processedGenerators = await (0, _DataGeneratorHelper.processGenerator)(generators, dataTable);
|
|
54
31
|
const apiPayload = await _assertClassBrand(_DataGenerator_brand, this, _constructApiPayload).call(this, scenarioName, processedGenerators, actorInfo);
|
|
55
|
-
const
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
'featureflags': featureFlags
|
|
59
|
-
};
|
|
60
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, 'Making request headers:', headers);
|
|
61
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Payload: ${JSON.stringify(apiPayload, null, 4)}`);
|
|
62
|
-
const response = await (0, _DataGeneratorHelper.makeRequest)(process.env.DG_SERVICE_DOMAIN + process.env.DG_SERVICE_API_PATH, apiPayload, headers);
|
|
63
|
-
const endMs = Date.now();
|
|
64
|
-
const endLabel = (0, _timeFormat.formatTimestamp)(endMs);
|
|
65
|
-
const totalLabel = (0, _timeFormat.formatDuration)(endMs - startMs);
|
|
66
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Generated response for the generator: ${generatorName} for scenario: ${scenarioName}, Response: ${JSON.stringify(response, null, 4)}`);
|
|
67
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Data generation completed | generator="${generatorName}" | scenario="${scenarioName}" | startTime=${startLabel} | endTime=${endLabel} | totalTime=${totalLabel}`);
|
|
68
|
-
return {
|
|
69
|
-
response,
|
|
70
|
-
generators
|
|
71
|
-
};
|
|
32
|
+
const response = await (0, _DataGeneratorHelper.makeRequest)(process.env.DG_SERVICE_DOMAIN + process.env.DG_SERVICE_API_PATH, apiPayload);
|
|
33
|
+
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Generated response for the generator: ${generatorName} for scenario: ${scenarioName}, Response: ${JSON.stringify(response)}`);
|
|
34
|
+
return response;
|
|
72
35
|
} catch (error) {
|
|
73
|
-
const endMs = Date.now();
|
|
74
|
-
const endLabel = (0, _timeFormat.formatTimestamp)(endMs);
|
|
75
|
-
const totalLabel = (0, _timeFormat.formatDuration)(endMs - startMs);
|
|
76
|
-
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `Data generation failed | generator="${generatorName}" | scenario="${scenarioName}" | startTime=${startLabel} | endTime=${endLabel} | totalTime=${totalLabel}`);
|
|
77
36
|
if (error instanceof _DataGeneratorError.DataGeneratorError) {
|
|
78
|
-
|
|
79
|
-
|
|
37
|
+
console.error(error.getMessage());
|
|
38
|
+
console.error("Stack trace:", error.stack);
|
|
80
39
|
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, error.getMessage());
|
|
81
40
|
} else {
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
41
|
+
console.error("Error Type:", error.constructor.name);
|
|
42
|
+
console.error("Error Message:", error.message);
|
|
43
|
+
console.error("Stack trace:", error.stack);
|
|
85
44
|
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `${error.constructor.name} - Message: ${error.message}`);
|
|
86
45
|
}
|
|
87
|
-
|
|
46
|
+
console.error('Data Generation failed for the generator: ', generatorName, "\n\nError response :", error);
|
|
88
47
|
throw error;
|
|
89
48
|
}
|
|
90
49
|
}
|
|
91
50
|
}
|
|
92
|
-
function
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
const fullPath = _path.default.join(dir, entry.name);
|
|
102
|
-
if (entry.isDirectory()) {
|
|
103
|
-
_assertClassBrand(_DataGenerator_brand, this, _scanDir).call(this, fullPath, index, pattern);
|
|
104
|
-
} else if (_assertClassBrand(_DataGenerator_brand, this, _matchesPattern).call(this, entry.name, pattern)) {
|
|
105
|
-
try {
|
|
106
|
-
const data = _fs.default.readFileSync(fullPath, 'utf8');
|
|
107
|
-
const obj = JSON.parse(data);
|
|
108
|
-
if (obj.generators) {
|
|
109
|
-
for (const [name, config] of Object.entries(obj.generators)) {
|
|
110
|
-
if (!index.has(name)) {
|
|
111
|
-
index.set(name, config);
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
} catch (err) {
|
|
116
|
-
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `Failed to parse generator file: ${fullPath} - ${err.message}`);
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
function _buildIndex(modulesRoot, pattern) {
|
|
122
|
-
const index = new Map();
|
|
123
|
-
_assertClassBrand(_DataGenerator_brand, this, _scanDir).call(this, modulesRoot, index, pattern);
|
|
124
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Generator index built: ${index.size} generators found`);
|
|
125
|
-
return index;
|
|
126
|
-
}
|
|
127
|
-
function _getModulesRoot() {
|
|
128
|
-
const stage = (0, _ConfigurationHelper.getRunStage)();
|
|
129
|
-
return _path.default.join(process.cwd(), _configConstants.default.TEST_SLICE_FOLDER, stage, 'modules');
|
|
130
|
-
}
|
|
131
|
-
async function _getGenerator(generatorName) {
|
|
132
|
-
if (!_classPrivateFieldGet(_generatorIndex, this)) {
|
|
133
|
-
const modulesRoot = _assertClassBrand(_DataGenerator_brand, this, _getModulesRoot).call(this);
|
|
134
|
-
const {
|
|
135
|
-
generatorFilePattern: pattern = '*.generators.json'
|
|
136
|
-
} = (0, _readConfigFile.generateConfigFromFile)();
|
|
137
|
-
if (modulesRoot) {
|
|
138
|
-
_classPrivateFieldSet(_generatorIndex, this, _assertClassBrand(_DataGenerator_brand, this, _buildIndex).call(this, modulesRoot, pattern));
|
|
51
|
+
async function _getGenerator(testInfo, generatorName) {
|
|
52
|
+
let generator = null;
|
|
53
|
+
let generatorFilePath = await (0, _DataGeneratorHelper.getGeneratorFilePath)(testInfo.file);
|
|
54
|
+
generatorFilePath = _path.default.join(generatorFilePath, "../../data-generators/generators.json");
|
|
55
|
+
if (_fs.default.existsSync(generatorFilePath)) {
|
|
56
|
+
const data = _fs.default.readFileSync(generatorFilePath, 'utf8');
|
|
57
|
+
const generatorObj = JSON.parse(data);
|
|
58
|
+
if (generatorName || generatorObj.generators) {
|
|
59
|
+
generator = generatorObj.generators[generatorName] || null;
|
|
139
60
|
}
|
|
140
61
|
}
|
|
141
|
-
if (
|
|
142
|
-
|
|
62
|
+
if (!generator) {
|
|
63
|
+
throw new _DataGeneratorError.GeneratorError(`Generator "${generatorName}" could not be found in the path located at "${generatorFilePath}"`);
|
|
143
64
|
}
|
|
144
|
-
|
|
65
|
+
return generator;
|
|
145
66
|
}
|
|
146
67
|
async function _generateAPIGenerator(operationId) {
|
|
147
68
|
return [{
|
|
@@ -6,54 +6,6 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
6
6
|
exports.getGeneratorFilePath = getGeneratorFilePath;
|
|
7
7
|
exports.makeRequest = makeRequest;
|
|
8
8
|
exports.processGenerator = processGenerator;
|
|
9
|
-
exports.resolveCacheReferences = resolveCacheReferences;
|
|
10
|
-
// Matches a single ${EntityName} or ${EntityName.field.path} cell value.
|
|
11
|
-
// Anchored — values like "foo ${E1.id} bar" are left as-is on purpose.
|
|
12
|
-
const CACHE_REF = /^\$\{([A-Za-z_][\w-]*)(?:\.([A-Za-z_][\w.-]*))?\}$/;
|
|
13
|
-
|
|
14
|
-
// Resolve ${EntityName} / ${EntityName.field.path} references in a dataTable
|
|
15
|
-
// against the cacheLayer. Lets a later `generate ...` step reference an entity
|
|
16
|
-
// that an earlier step cached (e.g. ${E1.id} for an event created in a prior
|
|
17
|
-
// generator invocation).
|
|
18
|
-
async function resolveCacheReferences(rows, cacheLayer) {
|
|
19
|
-
if (!rows || !rows.length || !cacheLayer || typeof cacheLayer.get !== 'function') {
|
|
20
|
-
return rows || [];
|
|
21
|
-
}
|
|
22
|
-
const out = [];
|
|
23
|
-
for (const row of rows) {
|
|
24
|
-
const resolved = {};
|
|
25
|
-
for (const [key, value] of Object.entries(row)) {
|
|
26
|
-
const match = typeof value === 'string' ? value.match(CACHE_REF) : null;
|
|
27
|
-
if (!match) {
|
|
28
|
-
resolved[key] = value;
|
|
29
|
-
continue;
|
|
30
|
-
}
|
|
31
|
-
const [, entityName, fieldPath] = match;
|
|
32
|
-
const entity = await cacheLayer.get(entityName);
|
|
33
|
-
if (entity === undefined || entity === null) {
|
|
34
|
-
throw new Error(`DataGenerator: ${value} references entity "${entityName}" which is not in the cache.`);
|
|
35
|
-
}
|
|
36
|
-
if (!fieldPath) {
|
|
37
|
-
resolved[key] = entity;
|
|
38
|
-
continue;
|
|
39
|
-
}
|
|
40
|
-
let acc = entity;
|
|
41
|
-
for (const part of fieldPath.split('.')) {
|
|
42
|
-
if (acc === undefined || acc === null) {
|
|
43
|
-
break;
|
|
44
|
-
}
|
|
45
|
-
acc = acc[part];
|
|
46
|
-
}
|
|
47
|
-
if (acc === undefined) {
|
|
48
|
-
throw new Error(`DataGenerator: ${value} resolved to undefined (entity "${entityName}" has no path "${fieldPath}").`);
|
|
49
|
-
}
|
|
50
|
-
resolved[key] = acc;
|
|
51
|
-
}
|
|
52
|
-
out.push(resolved);
|
|
53
|
-
}
|
|
54
|
-
return out;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
9
|
//Create payload for the generators
|
|
58
10
|
async function processGenerator(generators, dataTable) {
|
|
59
11
|
if (!dataTable) {
|
|
@@ -77,12 +29,12 @@ async function processGenerator(generators, dataTable) {
|
|
|
77
29
|
return generator;
|
|
78
30
|
});
|
|
79
31
|
}
|
|
80
|
-
async function makeRequest(url, payload
|
|
81
|
-
'Content-Type': 'application/json'
|
|
82
|
-
}) {
|
|
32
|
+
async function makeRequest(url, payload) {
|
|
83
33
|
const response = await fetch(url, {
|
|
84
34
|
method: 'POST',
|
|
85
|
-
headers:
|
|
35
|
+
headers: {
|
|
36
|
+
'Content-Type': 'application/json'
|
|
37
|
+
},
|
|
86
38
|
body: JSON.stringify(payload)
|
|
87
39
|
});
|
|
88
40
|
if (!response.ok) {
|
|
@@ -1,208 +1,13 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
|
-
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
3
|
Object.defineProperty(exports, "__esModule", {
|
|
5
4
|
value: true
|
|
6
5
|
});
|
|
7
6
|
exports.default = void 0;
|
|
8
|
-
|
|
9
|
-
var _fs = _interopRequireDefault(require("fs"));
|
|
10
|
-
var _logger = require("../../../utils/logger");
|
|
11
|
-
var _timeFormat = require("../../../utils/timeFormat");
|
|
12
|
-
var _DataGeneratorHelper = require("../../dataGenerator/DataGeneratorHelper");
|
|
13
|
-
var _readConfigFile = require("../readConfigFile");
|
|
14
|
-
var _jsonpath = _interopRequireDefault(require("jsonpath"));
|
|
15
|
-
let cleanupRegistry = null;
|
|
16
|
-
function getModulesRoot() {
|
|
17
|
-
const configConstants = require('../constants/configConstants');
|
|
18
|
-
const {
|
|
19
|
-
getRunStage
|
|
20
|
-
} = require('../configuration/ConfigurationHelper');
|
|
21
|
-
const stage = getRunStage();
|
|
22
|
-
return _path.default.join(process.cwd(), configConstants.TEST_SLICE_FOLDER, stage, 'modules');
|
|
23
|
-
}
|
|
24
|
-
function buildCleanupRegistry() {
|
|
25
|
-
const modulesRoot = getModulesRoot();
|
|
26
|
-
const registry = {};
|
|
27
|
-
if (!_fs.default.existsSync(modulesRoot)) return registry;
|
|
28
|
-
scanDir(modulesRoot, registry);
|
|
29
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup registry built: ${Object.keys(registry).length} generator chains from ${modulesRoot}`);
|
|
30
|
-
return registry;
|
|
31
|
-
}
|
|
32
|
-
function scanDir(dir, registry) {
|
|
33
|
-
const entries = _fs.default.readdirSync(dir, {
|
|
34
|
-
withFileTypes: true
|
|
35
|
-
});
|
|
36
|
-
for (const entry of entries) {
|
|
37
|
-
const fullPath = _path.default.join(dir, entry.name);
|
|
38
|
-
if (entry.isDirectory()) {
|
|
39
|
-
scanDir(fullPath, registry);
|
|
40
|
-
} else if (entry.name.endsWith('.cleanup.js')) {
|
|
41
|
-
try {
|
|
42
|
-
const cleanupModule = require(fullPath);
|
|
43
|
-
for (const [generatorName, chain] of Object.entries(cleanupModule)) {
|
|
44
|
-
if (registry[generatorName]) {
|
|
45
|
-
throw new Error(`Duplicate cleanup chain for generator "${generatorName}" in ${fullPath}. ` + `Each generator must have exactly one cleanup chain.`);
|
|
46
|
-
}
|
|
47
|
-
registry[generatorName] = chain;
|
|
48
|
-
}
|
|
49
|
-
} catch (err) {
|
|
50
|
-
if (err.message.includes('Duplicate cleanup chain')) {
|
|
51
|
-
throw err;
|
|
52
|
-
}
|
|
53
|
-
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `Failed to load cleanup file: ${fullPath} - ${err.message}`);
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
function extractId(data, idPath) {
|
|
59
|
-
const result = _jsonpath.default.query(data, idPath);
|
|
60
|
-
if (result.length === 0) {
|
|
61
|
-
throw new Error(`Could not extract ID using path "${idPath}"`);
|
|
62
|
-
}
|
|
63
|
-
return result[0];
|
|
64
|
-
}
|
|
65
|
-
function parseLogBody(stepLog) {
|
|
66
|
-
var _stepLog$response;
|
|
67
|
-
if (!(stepLog !== null && stepLog !== void 0 && (_stepLog$response = stepLog.response) !== null && _stepLog$response !== void 0 && _stepLog$response.body)) return null;
|
|
68
|
-
const body = stepLog.response.body;
|
|
69
|
-
return typeof body === 'string' ? JSON.parse(body) : body;
|
|
70
|
-
}
|
|
71
|
-
async function cleanupViaOAS(config, entityId, actorInfo) {
|
|
72
|
-
const dataGeneratorObj = actorInfo['data-generator'];
|
|
73
|
-
if (!dataGeneratorObj) {
|
|
74
|
-
throw new Error('No data-generator config available for cleanup');
|
|
75
|
-
}
|
|
76
|
-
const payload = {
|
|
77
|
-
scenario_name: 'cleanup',
|
|
78
|
-
data_generation_templates: [{
|
|
79
|
-
type: 'dynamic',
|
|
80
|
-
generatorOperationId: config.operationId,
|
|
81
|
-
dataPath: '$.response.body:$',
|
|
82
|
-
name: config.operationId,
|
|
83
|
-
params: {
|
|
84
|
-
[config.paramName || 'id']: String(entityId)
|
|
85
|
-
}
|
|
86
|
-
}],
|
|
87
|
-
...dataGeneratorObj
|
|
88
|
-
};
|
|
89
|
-
if (payload.account) {
|
|
90
|
-
payload.account.email = actorInfo.email;
|
|
91
|
-
payload.account.password = actorInfo.password;
|
|
92
|
-
}
|
|
93
|
-
const environmentDetails = payload.environmentDetails || {};
|
|
94
|
-
environmentDetails.iam_url = process.env.DG_IAM_DOMAIN;
|
|
95
|
-
environmentDetails.host = new URL(process.env.domain).origin;
|
|
96
|
-
payload.environmentDetails = environmentDetails;
|
|
97
|
-
const response = await (0, _DataGeneratorHelper.makeRequest)(process.env.DG_SERVICE_DOMAIN + process.env.DG_SERVICE_API_PATH, payload);
|
|
98
|
-
return response;
|
|
99
|
-
}
|
|
100
|
-
async function cleanupViaAPI(config, entityId) {
|
|
101
|
-
const url = `${new URL(process.env.domain).origin}${config.apiPath.replace('{id}', entityId)}`;
|
|
102
|
-
const options = {
|
|
103
|
-
method: config.method,
|
|
104
|
-
headers: {
|
|
105
|
-
'Content-Type': 'application/json'
|
|
106
|
-
}
|
|
107
|
-
};
|
|
108
|
-
if (config.body) {
|
|
109
|
-
options.body = JSON.stringify(config.body);
|
|
110
|
-
}
|
|
111
|
-
const response = await fetch(url, options);
|
|
112
|
-
const responseBody = await response.text();
|
|
113
|
-
if (!response.ok) {
|
|
114
|
-
throw new Error(`${config.method} ${config.apiPath} - status: ${response.status}, body: ${responseBody}`);
|
|
115
|
-
}
|
|
116
|
-
return {
|
|
117
|
-
status: response.status,
|
|
118
|
-
body: responseBody
|
|
119
|
-
};
|
|
120
|
-
}
|
|
7
|
+
const cacheMap = new Map();
|
|
121
8
|
var _default = exports.default = {
|
|
122
9
|
// eslint-disable-next-line no-empty-pattern
|
|
123
10
|
cacheLayer: async ({}, use) => {
|
|
124
|
-
|
|
125
|
-
const cleanupEntries = [];
|
|
126
|
-
cache._trackForCleanup = (entityName, data, generators, actorInfo, logs, generatorName) => {
|
|
127
|
-
cache.set(entityName, data);
|
|
128
|
-
cache.set(`${entityName}_logs`, logs);
|
|
129
|
-
cleanupEntries.push({
|
|
130
|
-
entityName,
|
|
131
|
-
data,
|
|
132
|
-
generators,
|
|
133
|
-
actorInfo,
|
|
134
|
-
logs: logs || [],
|
|
135
|
-
generatorName
|
|
136
|
-
});
|
|
137
|
-
};
|
|
138
|
-
await use(cache);
|
|
139
|
-
|
|
140
|
-
// TEARDOWN — runs after scenario ends (pass or fail)
|
|
141
|
-
const {
|
|
142
|
-
autoCleanup = true
|
|
143
|
-
} = (0, _readConfigFile.generateConfigFromFile)();
|
|
144
|
-
if (!autoCleanup || cleanupEntries.length === 0) {
|
|
145
|
-
cache.clear();
|
|
146
|
-
return;
|
|
147
|
-
}
|
|
148
|
-
if (!cleanupRegistry) {
|
|
149
|
-
cleanupRegistry = buildCleanupRegistry();
|
|
150
|
-
}
|
|
151
|
-
const cleanupStartMs = Date.now();
|
|
152
|
-
const cleanupStartLabel = (0, _timeFormat.formatTimestamp)(cleanupStartMs);
|
|
153
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup started | entities=${cleanupEntries.length} | startTime=${cleanupStartLabel}`);
|
|
154
|
-
let cleaned = 0;
|
|
155
|
-
let skipped = 0;
|
|
156
|
-
let failed = 0;
|
|
157
|
-
for (const entry of [...cleanupEntries].reverse()) {
|
|
158
|
-
const cleanupChain = cleanupRegistry[entry.generatorName];
|
|
159
|
-
if (!cleanupChain) {
|
|
160
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup: no chain for generator "${entry.generatorName}" — skipping`);
|
|
161
|
-
skipped++;
|
|
162
|
-
continue;
|
|
163
|
-
}
|
|
164
|
-
const entityStartMs = Date.now();
|
|
165
|
-
const entityStartLabel = (0, _timeFormat.formatTimestamp)(entityStartMs);
|
|
166
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup entity started | entity="${entry.entityName}" | generator="${entry.generatorName}" | steps=${cleanupChain.length} | startTime=${entityStartLabel}`);
|
|
167
|
-
for (const cleanupStep of cleanupChain) {
|
|
168
|
-
const stepStartMs = Date.now();
|
|
169
|
-
const stepStartLabel = (0, _timeFormat.formatTimestamp)(stepStartMs);
|
|
170
|
-
const actionDesc = cleanupStep.operationId || `${cleanupStep.method} ${cleanupStep.apiPath}`;
|
|
171
|
-
try {
|
|
172
|
-
// Find the step's response from logs by matching operationId
|
|
173
|
-
const stepLog = entry.logs.find(log => log.generationOperationId === cleanupStep.operationId || log.name === cleanupStep.operationId);
|
|
174
|
-
const stepData = parseLogBody(stepLog) || entry.data;
|
|
175
|
-
const entityId = extractId(stepData, cleanupStep.idPath);
|
|
176
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup step started | [${cleanupStep.type}] ${actionDesc} (id: ${entityId}) | startTime=${stepStartLabel}`);
|
|
177
|
-
let cleanupResponse;
|
|
178
|
-
if (cleanupStep.type === 'oas') {
|
|
179
|
-
cleanupResponse = await cleanupViaOAS(cleanupStep, entityId, entry.actorInfo);
|
|
180
|
-
} else if (cleanupStep.type === 'api') {
|
|
181
|
-
cleanupResponse = await cleanupViaAPI(cleanupStep, entityId);
|
|
182
|
-
}
|
|
183
|
-
cleaned++;
|
|
184
|
-
const stepEndMs = Date.now();
|
|
185
|
-
const stepEndLabel = (0, _timeFormat.formatTimestamp)(stepEndMs);
|
|
186
|
-
const stepTotalLabel = (0, _timeFormat.formatDuration)(stepEndMs - stepStartMs);
|
|
187
|
-
_logger.Logger.log(_logger.Logger.SUCCESS_TYPE, `Cleanup step success | ${actionDesc} (id: ${entityId}) | response=${JSON.stringify(cleanupResponse, null, 4)} | startTime=${stepStartLabel} | endTime=${stepEndLabel} | totalTime=${stepTotalLabel}`);
|
|
188
|
-
} catch (err) {
|
|
189
|
-
failed++;
|
|
190
|
-
const stepEndMs = Date.now();
|
|
191
|
-
const stepEndLabel = (0, _timeFormat.formatTimestamp)(stepEndMs);
|
|
192
|
-
const stepTotalLabel = (0, _timeFormat.formatDuration)(stepEndMs - stepStartMs);
|
|
193
|
-
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `Cleanup step failed | ${actionDesc} — ${err.message} | startTime=${stepStartLabel} | endTime=${stepEndLabel} | totalTime=${stepTotalLabel}`);
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
const entityEndMs = Date.now();
|
|
197
|
-
const entityEndLabel = (0, _timeFormat.formatTimestamp)(entityEndMs);
|
|
198
|
-
const entityTotalLabel = (0, _timeFormat.formatDuration)(entityEndMs - entityStartMs);
|
|
199
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup entity completed | entity="${entry.entityName}" | generator="${entry.generatorName}" | startTime=${entityStartLabel} | endTime=${entityEndLabel} | totalTime=${entityTotalLabel}`);
|
|
200
|
-
}
|
|
201
|
-
const cleanupEndMs = Date.now();
|
|
202
|
-
const cleanupEndLabel = (0, _timeFormat.formatTimestamp)(cleanupEndMs);
|
|
203
|
-
const cleanupTotalLabel = (0, _timeFormat.formatDuration)(cleanupEndMs - cleanupStartMs);
|
|
204
|
-
_logger.Logger.log(_logger.Logger.INFO_TYPE, `Cleanup completed | cleaned=${cleaned} | skipped=${skipped} (no chain) | failed=${failed} | startTime=${cleanupStartLabel} | endTime=${cleanupEndLabel} | totalTime=${cleanupTotalLabel}`);
|
|
205
|
-
cleanupEntries.length = 0;
|
|
206
|
-
cache.clear();
|
|
11
|
+
await use(cacheMap);
|
|
207
12
|
}
|
|
208
13
|
};
|
|
@@ -11,5 +11,6 @@ const stage = (0, _ConfigurationHelper.getRunStage)();
|
|
|
11
11
|
class ReporterConstants {
|
|
12
12
|
static DEFAULT_REPORTER_PATH = `${_configConstants.default.TEST_SLICE_FOLDER}/${stage}/test-results/playwright-test-results.json`;
|
|
13
13
|
static LAST_RUN_REPORTER_PATH = `${_configConstants.default.TEST_SLICE_FOLDER}/${stage}/test-results/.last-run.json`;
|
|
14
|
+
static DEFAULT_UNIT_TEST_REPORTER_PATH = `${_configConstants.default.TEST_SLICE_FOLDER}/unit-test/unit_reports/report.html`;
|
|
14
15
|
}
|
|
15
16
|
exports.default = ReporterConstants;
|
|
@@ -96,9 +96,9 @@ function getUserForSelectedEditionAndProfile(preferedEdition, preferredProfile,
|
|
|
96
96
|
throw new Error(`There is no "${edition}" edition configured.`);
|
|
97
97
|
}
|
|
98
98
|
if (testDataPortal !== null) {
|
|
99
|
-
testingPortal = userdata[edition].find(editionData => editionData.
|
|
99
|
+
testingPortal = userdata[edition].find(editionData => editionData.orgName === testDataPortal);
|
|
100
100
|
if (!testingPortal) {
|
|
101
|
-
throw new Error(`There is no "${testDataPortal}" portal
|
|
101
|
+
throw new Error(`There is no "${testDataPortal}" portal configured in "${edition}" edition.`);
|
|
102
102
|
}
|
|
103
103
|
} else {
|
|
104
104
|
testingPortal = userdata[edition] ? userdata[edition][0] : {};
|
|
@@ -55,9 +55,7 @@ function getDefaultConfig() {
|
|
|
55
55
|
featureFilesFolder: 'feature-files',
|
|
56
56
|
stepDefinitionsFolder: 'steps',
|
|
57
57
|
testSetup: {},
|
|
58
|
-
editionOrder: ['Free', 'Express', 'Standard', 'Professional', 'Enterprise']
|
|
59
|
-
generatorFilePattern: '*.generators.json',
|
|
60
|
-
autoCleanup: true
|
|
58
|
+
editionOrder: ['Free', 'Express', 'Standard', 'Professional', 'Enterprise']
|
|
61
59
|
};
|
|
62
60
|
}
|
|
63
61
|
function combineDefaultConfigWithUserConfig(userConfiguration) {
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
Object.defineProperty(exports, "__esModule", {
|
|
5
|
+
value: true
|
|
6
|
+
});
|
|
7
|
+
exports.default = void 0;
|
|
8
|
+
var _child_process = require("child_process");
|
|
9
|
+
var _path = _interopRequireDefault(require("path"));
|
|
10
|
+
var _logger = require("../../../utils/logger");
|
|
11
|
+
var _rootPath = require("../../../utils/rootPath");
|
|
12
|
+
var _readConfigFile = require("../readConfigFile");
|
|
13
|
+
class PlaywrightReporter {
|
|
14
|
+
static generate() {
|
|
15
|
+
const userArgs = process.argv.slice(3);
|
|
16
|
+
const playwrightPath = _path.default.resolve((0, _rootPath.getExecutableBinaryPath)('playwright'));
|
|
17
|
+
const command = playwrightPath;
|
|
18
|
+
const {
|
|
19
|
+
reportPath: htmlPath
|
|
20
|
+
} = (0, _readConfigFile.generateConfigFromFile)();
|
|
21
|
+
const args = ['show-report', htmlPath].concat(userArgs);
|
|
22
|
+
const childProcess = (0, _child_process.spawn)(command, args, {
|
|
23
|
+
stdio: 'inherit'
|
|
24
|
+
});
|
|
25
|
+
childProcess.on('error', error => {
|
|
26
|
+
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, error);
|
|
27
|
+
});
|
|
28
|
+
childProcess.on('exit', (code, signal) => {
|
|
29
|
+
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `Child Process Exited with Code ${code} and Signal ${signal}`);
|
|
30
|
+
process.exit();
|
|
31
|
+
});
|
|
32
|
+
process.on('exit', () => {
|
|
33
|
+
_logger.Logger.log(_logger.Logger.INFO_TYPE, 'Terminating Playwright Process...');
|
|
34
|
+
childProcess.kill();
|
|
35
|
+
return;
|
|
36
|
+
});
|
|
37
|
+
process.on('SIGINT', () => {
|
|
38
|
+
_logger.Logger.log(_logger.Logger.INFO_TYPE, 'Cleaning up...');
|
|
39
|
+
childProcess.kill();
|
|
40
|
+
process.exit();
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
exports.default = PlaywrightReporter;
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
Object.defineProperty(exports, "__esModule", {
|
|
5
|
+
value: true
|
|
6
|
+
});
|
|
7
|
+
exports.default = void 0;
|
|
8
|
+
var _child_process = require("child_process");
|
|
9
|
+
var _path = _interopRequireDefault(require("path"));
|
|
10
|
+
var _os = require("os");
|
|
11
|
+
var _logger = require("../../../utils/logger");
|
|
12
|
+
var _reporterConstants = _interopRequireDefault(require("../constants/reporterConstants"));
|
|
13
|
+
class UnitReporter {
|
|
14
|
+
static generate() {
|
|
15
|
+
const reportPath = _path.default.resolve(process.cwd(), _reporterConstants.default.DEFAULT_UNIT_TEST_REPORTER_PATH);
|
|
16
|
+
_logger.Logger.log(_logger.Logger.SUCCESS_TYPE, `Opening unit test report: ${reportPath}`);
|
|
17
|
+
const os = (0, _os.platform)();
|
|
18
|
+
const openCommand = os === 'darwin' ? 'open' : os === 'win32' ? 'start' : 'xdg-open';
|
|
19
|
+
(0, _child_process.execFile)(openCommand, [reportPath], error => {
|
|
20
|
+
if (error) {
|
|
21
|
+
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, `Failed to open report: ${error.message}`);
|
|
22
|
+
}
|
|
23
|
+
process.exit();
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
exports.default = UnitReporter;
|
|
@@ -5,15 +5,11 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
5
5
|
value: true
|
|
6
6
|
});
|
|
7
7
|
exports.default = void 0;
|
|
8
|
-
var _path = _interopRequireDefault(require("path"));
|
|
9
8
|
var _parseUserArgs = _interopRequireDefault(require("./helpers/parseUserArgs"));
|
|
10
9
|
var _readConfigFile = require("./readConfigFile");
|
|
11
10
|
var _tagProcessor = _interopRequireDefault(require("./tagProcessor"));
|
|
12
11
|
var _testRunner = require("./test-runner");
|
|
13
12
|
var _logger = require("../../utils/logger");
|
|
14
|
-
var _validateGenerators = require("../dataGenerator/validateGenerators");
|
|
15
|
-
var _configConstants = _interopRequireDefault(require("./constants/configConstants"));
|
|
16
|
-
var _ConfigurationHelper = require("./configuration/ConfigurationHelper");
|
|
17
13
|
const validateFeatureFiles = () => {
|
|
18
14
|
const userArgsObject = (0, _parseUserArgs.default)();
|
|
19
15
|
const uatConfig = (0, _readConfigFile.generateConfigFromFile)();
|
|
@@ -21,13 +17,6 @@ const validateFeatureFiles = () => {
|
|
|
21
17
|
editionOrder
|
|
22
18
|
} = uatConfig;
|
|
23
19
|
const configPath = (0, _readConfigFile.isUserConfigFileAvailable)() ? require.resolve('./setup/config-creator.js') : require.resolve('../../../playwright.config.js');
|
|
24
|
-
const stage = (0, _ConfigurationHelper.getRunStage)();
|
|
25
|
-
const modulesRoot = _path.default.join(process.cwd(), _configConstants.default.TEST_SLICE_FOLDER, stage, 'modules');
|
|
26
|
-
const generatorResult = (0, _validateGenerators.validateGenerators)(modulesRoot);
|
|
27
|
-
if (!generatorResult.valid) {
|
|
28
|
-
_logger.Logger.log(_logger.Logger.FAILURE_TYPE, 'Generator validation failed. Fix duplicate generator names before running tests.');
|
|
29
|
-
process.exit(1);
|
|
30
|
-
}
|
|
31
20
|
const tagProcessor = new _tagProcessor.default(editionOrder);
|
|
32
21
|
const tagArgs = tagProcessor.processTags(userArgsObject);
|
|
33
22
|
(0, _testRunner.runPreprocessing)(tagArgs, configPath).then(() => {
|