@cqa-lib/cqa-ui 1.1.194 → 1.1.196
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/esm2020/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.mjs +87 -4
- package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +531 -13
- package/esm2020/lib/ui-kit.module.mjs +9 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +640 -35
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +638 -35
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.d.ts +19 -3
- package/lib/step-builder/step-builder-database/step-builder-database.component.d.ts +105 -5
- package/lib/ui-kit.module.d.ts +7 -5
- package/package.json +1 -1
- package/styles.css +1 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
-
import { Validators } from '@angular/forms';
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ViewChild } from '@angular/core';
|
|
2
|
+
import { FormGroup, Validators } from '@angular/forms';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
import * as i1 from "@angular/forms";
|
|
5
5
|
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
@@ -7,10 +7,19 @@ import * as i3 from "../../button/button.component";
|
|
|
7
7
|
import * as i4 from "../../badge/badge.component";
|
|
8
8
|
import * as i5 from "../../custom-textarea/custom-textarea.component";
|
|
9
9
|
import * as i6 from "../../custom-input/custom-input.component";
|
|
10
|
-
import * as i7 from "@angular/
|
|
10
|
+
import * as i7 from "@angular/material/button";
|
|
11
|
+
import * as i8 from "@angular/material/icon";
|
|
12
|
+
import * as i9 from "@angular/material/table";
|
|
13
|
+
import * as i10 from "@angular/material/form-field";
|
|
14
|
+
import * as i11 from "@angular/material/select";
|
|
15
|
+
import * as i12 from "@angular/material/core";
|
|
16
|
+
import * as i13 from "@angular/common";
|
|
17
|
+
import * as i14 from "@angular/material/input";
|
|
18
|
+
import * as i15 from "@angular/material/tooltip";
|
|
11
19
|
export class StepBuilderDatabaseComponent {
|
|
12
|
-
constructor(fb) {
|
|
20
|
+
constructor(fb, cdr) {
|
|
13
21
|
this.fb = fb;
|
|
22
|
+
this.cdr = cdr;
|
|
14
23
|
/** Options for DB environment dropdown */
|
|
15
24
|
this.dbEnvironmentOptions = [];
|
|
16
25
|
/** Initial queries list */
|
|
@@ -29,8 +38,67 @@ export class StepBuilderDatabaseComponent {
|
|
|
29
38
|
this.addQuery = new EventEmitter();
|
|
30
39
|
/** Emit when query is deleted */
|
|
31
40
|
this.deleteQuery = new EventEmitter();
|
|
41
|
+
/** Emit when assertion data changes for a query */
|
|
42
|
+
this.assertionDataChange = new EventEmitter();
|
|
43
|
+
/** Emit when selected query index changes */
|
|
44
|
+
this.selectedQueryIndexChange = new EventEmitter();
|
|
45
|
+
/** Emit when active tab changes */
|
|
46
|
+
this.activeTabChange = new EventEmitter();
|
|
32
47
|
this.selectedTab = 'output';
|
|
33
48
|
this.selectedQueryIndex = 0;
|
|
49
|
+
// Assertions form management - one form per query
|
|
50
|
+
this.assertionsForms = new Map();
|
|
51
|
+
this.displayedColumns = ['jsonPath', 'verificationType', 'expectedType', 'expectedValue', 'actions'];
|
|
52
|
+
this.defaultExpectedType = 'string';
|
|
53
|
+
this.formSubscriptions = new Map();
|
|
54
|
+
/**
|
|
55
|
+
* Expected value validator
|
|
56
|
+
*/
|
|
57
|
+
this.expectedValueValidator = (control) => {
|
|
58
|
+
const expectedType = control.parent?.get('expectedType')?.value;
|
|
59
|
+
if (!expectedType || expectedType === 'null')
|
|
60
|
+
return null;
|
|
61
|
+
const value = control.value;
|
|
62
|
+
if (!value && value !== 0 && value !== false)
|
|
63
|
+
return null;
|
|
64
|
+
if (expectedType === 'number') {
|
|
65
|
+
if (isNaN(Number(value))) {
|
|
66
|
+
return { invalidType: true, message: 'Invalid number value' };
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
else if (expectedType === 'boolean') {
|
|
70
|
+
if (value !== 'true' && value !== 'false' && value !== true && value !== false) {
|
|
71
|
+
return { invalidType: true, message: 'Invalid boolean value' };
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
else if (expectedType === 'array') {
|
|
75
|
+
if (typeof value === 'string' && value.trim()) {
|
|
76
|
+
try {
|
|
77
|
+
const parsed = JSON.parse(value);
|
|
78
|
+
if (!Array.isArray(parsed)) {
|
|
79
|
+
return { invalidType: true, message: 'Value must be a valid JSON array' };
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
catch (e) {
|
|
83
|
+
return { invalidType: true, message: 'Invalid Array value - must be valid JSON' };
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
else if (expectedType === 'object') {
|
|
88
|
+
if (typeof value === 'string' && value.trim() && value.trim() !== '{}') {
|
|
89
|
+
try {
|
|
90
|
+
const parsed = JSON.parse(value);
|
|
91
|
+
if (parsed === null || typeof parsed !== 'object' || Array.isArray(parsed)) {
|
|
92
|
+
return { invalidType: true, message: 'Value must be a valid JSON object' };
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
catch (e) {
|
|
96
|
+
return { invalidType: true, message: 'Invalid Object value - must be valid JSON' };
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
return null;
|
|
101
|
+
};
|
|
34
102
|
this.databaseForm = this.fb.group({
|
|
35
103
|
dbEnvironment: ['', Validators.required],
|
|
36
104
|
queries: this.fb.array([])
|
|
@@ -46,9 +114,29 @@ export class StepBuilderDatabaseComponent {
|
|
|
46
114
|
while (queriesArray.length !== 0) {
|
|
47
115
|
queriesArray.removeAt(0);
|
|
48
116
|
}
|
|
49
|
-
|
|
117
|
+
// Clear old assertions forms
|
|
118
|
+
this.assertionsForms.forEach((form, index) => {
|
|
119
|
+
if (this.formSubscriptions.has(index)) {
|
|
120
|
+
this.formSubscriptions.get(index).unsubscribe();
|
|
121
|
+
}
|
|
122
|
+
});
|
|
123
|
+
this.assertionsForms.clear();
|
|
124
|
+
this.formSubscriptions.clear();
|
|
125
|
+
// Rebuild queries and assertions forms
|
|
126
|
+
this.queries.forEach((query, index) => {
|
|
127
|
+
// Ensure assertion data is initialized
|
|
128
|
+
if (!query.assertionData) {
|
|
129
|
+
query.assertionData = [];
|
|
130
|
+
}
|
|
50
131
|
this.addQueryToForm(query);
|
|
132
|
+
// Initialize assertions form for each query
|
|
133
|
+
this.ensureAssertionDataInitialized(index);
|
|
51
134
|
});
|
|
135
|
+
// Ensure selected query index is valid
|
|
136
|
+
if (this.selectedQueryIndex >= this.queries.length) {
|
|
137
|
+
this.selectedQueryIndex = Math.max(0, this.queries.length - 1);
|
|
138
|
+
}
|
|
139
|
+
this.cdr.detectChanges();
|
|
52
140
|
}
|
|
53
141
|
if (changes['initialDbEnvironment'] && changes['initialDbEnvironment'].currentValue) {
|
|
54
142
|
this.databaseForm.patchValue({ dbEnvironment: changes['initialDbEnvironment'].currentValue });
|
|
@@ -65,9 +153,16 @@ export class StepBuilderDatabaseComponent {
|
|
|
65
153
|
}
|
|
66
154
|
else {
|
|
67
155
|
this.queries.forEach((query, index) => {
|
|
156
|
+
// Ensure assertion data is initialized
|
|
157
|
+
if (!query.assertionData) {
|
|
158
|
+
query.assertionData = [];
|
|
159
|
+
}
|
|
68
160
|
this.addQueryToForm(query);
|
|
161
|
+
// Initialize assertions form for each query
|
|
162
|
+
this.ensureAssertionDataInitialized(index);
|
|
69
163
|
if (index === 0) {
|
|
70
164
|
this.selectedQueryIndex = 0;
|
|
165
|
+
this.selectedQueryIndexChange.emit(0);
|
|
71
166
|
}
|
|
72
167
|
});
|
|
73
168
|
}
|
|
@@ -80,12 +175,18 @@ export class StepBuilderDatabaseComponent {
|
|
|
80
175
|
id: `query_${Date.now()}`,
|
|
81
176
|
query: '',
|
|
82
177
|
variable: '',
|
|
83
|
-
status: 'pending'
|
|
178
|
+
status: 'pending',
|
|
179
|
+
assertionData: [] // Initialize with empty assertion data
|
|
84
180
|
};
|
|
85
181
|
this.queries.push(newQuery);
|
|
86
182
|
this.addQueryToForm(newQuery);
|
|
87
|
-
|
|
183
|
+
const newIndex = this.queries.length - 1;
|
|
184
|
+
this.selectedQueryIndex = newIndex;
|
|
185
|
+
// Initialize assertions form for the new query
|
|
186
|
+
this.ensureAssertionDataInitialized(newIndex);
|
|
187
|
+
this.selectedQueryIndexChange.emit(this.selectedQueryIndex);
|
|
88
188
|
this.addQuery.emit();
|
|
189
|
+
this.cdr.detectChanges();
|
|
89
190
|
}
|
|
90
191
|
addQueryToForm(query) {
|
|
91
192
|
const queryGroup = this.fb.group({
|
|
@@ -98,12 +199,44 @@ export class StepBuilderDatabaseComponent {
|
|
|
98
199
|
deleteQueryById(queryId) {
|
|
99
200
|
const index = this.queries.findIndex(q => q.id === queryId);
|
|
100
201
|
if (index !== -1) {
|
|
202
|
+
// Clean up assertions form and subscription for this query
|
|
203
|
+
if (this.formSubscriptions.has(index)) {
|
|
204
|
+
this.formSubscriptions.get(index).unsubscribe();
|
|
205
|
+
this.formSubscriptions.delete(index);
|
|
206
|
+
}
|
|
207
|
+
this.assertionsForms.delete(index);
|
|
208
|
+
// Reindex remaining forms and subscriptions
|
|
209
|
+
const formsToReindex = new Map();
|
|
210
|
+
const subscriptionsToReindex = new Map();
|
|
211
|
+
this.assertionsForms.forEach((form, oldIndex) => {
|
|
212
|
+
if (oldIndex > index) {
|
|
213
|
+
formsToReindex.set(oldIndex - 1, form);
|
|
214
|
+
}
|
|
215
|
+
else if (oldIndex < index) {
|
|
216
|
+
formsToReindex.set(oldIndex, form);
|
|
217
|
+
}
|
|
218
|
+
});
|
|
219
|
+
this.formSubscriptions.forEach((sub, oldIndex) => {
|
|
220
|
+
if (oldIndex > index) {
|
|
221
|
+
subscriptionsToReindex.set(oldIndex - 1, sub);
|
|
222
|
+
}
|
|
223
|
+
else if (oldIndex < index) {
|
|
224
|
+
subscriptionsToReindex.set(oldIndex, sub);
|
|
225
|
+
}
|
|
226
|
+
});
|
|
227
|
+
this.assertionsForms = formsToReindex;
|
|
228
|
+
this.formSubscriptions = subscriptionsToReindex;
|
|
101
229
|
this.queries.splice(index, 1);
|
|
102
230
|
this.queriesFormArray.removeAt(index);
|
|
103
231
|
if (this.selectedQueryIndex >= this.queries.length) {
|
|
104
232
|
this.selectedQueryIndex = Math.max(0, this.queries.length - 1);
|
|
233
|
+
// Ensure assertions form is initialized for the new selected query
|
|
234
|
+
if (this.selectedQueryIndex >= 0) {
|
|
235
|
+
this.ensureAssertionDataInitialized(this.selectedQueryIndex);
|
|
236
|
+
}
|
|
105
237
|
}
|
|
106
238
|
this.deleteQuery.emit(queryId);
|
|
239
|
+
this.cdr.detectChanges();
|
|
107
240
|
}
|
|
108
241
|
}
|
|
109
242
|
getDbEnvironmentConfig() {
|
|
@@ -145,10 +278,33 @@ export class StepBuilderDatabaseComponent {
|
|
|
145
278
|
}
|
|
146
279
|
}
|
|
147
280
|
onSelectQuery(index) {
|
|
281
|
+
// Ensure assertion data is initialized for the selected query
|
|
282
|
+
this.ensureAssertionDataInitialized(index);
|
|
148
283
|
this.selectedQueryIndex = index;
|
|
284
|
+
this.selectedQueryIndexChange.emit(index);
|
|
285
|
+
// Trigger change detection to ensure assertions form is displayed
|
|
286
|
+
this.cdr.detectChanges();
|
|
287
|
+
}
|
|
288
|
+
/**
|
|
289
|
+
* Get the currently selected query index
|
|
290
|
+
*/
|
|
291
|
+
getSelectedQueryIndex() {
|
|
292
|
+
return this.selectedQueryIndex;
|
|
293
|
+
}
|
|
294
|
+
/**
|
|
295
|
+
* Get the currently active tab
|
|
296
|
+
*/
|
|
297
|
+
getActiveTab() {
|
|
298
|
+
return this.selectedTab;
|
|
149
299
|
}
|
|
150
300
|
onTabChange(tab) {
|
|
151
301
|
this.selectedTab = tab;
|
|
302
|
+
this.activeTabChange.emit(tab);
|
|
303
|
+
// Ensure assertions form is initialized when switching to verification tab
|
|
304
|
+
if (tab === 'verification') {
|
|
305
|
+
this.ensureAssertionDataInitialized(this.selectedQueryIndex);
|
|
306
|
+
this.cdr.detectChanges();
|
|
307
|
+
}
|
|
152
308
|
}
|
|
153
309
|
onCopyResults() {
|
|
154
310
|
// Copy query results to clipboard
|
|
@@ -173,7 +329,8 @@ export class StepBuilderDatabaseComponent {
|
|
|
173
329
|
id: query.id,
|
|
174
330
|
query: queryFormGroup.get('query')?.value || '',
|
|
175
331
|
variable: queryFormGroup.get('variable')?.value || '',
|
|
176
|
-
status: query.status
|
|
332
|
+
status: query.status,
|
|
333
|
+
assertionData: query.assertionData || [] // Include assertion data
|
|
177
334
|
};
|
|
178
335
|
})
|
|
179
336
|
};
|
|
@@ -186,13 +343,365 @@ export class StepBuilderDatabaseComponent {
|
|
|
186
343
|
}
|
|
187
344
|
return Object.keys(this.queryResults[0]);
|
|
188
345
|
}
|
|
346
|
+
/**
|
|
347
|
+
* Get assertion data for a specific query
|
|
348
|
+
*/
|
|
349
|
+
getAssertionDataForQuery(queryIndex) {
|
|
350
|
+
if (this.queries[queryIndex] && this.queries[queryIndex].assertionData) {
|
|
351
|
+
return this.queries[queryIndex].assertionData;
|
|
352
|
+
}
|
|
353
|
+
return [];
|
|
354
|
+
}
|
|
355
|
+
/**
|
|
356
|
+
* Handle assertion data change from external component
|
|
357
|
+
*/
|
|
358
|
+
onAssertionDataChange(assertionData, queryIndex) {
|
|
359
|
+
if (this.queries[queryIndex]) {
|
|
360
|
+
// Create a deep copy to ensure each query has its own data
|
|
361
|
+
if (assertionData && assertionData.length > 0) {
|
|
362
|
+
this.queries[queryIndex].assertionData = assertionData.map((item) => {
|
|
363
|
+
// Ensure expectedValue is properly formatted
|
|
364
|
+
const expectedType = item.expectedType || 'string';
|
|
365
|
+
let expectedValue = item.expectedValue;
|
|
366
|
+
// For object/array types, ensure expectedValue is an object/array, not a string
|
|
367
|
+
if (expectedType === 'object' && typeof expectedValue === 'string' && expectedValue.trim()) {
|
|
368
|
+
try {
|
|
369
|
+
expectedValue = JSON.parse(expectedValue);
|
|
370
|
+
}
|
|
371
|
+
catch (e) {
|
|
372
|
+
// If parsing fails, keep as string
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
else if (expectedType === 'array' && typeof expectedValue === 'string' && expectedValue.trim()) {
|
|
376
|
+
try {
|
|
377
|
+
expectedValue = JSON.parse(expectedValue);
|
|
378
|
+
}
|
|
379
|
+
catch (e) {
|
|
380
|
+
// If parsing fails, keep as string
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
else if (expectedType === 'number' && typeof expectedValue === 'string') {
|
|
384
|
+
expectedValue = expectedValue !== '' ? Number(expectedValue) : null;
|
|
385
|
+
}
|
|
386
|
+
else if (expectedType === 'boolean') {
|
|
387
|
+
if (expectedValue === 'true' || expectedValue === true) {
|
|
388
|
+
expectedValue = true;
|
|
389
|
+
}
|
|
390
|
+
else if (expectedValue === 'false' || expectedValue === false) {
|
|
391
|
+
expectedValue = false;
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
return {
|
|
395
|
+
jsonPath: item.jsonPath || '',
|
|
396
|
+
verificationType: item.verificationType || 'equals',
|
|
397
|
+
expectedType: expectedType,
|
|
398
|
+
expectedValue: expectedValue
|
|
399
|
+
};
|
|
400
|
+
});
|
|
401
|
+
}
|
|
402
|
+
else {
|
|
403
|
+
this.queries[queryIndex].assertionData = [];
|
|
404
|
+
}
|
|
405
|
+
// Emit the change to parent
|
|
406
|
+
this.assertionDataChange.emit({
|
|
407
|
+
queryIndex: queryIndex,
|
|
408
|
+
assertionData: this.queries[queryIndex].assertionData || []
|
|
409
|
+
});
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
/**
|
|
413
|
+
* Initialize assertion data for a query if it doesn't exist
|
|
414
|
+
*/
|
|
415
|
+
ensureAssertionDataInitialized(queryIndex) {
|
|
416
|
+
if (this.queries[queryIndex] && !this.queries[queryIndex].assertionData) {
|
|
417
|
+
this.queries[queryIndex].assertionData = [];
|
|
418
|
+
}
|
|
419
|
+
// Initialize assertions form for this query if it doesn't exist
|
|
420
|
+
if (!this.assertionsForms.has(queryIndex)) {
|
|
421
|
+
this.initializeAssertionsForm(queryIndex);
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
/**
|
|
425
|
+
* Initialize assertions form for a specific query
|
|
426
|
+
*/
|
|
427
|
+
initializeAssertionsForm(queryIndex) {
|
|
428
|
+
const assertionsArray = this.fb.array([]);
|
|
429
|
+
const assertionsForm = this.fb.group({
|
|
430
|
+
assertions: assertionsArray
|
|
431
|
+
});
|
|
432
|
+
this.assertionsForms.set(queryIndex, assertionsForm);
|
|
433
|
+
// Load existing assertions if they exist
|
|
434
|
+
const query = this.queries[queryIndex];
|
|
435
|
+
if (query && query.assertionData && query.assertionData.length > 0) {
|
|
436
|
+
this.loadAssertionsForQuery(queryIndex);
|
|
437
|
+
}
|
|
438
|
+
// Subscribe to form changes
|
|
439
|
+
this.subscribeToAssertionsFormChanges(queryIndex);
|
|
440
|
+
}
|
|
441
|
+
/**
|
|
442
|
+
* Get assertions form for current query
|
|
443
|
+
*/
|
|
444
|
+
getCurrentAssertionsForm() {
|
|
445
|
+
if (this.assertionsForms.has(this.selectedQueryIndex)) {
|
|
446
|
+
return this.assertionsForms.get(this.selectedQueryIndex);
|
|
447
|
+
}
|
|
448
|
+
// Initialize if it doesn't exist
|
|
449
|
+
this.ensureAssertionDataInitialized(this.selectedQueryIndex);
|
|
450
|
+
return this.assertionsForms.get(this.selectedQueryIndex) || null;
|
|
451
|
+
}
|
|
452
|
+
/**
|
|
453
|
+
* Get assertions form array for current query
|
|
454
|
+
*/
|
|
455
|
+
getAssertionsFormArray() {
|
|
456
|
+
const form = this.getCurrentAssertionsForm();
|
|
457
|
+
if (form) {
|
|
458
|
+
return form.get('assertions');
|
|
459
|
+
}
|
|
460
|
+
return this.fb.array([]);
|
|
461
|
+
}
|
|
462
|
+
/**
|
|
463
|
+
* Load assertions into form for a specific query
|
|
464
|
+
*/
|
|
465
|
+
loadAssertionsForQuery(queryIndex) {
|
|
466
|
+
const form = this.assertionsForms.get(queryIndex);
|
|
467
|
+
if (!form)
|
|
468
|
+
return;
|
|
469
|
+
const formArray = form.get('assertions');
|
|
470
|
+
const query = this.queries[queryIndex];
|
|
471
|
+
if (!query || !query.assertionData)
|
|
472
|
+
return;
|
|
473
|
+
// Clear existing form array
|
|
474
|
+
while (formArray.length !== 0) {
|
|
475
|
+
formArray.removeAt(0);
|
|
476
|
+
}
|
|
477
|
+
// Load assertions
|
|
478
|
+
query.assertionData.forEach((assertion) => {
|
|
479
|
+
const expectedType = assertion.expectedType || 'string';
|
|
480
|
+
let expectedValue = '';
|
|
481
|
+
if (expectedType === 'null') {
|
|
482
|
+
expectedValue = null;
|
|
483
|
+
}
|
|
484
|
+
else if (expectedType === 'object' || expectedType === 'array') {
|
|
485
|
+
if (typeof assertion.expectedValue === 'object' && assertion.expectedValue !== null) {
|
|
486
|
+
expectedValue = JSON.stringify(assertion.expectedValue, null, 2);
|
|
487
|
+
}
|
|
488
|
+
else {
|
|
489
|
+
expectedValue = assertion.expectedValue || '';
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
else if (expectedType === 'boolean') {
|
|
493
|
+
expectedValue = (assertion.expectedValue === true || assertion.expectedValue === 'true') ? 'true' : 'false';
|
|
494
|
+
}
|
|
495
|
+
else if (expectedType === 'number') {
|
|
496
|
+
expectedValue = assertion.expectedValue !== null && assertion.expectedValue !== undefined
|
|
497
|
+
? String(assertion.expectedValue)
|
|
498
|
+
: '';
|
|
499
|
+
}
|
|
500
|
+
else {
|
|
501
|
+
expectedValue = assertion.expectedValue !== null && assertion.expectedValue !== undefined
|
|
502
|
+
? String(assertion.expectedValue)
|
|
503
|
+
: '';
|
|
504
|
+
}
|
|
505
|
+
const formGroup = this.fb.group({
|
|
506
|
+
jsonPath: [assertion.jsonPath || '', Validators.required],
|
|
507
|
+
verificationType: [assertion.verificationType || 'equals'],
|
|
508
|
+
expectedType: [expectedType],
|
|
509
|
+
expectedValue: [expectedValue, [Validators.required, this.expectedValueValidator]]
|
|
510
|
+
});
|
|
511
|
+
formArray.push(formGroup);
|
|
512
|
+
});
|
|
513
|
+
this.cdr.detectChanges();
|
|
514
|
+
}
|
|
515
|
+
/**
|
|
516
|
+
* Subscribe to assertions form changes
|
|
517
|
+
*/
|
|
518
|
+
subscribeToAssertionsFormChanges(queryIndex) {
|
|
519
|
+
const form = this.assertionsForms.get(queryIndex);
|
|
520
|
+
if (!form)
|
|
521
|
+
return;
|
|
522
|
+
// Unsubscribe from previous subscription if exists
|
|
523
|
+
if (this.formSubscriptions.has(queryIndex)) {
|
|
524
|
+
this.formSubscriptions.get(queryIndex).unsubscribe();
|
|
525
|
+
}
|
|
526
|
+
const formArray = form.get('assertions');
|
|
527
|
+
const subscription = formArray.valueChanges.subscribe(() => {
|
|
528
|
+
this.emitAssertionDataForQuery(queryIndex);
|
|
529
|
+
});
|
|
530
|
+
this.formSubscriptions.set(queryIndex, subscription);
|
|
531
|
+
}
|
|
532
|
+
/**
|
|
533
|
+
* Emit assertion data for a specific query
|
|
534
|
+
*/
|
|
535
|
+
emitAssertionDataForQuery(queryIndex) {
|
|
536
|
+
const form = this.assertionsForms.get(queryIndex);
|
|
537
|
+
if (!form)
|
|
538
|
+
return;
|
|
539
|
+
const formArray = form.get('assertions');
|
|
540
|
+
const assertionData = formArray.controls.map((control) => {
|
|
541
|
+
if (!(control instanceof FormGroup))
|
|
542
|
+
return null;
|
|
543
|
+
const value = control.value;
|
|
544
|
+
const expectedType = value.expectedType || 'string';
|
|
545
|
+
let expectedValue = value.expectedValue;
|
|
546
|
+
// Parse expectedValue based on type
|
|
547
|
+
if (expectedType === 'null') {
|
|
548
|
+
expectedValue = null;
|
|
549
|
+
}
|
|
550
|
+
else if (expectedType === 'number') {
|
|
551
|
+
expectedValue = expectedValue !== '' ? Number(expectedValue) : null;
|
|
552
|
+
}
|
|
553
|
+
else if (expectedType === 'boolean') {
|
|
554
|
+
expectedValue = (expectedValue === 'true' || expectedValue === true);
|
|
555
|
+
}
|
|
556
|
+
else if (expectedType === 'array' || expectedType === 'object') {
|
|
557
|
+
if (typeof expectedValue === 'string' && expectedValue.trim()) {
|
|
558
|
+
try {
|
|
559
|
+
expectedValue = JSON.parse(expectedValue);
|
|
560
|
+
}
|
|
561
|
+
catch (e) {
|
|
562
|
+
// Keep as string if parsing fails
|
|
563
|
+
}
|
|
564
|
+
}
|
|
565
|
+
}
|
|
566
|
+
else {
|
|
567
|
+
expectedValue = String(expectedValue || '');
|
|
568
|
+
}
|
|
569
|
+
return {
|
|
570
|
+
jsonPath: (value.jsonPath || '').trim(),
|
|
571
|
+
verificationType: value.verificationType || 'equals',
|
|
572
|
+
expectedType: expectedType,
|
|
573
|
+
expectedValue: expectedValue
|
|
574
|
+
};
|
|
575
|
+
}).filter(item => item !== null && item.jsonPath);
|
|
576
|
+
// Update query's assertion data
|
|
577
|
+
if (this.queries[queryIndex]) {
|
|
578
|
+
this.queries[queryIndex].assertionData = assertionData;
|
|
579
|
+
}
|
|
580
|
+
// Emit to parent
|
|
581
|
+
this.assertionDataChange.emit({
|
|
582
|
+
queryIndex: queryIndex,
|
|
583
|
+
assertionData: assertionData
|
|
584
|
+
});
|
|
585
|
+
}
|
|
586
|
+
/**
|
|
587
|
+
* Add a new assertion
|
|
588
|
+
*/
|
|
589
|
+
addAssertion() {
|
|
590
|
+
const formArray = this.getAssertionsFormArray();
|
|
591
|
+
const formGroup = this.fb.group({
|
|
592
|
+
jsonPath: ['', Validators.required],
|
|
593
|
+
verificationType: ['equals'],
|
|
594
|
+
expectedType: [this.defaultExpectedType],
|
|
595
|
+
expectedValue: ['', [Validators.required, this.expectedValueValidator]]
|
|
596
|
+
});
|
|
597
|
+
formArray.push(formGroup);
|
|
598
|
+
this.cdr.detectChanges();
|
|
599
|
+
if (this.assertionsTable) {
|
|
600
|
+
this.assertionsTable.renderRows();
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
/**
|
|
604
|
+
* Remove an assertion
|
|
605
|
+
*/
|
|
606
|
+
removeAssertion(index) {
|
|
607
|
+
const formArray = this.getAssertionsFormArray();
|
|
608
|
+
formArray.removeAt(index);
|
|
609
|
+
this.cdr.detectChanges();
|
|
610
|
+
if (this.assertionsTable) {
|
|
611
|
+
this.assertionsTable.renderRows();
|
|
612
|
+
}
|
|
613
|
+
this.emitAssertionDataForQuery(this.selectedQueryIndex);
|
|
614
|
+
}
|
|
615
|
+
/**
|
|
616
|
+
* Get form control at specific index
|
|
617
|
+
*/
|
|
618
|
+
getFormControlAt(index, controlName) {
|
|
619
|
+
const formArray = this.getAssertionsFormArray();
|
|
620
|
+
if (index < 0 || index >= formArray.length)
|
|
621
|
+
return null;
|
|
622
|
+
const formGroup = formArray.at(index);
|
|
623
|
+
return formGroup?.get(controlName) || null;
|
|
624
|
+
}
|
|
625
|
+
/**
|
|
626
|
+
* Get expected type for an assertion
|
|
627
|
+
*/
|
|
628
|
+
getExpectedType(index) {
|
|
629
|
+
const formArray = this.getAssertionsFormArray();
|
|
630
|
+
if (index < 0 || index >= formArray.length)
|
|
631
|
+
return 'string';
|
|
632
|
+
const formGroup = formArray.at(index);
|
|
633
|
+
return formGroup?.get('expectedType')?.value || 'string';
|
|
634
|
+
}
|
|
635
|
+
/**
|
|
636
|
+
* Get expected value placeholder
|
|
637
|
+
*/
|
|
638
|
+
getExpectedValuePlaceholder(index) {
|
|
639
|
+
const expectedType = this.getExpectedType(index);
|
|
640
|
+
switch (expectedType) {
|
|
641
|
+
case 'string': return 'Expected Value in String';
|
|
642
|
+
case 'array': return 'Expected Value in Array (JSON format)';
|
|
643
|
+
case 'object': return 'Expected Value in Object (JSON format)';
|
|
644
|
+
case 'null': return 'No value required for null type';
|
|
645
|
+
default: return 'Expected Value';
|
|
646
|
+
}
|
|
647
|
+
}
|
|
648
|
+
/**
|
|
649
|
+
* Handle expected type change
|
|
650
|
+
*/
|
|
651
|
+
onExpectedTypeChange(index, value) {
|
|
652
|
+
const formArray = this.getAssertionsFormArray();
|
|
653
|
+
if (index < 0 || index >= formArray.length)
|
|
654
|
+
return;
|
|
655
|
+
const formGroup = formArray.at(index);
|
|
656
|
+
const expectedValueControl = formGroup.get('expectedValue');
|
|
657
|
+
if (!expectedValueControl)
|
|
658
|
+
return;
|
|
659
|
+
let newValue = '';
|
|
660
|
+
if (value === 'boolean') {
|
|
661
|
+
newValue = 'true';
|
|
662
|
+
}
|
|
663
|
+
else if (value === 'null') {
|
|
664
|
+
newValue = null;
|
|
665
|
+
}
|
|
666
|
+
expectedValueControl.clearValidators();
|
|
667
|
+
expectedValueControl.setValue(newValue, { emitEvent: false });
|
|
668
|
+
if (value === 'null') {
|
|
669
|
+
expectedValueControl.setValidators([this.expectedValueValidator]);
|
|
670
|
+
}
|
|
671
|
+
else if (value === 'object' || value === 'array') {
|
|
672
|
+
expectedValueControl.setValidators([Validators.required]);
|
|
673
|
+
}
|
|
674
|
+
else {
|
|
675
|
+
expectedValueControl.setValidators([Validators.required, this.expectedValueValidator]);
|
|
676
|
+
}
|
|
677
|
+
setTimeout(() => {
|
|
678
|
+
expectedValueControl.updateValueAndValidity({ emitEvent: false });
|
|
679
|
+
}, 100);
|
|
680
|
+
}
|
|
681
|
+
/**
|
|
682
|
+
* Get assertions data source for table
|
|
683
|
+
*/
|
|
684
|
+
get assertionsDataSource() {
|
|
685
|
+
return this.getAssertionsFormArray().controls;
|
|
686
|
+
}
|
|
687
|
+
/**
|
|
688
|
+
* Track by function for table rows
|
|
689
|
+
*/
|
|
690
|
+
trackByIndex(index) {
|
|
691
|
+
return index;
|
|
692
|
+
}
|
|
693
|
+
ngOnDestroy() {
|
|
694
|
+
// Clean up subscriptions
|
|
695
|
+
this.formSubscriptions.forEach(sub => sub.unsubscribe());
|
|
696
|
+
this.formSubscriptions.clear();
|
|
697
|
+
}
|
|
189
698
|
}
|
|
190
|
-
StepBuilderDatabaseComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
191
|
-
StepBuilderDatabaseComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderDatabaseComponent, selector: "cqa-step-builder-database", inputs: { dbEnvironmentOptions: "dbEnvironmentOptions", queries: "queries", queryResults: "queryResults", isLoading: "isLoading", initialDbEnvironment: "initialDbEnvironment", editMode: "editMode" }, outputs: { createStep: "createStep", cancelled: "cancelled", runQuery: "runQuery", addQuery: "addQuery", deleteQuery: "deleteQuery" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n </h2>\n\n <!-- DB Environment Section (separator line only on bottom) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n </div>\n <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n </cqa-button>\n </div>\n <p class=\"cqa-text-xs cqa-text-gray-500\">\n Uses Database environments from Environments.\n </p>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n (click)=\"onSelectQuery(i)\"\n style=\"height: 49px; min-height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n [textareaInlineStyle]=\"'padding: 1rem;'\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons (pinned to bottom) -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n [iconColor]=\"'#414146'\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"text\"\n [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n <svg class=\"cqa-flex-shrink-0\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#6B7280\" />\n </svg>\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n [iconColor]=\"'#FBFCFF'\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i4.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { type: i5.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused"] }, { type: i6.CustomInputComponent, selector: "cqa-custom-input", inputs: ["label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }], directives: [{ type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
699
|
+
StepBuilderDatabaseComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
700
|
+
StepBuilderDatabaseComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderDatabaseComponent, selector: "cqa-step-builder-database", inputs: { dbEnvironmentOptions: "dbEnvironmentOptions", queries: "queries", queryResults: "queryResults", isLoading: "isLoading", initialDbEnvironment: "initialDbEnvironment", editMode: "editMode" }, outputs: { createStep: "createStep", cancelled: "cancelled", runQuery: "runQuery", addQuery: "addQuery", deleteQuery: "deleteQuery", assertionDataChange: "assertionDataChange", selectedQueryIndexChange: "selectedQueryIndexChange", activeTabChange: "activeTabChange" }, host: { classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "assertionsTable", first: true, predicate: ["assertionsTable"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n </h2>\n\n <!-- DB Environment Section (separator line only on bottom) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n </div>\n <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n </cqa-button>\n </div>\n <p class=\"cqa-text-xs cqa-text-gray-500\">\n Uses Database environments from Environments.\n </p>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n (click)=\"onSelectQuery(i)\"\n style=\"height: 49px; min-height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n [textareaInlineStyle]=\"'padding: 1rem;'\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons (pinned to bottom) -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n [iconColor]=\"'#414146'\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"text\"\n [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n <svg class=\"cqa-flex-shrink-0\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#6B7280\" />\n </svg>\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n [iconColor]=\"'#FBFCFF'\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\" class=\"assertions-container\">\n <!-- Header Section -->\n <div class=\"assertions-header\">\n <div class=\"header-content\">\n <h3 class=\"header-title\">Assertions</h3>\n <p class=\"header-subtitle\">Use JSONPath to target values in the query output.</p>\n </div>\n <button \n mat-stroked-button \n color=\"primary\" \n class=\"add-assertion-button\"\n (click)=\"addAssertion()\">\n <mat-icon>add</mat-icon>\n Add Assertion\n </button>\n </div>\n\n <!-- Table Section -->\n <form *ngIf=\"getCurrentAssertionsForm()\" [formGroup]=\"getCurrentAssertionsForm()!\" class=\"assertions-form\">\n <div formArrayName=\"assertions\">\n <div class=\"table-scroll-container\">\n <table mat-table [dataSource]=\"assertionsDataSource\" class=\"assertions-table\" #assertionsTable>\n <!-- JSONPath Column -->\n <ng-container matColumnDef=\"jsonPath\">\n <th mat-header-cell *matHeaderCellDef>jsonPath</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n <input \n matInput \n formControlName=\"jsonPath\"\n placeholder=\"username\"\n required>\n <mat-error *ngIf=\"getFormControlAt(i, 'jsonPath')?.hasError('required') && \n getFormControlAt(i, 'jsonPath')?.touched\">\n JSONPath is required\n </mat-error>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Verification Type Column -->\n <ng-container matColumnDef=\"verificationType\">\n <th mat-header-cell *matHeaderCellDef>verificationType</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n <mat-select formControlName=\"verificationType\">\n <mat-option value=\"equals\">Equals</mat-option>\n <mat-option value=\"not_equals\">Not Equals</mat-option>\n <mat-option value=\"contains\">Contains</mat-option>\n <mat-option value=\"not_contains\">Not Contains</mat-option>\n <mat-option value=\"greater_than\">Greater Than</mat-option>\n <mat-option value=\"less_than\">Less Than</mat-option>\n <mat-option value=\"greater_than_or_equals\">Greater Than Or Equals</mat-option>\n <mat-option value=\"less_than_or_equals\">Less Than Or Equals</mat-option>\n <mat-option value=\"exists\">Exists</mat-option>\n <mat-option value=\"not_exists\">Not Exists</mat-option>\n </mat-select>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Expected Type Column -->\n <ng-container matColumnDef=\"expectedType\">\n <th mat-header-cell *matHeaderCellDef>expectedType</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n <mat-select \n formControlName=\"expectedType\"\n (selectionChange)=\"onExpectedTypeChange(i, $event.value)\">\n <mat-option value=\"string\">String</mat-option>\n <mat-option value=\"number\">Number</mat-option>\n <mat-option value=\"boolean\">Boolean</mat-option>\n <mat-option value=\"object\">Object</mat-option>\n <mat-option value=\"array\">Array</mat-option>\n <mat-option value=\"null\">Null</mat-option>\n </mat-select>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Expected Value Column -->\n <ng-container matColumnDef=\"expectedValue\">\n <th mat-header-cell *matHeaderCellDef>expectedValue</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <!-- String, Array, Object Input -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'string' || getExpectedType(i) === 'array' || getExpectedType(i) === 'object'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <textarea \n *ngIf=\"getExpectedType(i) === 'array' || getExpectedType(i) === 'object'\"\n matInput \n formControlName=\"expectedValue\"\n [placeholder]=\"getExpectedValuePlaceholder(i)\"\n rows=\"3\"\n required></textarea>\n <input \n *ngIf=\"getExpectedType(i) === 'string'\"\n matInput \n formControlName=\"expectedValue\"\n [placeholder]=\"getExpectedValuePlaceholder(i)\"\n required>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n getFormControlAt(i, 'expectedValue')?.touched\">\n Expected value is required\n </mat-error>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('invalidType')\">\n {{ getFormControlAt(i, 'expectedValue')?.errors?.message }}\n </mat-error>\n </mat-form-field>\n\n <!-- Null Type Display -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'null'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <input \n matInput \n formControlName=\"expectedValue\"\n [placeholder]=\"getExpectedValuePlaceholder(i)\"\n [disabled]=\"true\">\n </mat-form-field>\n\n <!-- Number Input -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'number'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <input \n matInput \n type=\"number\"\n formControlName=\"expectedValue\"\n placeholder=\"Expected Value in Number\"\n required>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n getFormControlAt(i, 'expectedValue')?.touched\">\n Expected value is required\n </mat-error>\n </mat-form-field>\n\n <!-- Boolean Select -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'boolean'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <mat-select formControlName=\"expectedValue\" required>\n <mat-option value=\"true\">true</mat-option>\n <mat-option value=\"false\">false</mat-option>\n </mat-select>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n getFormControlAt(i, 'expectedValue')?.touched\">\n Expected value is required\n </mat-error>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Actions Column -->\n <ng-container matColumnDef=\"actions\">\n <th mat-header-cell *matHeaderCellDef></th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <button \n mat-icon-button \n color=\"warn\"\n class=\"delete-button\"\n (click)=\"removeAssertion(i)\"\n [matTooltip]=\"'Delete assertion'\">\n <mat-icon>delete</mat-icon>\n </button>\n </td>\n </ng-container>\n\n <tr mat-header-row *matHeaderRowDef=\"displayedColumns\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: displayedColumns;\"></tr>\n </table>\n </div>\n\n <!-- Empty State -->\n <div *ngIf=\"getAssertionsFormArray().length === 0\" class=\"empty-state\">\n <p>No assertions added. Click \"Add Assertion\" to create one.</p>\n </div>\n </div>\n </form>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i4.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { type: i5.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused"] }, { type: i6.CustomInputComponent, selector: "cqa-custom-input", inputs: ["label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i7.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i8.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i9.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { type: i10.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i11.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex"], exportAs: ["matSelect"] }, { type: i12.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { type: i9.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { type: i9.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }], directives: [{ type: i13.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i13.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i1.FormArrayName, selector: "[formArrayName]", inputs: ["formArrayName"] }, { type: i9.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { type: i9.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { type: i9.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { type: i9.MatCellDef, selector: "[matCellDef]" }, { type: i9.MatCell, selector: "mat-cell, td[mat-cell]" }, { type: i1.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { type: i14.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { type: i10.MatError, selector: "mat-error", inputs: ["id"] }, { type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { type: i15.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { type: i9.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { type: i9.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }] });
|
|
192
701
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, decorators: [{
|
|
193
702
|
type: Component,
|
|
194
|
-
args: [{ selector: 'cqa-step-builder-database', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n </h2>\n\n <!-- DB Environment Section (separator line only on bottom) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n </div>\n <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n </cqa-button>\n </div>\n <p class=\"cqa-text-xs cqa-text-gray-500\">\n Uses Database environments from Environments.\n </p>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n (click)=\"onSelectQuery(i)\"\n style=\"height: 49px; min-height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n [textareaInlineStyle]=\"'padding: 1rem;'\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons (pinned to bottom) -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n [iconColor]=\"'#414146'\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"text\"\n [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n <svg class=\"cqa-flex-shrink-0\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#6B7280\" />\n </svg>\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n [iconColor]=\"'#FBFCFF'\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", styles: [] }]
|
|
195
|
-
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { dbEnvironmentOptions: [{
|
|
703
|
+
args: [{ selector: 'cqa-step-builder-database', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n </h2>\n\n <!-- DB Environment Section (separator line only on bottom) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n </div>\n <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n </cqa-button>\n </div>\n <p class=\"cqa-text-xs cqa-text-gray-500\">\n Uses Database environments from Environments.\n </p>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n (click)=\"onSelectQuery(i)\"\n style=\"height: 49px; min-height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n [textareaInlineStyle]=\"'padding: 1rem;'\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons (pinned to bottom) -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n [iconColor]=\"'#414146'\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"text\"\n [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n <svg class=\"cqa-flex-shrink-0\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#6B7280\" />\n </svg>\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n [iconColor]=\"'#FBFCFF'\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\" class=\"assertions-container\">\n <!-- Header Section -->\n <div class=\"assertions-header\">\n <div class=\"header-content\">\n <h3 class=\"header-title\">Assertions</h3>\n <p class=\"header-subtitle\">Use JSONPath to target values in the query output.</p>\n </div>\n <button \n mat-stroked-button \n color=\"primary\" \n class=\"add-assertion-button\"\n (click)=\"addAssertion()\">\n <mat-icon>add</mat-icon>\n Add Assertion\n </button>\n </div>\n\n <!-- Table Section -->\n <form *ngIf=\"getCurrentAssertionsForm()\" [formGroup]=\"getCurrentAssertionsForm()!\" class=\"assertions-form\">\n <div formArrayName=\"assertions\">\n <div class=\"table-scroll-container\">\n <table mat-table [dataSource]=\"assertionsDataSource\" class=\"assertions-table\" #assertionsTable>\n <!-- JSONPath Column -->\n <ng-container matColumnDef=\"jsonPath\">\n <th mat-header-cell *matHeaderCellDef>jsonPath</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n <input \n matInput \n formControlName=\"jsonPath\"\n placeholder=\"username\"\n required>\n <mat-error *ngIf=\"getFormControlAt(i, 'jsonPath')?.hasError('required') && \n getFormControlAt(i, 'jsonPath')?.touched\">\n JSONPath is required\n </mat-error>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Verification Type Column -->\n <ng-container matColumnDef=\"verificationType\">\n <th mat-header-cell *matHeaderCellDef>verificationType</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n <mat-select formControlName=\"verificationType\">\n <mat-option value=\"equals\">Equals</mat-option>\n <mat-option value=\"not_equals\">Not Equals</mat-option>\n <mat-option value=\"contains\">Contains</mat-option>\n <mat-option value=\"not_contains\">Not Contains</mat-option>\n <mat-option value=\"greater_than\">Greater Than</mat-option>\n <mat-option value=\"less_than\">Less Than</mat-option>\n <mat-option value=\"greater_than_or_equals\">Greater Than Or Equals</mat-option>\n <mat-option value=\"less_than_or_equals\">Less Than Or Equals</mat-option>\n <mat-option value=\"exists\">Exists</mat-option>\n <mat-option value=\"not_exists\">Not Exists</mat-option>\n </mat-select>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Expected Type Column -->\n <ng-container matColumnDef=\"expectedType\">\n <th mat-header-cell *matHeaderCellDef>expectedType</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n <mat-select \n formControlName=\"expectedType\"\n (selectionChange)=\"onExpectedTypeChange(i, $event.value)\">\n <mat-option value=\"string\">String</mat-option>\n <mat-option value=\"number\">Number</mat-option>\n <mat-option value=\"boolean\">Boolean</mat-option>\n <mat-option value=\"object\">Object</mat-option>\n <mat-option value=\"array\">Array</mat-option>\n <mat-option value=\"null\">Null</mat-option>\n </mat-select>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Expected Value Column -->\n <ng-container matColumnDef=\"expectedValue\">\n <th mat-header-cell *matHeaderCellDef>expectedValue</th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <ng-container [formGroupName]=\"i\">\n <!-- String, Array, Object Input -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'string' || getExpectedType(i) === 'array' || getExpectedType(i) === 'object'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <textarea \n *ngIf=\"getExpectedType(i) === 'array' || getExpectedType(i) === 'object'\"\n matInput \n formControlName=\"expectedValue\"\n [placeholder]=\"getExpectedValuePlaceholder(i)\"\n rows=\"3\"\n required></textarea>\n <input \n *ngIf=\"getExpectedType(i) === 'string'\"\n matInput \n formControlName=\"expectedValue\"\n [placeholder]=\"getExpectedValuePlaceholder(i)\"\n required>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n getFormControlAt(i, 'expectedValue')?.touched\">\n Expected value is required\n </mat-error>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('invalidType')\">\n {{ getFormControlAt(i, 'expectedValue')?.errors?.message }}\n </mat-error>\n </mat-form-field>\n\n <!-- Null Type Display -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'null'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <input \n matInput \n formControlName=\"expectedValue\"\n [placeholder]=\"getExpectedValuePlaceholder(i)\"\n [disabled]=\"true\">\n </mat-form-field>\n\n <!-- Number Input -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'number'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <input \n matInput \n type=\"number\"\n formControlName=\"expectedValue\"\n placeholder=\"Expected Value in Number\"\n required>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n getFormControlAt(i, 'expectedValue')?.touched\">\n Expected value is required\n </mat-error>\n </mat-form-field>\n\n <!-- Boolean Select -->\n <mat-form-field \n *ngIf=\"getExpectedType(i) === 'boolean'\"\n appearance=\"outline\" \n class=\"full-width-field\">\n <mat-select formControlName=\"expectedValue\" required>\n <mat-option value=\"true\">true</mat-option>\n <mat-option value=\"false\">false</mat-option>\n </mat-select>\n <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n getFormControlAt(i, 'expectedValue')?.touched\">\n Expected value is required\n </mat-error>\n </mat-form-field>\n </ng-container>\n </td>\n </ng-container>\n\n <!-- Actions Column -->\n <ng-container matColumnDef=\"actions\">\n <th mat-header-cell *matHeaderCellDef></th>\n <td mat-cell *matCellDef=\"let assertion; let i = index\">\n <button \n mat-icon-button \n color=\"warn\"\n class=\"delete-button\"\n (click)=\"removeAssertion(i)\"\n [matTooltip]=\"'Delete assertion'\">\n <mat-icon>delete</mat-icon>\n </button>\n </td>\n </ng-container>\n\n <tr mat-header-row *matHeaderRowDef=\"displayedColumns\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: displayedColumns;\"></tr>\n </table>\n </div>\n\n <!-- Empty State -->\n <div *ngIf=\"getAssertionsFormArray().length === 0\" class=\"empty-state\">\n <p>No assertions added. Click \"Add Assertion\" to create one.</p>\n </div>\n </div>\n </form>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", styles: [] }]
|
|
704
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { dbEnvironmentOptions: [{
|
|
196
705
|
type: Input
|
|
197
706
|
}], queries: [{
|
|
198
707
|
type: Input
|
|
@@ -214,5 +723,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
214
723
|
type: Output
|
|
215
724
|
}], deleteQuery: [{
|
|
216
725
|
type: Output
|
|
726
|
+
}], assertionDataChange: [{
|
|
727
|
+
type: Output
|
|
728
|
+
}], selectedQueryIndexChange: [{
|
|
729
|
+
type: Output
|
|
730
|
+
}], activeTabChange: [{
|
|
731
|
+
type: Output
|
|
732
|
+
}], assertionsTable: [{
|
|
733
|
+
type: ViewChild,
|
|
734
|
+
args: ['assertionsTable']
|
|
217
735
|
}] } });
|
|
218
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-database.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.ts","../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges } from '@angular/core';\nimport { FormBuilder, FormGroup, FormArray, FormControl, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\n\nexport interface DatabaseQuery {\n  id: string;\n  query: string;\n  variable: string;\n  status?: 'passed' | 'failed' | 'pending';\n}\n\nexport interface DatabaseFormData {\n  dbEnvironment: string;\n  queries: DatabaseQuery[];\n}\n\n@Component({\n  selector: 'cqa-step-builder-database',\n  templateUrl: './step-builder-database.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderDatabaseComponent implements OnInit, OnChanges {\n  /** Options for DB environment dropdown */\n  @Input() dbEnvironmentOptions: SelectOption[] = [];\n\n  /** Initial queries list */\n  @Input() queries: DatabaseQuery[] = [];\n\n  /** Query results data */\n  @Input() queryResults: any[] = [];\n\n  /** Loading state */\n  @Input() isLoading: boolean = false;\n\n  /** Initial DB environment for edit mode */\n  @Input() initialDbEnvironment?: string;\n\n  /** Whether in edit mode */\n  @Input() editMode?: boolean;\n\n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<DatabaseFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Emit when query is run */\n  @Output() runQuery = new EventEmitter<{ query: string; variable: string; dbEnvironment: string }>();\n\n  /** Emit when query is added */\n  @Output() addQuery = new EventEmitter<void>();\n\n  /** Emit when query is deleted */\n  @Output() deleteQuery = new EventEmitter<string>();\n\n  databaseForm: FormGroup;\n  selectedTab: 'output' | 'verification' = 'output';\n  selectedQueryIndex: number = 0;\n\n  constructor(private fb: FormBuilder) {\n    this.databaseForm = this.fb.group({\n      dbEnvironment: ['', Validators.required],\n      queries: this.fb.array([])\n    });\n  }\n\n  ngOnInit(): void {\n    this.initializeFormWithInitialValues();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['queries'] && !changes['queries'].firstChange) {\n      // Rebuild form array when queries change\n      const queriesArray = this.databaseForm.get('queries') as FormArray;\n      while (queriesArray.length !== 0) {\n        queriesArray.removeAt(0);\n      }\n      this.queries.forEach(query => {\n        this.addQueryToForm(query);\n      });\n    }\n    if (changes['initialDbEnvironment'] && changes['initialDbEnvironment'].currentValue) {\n      this.databaseForm.patchValue({ dbEnvironment: changes['initialDbEnvironment'].currentValue });\n    }\n  }\n\n  private initializeFormWithInitialValues(): void {\n    // Set initial DB environment if provided\n    if (this.initialDbEnvironment) {\n      this.databaseForm.patchValue({ dbEnvironment: this.initialDbEnvironment });\n    }\n\n    // Initialize with queries if provided, otherwise add default query\n    if (this.queries.length === 0) {\n      this.addNewQuery();\n    } else {\n      this.queries.forEach((query, index) => {\n        this.addQueryToForm(query);\n        if (index === 0) {\n          this.selectedQueryIndex = 0;\n        }\n      });\n    }\n  }\n\n  get queriesFormArray(): FormArray {\n    return this.databaseForm.get('queries') as FormArray;\n  }\n\n  addNewQuery(): void {\n    const newQuery: DatabaseQuery = {\n      id: `query_${Date.now()}`,\n      query: '',\n      variable: '',\n      status: 'pending'\n    };\n    this.queries.push(newQuery);\n    this.addQueryToForm(newQuery);\n    this.selectedQueryIndex = this.queries.length - 1;\n    this.addQuery.emit();\n  }\n\n  private addQueryToForm(query: DatabaseQuery): void {\n    const queryGroup = this.fb.group({\n      id: [query.id],\n      query: [query.query, Validators.required],\n      variable: [query.variable, Validators.required]\n    });\n    this.queriesFormArray.push(queryGroup);\n  }\n\n  deleteQueryById(queryId: string): void {\n    const index = this.queries.findIndex(q => q.id === queryId);\n    if (index !== -1) {\n      this.queries.splice(index, 1);\n      this.queriesFormArray.removeAt(index);\n      if (this.selectedQueryIndex >= this.queries.length) {\n        this.selectedQueryIndex = Math.max(0, this.queries.length - 1);\n      }\n      this.deleteQuery.emit(queryId);\n    }\n  }\n\n  getDbEnvironmentConfig(): DynamicSelectFieldConfig {\n    return {\n      key: 'dbEnvironment',\n      placeholder: 'Select DB environment',\n      multiple: false,\n      searchable: false,\n      options: this.dbEnvironmentOptions\n    };\n  }\n\n  getCurrentQuery(): DatabaseQuery | null {\n    if (this.queries.length > 0 && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queries.length) {\n      return this.queries[this.selectedQueryIndex];\n    }\n    return null;\n  }\n\n  getCurrentQueryFormGroup(): FormGroup | null {\n    if (this.queriesFormArray && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queriesFormArray.length) {\n      return this.queriesFormArray.at(this.selectedQueryIndex) as FormGroup;\n    }\n    return null;\n  }\n\n  onRunQuery(): void {\n    const dbEnvironment = this.databaseForm.get('dbEnvironment')?.value;\n    const currentQuery = this.getCurrentQuery();\n    const queryFormGroup = this.getCurrentQueryFormGroup();\n\n    if (currentQuery && queryFormGroup && dbEnvironment) {\n      const query = queryFormGroup.get('query')?.value || '';\n      const variable = queryFormGroup.get('variable')?.value || '';\n      \n      // Update the query object\n      currentQuery.query = query;\n      currentQuery.variable = variable;\n\n      this.runQuery.emit({\n        query,\n        variable,\n        dbEnvironment\n      });\n    }\n  }\n\n  onSelectQuery(index: number): void {\n    this.selectedQueryIndex = index;\n  }\n\n  onTabChange(tab: 'output' | 'verification'): void {\n    this.selectedTab = tab;\n  }\n\n  onCopyResults(): void {\n    // Copy query results to clipboard\n    if (this.queryResults && this.queryResults.length > 0) {\n      const text = JSON.stringify(this.queryResults, null, 2);\n      navigator.clipboard.writeText(text).then(() => {\n        // Could emit an event or show a toast notification\n      });\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    if (this.databaseForm.valid) {\n      const formValue = this.databaseForm.value;\n      const stepData: DatabaseFormData = {\n        dbEnvironment: formValue.dbEnvironment,\n        queries: this.queries.map((query, index) => {\n          const queryFormGroup = this.queriesFormArray.at(index) as FormGroup;\n          return {\n            id: query.id,\n            query: queryFormGroup.get('query')?.value || '',\n            variable: queryFormGroup.get('variable')?.value || '',\n            status: query.status\n          };\n        })\n      };\n      this.createStep.emit(stepData);\n    }\n  }\n\n  getTableColumns(): string[] {\n    if (!this.queryResults || this.queryResults.length === 0) {\n      return [];\n    }\n    return Object.keys(this.queryResults[0]);\n  }\n}\n\n","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n  </h2>\n\n  <!-- DB Environment Section (separator line only on bottom) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n      DB Environment <span class=\"cqa-text-red-500\">*</span>\n    </label>\n    <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n      <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n        <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n        </cqa-dynamic-select>\n      </div>\n      <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n        [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n        [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n      </cqa-button>\n    </div>\n    <p class=\"cqa-text-xs cqa-text-gray-500\">\n      Uses Database environments from Environments.\n    </p>\n  </div>\n\n  <!-- Main Content: Two Column Layout -->\n  <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n    <!-- Left Panel: Query List -->\n    <div\n      class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n      <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n        <div *ngFor=\"let query of queries; let i = index\"\n          class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n          [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n          [class.cqa-border]=\"selectedQueryIndex === i\"\n          [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n          [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n          (click)=\"onSelectQuery(i)\"\n          style=\"height: 49px; min-height: 49px;\">\n          <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n          <cqa-badge *ngIf=\"query.status\"\n            [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n            [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n            size=\"small\">\n          </cqa-badge>\n        </div>\n      </div>\n    </div>\n\n    <!-- Right Panel: Query Editor -->\n    <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n      <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n          <div class=\"\">\n            <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n          </div>\n\n          <div class=\"cqa-flex cqa-justify-between\">\n            <!-- SQL Query Textarea -->\n            <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n              <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n                [textareaInlineStyle]=\"'padding: 1rem;'\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n              </cqa-custom-textarea>\n            </div>\n          </div>\n        </div>\n\n        <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n          <!-- Variable Input -->\n          <div class=\"cqa-flex cqa-gap-4\">\n            <div class=\"cqa-flex-1\">\n              <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n                Variable\n              </label>\n              <cqa-custom-input [placeholder]=\"'Variable name'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n              </cqa-custom-input>\n              <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n                Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n              </p>\n            </div>\n          </div>\n          <!-- Action Buttons (pinned to bottom) -->\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n            <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n              [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n              [iconColor]=\"'#414146'\">\n            </cqa-button>\n            <cqa-button type=\"button\" variant=\"text\"\n              [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n              [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n              [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n              <svg class=\"cqa-flex-shrink-0\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                <path\n                  d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                  fill=\"#6B7280\" />\n              </svg>\n            </cqa-button>\n            <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n              [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n              [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n              [iconColor]=\"'#FBFCFF'\">\n            </cqa-button>\n          </div>\n        </div>\n      </div>\n      <!-- Results Section -->\n      <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n        <!-- Tabs -->\n        <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n            Output\n          </button>\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n            Verification\n          </button>\n        </div>\n\n        <!-- Tab Content -->\n        <div class=\"cqa-p-4\">\n          <!-- Output Tab -->\n          <div *ngIf=\"selectedTab === 'output'\">\n            <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n              <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n              <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n                (clicked)=\"onCopyResults()\">\n              </cqa-button>\n            </div>\n\n            <!-- Results Table -->\n            <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n              <table class=\"cqa-w-full cqa-border-collapse\">\n                <thead>\n                  <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n                    <th *ngFor=\"let key of getTableColumns()\"\n                      class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n                      {{ key }}\n                    </th>\n                  </tr>\n                </thead>\n                <tbody>\n                  <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n                    <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n                      {{ row[key] }}\n                    </td>\n                  </tr>\n                </tbody>\n              </table>\n            </div>\n\n            <div *ngIf=\"!queryResults || queryResults.length === 0\"\n              class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n              No results yet. Run a query to see results.\n            </div>\n          </div>\n\n          <!-- Verification Tab -->\n          <div *ngIf=\"selectedTab === 'verification'\">\n            <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n\n\n  <!-- Action Buttons -->\n  <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n    <cqa-button class=\"cqa-w-1/2\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>"]}
|
|
736
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-database.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.ts","../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges, ViewChild, ChangeDetectorRef, OnDestroy } from '@angular/core';\nimport { FormBuilder, FormGroup, FormArray, FormControl, Validators, AbstractControl } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\nimport { MatTable } from '@angular/material/table';\nimport { Subscription } from 'rxjs';\n\nexport interface DatabaseQuery {\n  id: string;\n  query: string;\n  variable: string;\n  status?: 'passed' | 'failed' | 'pending';\n  assertionData?: any[]; // Assertion data for verification\n}\n\nexport interface DatabaseFormData {\n  dbEnvironment: string;\n  queries: DatabaseQuery[];\n}\n\n@Component({\n  selector: 'cqa-step-builder-database',\n  templateUrl: './step-builder-database.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderDatabaseComponent implements OnInit, OnChanges, OnDestroy {\n  /** Options for DB environment dropdown */\n  @Input() dbEnvironmentOptions: SelectOption[] = [];\n\n  /** Initial queries list */\n  @Input() queries: DatabaseQuery[] = [];\n\n  /** Query results data */\n  @Input() queryResults: any[] = [];\n\n  /** Loading state */\n  @Input() isLoading: boolean = false;\n\n  /** Initial DB environment for edit mode */\n  @Input() initialDbEnvironment?: string;\n\n  /** Whether in edit mode */\n  @Input() editMode?: boolean;\n\n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<DatabaseFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Emit when query is run */\n  @Output() runQuery = new EventEmitter<{ query: string; variable: string; dbEnvironment: string }>();\n\n  /** Emit when query is added */\n  @Output() addQuery = new EventEmitter<void>();\n\n  /** Emit when query is deleted */\n  @Output() deleteQuery = new EventEmitter<string>();\n\n  /** Emit when assertion data changes for a query */\n  @Output() assertionDataChange = new EventEmitter<{ queryIndex: number; assertionData: any[] }>();\n\n  /** Emit when selected query index changes */\n  @Output() selectedQueryIndexChange = new EventEmitter<number>();\n\n  /** Emit when active tab changes */\n  @Output() activeTabChange = new EventEmitter<'output' | 'verification'>();\n\n  databaseForm: FormGroup;\n  selectedTab: 'output' | 'verification' = 'output';\n  selectedQueryIndex: number = 0;\n\n  // Assertions form management - one form per query\n  assertionsForms: Map<number, FormGroup> = new Map();\n  displayedColumns: string[] = ['jsonPath', 'verificationType', 'expectedType', 'expectedValue', 'actions'];\n  defaultExpectedType = 'string';\n  private formSubscriptions: Map<number, Subscription> = new Map();\n  @ViewChild('assertionsTable') assertionsTable?: MatTable<any>;\n\n  constructor(\n    private fb: FormBuilder,\n    private cdr: ChangeDetectorRef\n  ) {\n    this.databaseForm = this.fb.group({\n      dbEnvironment: ['', Validators.required],\n      queries: this.fb.array([])\n    });\n  }\n\n  ngOnInit(): void {\n    this.initializeFormWithInitialValues();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['queries'] && !changes['queries'].firstChange) {\n      // Rebuild form array when queries change\n      const queriesArray = this.databaseForm.get('queries') as FormArray;\n      while (queriesArray.length !== 0) {\n        queriesArray.removeAt(0);\n      }\n      // Clear old assertions forms\n      this.assertionsForms.forEach((form, index) => {\n        if (this.formSubscriptions.has(index)) {\n          this.formSubscriptions.get(index)!.unsubscribe();\n        }\n      });\n      this.assertionsForms.clear();\n      this.formSubscriptions.clear();\n\n      // Rebuild queries and assertions forms\n      this.queries.forEach((query, index) => {\n        // Ensure assertion data is initialized\n        if (!query.assertionData) {\n          query.assertionData = [];\n        }\n        this.addQueryToForm(query);\n        // Initialize assertions form for each query\n        this.ensureAssertionDataInitialized(index);\n      });\n\n      // Ensure selected query index is valid\n      if (this.selectedQueryIndex >= this.queries.length) {\n        this.selectedQueryIndex = Math.max(0, this.queries.length - 1);\n      }\n      this.cdr.detectChanges();\n    }\n    if (changes['initialDbEnvironment'] && changes['initialDbEnvironment'].currentValue) {\n      this.databaseForm.patchValue({ dbEnvironment: changes['initialDbEnvironment'].currentValue });\n    }\n  }\n\n  private initializeFormWithInitialValues(): void {\n    // Set initial DB environment if provided\n    if (this.initialDbEnvironment) {\n      this.databaseForm.patchValue({ dbEnvironment: this.initialDbEnvironment });\n    }\n\n    // Initialize with queries if provided, otherwise add default query\n    if (this.queries.length === 0) {\n      this.addNewQuery();\n    } else {\n      this.queries.forEach((query, index) => {\n        // Ensure assertion data is initialized\n        if (!query.assertionData) {\n          query.assertionData = [];\n        }\n        this.addQueryToForm(query);\n        // Initialize assertions form for each query\n        this.ensureAssertionDataInitialized(index);\n        if (index === 0) {\n          this.selectedQueryIndex = 0;\n          this.selectedQueryIndexChange.emit(0);\n        }\n      });\n    }\n  }\n\n  get queriesFormArray(): FormArray {\n    return this.databaseForm.get('queries') as FormArray;\n  }\n\n  addNewQuery(): void {\n    const newQuery: DatabaseQuery = {\n      id: `query_${Date.now()}`,\n      query: '',\n      variable: '',\n      status: 'pending',\n      assertionData: [] // Initialize with empty assertion data\n    };\n    this.queries.push(newQuery);\n    this.addQueryToForm(newQuery);\n    const newIndex = this.queries.length - 1;\n    this.selectedQueryIndex = newIndex;\n    // Initialize assertions form for the new query\n    this.ensureAssertionDataInitialized(newIndex);\n    this.selectedQueryIndexChange.emit(this.selectedQueryIndex);\n    this.addQuery.emit();\n    this.cdr.detectChanges();\n  }\n\n  private addQueryToForm(query: DatabaseQuery): void {\n    const queryGroup = this.fb.group({\n      id: [query.id],\n      query: [query.query, Validators.required],\n      variable: [query.variable, Validators.required]\n    });\n    this.queriesFormArray.push(queryGroup);\n  }\n\n  deleteQueryById(queryId: string): void {\n    const index = this.queries.findIndex(q => q.id === queryId);\n    if (index !== -1) {\n      // Clean up assertions form and subscription for this query\n      if (this.formSubscriptions.has(index)) {\n        this.formSubscriptions.get(index)!.unsubscribe();\n        this.formSubscriptions.delete(index);\n      }\n      this.assertionsForms.delete(index);\n\n      // Reindex remaining forms and subscriptions\n      const formsToReindex = new Map<number, FormGroup>();\n      const subscriptionsToReindex = new Map<number, Subscription>();\n      \n      this.assertionsForms.forEach((form, oldIndex) => {\n        if (oldIndex > index) {\n          formsToReindex.set(oldIndex - 1, form);\n        } else if (oldIndex < index) {\n          formsToReindex.set(oldIndex, form);\n        }\n      });\n\n      this.formSubscriptions.forEach((sub, oldIndex) => {\n        if (oldIndex > index) {\n          subscriptionsToReindex.set(oldIndex - 1, sub);\n        } else if (oldIndex < index) {\n          subscriptionsToReindex.set(oldIndex, sub);\n        }\n      });\n\n      this.assertionsForms = formsToReindex;\n      this.formSubscriptions = subscriptionsToReindex;\n\n      this.queries.splice(index, 1);\n      this.queriesFormArray.removeAt(index);\n      if (this.selectedQueryIndex >= this.queries.length) {\n        this.selectedQueryIndex = Math.max(0, this.queries.length - 1);\n        // Ensure assertions form is initialized for the new selected query\n        if (this.selectedQueryIndex >= 0) {\n          this.ensureAssertionDataInitialized(this.selectedQueryIndex);\n        }\n      }\n      this.deleteQuery.emit(queryId);\n      this.cdr.detectChanges();\n    }\n  }\n\n  getDbEnvironmentConfig(): DynamicSelectFieldConfig {\n    return {\n      key: 'dbEnvironment',\n      placeholder: 'Select DB environment',\n      multiple: false,\n      searchable: false,\n      options: this.dbEnvironmentOptions\n    };\n  }\n\n  getCurrentQuery(): DatabaseQuery | null {\n    if (this.queries.length > 0 && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queries.length) {\n      return this.queries[this.selectedQueryIndex];\n    }\n    return null;\n  }\n\n  getCurrentQueryFormGroup(): FormGroup | null {\n    if (this.queriesFormArray && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queriesFormArray.length) {\n      return this.queriesFormArray.at(this.selectedQueryIndex) as FormGroup;\n    }\n    return null;\n  }\n\n  onRunQuery(): void {\n    const dbEnvironment = this.databaseForm.get('dbEnvironment')?.value;\n    const currentQuery = this.getCurrentQuery();\n    const queryFormGroup = this.getCurrentQueryFormGroup();\n\n    if (currentQuery && queryFormGroup && dbEnvironment) {\n      const query = queryFormGroup.get('query')?.value || '';\n      const variable = queryFormGroup.get('variable')?.value || '';\n      \n      // Update the query object\n      currentQuery.query = query;\n      currentQuery.variable = variable;\n\n      this.runQuery.emit({\n        query,\n        variable,\n        dbEnvironment\n      });\n    }\n  }\n\n  onSelectQuery(index: number): void {\n    // Ensure assertion data is initialized for the selected query\n    this.ensureAssertionDataInitialized(index);\n    this.selectedQueryIndex = index;\n    this.selectedQueryIndexChange.emit(index);\n    // Trigger change detection to ensure assertions form is displayed\n    this.cdr.detectChanges();\n  }\n\n  /**\n   * Get the currently selected query index\n   */\n  getSelectedQueryIndex(): number {\n    return this.selectedQueryIndex;\n  }\n\n  /**\n   * Get the currently active tab\n   */\n  getActiveTab(): 'output' | 'verification' {\n    return this.selectedTab;\n  }\n\n  onTabChange(tab: 'output' | 'verification'): void {\n    this.selectedTab = tab;\n    this.activeTabChange.emit(tab);\n    // Ensure assertions form is initialized when switching to verification tab\n    if (tab === 'verification') {\n      this.ensureAssertionDataInitialized(this.selectedQueryIndex);\n      this.cdr.detectChanges();\n    }\n  }\n\n  onCopyResults(): void {\n    // Copy query results to clipboard\n    if (this.queryResults && this.queryResults.length > 0) {\n      const text = JSON.stringify(this.queryResults, null, 2);\n      navigator.clipboard.writeText(text).then(() => {\n        // Could emit an event or show a toast notification\n      });\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    if (this.databaseForm.valid) {\n      const formValue = this.databaseForm.value;\n      const stepData: DatabaseFormData = {\n        dbEnvironment: formValue.dbEnvironment,\n        queries: this.queries.map((query, index) => {\n          const queryFormGroup = this.queriesFormArray.at(index) as FormGroup;\n          return {\n            id: query.id,\n            query: queryFormGroup.get('query')?.value || '',\n            variable: queryFormGroup.get('variable')?.value || '',\n            status: query.status,\n            assertionData: query.assertionData || [] // Include assertion data\n          };\n        })\n      };\n      this.createStep.emit(stepData);\n    }\n  }\n\n  getTableColumns(): string[] {\n    if (!this.queryResults || this.queryResults.length === 0) {\n      return [];\n    }\n    return Object.keys(this.queryResults[0]);\n  }\n\n  /**\n   * Get assertion data for a specific query\n   */\n  getAssertionDataForQuery(queryIndex: number): any[] {\n    if (this.queries[queryIndex] && this.queries[queryIndex].assertionData) {\n      return this.queries[queryIndex].assertionData;\n    }\n    return [];\n  }\n\n  /**\n   * Handle assertion data change from external component\n   */\n  onAssertionDataChange(assertionData: any[], queryIndex: number): void {\n    if (this.queries[queryIndex]) {\n      // Create a deep copy to ensure each query has its own data\n      if (assertionData && assertionData.length > 0) {\n        this.queries[queryIndex].assertionData = assertionData.map((item: any) => {\n          // Ensure expectedValue is properly formatted\n          const expectedType = item.expectedType || 'string';\n          let expectedValue = item.expectedValue;\n\n          // For object/array types, ensure expectedValue is an object/array, not a string\n          if (expectedType === 'object' && typeof expectedValue === 'string' && expectedValue.trim()) {\n            try {\n              expectedValue = JSON.parse(expectedValue);\n            } catch (e) {\n              // If parsing fails, keep as string\n            }\n          } else if (expectedType === 'array' && typeof expectedValue === 'string' && expectedValue.trim()) {\n            try {\n              expectedValue = JSON.parse(expectedValue);\n            } catch (e) {\n              // If parsing fails, keep as string\n            }\n          } else if (expectedType === 'number' && typeof expectedValue === 'string') {\n            expectedValue = expectedValue !== '' ? Number(expectedValue) : null;\n          } else if (expectedType === 'boolean') {\n            if (expectedValue === 'true' || expectedValue === true) {\n              expectedValue = true;\n            } else if (expectedValue === 'false' || expectedValue === false) {\n              expectedValue = false;\n            }\n          }\n\n          return {\n            jsonPath: item.jsonPath || '',\n            verificationType: item.verificationType || 'equals',\n            expectedType: expectedType,\n            expectedValue: expectedValue\n          };\n        });\n      } else {\n        this.queries[queryIndex].assertionData = [];\n      }\n\n      // Emit the change to parent\n      this.assertionDataChange.emit({\n        queryIndex: queryIndex,\n        assertionData: this.queries[queryIndex].assertionData || []\n      });\n    }\n  }\n\n  /**\n   * Initialize assertion data for a query if it doesn't exist\n   */\n  private ensureAssertionDataInitialized(queryIndex: number): void {\n    if (this.queries[queryIndex] && !this.queries[queryIndex].assertionData) {\n      this.queries[queryIndex].assertionData = [];\n    }\n    // Initialize assertions form for this query if it doesn't exist\n    if (!this.assertionsForms.has(queryIndex)) {\n      this.initializeAssertionsForm(queryIndex);\n    }\n  }\n\n  /**\n   * Initialize assertions form for a specific query\n   */\n  private initializeAssertionsForm(queryIndex: number): void {\n    const assertionsArray = this.fb.array([]);\n    const assertionsForm = this.fb.group({\n      assertions: assertionsArray\n    });\n    this.assertionsForms.set(queryIndex, assertionsForm);\n\n    // Load existing assertions if they exist\n    const query = this.queries[queryIndex];\n    if (query && query.assertionData && query.assertionData.length > 0) {\n      this.loadAssertionsForQuery(queryIndex);\n    }\n\n    // Subscribe to form changes\n    this.subscribeToAssertionsFormChanges(queryIndex);\n  }\n\n  /**\n   * Get assertions form for current query\n   */\n  getCurrentAssertionsForm(): FormGroup | null {\n    if (this.assertionsForms.has(this.selectedQueryIndex)) {\n      return this.assertionsForms.get(this.selectedQueryIndex)!;\n    }\n    // Initialize if it doesn't exist\n    this.ensureAssertionDataInitialized(this.selectedQueryIndex);\n    return this.assertionsForms.get(this.selectedQueryIndex) || null;\n  }\n\n  /**\n   * Get assertions form array for current query\n   */\n  getAssertionsFormArray(): FormArray {\n    const form = this.getCurrentAssertionsForm();\n    if (form) {\n      return form.get('assertions') as FormArray;\n    }\n    return this.fb.array([]);\n  }\n\n  /**\n   * Load assertions into form for a specific query\n   */\n  private loadAssertionsForQuery(queryIndex: number): void {\n    const form = this.assertionsForms.get(queryIndex);\n    if (!form) return;\n\n    const formArray = form.get('assertions') as FormArray;\n    const query = this.queries[queryIndex];\n    if (!query || !query.assertionData) return;\n\n    // Clear existing form array\n    while (formArray.length !== 0) {\n      formArray.removeAt(0);\n    }\n\n    // Load assertions\n    query.assertionData.forEach((assertion: any) => {\n      const expectedType = assertion.expectedType || 'string';\n      let expectedValue: any = '';\n\n      if (expectedType === 'null') {\n        expectedValue = null;\n      } else if (expectedType === 'object' || expectedType === 'array') {\n        if (typeof assertion.expectedValue === 'object' && assertion.expectedValue !== null) {\n          expectedValue = JSON.stringify(assertion.expectedValue, null, 2);\n        } else {\n          expectedValue = assertion.expectedValue || '';\n        }\n      } else if (expectedType === 'boolean') {\n        expectedValue = (assertion.expectedValue === true || assertion.expectedValue === 'true') ? 'true' : 'false';\n      } else if (expectedType === 'number') {\n        expectedValue = assertion.expectedValue !== null && assertion.expectedValue !== undefined\n          ? String(assertion.expectedValue)\n          : '';\n      } else {\n        expectedValue = assertion.expectedValue !== null && assertion.expectedValue !== undefined\n          ? String(assertion.expectedValue)\n          : '';\n      }\n\n      const formGroup = this.fb.group({\n        jsonPath: [assertion.jsonPath || '', Validators.required],\n        verificationType: [assertion.verificationType || 'equals'],\n        expectedType: [expectedType],\n        expectedValue: [expectedValue, [Validators.required, this.expectedValueValidator]]\n      });\n\n      formArray.push(formGroup);\n    });\n\n    this.cdr.detectChanges();\n  }\n\n  /**\n   * Subscribe to assertions form changes\n   */\n  private subscribeToAssertionsFormChanges(queryIndex: number): void {\n    const form = this.assertionsForms.get(queryIndex);\n    if (!form) return;\n\n    // Unsubscribe from previous subscription if exists\n    if (this.formSubscriptions.has(queryIndex)) {\n      this.formSubscriptions.get(queryIndex)!.unsubscribe();\n    }\n\n    const formArray = form.get('assertions') as FormArray;\n    const subscription = formArray.valueChanges.subscribe(() => {\n      this.emitAssertionDataForQuery(queryIndex);\n    });\n\n    this.formSubscriptions.set(queryIndex, subscription);\n  }\n\n  /**\n   * Emit assertion data for a specific query\n   */\n  private emitAssertionDataForQuery(queryIndex: number): void {\n    const form = this.assertionsForms.get(queryIndex);\n    if (!form) return;\n\n    const formArray = form.get('assertions') as FormArray;\n    const assertionData = formArray.controls.map((control) => {\n      if (!(control instanceof FormGroup)) return null;\n\n      const value = control.value;\n      const expectedType = value.expectedType || 'string';\n      let expectedValue: any = value.expectedValue;\n\n      // Parse expectedValue based on type\n      if (expectedType === 'null') {\n        expectedValue = null;\n      } else if (expectedType === 'number') {\n        expectedValue = expectedValue !== '' ? Number(expectedValue) : null;\n      } else if (expectedType === 'boolean') {\n        expectedValue = (expectedValue === 'true' || expectedValue === true);\n      } else if (expectedType === 'array' || expectedType === 'object') {\n        if (typeof expectedValue === 'string' && expectedValue.trim()) {\n          try {\n            expectedValue = JSON.parse(expectedValue);\n          } catch (e) {\n            // Keep as string if parsing fails\n          }\n        }\n      } else {\n        expectedValue = String(expectedValue || '');\n      }\n\n      return {\n        jsonPath: (value.jsonPath || '').trim(),\n        verificationType: value.verificationType || 'equals',\n        expectedType: expectedType,\n        expectedValue: expectedValue\n      };\n    }).filter(item => item !== null && item!.jsonPath) as any[];\n\n    // Update query's assertion data\n    if (this.queries[queryIndex]) {\n      this.queries[queryIndex].assertionData = assertionData;\n    }\n\n    // Emit to parent\n    this.assertionDataChange.emit({\n      queryIndex: queryIndex,\n      assertionData: assertionData\n    });\n  }\n\n  /**\n   * Add a new assertion\n   */\n  addAssertion(): void {\n    const formArray = this.getAssertionsFormArray();\n    const formGroup = this.fb.group({\n      jsonPath: ['', Validators.required],\n      verificationType: ['equals'],\n      expectedType: [this.defaultExpectedType],\n      expectedValue: ['', [Validators.required, this.expectedValueValidator]]\n    });\n    formArray.push(formGroup);\n    this.cdr.detectChanges();\n    if (this.assertionsTable) {\n      this.assertionsTable.renderRows();\n    }\n  }\n\n  /**\n   * Remove an assertion\n   */\n  removeAssertion(index: number): void {\n    const formArray = this.getAssertionsFormArray();\n    formArray.removeAt(index);\n    this.cdr.detectChanges();\n    if (this.assertionsTable) {\n      this.assertionsTable.renderRows();\n    }\n    this.emitAssertionDataForQuery(this.selectedQueryIndex);\n  }\n\n  /**\n   * Get form control at specific index\n   */\n  getFormControlAt(index: number, controlName: string): AbstractControl | null {\n    const formArray = this.getAssertionsFormArray();\n    if (index < 0 || index >= formArray.length) return null;\n    const formGroup = formArray.at(index) as FormGroup;\n    return formGroup?.get(controlName) || null;\n  }\n\n  /**\n   * Get expected type for an assertion\n   */\n  getExpectedType(index: number): string {\n    const formArray = this.getAssertionsFormArray();\n    if (index < 0 || index >= formArray.length) return 'string';\n    const formGroup = formArray.at(index) as FormGroup;\n    return formGroup?.get('expectedType')?.value || 'string';\n  }\n\n  /**\n   * Get expected value placeholder\n   */\n  getExpectedValuePlaceholder(index: number): string {\n    const expectedType = this.getExpectedType(index);\n    switch (expectedType) {\n      case 'string': return 'Expected Value in String';\n      case 'array': return 'Expected Value in Array (JSON format)';\n      case 'object': return 'Expected Value in Object (JSON format)';\n      case 'null': return 'No value required for null type';\n      default: return 'Expected Value';\n    }\n  }\n\n  /**\n   * Handle expected type change\n   */\n  onExpectedTypeChange(index: number, value: string): void {\n    const formArray = this.getAssertionsFormArray();\n    if (index < 0 || index >= formArray.length) return;\n\n    const formGroup = formArray.at(index) as FormGroup;\n    const expectedValueControl = formGroup.get('expectedValue');\n    if (!expectedValueControl) return;\n\n    let newValue: any = '';\n    if (value === 'boolean') {\n      newValue = 'true';\n    } else if (value === 'null') {\n      newValue = null;\n    }\n\n    expectedValueControl.clearValidators();\n    expectedValueControl.setValue(newValue, { emitEvent: false });\n\n    if (value === 'null') {\n      expectedValueControl.setValidators([this.expectedValueValidator]);\n    } else if (value === 'object' || value === 'array') {\n      expectedValueControl.setValidators([Validators.required]);\n    } else {\n      expectedValueControl.setValidators([Validators.required, this.expectedValueValidator]);\n    }\n\n    setTimeout(() => {\n      expectedValueControl.updateValueAndValidity({ emitEvent: false });\n    }, 100);\n  }\n\n  /**\n   * Expected value validator\n   */\n  expectedValueValidator = (control: AbstractControl): { [key: string]: any } | null => {\n    const expectedType = control.parent?.get('expectedType')?.value;\n    if (!expectedType || expectedType === 'null') return null;\n\n    const value = control.value;\n    if (!value && value !== 0 && value !== false) return null;\n\n    if (expectedType === 'number') {\n      if (isNaN(Number(value))) {\n        return { invalidType: true, message: 'Invalid number value' };\n      }\n    } else if (expectedType === 'boolean') {\n      if (value !== 'true' && value !== 'false' && value !== true && value !== false) {\n        return { invalidType: true, message: 'Invalid boolean value' };\n      }\n    } else if (expectedType === 'array') {\n      if (typeof value === 'string' && value.trim()) {\n        try {\n          const parsed = JSON.parse(value);\n          if (!Array.isArray(parsed)) {\n            return { invalidType: true, message: 'Value must be a valid JSON array' };\n          }\n        } catch (e) {\n          return { invalidType: true, message: 'Invalid Array value - must be valid JSON' };\n        }\n      }\n    } else if (expectedType === 'object') {\n      if (typeof value === 'string' && value.trim() && value.trim() !== '{}') {\n        try {\n          const parsed = JSON.parse(value);\n          if (parsed === null || typeof parsed !== 'object' || Array.isArray(parsed)) {\n            return { invalidType: true, message: 'Value must be a valid JSON object' };\n          }\n        } catch (e) {\n          return { invalidType: true, message: 'Invalid Object value - must be valid JSON' };\n        }\n      }\n    }\n\n    return null;\n  };\n\n  /**\n   * Get assertions data source for table\n   */\n  get assertionsDataSource() {\n    return this.getAssertionsFormArray().controls;\n  }\n\n  /**\n   * Track by function for table rows\n   */\n  trackByIndex(index: number): any {\n    return index;\n  }\n\n  ngOnDestroy(): void {\n    // Clean up subscriptions\n    this.formSubscriptions.forEach(sub => sub.unsubscribe());\n    this.formSubscriptions.clear();\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n  </h2>\n\n  <!-- DB Environment Section (separator line only on bottom) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n      DB Environment <span class=\"cqa-text-red-500\">*</span>\n    </label>\n    <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n      <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n        <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n        </cqa-dynamic-select>\n      </div>\n      <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n        [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n        [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n      </cqa-button>\n    </div>\n    <p class=\"cqa-text-xs cqa-text-gray-500\">\n      Uses Database environments from Environments.\n    </p>\n  </div>\n\n  <!-- Main Content: Two Column Layout -->\n  <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n    <!-- Left Panel: Query List -->\n    <div\n      class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n      <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n        <div *ngFor=\"let query of queries; let i = index\"\n          class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n          [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n          [class.cqa-border]=\"selectedQueryIndex === i\"\n          [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n          [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n          (click)=\"onSelectQuery(i)\"\n          style=\"height: 49px; min-height: 49px;\">\n          <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n          <cqa-badge *ngIf=\"query.status\"\n            [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n            [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n            size=\"small\">\n          </cqa-badge>\n        </div>\n      </div>\n    </div>\n\n    <!-- Right Panel: Query Editor -->\n    <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n      <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n          <div class=\"\">\n            <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n          </div>\n\n          <div class=\"cqa-flex cqa-justify-between\">\n            <!-- SQL Query Textarea -->\n            <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n              <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n                [textareaInlineStyle]=\"'padding: 1rem;'\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n              </cqa-custom-textarea>\n            </div>\n          </div>\n        </div>\n\n        <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n          <!-- Variable Input -->\n          <div class=\"cqa-flex cqa-gap-4\">\n            <div class=\"cqa-flex-1\">\n              <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n                Variable\n              </label>\n              <cqa-custom-input [placeholder]=\"'Variable name'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n              </cqa-custom-input>\n              <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n                Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n              </p>\n            </div>\n          </div>\n          <!-- Action Buttons (pinned to bottom) -->\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n            <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n              [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n              [iconColor]=\"'#414146'\">\n            </cqa-button>\n            <cqa-button type=\"button\" variant=\"text\"\n              [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n              [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n              [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n              <svg class=\"cqa-flex-shrink-0\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                <path\n                  d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                  fill=\"#6B7280\" />\n              </svg>\n            </cqa-button>\n            <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n              [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n              [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n              [iconColor]=\"'#FBFCFF'\">\n            </cqa-button>\n          </div>\n        </div>\n      </div>\n      <!-- Results Section -->\n      <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n        <!-- Tabs -->\n        <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n            Output\n          </button>\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n            Verification\n          </button>\n        </div>\n\n        <!-- Tab Content -->\n        <div class=\"cqa-p-4\">\n          <!-- Output Tab -->\n          <div *ngIf=\"selectedTab === 'output'\">\n            <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n              <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n              <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n                (clicked)=\"onCopyResults()\">\n              </cqa-button>\n            </div>\n\n            <!-- Results Table -->\n            <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n              <table class=\"cqa-w-full cqa-border-collapse\">\n                <thead>\n                  <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n                    <th *ngFor=\"let key of getTableColumns()\"\n                      class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n                      {{ key }}\n                    </th>\n                  </tr>\n                </thead>\n                <tbody>\n                  <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n                    <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n                      {{ row[key] }}\n                    </td>\n                  </tr>\n                </tbody>\n              </table>\n            </div>\n\n            <div *ngIf=\"!queryResults || queryResults.length === 0\"\n              class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n              No results yet. Run a query to see results.\n            </div>\n          </div>\n\n          <!-- Verification Tab -->\n          <div *ngIf=\"selectedTab === 'verification'\" class=\"assertions-container\">\n            <!-- Header Section -->\n            <div class=\"assertions-header\">\n              <div class=\"header-content\">\n                <h3 class=\"header-title\">Assertions</h3>\n                <p class=\"header-subtitle\">Use JSONPath to target values in the query output.</p>\n              </div>\n              <button \n                mat-stroked-button \n                color=\"primary\" \n                class=\"add-assertion-button\"\n                (click)=\"addAssertion()\">\n                <mat-icon>add</mat-icon>\n                Add Assertion\n              </button>\n            </div>\n\n            <!-- Table Section -->\n            <form *ngIf=\"getCurrentAssertionsForm()\" [formGroup]=\"getCurrentAssertionsForm()!\" class=\"assertions-form\">\n              <div formArrayName=\"assertions\">\n                <div class=\"table-scroll-container\">\n                  <table mat-table [dataSource]=\"assertionsDataSource\" class=\"assertions-table\" #assertionsTable>\n                    <!-- JSONPath Column -->\n                    <ng-container matColumnDef=\"jsonPath\">\n                      <th mat-header-cell *matHeaderCellDef>jsonPath</th>\n                      <td mat-cell *matCellDef=\"let assertion; let i = index\">\n                        <ng-container [formGroupName]=\"i\">\n                          <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n                            <input \n                              matInput \n                              formControlName=\"jsonPath\"\n                              placeholder=\"username\"\n                              required>\n                            <mat-error *ngIf=\"getFormControlAt(i, 'jsonPath')?.hasError('required') && \n                                              getFormControlAt(i, 'jsonPath')?.touched\">\n                              JSONPath is required\n                            </mat-error>\n                          </mat-form-field>\n                        </ng-container>\n                      </td>\n                    </ng-container>\n\n                    <!-- Verification Type Column -->\n                    <ng-container matColumnDef=\"verificationType\">\n                      <th mat-header-cell *matHeaderCellDef>verificationType</th>\n                      <td mat-cell *matCellDef=\"let assertion; let i = index\">\n                        <ng-container [formGroupName]=\"i\">\n                          <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n                            <mat-select formControlName=\"verificationType\">\n                              <mat-option value=\"equals\">Equals</mat-option>\n                              <mat-option value=\"not_equals\">Not Equals</mat-option>\n                              <mat-option value=\"contains\">Contains</mat-option>\n                              <mat-option value=\"not_contains\">Not Contains</mat-option>\n                              <mat-option value=\"greater_than\">Greater Than</mat-option>\n                              <mat-option value=\"less_than\">Less Than</mat-option>\n                              <mat-option value=\"greater_than_or_equals\">Greater Than Or Equals</mat-option>\n                              <mat-option value=\"less_than_or_equals\">Less Than Or Equals</mat-option>\n                              <mat-option value=\"exists\">Exists</mat-option>\n                              <mat-option value=\"not_exists\">Not Exists</mat-option>\n                            </mat-select>\n                          </mat-form-field>\n                        </ng-container>\n                      </td>\n                    </ng-container>\n\n                    <!-- Expected Type Column -->\n                    <ng-container matColumnDef=\"expectedType\">\n                      <th mat-header-cell *matHeaderCellDef>expectedType</th>\n                      <td mat-cell *matCellDef=\"let assertion; let i = index\">\n                        <ng-container [formGroupName]=\"i\">\n                          <mat-form-field appearance=\"outline\" class=\"full-width-field\">\n                            <mat-select \n                              formControlName=\"expectedType\"\n                              (selectionChange)=\"onExpectedTypeChange(i, $event.value)\">\n                              <mat-option value=\"string\">String</mat-option>\n                              <mat-option value=\"number\">Number</mat-option>\n                              <mat-option value=\"boolean\">Boolean</mat-option>\n                              <mat-option value=\"object\">Object</mat-option>\n                              <mat-option value=\"array\">Array</mat-option>\n                              <mat-option value=\"null\">Null</mat-option>\n                            </mat-select>\n                          </mat-form-field>\n                        </ng-container>\n                      </td>\n                    </ng-container>\n\n                    <!-- Expected Value Column -->\n                    <ng-container matColumnDef=\"expectedValue\">\n                      <th mat-header-cell *matHeaderCellDef>expectedValue</th>\n                      <td mat-cell *matCellDef=\"let assertion; let i = index\">\n                        <ng-container [formGroupName]=\"i\">\n                          <!-- String, Array, Object Input -->\n                          <mat-form-field \n                            *ngIf=\"getExpectedType(i) === 'string' || getExpectedType(i) === 'array' || getExpectedType(i) === 'object'\"\n                            appearance=\"outline\" \n                            class=\"full-width-field\">\n                            <textarea \n                              *ngIf=\"getExpectedType(i) === 'array' || getExpectedType(i) === 'object'\"\n                              matInput \n                              formControlName=\"expectedValue\"\n                              [placeholder]=\"getExpectedValuePlaceholder(i)\"\n                              rows=\"3\"\n                              required></textarea>\n                            <input \n                              *ngIf=\"getExpectedType(i) === 'string'\"\n                              matInput \n                              formControlName=\"expectedValue\"\n                              [placeholder]=\"getExpectedValuePlaceholder(i)\"\n                              required>\n                            <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n                                              getFormControlAt(i, 'expectedValue')?.touched\">\n                              Expected value is required\n                            </mat-error>\n                            <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('invalidType')\">\n                              {{ getFormControlAt(i, 'expectedValue')?.errors?.message }}\n                            </mat-error>\n                          </mat-form-field>\n\n                          <!-- Null Type Display -->\n                          <mat-form-field \n                            *ngIf=\"getExpectedType(i) === 'null'\"\n                            appearance=\"outline\" \n                            class=\"full-width-field\">\n                            <input \n                              matInput \n                              formControlName=\"expectedValue\"\n                              [placeholder]=\"getExpectedValuePlaceholder(i)\"\n                              [disabled]=\"true\">\n                          </mat-form-field>\n\n                          <!-- Number Input -->\n                          <mat-form-field \n                            *ngIf=\"getExpectedType(i) === 'number'\"\n                            appearance=\"outline\" \n                            class=\"full-width-field\">\n                            <input \n                              matInput \n                              type=\"number\"\n                              formControlName=\"expectedValue\"\n                              placeholder=\"Expected Value in Number\"\n                              required>\n                            <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n                                              getFormControlAt(i, 'expectedValue')?.touched\">\n                              Expected value is required\n                            </mat-error>\n                          </mat-form-field>\n\n                          <!-- Boolean Select -->\n                          <mat-form-field \n                            *ngIf=\"getExpectedType(i) === 'boolean'\"\n                            appearance=\"outline\" \n                            class=\"full-width-field\">\n                            <mat-select formControlName=\"expectedValue\" required>\n                              <mat-option value=\"true\">true</mat-option>\n                              <mat-option value=\"false\">false</mat-option>\n                            </mat-select>\n                            <mat-error *ngIf=\"getFormControlAt(i, 'expectedValue')?.hasError('required') && \n                                              getFormControlAt(i, 'expectedValue')?.touched\">\n                              Expected value is required\n                            </mat-error>\n                          </mat-form-field>\n                        </ng-container>\n                      </td>\n                    </ng-container>\n\n                    <!-- Actions Column -->\n                    <ng-container matColumnDef=\"actions\">\n                      <th mat-header-cell *matHeaderCellDef></th>\n                      <td mat-cell *matCellDef=\"let assertion; let i = index\">\n                        <button \n                          mat-icon-button \n                          color=\"warn\"\n                          class=\"delete-button\"\n                          (click)=\"removeAssertion(i)\"\n                          [matTooltip]=\"'Delete assertion'\">\n                          <mat-icon>delete</mat-icon>\n                        </button>\n                      </td>\n                    </ng-container>\n\n                    <tr mat-header-row *matHeaderRowDef=\"displayedColumns\"></tr>\n                    <tr mat-row *matRowDef=\"let row; columns: displayedColumns;\"></tr>\n                  </table>\n                </div>\n\n                <!-- Empty State -->\n                <div *ngIf=\"getAssertionsFormArray().length === 0\" class=\"empty-state\">\n                  <p>No assertions added. Click \"Add Assertion\" to create one.</p>\n                </div>\n              </div>\n            </form>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n\n\n  <!-- Action Buttons -->\n  <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n    <cqa-button class=\"cqa-w-1/2\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>"]}
|