@dev-blinq/cucumber_client 1.0.1444-dev → 1.0.1444-stage
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/bin/assets/bundled_scripts/recorder.js +73 -73
- package/bin/assets/preload/css_gen.js +10 -10
- package/bin/assets/preload/toolbar.js +27 -29
- package/bin/assets/preload/unique_locators.js +1 -1
- package/bin/assets/preload/yaml.js +288 -275
- package/bin/assets/scripts/aria_snapshot.js +223 -220
- package/bin/assets/scripts/dom_attr.js +329 -329
- package/bin/assets/scripts/dom_parent.js +169 -174
- package/bin/assets/scripts/event_utils.js +94 -94
- package/bin/assets/scripts/pw.js +2050 -1949
- package/bin/assets/scripts/recorder.js +70 -45
- package/bin/assets/scripts/snapshot_capturer.js +147 -147
- package/bin/assets/scripts/unique_locators.js +170 -49
- package/bin/assets/scripts/yaml.js +796 -783
- package/bin/assets/templates/_hooks_template.txt +6 -2
- package/bin/assets/templates/utils_template.txt +16 -16
- package/bin/client/code_cleanup/find_step_definition_references.js +0 -1
- package/bin/client/code_cleanup/utils.js +16 -7
- package/bin/client/code_gen/api_codegen.js +2 -2
- package/bin/client/code_gen/code_inversion.js +119 -2
- package/bin/client/code_gen/duplication_analysis.js +2 -1
- package/bin/client/code_gen/function_signature.js +4 -0
- package/bin/client/code_gen/page_reflection.js +52 -11
- package/bin/client/code_gen/playwright_codeget.js +163 -75
- package/bin/client/cucumber/feature.js +4 -17
- package/bin/client/cucumber/feature_data.js +2 -2
- package/bin/client/cucumber/project_to_document.js +8 -2
- package/bin/client/cucumber/steps_definitions.js +19 -3
- package/bin/client/local_agent.js +1 -0
- package/bin/client/parse_feature_file.js +23 -26
- package/bin/client/playground/projects/env.json +2 -2
- package/bin/client/recorderv3/bvt_init.js +305 -0
- package/bin/client/recorderv3/bvt_recorder.js +1024 -58
- package/bin/client/recorderv3/implemented_steps.js +2 -0
- package/bin/client/recorderv3/index.js +3 -283
- package/bin/client/recorderv3/services.js +818 -142
- package/bin/client/recorderv3/step_runner.js +20 -6
- package/bin/client/recorderv3/step_utils.js +542 -73
- package/bin/client/recorderv3/update_feature.js +87 -39
- package/bin/client/recorderv3/wbr_entry.js +61 -0
- package/bin/client/recording.js +1 -0
- package/bin/client/upload-service.js +4 -2
- package/bin/client/utils/app_dir.js +21 -0
- package/bin/client/utils/socket_logger.js +87 -125
- package/bin/index.js +4 -1
- package/package.json +11 -5
- package/bin/client/recorderv3/app_dir.js +0 -23
- package/bin/client/recorderv3/network.js +0 -299
- package/bin/client/recorderv3/scriptTest.js +0 -5
- package/bin/client/recorderv3/ws_server.js +0 -72
|
@@ -1,3 +1,7 @@
|
|
|
1
|
+
import dotenv from "dotenv";
|
|
2
|
+
// Load .env into process.env
|
|
3
|
+
dotenv.config();
|
|
4
|
+
|
|
1
5
|
import {
|
|
2
6
|
After,
|
|
3
7
|
setDefaultTimeout,
|
|
@@ -30,8 +34,8 @@ BeforeStep(async function (step) {
|
|
|
30
34
|
}
|
|
31
35
|
});
|
|
32
36
|
|
|
33
|
-
AfterStep(async function (
|
|
37
|
+
AfterStep(async function ({ result, pickleStep }) {
|
|
34
38
|
if (context) {
|
|
35
|
-
await context.web.afterStep(this,
|
|
39
|
+
await context.web.afterStep(this, pickleStep, result);
|
|
36
40
|
}
|
|
37
41
|
});
|
|
@@ -12,7 +12,7 @@ import path from "path";
|
|
|
12
12
|
* @param {string} text the text to verify exists in page
|
|
13
13
|
* @protect
|
|
14
14
|
*/
|
|
15
|
-
async function verifyTextExistsInPage(text) {
|
|
15
|
+
export async function verifyTextExistsInPage(text) {
|
|
16
16
|
await context.web.verifyTextExistInPage(text, null, this);
|
|
17
17
|
}
|
|
18
18
|
Then("Verify the text {string} can be found in the page", verifyTextExistsInPage);
|
|
@@ -22,7 +22,7 @@ Then("Verify the text {string} can be found in the page", verifyTextExistsInPage
|
|
|
22
22
|
* @param {string} elementDescription element description
|
|
23
23
|
* @protect
|
|
24
24
|
*/
|
|
25
|
-
async function clickOnElement(elementDescription) {
|
|
25
|
+
export async function clickOnElement(elementDescription) {
|
|
26
26
|
await context.web.simpleClick(elementDescription, null, null, this);
|
|
27
27
|
}
|
|
28
28
|
When("click on {string}", clickOnElement);
|
|
@@ -36,7 +36,7 @@ When("Click {string}", clickOnElement);
|
|
|
36
36
|
* @param {string} value value to fill the element with
|
|
37
37
|
* @protect
|
|
38
38
|
*/
|
|
39
|
-
async function fillElement(elementDescription, value) {
|
|
39
|
+
export async function fillElement(elementDescription, value) {
|
|
40
40
|
await context.web.simpleClickType(elementDescription, value, null, null, this);
|
|
41
41
|
}
|
|
42
42
|
When("fill {string} with {string}", fillElement);
|
|
@@ -47,7 +47,7 @@ When("Fill {string} with {string}", fillElement);
|
|
|
47
47
|
* @param {string} text the text to verify does not exist in page
|
|
48
48
|
* @protect
|
|
49
49
|
*/
|
|
50
|
-
async function verifyTextNotExistsInPage(text) {
|
|
50
|
+
export async function verifyTextNotExistsInPage(text) {
|
|
51
51
|
await context.web.waitForTextToDisappear(text, null, this);
|
|
52
52
|
}
|
|
53
53
|
Then("Verify the text {string} cannot be found in the page", verifyTextNotExistsInPage);
|
|
@@ -57,7 +57,7 @@ Then("Verify the text {string} cannot be found in the page", verifyTextNotExists
|
|
|
57
57
|
* @param {string} url URL to navigate
|
|
58
58
|
* @protect
|
|
59
59
|
*/
|
|
60
|
-
async function navigateTo(url) {
|
|
60
|
+
export async function navigateTo(url) {
|
|
61
61
|
await context.web.goto(url, this);
|
|
62
62
|
}
|
|
63
63
|
When("Navigate to {string}", navigateTo);
|
|
@@ -66,7 +66,7 @@ When("Navigate to {string}", navigateTo);
|
|
|
66
66
|
* Navigate to the current page
|
|
67
67
|
* @protect
|
|
68
68
|
*/
|
|
69
|
-
async function browserNavigateBack() {
|
|
69
|
+
export async function browserNavigateBack() {
|
|
70
70
|
await context.web.goBack({}, this);
|
|
71
71
|
}
|
|
72
72
|
Then("Browser navigate back", browserNavigateBack);
|
|
@@ -75,7 +75,7 @@ Then("Browser navigate back", browserNavigateBack);
|
|
|
75
75
|
* Navigate forward in browser history
|
|
76
76
|
* @protect
|
|
77
77
|
*/
|
|
78
|
-
async function browserNavigateForward() {
|
|
78
|
+
export async function browserNavigateForward() {
|
|
79
79
|
await context.web.goForward({}, this);
|
|
80
80
|
}
|
|
81
81
|
Then("Browser navigate forward", browserNavigateForward);
|
|
@@ -85,7 +85,7 @@ Then("Browser navigate forward", browserNavigateForward);
|
|
|
85
85
|
* @param {string} filePath the file path or empty to store in the test data file
|
|
86
86
|
* @protect
|
|
87
87
|
*/
|
|
88
|
-
async function storeBrowserSession(filePath) {
|
|
88
|
+
export async function storeBrowserSession(filePath) {
|
|
89
89
|
await context.web.saveStoreState(filePath, this);
|
|
90
90
|
}
|
|
91
91
|
When("Store browser session {string}", storeBrowserSession);
|
|
@@ -95,7 +95,7 @@ When("Store browser session {string}", storeBrowserSession);
|
|
|
95
95
|
* @param {string} filePath the file path or empty
|
|
96
96
|
* @protect
|
|
97
97
|
*/
|
|
98
|
-
async function resetBrowserSession(filePath) {
|
|
98
|
+
export async function resetBrowserSession(filePath) {
|
|
99
99
|
await context.web.restoreSaveState(filePath, this);
|
|
100
100
|
}
|
|
101
101
|
When("Reset browser session {string}", resetBrowserSession);
|
|
@@ -107,7 +107,7 @@ When("Reset browser session {string}", resetBrowserSession);
|
|
|
107
107
|
* @param {string} textToVerify the target text to verify
|
|
108
108
|
* @protect
|
|
109
109
|
*/
|
|
110
|
-
async function verifyTextRelatedToText(textAnchor, climb, textToVerify) {
|
|
110
|
+
export async function verifyTextRelatedToText(textAnchor, climb, textToVerify) {
|
|
111
111
|
await context.web.verifyTextRelatedToText(textAnchor, climb, textToVerify, null, this);
|
|
112
112
|
}
|
|
113
113
|
Then(
|
|
@@ -120,7 +120,7 @@ Then(
|
|
|
120
120
|
* @requestName the name of the bruno request file
|
|
121
121
|
* @protect
|
|
122
122
|
*/
|
|
123
|
-
async function runBrunoRequest(requestName) {
|
|
123
|
+
export async function runBrunoRequest(requestName) {
|
|
124
124
|
await executeBrunoRequest(requestName, {}, context, this);
|
|
125
125
|
}
|
|
126
126
|
When("Bruno - {string}", runBrunoRequest);
|
|
@@ -131,7 +131,7 @@ When("bruno - {string}", runBrunoRequest);
|
|
|
131
131
|
* @param {string} fileName the downloaded file to verify
|
|
132
132
|
* @protect
|
|
133
133
|
*/
|
|
134
|
-
async function verify_the_downloaded_file_exists(fileName) {
|
|
134
|
+
export async function verify_the_downloaded_file_exists(fileName) {
|
|
135
135
|
const downloadFolder = path.join(context.reportFolder, "downloads");
|
|
136
136
|
const downloadFile = path.join(downloadFolder, fileName);
|
|
137
137
|
await verifyFileExists(downloadFile, {}, context, this);
|
|
@@ -148,7 +148,7 @@ When("Noop", async function () {});
|
|
|
148
148
|
* @param {string} url URL to be verified against current URL
|
|
149
149
|
* @protect
|
|
150
150
|
*/
|
|
151
|
-
async function verify_page_url(url) {
|
|
151
|
+
export async function verify_page_url(url) {
|
|
152
152
|
await context.web.verifyPagePath(url, {}, this);
|
|
153
153
|
}
|
|
154
154
|
Then("Verify the page url is {string}", verify_page_url);
|
|
@@ -158,7 +158,7 @@ Then("Verify the page url is {string}", verify_page_url);
|
|
|
158
158
|
* @param {string} title Title to be verified against current Title
|
|
159
159
|
* @protect
|
|
160
160
|
*/
|
|
161
|
-
async function verify_page_title(title) {
|
|
161
|
+
export async function verify_page_title(title) {
|
|
162
162
|
await context.web.verifyPageTitle(title, {}, this);
|
|
163
163
|
}
|
|
164
164
|
Then("Verify the page title is {string}", verify_page_title);
|
|
@@ -170,7 +170,7 @@ Then("Verify the page title is {string}", verify_page_title);
|
|
|
170
170
|
* @param {world} - Optional world context
|
|
171
171
|
* @returns Promise that resolves after the specified duration
|
|
172
172
|
*/
|
|
173
|
-
async function sleep(duration) {
|
|
174
|
-
await context.web.sleep(duration, {},
|
|
173
|
+
export async function sleep(duration) {
|
|
174
|
+
await context.web.sleep(duration, {}, this);
|
|
175
175
|
}
|
|
176
176
|
Then("Sleep for {string} ms", { timeout: -1 }, sleep);
|
|
@@ -10,7 +10,7 @@ import * as t from "@babel/types";
|
|
|
10
10
|
import { CucumberExpression, ParameterTypeRegistry } from "@cucumber/cucumber-expressions";
|
|
11
11
|
import { existsSync, readFileSync, writeFileSync } from "fs";
|
|
12
12
|
import { getAiConfig } from "../code_gen/page_reflection.js";
|
|
13
|
-
import socketLogger from "../utils/socket_logger.js";
|
|
13
|
+
import socketLogger, { getErrorMessage } from "../utils/socket_logger.js";
|
|
14
14
|
|
|
15
15
|
const STEP_KEYWORDS = new Set(["Given", "When", "Then"]);
|
|
16
16
|
|
|
@@ -308,18 +308,27 @@ export function getDefaultPrettierConfig() {
|
|
|
308
308
|
const configContent = readFileSync(prettierConfigPath, "utf-8");
|
|
309
309
|
prettierConfig = JSON.parse(configContent);
|
|
310
310
|
} catch (error) {
|
|
311
|
-
socketLogger.error(
|
|
312
|
-
|
|
311
|
+
socketLogger.error(
|
|
312
|
+
`Error parsing Prettier config file: ${getErrorMessage(error)}`,
|
|
313
|
+
undefined,
|
|
314
|
+
"getDefaultPrettierConfig"
|
|
315
|
+
);
|
|
313
316
|
}
|
|
314
317
|
} else {
|
|
315
318
|
// save the default config to .prettierrc
|
|
316
319
|
try {
|
|
317
320
|
writeFileSync(prettierConfigPath, JSON.stringify(prettierConfig, null, 2), "utf-8");
|
|
318
|
-
socketLogger.info(
|
|
319
|
-
|
|
321
|
+
socketLogger.info(
|
|
322
|
+
`Created default Prettier config at ${prettierConfigPath}`,
|
|
323
|
+
undefined,
|
|
324
|
+
"getDefaultPrettierConfig"
|
|
325
|
+
);
|
|
320
326
|
} catch (error) {
|
|
321
|
-
socketLogger.error(
|
|
322
|
-
|
|
327
|
+
socketLogger.error(
|
|
328
|
+
`Error writing Prettier config file: ${getErrorMessage(error)}`,
|
|
329
|
+
undefined,
|
|
330
|
+
"getDefaultPrettierConfig"
|
|
331
|
+
);
|
|
323
332
|
}
|
|
324
333
|
}
|
|
325
334
|
return prettierConfig;
|
|
@@ -50,7 +50,7 @@ const generateApiCode = async (
|
|
|
50
50
|
// create new codePage
|
|
51
51
|
mjsFullPath = path.join(
|
|
52
52
|
projectDir,
|
|
53
|
-
stepsDefinitions.isTemp ? process.env.tempFeaturesFolderPath ?? "__temp_features" : "features",
|
|
53
|
+
stepsDefinitions.isTemp ? (process.env.tempFeaturesFolderPath ?? "__temp_features") : "features",
|
|
54
54
|
"step_definitions",
|
|
55
55
|
`api_tests_page.mjs`
|
|
56
56
|
);
|
|
@@ -60,7 +60,7 @@ const generateApiCode = async (
|
|
|
60
60
|
if (!codePage) {
|
|
61
61
|
mjsFullPath = path.join(
|
|
62
62
|
projectDir,
|
|
63
|
-
stepsDefinitions.isTemp ? process.env.tempFeaturesFolderPath ?? "__temp_features" : "features",
|
|
63
|
+
stepsDefinitions.isTemp ? (process.env.tempFeaturesFolderPath ?? "__temp_features") : "features",
|
|
64
64
|
"step_definitions",
|
|
65
65
|
`api_tests_page.mjs`
|
|
66
66
|
);
|
|
@@ -86,6 +86,30 @@ const parseDataSource = (paramNode, stepParams) => {
|
|
|
86
86
|
// }
|
|
87
87
|
throw new Error("Unknown parameter type");
|
|
88
88
|
};
|
|
89
|
+
|
|
90
|
+
|
|
91
|
+
const parseOptions = (optionsNode) => {
|
|
92
|
+
const properties = optionsNode.properties;
|
|
93
|
+
const options = {}
|
|
94
|
+
for (const prop of properties) {
|
|
95
|
+
let key = "";
|
|
96
|
+
if (prop.key.type === 'Identifier') {
|
|
97
|
+
key = prop.key.name;
|
|
98
|
+
} else if (prop.key.type === 'StringLiteral') {
|
|
99
|
+
key = prop.key.value;
|
|
100
|
+
}
|
|
101
|
+
if (key === 'context') continue;
|
|
102
|
+
options[key] = prop.value.value;
|
|
103
|
+
}
|
|
104
|
+
return options;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
const parseOptionsFromCallNode = (call, index) => {
|
|
108
|
+
if (call.arguments[index] && call.arguments[index].type === "ObjectExpression") {
|
|
109
|
+
return parseOptions(call.arguments[index]);
|
|
110
|
+
}
|
|
111
|
+
return null
|
|
112
|
+
}
|
|
89
113
|
const invertStableCommand = (call, elements, stepParams) => {
|
|
90
114
|
const methodName = call.callee.property.name;
|
|
91
115
|
const step = { type: null, parameters: [], element: null, locators: null };
|
|
@@ -104,6 +128,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
104
128
|
step.dataSource = inputParam.dataSource;
|
|
105
129
|
step.dataKey = inputParam.dataKey;
|
|
106
130
|
}
|
|
131
|
+
step.options = parseOptionsFromCallNode(call, 3);
|
|
107
132
|
break;
|
|
108
133
|
}
|
|
109
134
|
|
|
@@ -130,6 +155,8 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
130
155
|
step.count = clickCountProp.value.value;
|
|
131
156
|
}
|
|
132
157
|
}
|
|
158
|
+
|
|
159
|
+
step.options = parseOptionsFromCallNode(call, 2);
|
|
133
160
|
break;
|
|
134
161
|
|
|
135
162
|
case "setDateTime": {
|
|
@@ -147,6 +174,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
147
174
|
step.dataKey = dateParam.dataKey;
|
|
148
175
|
step.parameters = [toVariableName(dateParam.dataKey), format, enterParam];
|
|
149
176
|
}
|
|
177
|
+
step.options = parseOptionsFromCallNode(call, 5);
|
|
150
178
|
break;
|
|
151
179
|
}
|
|
152
180
|
|
|
@@ -163,6 +191,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
163
191
|
step.dataKey = selectParam.dataKey;
|
|
164
192
|
step.parameters = [toVariableName(selectParam.dataKey)];
|
|
165
193
|
}
|
|
194
|
+
step.options = parseOptionsFromCallNode(call, 3);
|
|
166
195
|
break;
|
|
167
196
|
}
|
|
168
197
|
|
|
@@ -176,6 +205,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
176
205
|
step.dataKey = text.dataKey;
|
|
177
206
|
step.parameters = [toVariableName(text.dataKey)];
|
|
178
207
|
}
|
|
208
|
+
step.options = parseOptionsFromCallNode(call, 2);
|
|
179
209
|
break;
|
|
180
210
|
}
|
|
181
211
|
case "waitForTextToDisappear": {
|
|
@@ -188,6 +218,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
188
218
|
step.dataKey = text.dataKey;
|
|
189
219
|
step.parameters = [toVariableName(text.dataKey)];
|
|
190
220
|
}
|
|
221
|
+
step.options = parseOptionsFromCallNode(call, 2);
|
|
191
222
|
break;
|
|
192
223
|
}
|
|
193
224
|
|
|
@@ -203,6 +234,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
203
234
|
step.dataKey = text.dataKey;
|
|
204
235
|
step.parameters = [toVariableName(text.dataKey), climb ?? null];
|
|
205
236
|
}
|
|
237
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
206
238
|
break;
|
|
207
239
|
}
|
|
208
240
|
case "extractAttribute": {
|
|
@@ -233,6 +265,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
233
265
|
step.regex = "";
|
|
234
266
|
step.trimSpaces = false;
|
|
235
267
|
}
|
|
268
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
236
269
|
break;
|
|
237
270
|
}
|
|
238
271
|
case "verifyAttribute": {
|
|
@@ -249,6 +282,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
249
282
|
step.dataKey = value.dataKey;
|
|
250
283
|
step.parameters = [attribute, toVariableName(value.dataKey)];
|
|
251
284
|
}
|
|
285
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
252
286
|
break;
|
|
253
287
|
}
|
|
254
288
|
|
|
@@ -277,6 +311,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
277
311
|
step.parameters = [toVariableName(fillParam.dataKey), enterValue];
|
|
278
312
|
}
|
|
279
313
|
}
|
|
314
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
280
315
|
break;
|
|
281
316
|
|
|
282
317
|
case "loadTestDataAsync": {
|
|
@@ -307,23 +342,28 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
307
342
|
step.dataKey = url.dataKey;
|
|
308
343
|
step.parameters = [toVariableName(url.dataKey)];
|
|
309
344
|
}
|
|
345
|
+
step.options = parseOptionsFromCallNode(call, 2);
|
|
310
346
|
break;
|
|
311
347
|
}
|
|
312
348
|
|
|
313
349
|
case "goBack":
|
|
314
350
|
step.type = Types.GO_BACK;
|
|
351
|
+
step.options = parseOptionsFromCallNode(call, 0);
|
|
315
352
|
break;
|
|
316
353
|
|
|
317
354
|
case "goForward":
|
|
318
355
|
step.type = Types.GO_FORWARD;
|
|
356
|
+
step.options = parseOptionsFromCallNode(call, 0);
|
|
319
357
|
break;
|
|
320
358
|
|
|
321
359
|
case "reloadPage":
|
|
322
360
|
step.type = Types.RELOAD;
|
|
361
|
+
step.options = parseOptionsFromCallNode(call, 0);
|
|
323
362
|
break;
|
|
324
363
|
|
|
325
364
|
case "closePage":
|
|
326
365
|
step.type = Types.CLOSE_PAGE;
|
|
366
|
+
step.options = parseOptionsFromCallNode(call, 0);
|
|
327
367
|
break;
|
|
328
368
|
|
|
329
369
|
case "simpleClick":
|
|
@@ -341,12 +381,15 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
341
381
|
case "hover":
|
|
342
382
|
step.type = Types.HOVER;
|
|
343
383
|
step.element = extractElement(call.arguments[0]);
|
|
384
|
+
step.options = parseOptionsFromCallNode(call, 2);
|
|
344
385
|
break;
|
|
345
386
|
|
|
346
387
|
case "setCheck":
|
|
347
388
|
step.type = Types.CHECK;
|
|
348
389
|
step.element = extractElement(call.arguments[0]);
|
|
349
390
|
step.check = call.arguments[1].value;
|
|
391
|
+
///
|
|
392
|
+
step.options = parseOptionsFromCallNode(call, 3);
|
|
350
393
|
break;
|
|
351
394
|
|
|
352
395
|
case "setViewportSize": {
|
|
@@ -354,12 +397,14 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
354
397
|
const width = call.arguments[0].value;
|
|
355
398
|
const height = call.arguments[1].value;
|
|
356
399
|
step.parameters = [width, height];
|
|
400
|
+
step.options = parseOptionsFromCallNode(call, 2);
|
|
357
401
|
break;
|
|
358
402
|
}
|
|
359
403
|
|
|
360
404
|
case "visualVerification": {
|
|
361
405
|
step.type = Types.VISUAL_VERIFICATION;
|
|
362
406
|
step.parameters = [call.arguments[0].value];
|
|
407
|
+
step.options = parseOptionsFromCallNode(call, 1);
|
|
363
408
|
break;
|
|
364
409
|
}
|
|
365
410
|
case "waitForUserInput": {
|
|
@@ -370,17 +415,21 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
370
415
|
case "extractEmailData": {
|
|
371
416
|
step.type = Types.MAIL_GET_CODE_OR_URL;
|
|
372
417
|
step.parameters = [call.arguments[0].value];
|
|
418
|
+
step.options = parseOptionsFromCallNode(call, 1);
|
|
373
419
|
break;
|
|
374
420
|
}
|
|
375
421
|
case "verifyTextRelatedToText": {
|
|
376
422
|
step.type = Types.VERIFY_TEXT_RELATED_TO_TEXT;
|
|
377
423
|
const textAnchorParse = parseDataSource(call.arguments[0], stepParams);
|
|
378
|
-
const textAnchor =
|
|
424
|
+
const textAnchor =
|
|
425
|
+
textAnchorParse.type === "literal" ? textAnchorParse.value : toVariableName(textAnchorParse.dataKey);
|
|
379
426
|
const climbParse = parseDataSource(call.arguments[1], stepParams);
|
|
380
427
|
const climb = climbParse.type === "literal" ? climbParse.value : toVariableName(climbParse.dataKey);
|
|
381
428
|
const textToVerifyParse = parseDataSource(call.arguments[2], stepParams);
|
|
382
|
-
const textToVerify =
|
|
429
|
+
const textToVerify =
|
|
430
|
+
textToVerifyParse.type === "literal" ? textToVerifyParse.value : toVariableName(textToVerifyParse.dataKey);
|
|
383
431
|
step.parameters = [textAnchor, climb, textToVerify];
|
|
432
|
+
step.options = parseOptionsFromCallNode(call, 3);
|
|
384
433
|
break;
|
|
385
434
|
}
|
|
386
435
|
case "setInputFiles": {
|
|
@@ -394,6 +443,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
394
443
|
step.dataKey = fileParam.dataKey;
|
|
395
444
|
step.parameters = [toVariableName(fileParam.dataKey)];
|
|
396
445
|
}
|
|
446
|
+
step.options = parseOptionsFromCallNode(call, 3);
|
|
397
447
|
break;
|
|
398
448
|
}
|
|
399
449
|
case "snapshotValidation": {
|
|
@@ -403,6 +453,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
403
453
|
step.parameters = [inputParam.value];
|
|
404
454
|
}
|
|
405
455
|
step.selectors = call.arguments[0].value;
|
|
456
|
+
step.options = parseOptionsFromCallNode(call, 3);
|
|
406
457
|
break;
|
|
407
458
|
}
|
|
408
459
|
case "verifyPageTitle": {
|
|
@@ -415,6 +466,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
415
466
|
step.dataKey = text.dataKey;
|
|
416
467
|
step.parameters = [toVariableName(text.dataKey)];
|
|
417
468
|
}
|
|
469
|
+
step.options = parseOptionsFromCallNode(call, 1);
|
|
418
470
|
break;
|
|
419
471
|
}
|
|
420
472
|
case "verifyPagePath": {
|
|
@@ -427,6 +479,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
427
479
|
step.dataKey = path.dataKey;
|
|
428
480
|
step.parameters = [toVariableName(path.dataKey)];
|
|
429
481
|
}
|
|
482
|
+
step.options = parseOptionsFromCallNode(call, 1);
|
|
430
483
|
break;
|
|
431
484
|
}
|
|
432
485
|
case "verifyProperty": {
|
|
@@ -441,6 +494,7 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
441
494
|
step.dataKey = value.dataKey;
|
|
442
495
|
step.parameters = [property, toVariableName(value.dataKey)];
|
|
443
496
|
}
|
|
497
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
444
498
|
break;
|
|
445
499
|
}
|
|
446
500
|
case "extractProperty": {
|
|
@@ -471,8 +525,71 @@ const invertStableCommand = (call, elements, stepParams) => {
|
|
|
471
525
|
step.regex = "";
|
|
472
526
|
step.trimSpaces = false;
|
|
473
527
|
}
|
|
528
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
529
|
+
break;
|
|
530
|
+
}
|
|
531
|
+
|
|
532
|
+
case "conditionalWait": {
|
|
533
|
+
step.type = Types.CONDITIONAL_WAIT;
|
|
534
|
+
step.element = extractElement(call.arguments[0]);
|
|
535
|
+
const condition = call.arguments[1].value;
|
|
536
|
+
|
|
537
|
+
const _timeout = parseDataSource(call.arguments[2], stepParams);
|
|
538
|
+
let timeout = 30;
|
|
539
|
+
if (_timeout.type === "literal") {
|
|
540
|
+
if (isNaN(_timeout.value)) {
|
|
541
|
+
throw new Error(`Timeout value must be a number, got ${_timeout.value}`);
|
|
542
|
+
}
|
|
543
|
+
timeout = Number(_timeout.value) * 1000;
|
|
544
|
+
} else {
|
|
545
|
+
step.dataSource = _timeout.dataSource;
|
|
546
|
+
step.dataKey = _timeout.dataKey;
|
|
547
|
+
timeout = toVariableName(_timeout.dataKey);
|
|
548
|
+
}
|
|
549
|
+
// step.timeout = timeout;
|
|
550
|
+
// step.value = "true";
|
|
551
|
+
step.parameters = [timeout, condition, step.value];
|
|
552
|
+
// step.condition = condition;
|
|
553
|
+
step.options = parseOptionsFromCallNode(call, 4);
|
|
474
554
|
break;
|
|
475
555
|
}
|
|
556
|
+
|
|
557
|
+
case "sleep": {
|
|
558
|
+
step.type = Types.SLEEP;
|
|
559
|
+
const duration = parseDataSource(call.arguments[0], stepParams);
|
|
560
|
+
if (duration.type === "literal") {
|
|
561
|
+
if (isNaN(duration.value)) {
|
|
562
|
+
throw new Error(`Sleep duration must be a number, got ${duration.value}`);
|
|
563
|
+
}
|
|
564
|
+
step.parameters = [Number(duration.value)];
|
|
565
|
+
} else {
|
|
566
|
+
step.dataSource = duration.dataSource;
|
|
567
|
+
step.dataKey = duration.dataKey;
|
|
568
|
+
step.parameters = [toVariableName(duration.dataKey)];
|
|
569
|
+
}
|
|
570
|
+
step.options = parseOptionsFromCallNode(call, 1);
|
|
571
|
+
break;
|
|
572
|
+
}
|
|
573
|
+
case "verify_file_exists": {
|
|
574
|
+
step.type = Types.VERIFY_FILE_EXISTS;
|
|
575
|
+
const filePath = parseDataSource(call.arguments[0], stepParams);
|
|
576
|
+
if (filePath.type === "literal") {
|
|
577
|
+
step.parameters = [filePath.value];
|
|
578
|
+
} else {
|
|
579
|
+
step.dataSource = filePath.dataSource;
|
|
580
|
+
step.dataKey = filePath.dataKey;
|
|
581
|
+
step.parameters = [toVariableName(filePath.dataKey)];
|
|
582
|
+
}
|
|
583
|
+
step.options = parseOptionsFromCallNode(call, 1);
|
|
584
|
+
break;
|
|
585
|
+
}
|
|
586
|
+
|
|
587
|
+
// case "verifyPagePath":
|
|
588
|
+
// {
|
|
589
|
+
// step.type = Types.VERIFY_PAGE_PATH;
|
|
590
|
+
// step.parameters = [call.arguments[0].value];
|
|
591
|
+
// break;
|
|
592
|
+
// }
|
|
476
593
|
default:
|
|
477
594
|
return; // Skip if no matching method
|
|
478
595
|
}
|
|
@@ -4,6 +4,7 @@ import path from "path";
|
|
|
4
4
|
import { CodePage } from "./page_reflection.js";
|
|
5
5
|
import { compareElements, generateSignature } from "./function_signature.js";
|
|
6
6
|
import { updateStepDefinitions } from "../recorderv3/step_utils.js";
|
|
7
|
+
import socketLoggerInstance from "../utils/socket_logger.js";
|
|
7
8
|
/**
|
|
8
9
|
* DuplicationAnalizer class
|
|
9
10
|
* How to use:
|
|
@@ -101,7 +102,7 @@ async function compareWithScenario(projectDir, scenario) {
|
|
|
101
102
|
analyzerExisting.load();
|
|
102
103
|
// generate a temporary folder
|
|
103
104
|
const folder = fs.mkdtempSync(path.join(os.tmpdir(), "blinq-"));
|
|
104
|
-
await updateStepDefinitions({ scenario, featureName: "temp", projectDir: folder });
|
|
105
|
+
await updateStepDefinitions({ scenario, featureName: "temp", projectDir: folder, logger: socketLoggerInstance });
|
|
105
106
|
const analyzerScenario = new DuplicationAnalizer(folder);
|
|
106
107
|
analyzerScenario.load();
|
|
107
108
|
const compareResults = [];
|
|
@@ -11,6 +11,8 @@ async function verify_the_opportunity_name_was_created(_name) {
|
|
|
11
11
|
}
|
|
12
12
|
*/
|
|
13
13
|
|
|
14
|
+
import strip from "strip-comments";
|
|
15
|
+
|
|
14
16
|
function generateSignature(page, functionName) {
|
|
15
17
|
let functionCode = null;
|
|
16
18
|
let method = null;
|
|
@@ -28,6 +30,8 @@ function generateSignature(page, functionName) {
|
|
|
28
30
|
method.node.body.parent.params.forEach((param) => {
|
|
29
31
|
parameters.push(param.name);
|
|
30
32
|
});
|
|
33
|
+
const strippedFunctionCode = strip(functionCode);
|
|
34
|
+
functionCode = strippedFunctionCode !== "" ? strippedFunctionCode : functionCode;
|
|
31
35
|
let lines = functionCode.split("\n");
|
|
32
36
|
// trim all lines
|
|
33
37
|
lines = lines.map((line) => line.trim());
|