@cqa-lib/cqa-ui 1.1.93 → 1.1.94

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.
@@ -0,0 +1,289 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import { BaseStepComponent } from '../base-step.component';
3
+ import { EMPTY_STATE_IMAGES } from '../../assets/images/image-assets.constants';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "../../badge/badge.component";
6
+ import * as i2 from "../db-query-execution-item/db-query-execution-item.component";
7
+ import * as i3 from "../../segment-control/segment-control.component";
8
+ import * as i4 from "../../templates/table-template.component";
9
+ import * as i5 from "@angular/common";
10
+ export class DbVerificationStepComponent extends BaseStepComponent {
11
+ constructor() {
12
+ super(...arguments);
13
+ this.verificationFilter = 'all';
14
+ this.filterSegments = [
15
+ { label: 'All', value: 'all' },
16
+ { label: 'Failed only', value: 'failed' },
17
+ { label: 'Passed only', value: 'passed' },
18
+ ];
19
+ this.verificationTableColumns = [];
20
+ this.emptyStateConfig = {
21
+ title: 'No Assertions',
22
+ description: 'No verification assertions found for this database query.',
23
+ imageUrl: EMPTY_STATE_IMAGES.DASHBOARD,
24
+ actions: [],
25
+ };
26
+ this.pageIndex = 0;
27
+ this.pageSize = 10;
28
+ // Cached query results to avoid recreating components on every change detection
29
+ this.cachedQueryResults = [];
30
+ }
31
+ ngOnInit() {
32
+ this.updateConfig();
33
+ this.setupTableColumns();
34
+ this.updateCachedQueryResults();
35
+ super.ngOnInit();
36
+ if (this.expanded !== undefined) {
37
+ this.isExpanded = this.expanded;
38
+ }
39
+ }
40
+ getStatus() {
41
+ return this.config?.status || this.status;
42
+ }
43
+ ngOnChanges(changes) {
44
+ // Update config when inputs change
45
+ if (changes['dbTestResult'] || changes['dbConfig'] || changes['expanded'] || changes['status'] || changes['duration']) {
46
+ this.updateConfig();
47
+ if (changes['dbTestResult']) {
48
+ this.setupTableColumns();
49
+ this.updateCachedQueryResults();
50
+ }
51
+ }
52
+ // Update expanded state if changed
53
+ if (changes['expanded'] && this.expanded !== undefined) {
54
+ this.isExpanded = this.expanded;
55
+ }
56
+ }
57
+ updateConfig() {
58
+ this.config = {
59
+ id: this.id,
60
+ testStepResultId: this.testStepResultId,
61
+ stepNumber: this.stepNumber,
62
+ title: this.title,
63
+ status: this.status,
64
+ duration: this.duration,
65
+ type: 'db-verification',
66
+ dbTestResult: this.dbTestResult,
67
+ dbConfig: this.dbConfig,
68
+ timingBreakdown: this.timingBreakdown,
69
+ expanded: this.expanded,
70
+ };
71
+ }
72
+ toggleHeader() {
73
+ this.toggle();
74
+ }
75
+ updateCachedQueryResults() {
76
+ if (!this.dbTestResult?.results) {
77
+ this.cachedQueryResults = [];
78
+ return;
79
+ }
80
+ this.cachedQueryResults = Object.entries(this.dbTestResult.results).map(([key, result]) => ({ key, result }));
81
+ }
82
+ getQueryResults() {
83
+ return this.cachedQueryResults;
84
+ }
85
+ trackByQueryKey(index, item) {
86
+ return item.key;
87
+ }
88
+ getTotalRowsReturned() {
89
+ if (!this.dbTestResult?.results)
90
+ return 0;
91
+ return Object.values(this.dbTestResult.results).reduce((total, result) => {
92
+ return total + (Array.isArray(result.data) ? result.data.length : 0);
93
+ }, 0);
94
+ }
95
+ getQueryStatus(queryKey) {
96
+ if (!this.dbTestResult?.assertionResults)
97
+ return 'passed';
98
+ const queryAssertions = this.dbTestResult.assertionResults.filter(assertion => assertion.variableName === queryKey);
99
+ if (queryAssertions.length === 0)
100
+ return 'passed';
101
+ return queryAssertions.every(a => a.passed) ? 'passed' : 'failed';
102
+ }
103
+ getFilteredAssertions() {
104
+ if (!this.dbTestResult?.assertionResults)
105
+ return [];
106
+ switch (this.verificationFilter) {
107
+ case 'failed':
108
+ return this.dbTestResult.assertionResults.filter(a => !a.passed);
109
+ case 'passed':
110
+ return this.dbTestResult.assertionResults.filter(a => a.passed);
111
+ default:
112
+ return this.dbTestResult.assertionResults;
113
+ }
114
+ }
115
+ setVerificationFilter(filter) {
116
+ this.verificationFilter = filter;
117
+ }
118
+ onFilterChange(value) {
119
+ this.verificationFilter = value;
120
+ }
121
+ setupTableColumns() {
122
+ this.verificationTableColumns = [
123
+ {
124
+ fieldId: 'jsonPath',
125
+ fieldName: 'JSON path',
126
+ fieldValue: 'jsonPath',
127
+ isShow: true,
128
+ weight: 2,
129
+ render: (row) => {
130
+ return `
131
+ <div class="cqa-text-[12px] cqa-text-[#4B5563] cqa-font-mono">
132
+ ${this.escapeHtml(row.jsonPath)}
133
+ </div>
134
+ `;
135
+ }
136
+ },
137
+ {
138
+ fieldId: 'verificationType',
139
+ fieldName: 'Verification type',
140
+ fieldValue: 'verificationType',
141
+ isShow: true,
142
+ weight: 1.5,
143
+ render: (row) => {
144
+ return `
145
+ <div class="cqa-text-[12px] cqa-text-[#4B5563]">
146
+ ${this.escapeHtml(this.formatVerificationType(row.verificationType))}
147
+ </div>
148
+ `;
149
+ }
150
+ },
151
+ {
152
+ fieldId: 'actualValue',
153
+ fieldName: 'Actual value',
154
+ fieldValue: 'actualValue',
155
+ isShow: true,
156
+ weight: 1.5,
157
+ render: (row) => {
158
+ const value = typeof row.actualValue === 'object' ? JSON.stringify(row.actualValue) : String(row.actualValue);
159
+ return `
160
+ <div class="cqa-text-[12px] cqa-text-[#4B5563]">
161
+ ${this.escapeHtml(value)}
162
+ </div>
163
+ `;
164
+ }
165
+ },
166
+ {
167
+ fieldId: 'expectedValue',
168
+ fieldName: 'Expected value',
169
+ fieldValue: 'expectedValue',
170
+ isShow: true,
171
+ weight: 1.5,
172
+ render: (row) => {
173
+ const value = typeof row.expectedValue === 'object' ? JSON.stringify(row.expectedValue) : String(row.expectedValue);
174
+ return `
175
+ <div class="cqa-text-[12px] cqa-text-[#4B5563]">
176
+ ${this.escapeHtml(value)}
177
+ </div>
178
+ `;
179
+ }
180
+ },
181
+ {
182
+ fieldId: 'result',
183
+ fieldName: 'Result',
184
+ fieldValue: 'passed',
185
+ isShow: true,
186
+ weight: 1,
187
+ render: (row) => {
188
+ if (row.passed) {
189
+ return `
190
+ <span class="cqa-inline-block cqa-text-center cqa-w-11 cqa-px-2 cqa-py-1 cqa-rounded-[5px] cqa-text-[12px] cqa-bg-[#2E7D32] cqa-text-[#FFFFFF]">
191
+ Pass
192
+ </span>
193
+ `;
194
+ }
195
+ else {
196
+ return `
197
+ <span class="cqa-inline-block cqa-text-center cqa-w-11 cqa-px-2 cqa-py-1 cqa-rounded-[5px] cqa-text-[12px] cqa-bg-[#DC2626] cqa-text-[#FFFFFF]">
198
+ Fail
199
+ </span>
200
+ `;
201
+ }
202
+ }
203
+ },
204
+ ];
205
+ }
206
+ escapeHtml(text) {
207
+ const div = document.createElement('div');
208
+ div.textContent = text;
209
+ return div.innerHTML;
210
+ }
211
+ getTableData() {
212
+ return this.getFilteredAssertions();
213
+ }
214
+ get isEmptyTable() {
215
+ return this.getTableData().length === 0;
216
+ }
217
+ onPageChange(event) {
218
+ this.pageIndex = event.pageIndex;
219
+ this.pageSize = event.pageSize;
220
+ }
221
+ copyQuery(query) {
222
+ navigator.clipboard.writeText(query).then(() => {
223
+ });
224
+ }
225
+ copyResponse(queryResult) {
226
+ const response = JSON.stringify(queryResult.data, null, 2);
227
+ navigator.clipboard.writeText(response).then(() => {
228
+ });
229
+ }
230
+ copyEnvironment() {
231
+ if (this.dbConfig?.name) {
232
+ navigator.clipboard.writeText(this.dbConfig.name).then(() => {
233
+ });
234
+ }
235
+ }
236
+ getFailureMessage() {
237
+ if (!this.dbTestResult || this.dbTestResult.allAssertionsPassed)
238
+ return null;
239
+ const failedAssertion = this.dbTestResult.assertionResults?.find(a => !a.passed);
240
+ if (failedAssertion) {
241
+ const queryKey = failedAssertion.variableName;
242
+ const queryIndex = this.getQueryResults().findIndex(q => q.key === queryKey) + 1;
243
+ return `Assertion failed in Query ${queryIndex}. Expected value "${failedAssertion.expectedValue}" but got "${failedAssertion.actualValue}".`;
244
+ }
245
+ return null;
246
+ }
247
+ formatVerificationType(type) {
248
+ const typeMap = {
249
+ 'equals': 'Equals',
250
+ 'not_equals': 'Not Equals',
251
+ 'contains': 'Contains',
252
+ 'not_contains': 'Not Contains',
253
+ 'greater_than': 'Greater Than',
254
+ 'less_than': 'Less Than',
255
+ 'greater_than_or_equals': 'Greater Than Or Equals',
256
+ 'less_than_or_equals': 'Less Than Or Equals',
257
+ 'exists': 'Exists',
258
+ 'not_exists': 'Not Exists',
259
+ };
260
+ return typeMap[type] || type;
261
+ }
262
+ }
263
+ DbVerificationStepComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DbVerificationStepComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
264
+ DbVerificationStepComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DbVerificationStepComponent, selector: "cqa-db-verification-step", inputs: { id: "id", testStepResultId: "testStepResultId", stepNumber: "stepNumber", title: "title", status: "status", duration: "duration", timingBreakdown: "timingBreakdown", expanded: "expanded", dbTestResult: "dbTestResult", dbConfig: "dbConfig" }, host: { classAttribute: "cqa-ui-root" }, usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div>\n <!-- Header -->\n <div\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-p-2 cqa-cursor-pointer\"\n (click)=\"toggleHeader()\"\n style=\"border-bottom: 1px solid #F3F4F6\">\n \n <!-- Status Icon -->\n <!-- Success -->\n <div *ngIf=\"getStatus().toLowerCase() === 'success'\">\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10.9005 4.99999C11.1289 6.12064 10.9662 7.28571 10.4395 8.30089C9.91279 9.31608 9.054 10.12 8.00631 10.5787C6.95862 11.0373 5.78536 11.1229 4.6822 10.8212C3.57904 10.5195 2.61265 9.84869 1.94419 8.92071C1.27573 7.99272 0.945611 6.86361 1.00888 5.72169C1.07215 4.57976 1.52499 3.49404 2.29188 2.64558C3.05876 1.79712 4.09334 1.23721 5.22308 1.05922C6.35282 0.881233 7.50944 1.09592 8.50005 1.66749\" stroke=\"#22C55E\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4.5 5.5L6 7L11 2\" stroke=\"#22C55E\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n <!-- Failure -->\n <div *ngIf=\"getStatus().toLowerCase() === 'failure' || getStatus().toLowerCase() === 'failed'\">\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M7.5 4.5L4.5 7.5\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4.5 4.5L7.5 7.5\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n <!-- Pending -->\n <div *ngIf=\"getStatus().toLowerCase() === 'pending'\">\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M6 3V6L8 7\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n <!-- Running - Show spinner -->\n <div *ngIf=\"getStatus().toLowerCase() === 'running'\">\n <svg class=\"cqa-animate-spin cqa-text-[#3B82F6]\" width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"6\" r=\"5\" stroke=\"currentColor\" stroke-width=\"1.5\" fill=\"none\" opacity=\"0.25\"/>\n <path d=\"M6 1A5 5 0 0 1 11 6\" stroke=\"currentColor\" stroke-width=\"1.5\" fill=\"none\" stroke-linecap=\"round\"/>\n </svg>\n </div>\n\n <div><svg xmlns=\"http://www.w3.org/2000/svg\" width=\"20\" height=\"16\" viewBox=\"0 0 20 16\" fill=\"none\">\n <rect width=\"20\" height=\"16\" rx=\"4\" fill=\"#F0F0F1\"/>\n <path d=\"M13.5 3.5H6.5C5.95 3.5 5.5 3.95 5.5 4.5V11.5C5.5 12.05 5.95 12.5 6.5 12.5H13.5C14.05 12.5 14.5 12.05 14.5 11.5V4.5C14.5 3.95 14.05 3.5 13.5 3.5ZM13.5 4.5V6H6.5V4.5H13.5ZM13.5 7V9H6.5V7H13.5ZM6.5 11.5V10H13.5V11.5H6.5Z\" fill=\"#212122\"/>\n </svg></div>\n\n <!-- Step Number and Title -->\n <div class=\"cqa-font-bold cqa-flex-1 cqa-text-[#334155] cqa-text-[11px] cqa-leading-[13px]\">\n {{ stepNumber }}. <span [innerHTML]=\"title\"></span> \n \n <span class=\"cqa-ml-1 cqa-px-1.5 cqa-py-0.5 cqa-rounded-full cqa-font-medium cqa-text-[#212122] cqa-bg-[#F0F0F1] cqa-text-[10px] cqa-leading-[15px] cqa-min-w-max\">\n Database\n </span>\n\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-1\">\n <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#9CA3AF]\">\n {{ formatDuration(duration) }}\n </span>\n <svg [class.cqa-rotate-180]=\"isExpanded\" class=\"cqa-transition-transform\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M3.5 5L7 8.5L10.5 5\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n </div>\n\n <!-- Expanded Content -->\n <div *ngIf=\"isExpanded && dbTestResult\" class=\"cqa-p-2 !cqa-pl-9 !cqa-pr-6 cqa-bg-white\">\n <!-- Summary Bar -->\n <div class=\"cqa-mb-1.5 cqa-p-3 cqa-bg-[#FCFCFC] cqa-rounded-[6px] cqa-flex cqa-flex-col cqa-gap-2\" style=\"border: 1px solid #E5E7EB;\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-4\">\n <!-- Status Badge -->\n <cqa-badge\n *ngIf=\"getStatus().toLowerCase() === 'failure' || getStatus().toLowerCase() === 'failed'\"\n label=\"FAIL\"\n backgroundColor=\"#DC2626\"\n textColor=\"#FFFFFF\"\n size=\"small\">\n </cqa-badge>\n <cqa-badge\n *ngIf=\"getStatus().toLowerCase() === 'success'\"\n label=\"PASS\"\n backgroundColor=\"#22C55E\"\n textColor=\"#FFFFFF\"\n size=\"small\">\n </cqa-badge>\n\n <div class=\"cqa-h-[19px] cqa-w-[1px] cqa-bg-[#E5E7EB]\"></div>\n \n <!-- Summary Stats -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-4 cqa-text-[11px] cqa-leading-[16px] cqa-text-[#6B7280]\">\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Total Time: <span class=\"cqa-text-[#4B5563]\">{{ formatDuration(duration) }}</span></span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Queries: <span class=\"cqa-text-[#4B5563]\">{{ cachedQueryResults?.length || 0 }}</span></span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Assertions: <span class=\"cqa-text-[#4B5563]\">{{ dbTestResult?.assertionResults?.length || 0 }}</span></span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Rows Returned: <span class=\"cqa-text-[#4B5563]\">{{ getTotalRowsReturned() }}</span></span>\n </div>\n </div>\n\n <!-- Failure Banner -->\n <div \n *ngIf=\"getFailureMessage()\"\n class=\"cqa-flex cqa-px-2 cqa-py-1 cqa-bg-[#FEF2F2] cqa-rounded-[4px]\" style=\"border: 1px solid #FEE2E2;\">\n <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#B91C1C]\">\n {{ getFailureMessage() }}\n </span>\n </div>\n </div>\n\n <!-- Database Environment Section -->\n <div *ngIf=\"dbConfig\" class=\"cqa-mb-1.5\">\n <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Database environment</div>\n <div class=\"cqa-bg-white cqa-rounded-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-px-4 cqa-py-2 cqa-relative cqa-flex cqa-flex-wrap cqa-gap-3\">\n <!-- Environment -->\n <div class=\"\">\n <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#45556C] cqa-mb-1\">Environment</div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-4\">\n <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-text-[#0F172B]\">{{ dbConfig.name }}</span>\n <span (click)=\"copyEnvironment()\" class=\"cqa-cursor-pointer\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M5.33332 10.6667H3.99999C3.64637 10.6667 3.30723 10.5263 3.05718 10.2762C2.80713 10.0262 2.66666 9.68704 2.66666 9.33341V4.00008C2.66666 3.64646 2.80713 3.30732 3.05718 3.05727C3.30723 2.80722 3.64637 2.66675 3.99999 2.66675H9.33332C9.68695 2.66675 10.0261 2.80722 10.2761 3.05727C10.5262 3.30732 10.6667 3.64646 10.6667 4.00008V5.33341M6.66666 13.3334H12C12.3536 13.3334 12.6928 13.1929 12.9428 12.9429C13.1928 12.6928 13.3333 12.3537 13.3333 12.0001V6.66675C13.3333 6.31313 13.1928 5.97399 12.9428 5.72394C12.6928 5.47389 12.3536 5.33341 12 5.33341H6.66666C6.31303 5.33341 5.9739 5.47389 5.72385 5.72394C5.4738 5.97399 5.33332 6.31313 5.33332 6.66675V12.0001C5.33332 12.3537 5.4738 12.6928 5.72385 12.9429C5.9739 13.1929 6.31303 13.3334 6.66666 13.3334Z\" stroke=\"#636363\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </span>\n </div>\n </div>\n \n <div class=\"\">\n <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#45556C] cqa-mb-1\">Type</div>\n <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-text-[#0F172B]\">{{ dbConfig.dbType }}</div>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-mb-1.5\">\n <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Query Execution</div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-px-4 cqa-py-2 cqa-bg-white cqa-rounded-[10px]\" style=\"border: 1px solid #E2E8F0;\">\n <cqa-db-query-execution-item\n *ngFor=\"let queryItem of cachedQueryResults; let i = index; trackBy: trackByQueryKey\"\n [queryNumber]=\"i + 1\"\n [queryKey]=\"queryItem.key\"\n [queryResult]=\"queryItem.result\"\n [status]=\"getQueryStatus(queryItem.key)\"\n [variableName]=\"queryItem.key\">\n </cqa-db-query-execution-item>\n </div>\n </div>\n\n <!-- Verification Section -->\n <div *ngIf=\"dbTestResult.assertionResults && dbTestResult.assertionResults.length > 0\">\n <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Verification</div> \n <!-- Verification Table -->\n <div class=\"cqa-bg-white cqa-rounded-[10px] cqa-px-4 cqa-py-2 cqa-overflow-hidden\" style=\"border: 1px solid #E2E8F0;\">\n <!-- Segment Control for Filters -->\n <cqa-segment-control\n [segments]=\"filterSegments\"\n [value]=\"verificationFilter\"\n (valueChange)=\"onFilterChange($event)\">\n </cqa-segment-control>\n <cqa-table-template\n [columns]=\"verificationTableColumns\"\n [data]=\"getTableData()\"\n [pageIndex]=\"pageIndex\"\n [pageSize]=\"pageSize\"\n [isEmptyState]=\"isEmptyTable\"\n [emptyStateConfig]=\"emptyStateConfig\"\n [showSearchBar]=\"false\"\n [showFilterButton]=\"false\"\n [showSettingsButton]=\"false\"\n [showAutoRefreshButton]=\"false\"\n [showOtherButton]=\"false\"\n [showFilterPanel]=\"false\"\n [serverSidePagination]=\"false\"\n (pageChange)=\"onPageChange($event)\">\n </cqa-table-template>\n </div>\n </div>\n\n <!-- Timing Breakdown -->\n <div *ngIf=\"timingBreakdown\" class=\"cqa-flex cqa-items-center cqa-justify-end cqa-gap-5 cqa-pt-4 cqa-px-4 cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#9CA3AF]\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div><svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M6 3V6L8 7\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg></div>\n <span>Timing breakdown</span>\n </div>\n <span class=\"cqa-text-dialog-muted cqa-flex cqa-items-center cqa-gap-3\">\n <div>\n App <span class=\"cqa-text-[#374151]\">{{ formatDuration(timingBreakdown.app) }}</span>\n </div>\n <div><svg width=\"1\" height=\"11\" viewBox=\"0 0 1 11\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M-3.8147e-06 10.32V-7.15256e-07H0.959996V10.32H-3.8147e-06Z\" fill=\"#E5E7EB\"/></svg></div>\n <div>\n Tool <span class=\"cqa-text-[#374151]\">{{ formatDuration(timingBreakdown.tool) }}</span>\n </div>\n </span>\n </div>\n </div>\n</div>\n\n", components: [{ type: i1.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { type: i2.DbQueryExecutionItemComponent, selector: "cqa-db-query-execution-item", inputs: ["queryNumber", "queryKey", "queryResult", "status", "variableName"] }, { type: i3.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled"], outputs: ["valueChange"] }, { type: i4.TableTemplateComponent, selector: "cqa-table-template", inputs: ["searchPlaceholder", "searchValue", "showClear", "showSearchBar", "filterConfig", "showFilterPanel", "showFilterButton", "otherButtonLabel", "otherButtonVariant", "showOtherButton", "showActionButton", "showSettingsButton", "showAutoRefreshButton", "data", "isEmptyState", "emptyStateConfig", "actions", "chips", "filterApplied", "columns", "selectedAutoRefreshInterval", "pageIndex", "pageSize", "serverSidePagination", "totalElements", "isTableLoading", "isTableDataLoading"], outputs: ["onSearchChange", "onApplyFilterClick", "onResetFilterClick", "onClearAll", "removeChip", "pageChange", "onReload", "onAutoRefreshClick"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
265
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DbVerificationStepComponent, decorators: [{
266
+ type: Component,
267
+ args: [{ selector: 'cqa-db-verification-step', host: { class: 'cqa-ui-root' }, template: "<div>\n <!-- Header -->\n <div\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-p-2 cqa-cursor-pointer\"\n (click)=\"toggleHeader()\"\n style=\"border-bottom: 1px solid #F3F4F6\">\n \n <!-- Status Icon -->\n <!-- Success -->\n <div *ngIf=\"getStatus().toLowerCase() === 'success'\">\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10.9005 4.99999C11.1289 6.12064 10.9662 7.28571 10.4395 8.30089C9.91279 9.31608 9.054 10.12 8.00631 10.5787C6.95862 11.0373 5.78536 11.1229 4.6822 10.8212C3.57904 10.5195 2.61265 9.84869 1.94419 8.92071C1.27573 7.99272 0.945611 6.86361 1.00888 5.72169C1.07215 4.57976 1.52499 3.49404 2.29188 2.64558C3.05876 1.79712 4.09334 1.23721 5.22308 1.05922C6.35282 0.881233 7.50944 1.09592 8.50005 1.66749\" stroke=\"#22C55E\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4.5 5.5L6 7L11 2\" stroke=\"#22C55E\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n <!-- Failure -->\n <div *ngIf=\"getStatus().toLowerCase() === 'failure' || getStatus().toLowerCase() === 'failed'\">\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M7.5 4.5L4.5 7.5\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4.5 4.5L7.5 7.5\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n <!-- Pending -->\n <div *ngIf=\"getStatus().toLowerCase() === 'pending'\">\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M6 3V6L8 7\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n <!-- Running - Show spinner -->\n <div *ngIf=\"getStatus().toLowerCase() === 'running'\">\n <svg class=\"cqa-animate-spin cqa-text-[#3B82F6]\" width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"6\" r=\"5\" stroke=\"currentColor\" stroke-width=\"1.5\" fill=\"none\" opacity=\"0.25\"/>\n <path d=\"M6 1A5 5 0 0 1 11 6\" stroke=\"currentColor\" stroke-width=\"1.5\" fill=\"none\" stroke-linecap=\"round\"/>\n </svg>\n </div>\n\n <div><svg xmlns=\"http://www.w3.org/2000/svg\" width=\"20\" height=\"16\" viewBox=\"0 0 20 16\" fill=\"none\">\n <rect width=\"20\" height=\"16\" rx=\"4\" fill=\"#F0F0F1\"/>\n <path d=\"M13.5 3.5H6.5C5.95 3.5 5.5 3.95 5.5 4.5V11.5C5.5 12.05 5.95 12.5 6.5 12.5H13.5C14.05 12.5 14.5 12.05 14.5 11.5V4.5C14.5 3.95 14.05 3.5 13.5 3.5ZM13.5 4.5V6H6.5V4.5H13.5ZM13.5 7V9H6.5V7H13.5ZM6.5 11.5V10H13.5V11.5H6.5Z\" fill=\"#212122\"/>\n </svg></div>\n\n <!-- Step Number and Title -->\n <div class=\"cqa-font-bold cqa-flex-1 cqa-text-[#334155] cqa-text-[11px] cqa-leading-[13px]\">\n {{ stepNumber }}. <span [innerHTML]=\"title\"></span> \n \n <span class=\"cqa-ml-1 cqa-px-1.5 cqa-py-0.5 cqa-rounded-full cqa-font-medium cqa-text-[#212122] cqa-bg-[#F0F0F1] cqa-text-[10px] cqa-leading-[15px] cqa-min-w-max\">\n Database\n </span>\n\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-1\">\n <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#9CA3AF]\">\n {{ formatDuration(duration) }}\n </span>\n <svg [class.cqa-rotate-180]=\"isExpanded\" class=\"cqa-transition-transform\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M3.5 5L7 8.5L10.5 5\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </div>\n </div>\n\n <!-- Expanded Content -->\n <div *ngIf=\"isExpanded && dbTestResult\" class=\"cqa-p-2 !cqa-pl-9 !cqa-pr-6 cqa-bg-white\">\n <!-- Summary Bar -->\n <div class=\"cqa-mb-1.5 cqa-p-3 cqa-bg-[#FCFCFC] cqa-rounded-[6px] cqa-flex cqa-flex-col cqa-gap-2\" style=\"border: 1px solid #E5E7EB;\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-4\">\n <!-- Status Badge -->\n <cqa-badge\n *ngIf=\"getStatus().toLowerCase() === 'failure' || getStatus().toLowerCase() === 'failed'\"\n label=\"FAIL\"\n backgroundColor=\"#DC2626\"\n textColor=\"#FFFFFF\"\n size=\"small\">\n </cqa-badge>\n <cqa-badge\n *ngIf=\"getStatus().toLowerCase() === 'success'\"\n label=\"PASS\"\n backgroundColor=\"#22C55E\"\n textColor=\"#FFFFFF\"\n size=\"small\">\n </cqa-badge>\n\n <div class=\"cqa-h-[19px] cqa-w-[1px] cqa-bg-[#E5E7EB]\"></div>\n \n <!-- Summary Stats -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-4 cqa-text-[11px] cqa-leading-[16px] cqa-text-[#6B7280]\">\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Total Time: <span class=\"cqa-text-[#4B5563]\">{{ formatDuration(duration) }}</span></span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Queries: <span class=\"cqa-text-[#4B5563]\">{{ cachedQueryResults?.length || 0 }}</span></span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Assertions: <span class=\"cqa-text-[#4B5563]\">{{ dbTestResult?.assertionResults?.length || 0 }}</span></span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Rows Returned: <span class=\"cqa-text-[#4B5563]\">{{ getTotalRowsReturned() }}</span></span>\n </div>\n </div>\n\n <!-- Failure Banner -->\n <div \n *ngIf=\"getFailureMessage()\"\n class=\"cqa-flex cqa-px-2 cqa-py-1 cqa-bg-[#FEF2F2] cqa-rounded-[4px]\" style=\"border: 1px solid #FEE2E2;\">\n <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#B91C1C]\">\n {{ getFailureMessage() }}\n </span>\n </div>\n </div>\n\n <!-- Database Environment Section -->\n <div *ngIf=\"dbConfig\" class=\"cqa-mb-1.5\">\n <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Database environment</div>\n <div class=\"cqa-bg-white cqa-rounded-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-px-4 cqa-py-2 cqa-relative cqa-flex cqa-flex-wrap cqa-gap-3\">\n <!-- Environment -->\n <div class=\"\">\n <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#45556C] cqa-mb-1\">Environment</div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-4\">\n <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-text-[#0F172B]\">{{ dbConfig.name }}</span>\n <span (click)=\"copyEnvironment()\" class=\"cqa-cursor-pointer\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M5.33332 10.6667H3.99999C3.64637 10.6667 3.30723 10.5263 3.05718 10.2762C2.80713 10.0262 2.66666 9.68704 2.66666 9.33341V4.00008C2.66666 3.64646 2.80713 3.30732 3.05718 3.05727C3.30723 2.80722 3.64637 2.66675 3.99999 2.66675H9.33332C9.68695 2.66675 10.0261 2.80722 10.2761 3.05727C10.5262 3.30732 10.6667 3.64646 10.6667 4.00008V5.33341M6.66666 13.3334H12C12.3536 13.3334 12.6928 13.1929 12.9428 12.9429C13.1928 12.6928 13.3333 12.3537 13.3333 12.0001V6.66675C13.3333 6.31313 13.1928 5.97399 12.9428 5.72394C12.6928 5.47389 12.3536 5.33341 12 5.33341H6.66666C6.31303 5.33341 5.9739 5.47389 5.72385 5.72394C5.4738 5.97399 5.33332 6.31313 5.33332 6.66675V12.0001C5.33332 12.3537 5.4738 12.6928 5.72385 12.9429C5.9739 13.1929 6.31303 13.3334 6.66666 13.3334Z\" stroke=\"#636363\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </span>\n </div>\n </div>\n \n <div class=\"\">\n <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#45556C] cqa-mb-1\">Type</div>\n <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-text-[#0F172B]\">{{ dbConfig.dbType }}</div>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-mb-1.5\">\n <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Query Execution</div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-px-4 cqa-py-2 cqa-bg-white cqa-rounded-[10px]\" style=\"border: 1px solid #E2E8F0;\">\n <cqa-db-query-execution-item\n *ngFor=\"let queryItem of cachedQueryResults; let i = index; trackBy: trackByQueryKey\"\n [queryNumber]=\"i + 1\"\n [queryKey]=\"queryItem.key\"\n [queryResult]=\"queryItem.result\"\n [status]=\"getQueryStatus(queryItem.key)\"\n [variableName]=\"queryItem.key\">\n </cqa-db-query-execution-item>\n </div>\n </div>\n\n <!-- Verification Section -->\n <div *ngIf=\"dbTestResult.assertionResults && dbTestResult.assertionResults.length > 0\">\n <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Verification</div> \n <!-- Verification Table -->\n <div class=\"cqa-bg-white cqa-rounded-[10px] cqa-px-4 cqa-py-2 cqa-overflow-hidden\" style=\"border: 1px solid #E2E8F0;\">\n <!-- Segment Control for Filters -->\n <cqa-segment-control\n [segments]=\"filterSegments\"\n [value]=\"verificationFilter\"\n (valueChange)=\"onFilterChange($event)\">\n </cqa-segment-control>\n <cqa-table-template\n [columns]=\"verificationTableColumns\"\n [data]=\"getTableData()\"\n [pageIndex]=\"pageIndex\"\n [pageSize]=\"pageSize\"\n [isEmptyState]=\"isEmptyTable\"\n [emptyStateConfig]=\"emptyStateConfig\"\n [showSearchBar]=\"false\"\n [showFilterButton]=\"false\"\n [showSettingsButton]=\"false\"\n [showAutoRefreshButton]=\"false\"\n [showOtherButton]=\"false\"\n [showFilterPanel]=\"false\"\n [serverSidePagination]=\"false\"\n (pageChange)=\"onPageChange($event)\">\n </cqa-table-template>\n </div>\n </div>\n\n <!-- Timing Breakdown -->\n <div *ngIf=\"timingBreakdown\" class=\"cqa-flex cqa-items-center cqa-justify-end cqa-gap-5 cqa-pt-4 cqa-px-4 cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#9CA3AF]\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div><svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M6 3V6L8 7\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg></div>\n <span>Timing breakdown</span>\n </div>\n <span class=\"cqa-text-dialog-muted cqa-flex cqa-items-center cqa-gap-3\">\n <div>\n App <span class=\"cqa-text-[#374151]\">{{ formatDuration(timingBreakdown.app) }}</span>\n </div>\n <div><svg width=\"1\" height=\"11\" viewBox=\"0 0 1 11\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M-3.8147e-06 10.32V-7.15256e-07H0.959996V10.32H-3.8147e-06Z\" fill=\"#E5E7EB\"/></svg></div>\n <div>\n Tool <span class=\"cqa-text-[#374151]\">{{ formatDuration(timingBreakdown.tool) }}</span>\n </div>\n </span>\n </div>\n </div>\n</div>\n\n", styles: [] }]
268
+ }], propDecorators: { id: [{
269
+ type: Input
270
+ }], testStepResultId: [{
271
+ type: Input
272
+ }], stepNumber: [{
273
+ type: Input
274
+ }], title: [{
275
+ type: Input
276
+ }], status: [{
277
+ type: Input
278
+ }], duration: [{
279
+ type: Input
280
+ }], timingBreakdown: [{
281
+ type: Input
282
+ }], expanded: [{
283
+ type: Input
284
+ }], dbTestResult: [{
285
+ type: Input
286
+ }], dbConfig: [{
287
+ type: Input
288
+ }] } });
289
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"db-verification-step.component.js","sourceRoot":"","sources":["../../../../../../src/lib/execution-screen/db-verification-step/db-verification-step.component.ts","../../../../../../src/lib/execution-screen/db-verification-step/db-verification-step.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, OnChanges, SimpleChanges } from '@angular/core';\nimport { BaseStepComponent } from '../base-step.component';\nimport { DbVerificationStepConfig, DbTestResult, DbQueryResult, DbAssertionResult, StepStatus, TimingBreakdown } from '../execution-step.models';\nimport { DynamicTableColumn } from '../../table/dynamic-table/dynamic-table.component';\nimport { EmptyStateConfig } from '../../empty-state/empty-state-config.interface';\nimport { EMPTY_STATE_IMAGES } from '../../assets/images/image-assets.constants';\n\n@Component({\n  selector: 'cqa-db-verification-step',\n  templateUrl: './db-verification-step.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class DbVerificationStepComponent extends BaseStepComponent implements OnInit, OnChanges {\n  @Input() id!: string;\n  @Input() testStepResultId!: string;\n  @Input() stepNumber!: string;\n  @Input() title!: string;\n  @Input() status!: StepStatus;\n  @Input() duration!: number;\n  @Input() timingBreakdown?: TimingBreakdown;\n  @Input() expanded?: boolean;\n  @Input() dbTestResult!: DbTestResult;\n  @Input() dbConfig?: {\n    name: string;\n    dbType: string;\n  };\n\n  override config!: DbVerificationStepConfig;\n\n  verificationFilter: 'all' | 'failed' | 'passed' = 'all';\n\n  filterSegments = [\n    { label: 'All', value: 'all' },\n    { label: 'Failed only', value: 'failed' },\n    { label: 'Passed only', value: 'passed' },\n  ];\n\n  verificationTableColumns: DynamicTableColumn[] = [];\n  \n  emptyStateConfig: EmptyStateConfig = {\n    title: 'No Assertions',\n    description: 'No verification assertions found for this database query.',\n    imageUrl: EMPTY_STATE_IMAGES.DASHBOARD,\n    actions: [],\n  };\n\n  pageIndex: number = 0;\n  pageSize: number = 10;\n  \n  // Cached query results to avoid recreating components on every change detection\n  cachedQueryResults: Array<{ key: string; result: DbQueryResult }> = [];\n\n  override ngOnInit(): void {\n    this.updateConfig();\n    this.setupTableColumns();\n    this.updateCachedQueryResults();\n    super.ngOnInit();\n    if (this.expanded !== undefined) {\n      this.isExpanded = this.expanded;\n    }\n  }\n\n  getStatus(): StepStatus {\n    return this.config?.status || this.status;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Update config when inputs change\n    if (changes['dbTestResult'] || changes['dbConfig'] || changes['expanded'] || changes['status'] || changes['duration']) {\n      this.updateConfig();\n      if (changes['dbTestResult']) {\n        this.setupTableColumns();\n        this.updateCachedQueryResults();\n      }\n    }\n    \n    // Update expanded state if changed\n    if (changes['expanded'] && this.expanded !== undefined) {\n      this.isExpanded = this.expanded;\n    }\n  }\n\n  private updateConfig(): void {\n    this.config = {\n      id: this.id,\n      testStepResultId: this.testStepResultId,\n      stepNumber: this.stepNumber,\n      title: this.title,\n      status: this.status,\n      duration: this.duration,\n      type: 'db-verification',\n      dbTestResult: this.dbTestResult,\n      dbConfig: this.dbConfig,\n      timingBreakdown: this.timingBreakdown,\n      expanded: this.expanded,\n    } as DbVerificationStepConfig;\n  }\n\n  toggleHeader() {\n    this.toggle();\n  }\n\n  private updateCachedQueryResults(): void {\n    if (!this.dbTestResult?.results) {\n      this.cachedQueryResults = [];\n      return;\n    }\n    this.cachedQueryResults = Object.entries(this.dbTestResult.results).map(([key, result]) => ({ key, result }));\n  }\n\n  getQueryResults(): Array<{ key: string; result: DbQueryResult }> {\n    return this.cachedQueryResults;\n  }\n\n  trackByQueryKey(index: number, item: { key: string; result: DbQueryResult }): string {\n    return item.key;\n  }\n\n  getTotalRowsReturned(): number {\n    if (!this.dbTestResult?.results) return 0;\n    return Object.values(this.dbTestResult.results).reduce((total, result) => {\n      return total + (Array.isArray(result.data) ? result.data.length : 0);\n    }, 0);\n  }\n\n  getQueryStatus(queryKey: string): 'passed' | 'failed' {\n    if (!this.dbTestResult?.assertionResults) return 'passed';\n    const queryAssertions = this.dbTestResult.assertionResults.filter(\n      assertion => assertion.variableName === queryKey\n    );\n    if (queryAssertions.length === 0) return 'passed';\n    return queryAssertions.every(a => a.passed) ? 'passed' : 'failed';\n  }\n\n  getFilteredAssertions(): DbAssertionResult[] {\n    if (!this.dbTestResult?.assertionResults) return [];\n    switch (this.verificationFilter) {\n      case 'failed':\n        return this.dbTestResult.assertionResults.filter(a => !a.passed);\n      case 'passed':\n        return this.dbTestResult.assertionResults.filter(a => a.passed);\n      default:\n        return this.dbTestResult.assertionResults;\n    }\n  }\n\n  setVerificationFilter(filter: 'all' | 'failed' | 'passed') {\n    this.verificationFilter = filter;\n  }\n\n  onFilterChange(value: string) {\n    this.verificationFilter = value as 'all' | 'failed' | 'passed';\n  }\n\n  private setupTableColumns(): void {\n    this.verificationTableColumns = [\n      {\n        fieldId: 'jsonPath',\n        fieldName: 'JSON path',\n        fieldValue: 'jsonPath',\n        isShow: true,\n        weight: 2,\n        render: (row: DbAssertionResult) => {\n          return `\n            <div class=\"cqa-text-[12px] cqa-text-[#4B5563] cqa-font-mono\">\n              ${this.escapeHtml(row.jsonPath)}\n            </div>\n          `;\n        }\n      },\n      {\n        fieldId: 'verificationType',\n        fieldName: 'Verification type',\n        fieldValue: 'verificationType',\n        isShow: true,\n        weight: 1.5,\n        render: (row: DbAssertionResult) => {\n          return `\n            <div class=\"cqa-text-[12px] cqa-text-[#4B5563]\">\n              ${this.escapeHtml(this.formatVerificationType(row.verificationType))}\n            </div>\n          `;\n        }\n      },\n      {\n        fieldId: 'actualValue',\n        fieldName: 'Actual value',\n        fieldValue: 'actualValue',\n        isShow: true,\n        weight: 1.5,\n        render: (row: DbAssertionResult) => {\n          const value = typeof row.actualValue === 'object' ? JSON.stringify(row.actualValue) : String(row.actualValue);\n          return `\n            <div class=\"cqa-text-[12px] cqa-text-[#4B5563]\">\n              ${this.escapeHtml(value)}\n            </div>\n          `;\n        }\n      },\n      {\n        fieldId: 'expectedValue',\n        fieldName: 'Expected value',\n        fieldValue: 'expectedValue',\n        isShow: true,\n        weight: 1.5,\n        render: (row: DbAssertionResult) => {\n          const value = typeof row.expectedValue === 'object' ? JSON.stringify(row.expectedValue) : String(row.expectedValue);\n          return `\n            <div class=\"cqa-text-[12px] cqa-text-[#4B5563]\">\n              ${this.escapeHtml(value)}\n            </div>\n          `;\n        }\n      },\n      {\n        fieldId: 'result',\n        fieldName: 'Result',\n        fieldValue: 'passed',\n        isShow: true,\n        weight: 1,\n        render: (row: DbAssertionResult) => {\n          if (row.passed) {\n            return `\n              <span class=\"cqa-inline-block cqa-text-center cqa-w-11 cqa-px-2 cqa-py-1 cqa-rounded-[5px] cqa-text-[12px] cqa-bg-[#2E7D32] cqa-text-[#FFFFFF]\">\n                Pass\n              </span>\n            `;\n          } else {\n            return `\n              <span class=\"cqa-inline-block cqa-text-center cqa-w-11 cqa-px-2 cqa-py-1 cqa-rounded-[5px] cqa-text-[12px] cqa-bg-[#DC2626] cqa-text-[#FFFFFF]\">\n                Fail\n              </span>\n            `;\n          }\n        }\n      },\n    ];\n  }\n\n  private escapeHtml(text: string): string {\n    const div = document.createElement('div');\n    div.textContent = text;\n    return div.innerHTML;\n  }\n\n  getTableData(): DbAssertionResult[] {\n    return this.getFilteredAssertions();\n  }\n\n  get isEmptyTable(): boolean {\n    return this.getTableData().length === 0;\n  }\n\n  onPageChange(event: { pageIndex: number; pageSize: number }): void {\n    this.pageIndex = event.pageIndex;\n    this.pageSize = event.pageSize;\n  }\n\n  copyQuery(query: string) {\n    navigator.clipboard.writeText(query).then(() => {\n    });\n  }\n\n  copyResponse(queryResult: DbQueryResult) {\n    const response = JSON.stringify(queryResult.data, null, 2);\n    navigator.clipboard.writeText(response).then(() => {\n    });\n  }\n\n  copyEnvironment() {\n    if (this.dbConfig?.name) {\n      navigator.clipboard.writeText(this.dbConfig.name).then(() => {\n      });\n    }\n  }\n\n  getFailureMessage(): string | null {\n    if (!this.dbTestResult || this.dbTestResult.allAssertionsPassed) return null;\n    \n    const failedAssertion = this.dbTestResult.assertionResults?.find(a => !a.passed);\n    if (failedAssertion) {\n      const queryKey = failedAssertion.variableName;\n      const queryIndex = this.getQueryResults().findIndex(q => q.key === queryKey) + 1;\n      return `Assertion failed in Query ${queryIndex}. Expected value \"${failedAssertion.expectedValue}\" but got \"${failedAssertion.actualValue}\".`;\n    }\n    return null;\n  }\n\n  formatVerificationType(type: string): string {\n    const typeMap: { [key: string]: string } = {\n      'equals': 'Equals',\n      'not_equals': 'Not Equals',\n      'contains': 'Contains',\n      'not_contains': 'Not Contains',\n      'greater_than': 'Greater Than',\n      'less_than': 'Less Than',\n      'greater_than_or_equals': 'Greater Than Or Equals',\n      'less_than_or_equals': 'Less Than Or Equals',\n      'exists': 'Exists',\n      'not_exists': 'Not Exists',\n    };\n    return typeMap[type] || type;\n  }\n}\n","<div>\n  <!-- Header -->\n  <div\n    class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-p-2 cqa-cursor-pointer\"\n    (click)=\"toggleHeader()\"\n    style=\"border-bottom: 1px solid #F3F4F6\">\n    \n    <!-- Status Icon -->\n    <!-- Success -->\n    <div *ngIf=\"getStatus().toLowerCase() === 'success'\">\n      <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10.9005 4.99999C11.1289 6.12064 10.9662 7.28571 10.4395 8.30089C9.91279 9.31608 9.054 10.12 8.00631 10.5787C6.95862 11.0373 5.78536 11.1229 4.6822 10.8212C3.57904 10.5195 2.61265 9.84869 1.94419 8.92071C1.27573 7.99272 0.945611 6.86361 1.00888 5.72169C1.07215 4.57976 1.52499 3.49404 2.29188 2.64558C3.05876 1.79712 4.09334 1.23721 5.22308 1.05922C6.35282 0.881233 7.50944 1.09592 8.50005 1.66749\" stroke=\"#22C55E\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4.5 5.5L6 7L11 2\" stroke=\"#22C55E\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n    </div>\n    <!-- Failure -->\n    <div *ngIf=\"getStatus().toLowerCase() === 'failure' || getStatus().toLowerCase() === 'failed'\">\n      <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M7.5 4.5L4.5 7.5\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M4.5 4.5L7.5 7.5\" stroke=\"#DC2626\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n    </div>\n    <!-- Pending -->\n    <div *ngIf=\"getStatus().toLowerCase() === 'pending'\">\n      <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M6 3V6L8 7\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n    </div>\n    <!-- Running - Show spinner -->\n    <div *ngIf=\"getStatus().toLowerCase() === 'running'\">\n      <svg class=\"cqa-animate-spin cqa-text-[#3B82F6]\" width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <circle cx=\"6\" cy=\"6\" r=\"5\" stroke=\"currentColor\" stroke-width=\"1.5\" fill=\"none\" opacity=\"0.25\"/>\n        <path d=\"M6 1A5 5 0 0 1 11 6\" stroke=\"currentColor\" stroke-width=\"1.5\" fill=\"none\" stroke-linecap=\"round\"/>\n      </svg>\n    </div>\n\n    <div><svg xmlns=\"http://www.w3.org/2000/svg\" width=\"20\" height=\"16\" viewBox=\"0 0 20 16\" fill=\"none\">\n      <rect width=\"20\" height=\"16\" rx=\"4\" fill=\"#F0F0F1\"/>\n      <path d=\"M13.5 3.5H6.5C5.95 3.5 5.5 3.95 5.5 4.5V11.5C5.5 12.05 5.95 12.5 6.5 12.5H13.5C14.05 12.5 14.5 12.05 14.5 11.5V4.5C14.5 3.95 14.05 3.5 13.5 3.5ZM13.5 4.5V6H6.5V4.5H13.5ZM13.5 7V9H6.5V7H13.5ZM6.5 11.5V10H13.5V11.5H6.5Z\" fill=\"#212122\"/>\n      </svg></div>\n\n    <!-- Step Number and Title -->\n    <div class=\"cqa-font-bold cqa-flex-1 cqa-text-[#334155] cqa-text-[11px] cqa-leading-[13px]\">\n      {{ stepNumber }}. <span [innerHTML]=\"title\"></span>    \n      \n      <span class=\"cqa-ml-1 cqa-px-1.5 cqa-py-0.5 cqa-rounded-full cqa-font-medium cqa-text-[#212122] cqa-bg-[#F0F0F1] cqa-text-[10px] cqa-leading-[15px] cqa-min-w-max\">\n        Database\n      </span>\n\n    </div>\n    <div class=\"cqa-flex cqa-items-center cqa-gap-1\">\n      <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#9CA3AF]\">\n        {{ formatDuration(duration) }}\n      </span>\n      <svg [class.cqa-rotate-180]=\"isExpanded\" class=\"cqa-transition-transform\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M3.5 5L7 8.5L10.5 5\" stroke=\"#9CA3AF\" stroke-width=\"0.833333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n    </div>\n  </div>\n\n  <!-- Expanded Content -->\n  <div *ngIf=\"isExpanded && dbTestResult\" class=\"cqa-p-2 !cqa-pl-9 !cqa-pr-6 cqa-bg-white\">\n    <!-- Summary Bar -->\n    <div class=\"cqa-mb-1.5 cqa-p-3 cqa-bg-[#FCFCFC] cqa-rounded-[6px] cqa-flex cqa-flex-col cqa-gap-2\" style=\"border: 1px solid #E5E7EB;\">\n      <div class=\"cqa-flex cqa-items-center cqa-gap-4\">\n        <!-- Status Badge -->\n        <cqa-badge\n          *ngIf=\"getStatus().toLowerCase() === 'failure' || getStatus().toLowerCase() === 'failed'\"\n          label=\"FAIL\"\n          backgroundColor=\"#DC2626\"\n          textColor=\"#FFFFFF\"\n          size=\"small\">\n        </cqa-badge>\n        <cqa-badge\n          *ngIf=\"getStatus().toLowerCase() === 'success'\"\n          label=\"PASS\"\n          backgroundColor=\"#22C55E\"\n          textColor=\"#FFFFFF\"\n          size=\"small\">\n        </cqa-badge>\n\n        <div class=\"cqa-h-[19px] cqa-w-[1px] cqa-bg-[#E5E7EB]\"></div>\n        \n        <!-- Summary Stats -->\n        <div class=\"cqa-flex cqa-items-center cqa-gap-4 cqa-text-[11px] cqa-leading-[16px] cqa-text-[#6B7280]\">\n          <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Total Time: <span class=\"cqa-text-[#4B5563]\">{{ formatDuration(duration) }}</span></span>\n          <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Queries: <span class=\"cqa-text-[#4B5563]\">{{ cachedQueryResults?.length || 0 }}</span></span>\n          <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Assertions: <span class=\"cqa-text-[#4B5563]\">{{ dbTestResult?.assertionResults?.length || 0 }}</span></span>\n          <span class=\"cqa-font-medium cqa-text-[10px] cqa-text-[#111827]\">Rows Returned: <span class=\"cqa-text-[#4B5563]\">{{ getTotalRowsReturned() }}</span></span>\n        </div>\n      </div>\n\n      <!-- Failure Banner -->\n      <div \n        *ngIf=\"getFailureMessage()\"\n        class=\"cqa-flex cqa-px-2 cqa-py-1 cqa-bg-[#FEF2F2] cqa-rounded-[4px]\" style=\"border: 1px solid #FEE2E2;\">\n        <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#B91C1C]\">\n          {{ getFailureMessage() }}\n        </span>\n      </div>\n    </div>\n\n    <!-- Database Environment Section -->\n    <div *ngIf=\"dbConfig\" class=\"cqa-mb-1.5\">\n      <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Database environment</div>\n      <div class=\"cqa-bg-white cqa-rounded-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-px-4 cqa-py-2 cqa-relative cqa-flex cqa-flex-wrap cqa-gap-3\">\n        <!-- Environment -->\n        <div class=\"\">\n          <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#45556C] cqa-mb-1\">Environment</div>\n          <div class=\"cqa-flex cqa-items-center cqa-gap-4\">\n            <span class=\"cqa-text-[10px] cqa-leading-[15px] cqa-text-[#0F172B]\">{{ dbConfig.name }}</span>\n            <span (click)=\"copyEnvironment()\" class=\"cqa-cursor-pointer\">\n              <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M5.33332 10.6667H3.99999C3.64637 10.6667 3.30723 10.5263 3.05718 10.2762C2.80713 10.0262 2.66666 9.68704 2.66666 9.33341V4.00008C2.66666 3.64646 2.80713 3.30732 3.05718 3.05727C3.30723 2.80722 3.64637 2.66675 3.99999 2.66675H9.33332C9.68695 2.66675 10.0261 2.80722 10.2761 3.05727C10.5262 3.30732 10.6667 3.64646 10.6667 4.00008V5.33341M6.66666 13.3334H12C12.3536 13.3334 12.6928 13.1929 12.9428 12.9429C13.1928 12.6928 13.3333 12.3537 13.3333 12.0001V6.66675C13.3333 6.31313 13.1928 5.97399 12.9428 5.72394C12.6928 5.47389 12.3536 5.33341 12 5.33341H6.66666C6.31303 5.33341 5.9739 5.47389 5.72385 5.72394C5.4738 5.97399 5.33332 6.31313 5.33332 6.66675V12.0001C5.33332 12.3537 5.4738 12.6928 5.72385 12.9429C5.9739 13.1929 6.31303 13.3334 6.66666 13.3334Z\" stroke=\"#636363\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n            </span>\n          </div>\n        </div>\n        \n        <div class=\"\">\n          <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#45556C] cqa-mb-1\">Type</div>\n          <div class=\"cqa-text-[10px] cqa-leading-[15px] cqa-text-[#0F172B]\">{{ dbConfig.dbType }}</div>\n        </div>\n      </div>\n    </div>\n\n    <div class=\"cqa-mb-1.5\">\n      <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Query Execution</div>\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-px-4 cqa-py-2 cqa-bg-white cqa-rounded-[10px]\" style=\"border: 1px solid #E2E8F0;\">\n        <cqa-db-query-execution-item\n          *ngFor=\"let queryItem of cachedQueryResults; let i = index; trackBy: trackByQueryKey\"\n          [queryNumber]=\"i + 1\"\n          [queryKey]=\"queryItem.key\"\n          [queryResult]=\"queryItem.result\"\n          [status]=\"getQueryStatus(queryItem.key)\"\n          [variableName]=\"queryItem.key\">\n        </cqa-db-query-execution-item>\n      </div>\n    </div>\n\n    <!-- Verification Section -->\n    <div *ngIf=\"dbTestResult.assertionResults && dbTestResult.assertionResults.length > 0\">\n      <div class=\"cqa-text-[12px] cqa-font-semibold cqa-text-[#0B0B0B] cqa-mb-2\">Verification</div>      \n      <!-- Verification Table -->\n      <div class=\"cqa-bg-white cqa-rounded-[10px] cqa-px-4 cqa-py-2 cqa-overflow-hidden\" style=\"border: 1px solid #E2E8F0;\">\n        <!-- Segment Control for Filters -->\n        <cqa-segment-control\n          [segments]=\"filterSegments\"\n          [value]=\"verificationFilter\"\n          (valueChange)=\"onFilterChange($event)\">\n        </cqa-segment-control>\n        <cqa-table-template\n          [columns]=\"verificationTableColumns\"\n          [data]=\"getTableData()\"\n          [pageIndex]=\"pageIndex\"\n          [pageSize]=\"pageSize\"\n          [isEmptyState]=\"isEmptyTable\"\n          [emptyStateConfig]=\"emptyStateConfig\"\n          [showSearchBar]=\"false\"\n          [showFilterButton]=\"false\"\n          [showSettingsButton]=\"false\"\n          [showAutoRefreshButton]=\"false\"\n          [showOtherButton]=\"false\"\n          [showFilterPanel]=\"false\"\n          [serverSidePagination]=\"false\"\n          (pageChange)=\"onPageChange($event)\">\n        </cqa-table-template>\n      </div>\n    </div>\n\n    <!-- Timing Breakdown -->\n    <div *ngIf=\"timingBreakdown\" class=\"cqa-flex cqa-items-center cqa-justify-end cqa-gap-5 cqa-pt-4 cqa-px-4 cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-text-[#9CA3AF]\">\n      <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n        <div><svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 11C8.76142 11 11 8.76142 11 6C11 3.23858 8.76142 1 6 1C3.23858 1 1 3.23858 1 6C1 8.76142 3.23858 11 6 11Z\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/><path d=\"M6 3V6L8 7\" stroke=\"#9CA3AF\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg></div>\n        <span>Timing breakdown</span>\n      </div>\n      <span class=\"cqa-text-dialog-muted cqa-flex cqa-items-center cqa-gap-3\">\n        <div>\n          App <span class=\"cqa-text-[#374151]\">{{ formatDuration(timingBreakdown.app) }}</span>\n        </div>\n        <div><svg width=\"1\" height=\"11\" viewBox=\"0 0 1 11\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M-3.8147e-06 10.32V-7.15256e-07H0.959996V10.32H-3.8147e-06Z\" fill=\"#E5E7EB\"/></svg></div>\n        <div>\n          Tool <span class=\"cqa-text-[#374151]\">{{ formatDuration(timingBreakdown.tool) }}</span>\n        </div>\n      </span>\n    </div>\n  </div>\n</div>\n\n"]}
@@ -2,4 +2,4 @@
2
2
  * Models and interfaces for execution screen step components
3
3
  */
4
4
  export {};
5
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution-step.models.js","sourceRoot":"","sources":["../../../../../src/lib/execution-screen/execution-step.models.ts"],"names":[],"mappings":"AAAA;;GAEG","sourcesContent":["/**\n * Models and interfaces for execution screen step components\n */\n\nexport type StepStatus = 'success' | 'failed' | 'pending' | 'running' | 'SUCCESS' | 'FAILED' | 'PENDING' | 'RUNNING';\n\nexport interface TimingBreakdown {\n  app: number; // in seconds\n  tool: number; // in seconds\n}\n\nexport interface SubStep {\n  id: string;\n  description: string;\n  status: StepStatus;\n  duration: number; // in seconds\n  timestamp?: string; // optional timestamp like \"00:12\"\n}\n\nexport interface LoopIteration {\n  id: string;\n  label: string; // e.g., \"Iter 2 (production)\"\n  status: StepStatus;\n  subSteps?: SubStep[];\n}\n\nexport interface ConditionBranch {\n  type: 'if' | 'else' | 'else if';\n  label: string; // e.g., \"Premium user\" or \"Free user\"\n  executed: boolean;\n  subSteps?: SubStep[];\n  branchStepId?: string | number; // ID of the branch step to load children from\n  isIfBranch?: boolean; // Whether this is the main IF branch\n  branchStep: any;\n}\n\nexport interface ScreenshotData {\n  baseline?: string; // URL or base64\n  current?: string; // URL or base64\n  difference?: string; // URL or base64\n}\n\nexport interface LogEntry {\n  level: 'info' | 'warning' | 'error';\n  message: string;\n  timestamp?: string;\n}\n\nexport interface FailureDetails {\n  testStepId: string;\n  expected: string;\n  actual: string;\n  failedAction?: string; // The action that failed\n  aiFixApplied?: boolean;\n  aiFixMessage?: string;\n  screenshots?: ScreenshotData;\n  logs?: LogEntry[];\n}\n\nexport interface BaseStepConfig {\n  id: string;\n  testStepResultId: string;\n  stepNumber: string; // e.g., \"1\", \"2.1\", \"3.2.1\"\n  title: string;\n  status: StepStatus;\n  duration: number; // total duration in seconds\n  timingBreakdown?: TimingBreakdown;\n  expanded?: boolean; // whether the step is expanded by default\n  failureDetails?: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n  // Additional properties for live execution and runtime use\n  selectedIterationId?: string; // which iteration to display\n}\n\nexport interface SelfHealAnalysisData {\n  originalLocator: string;\n  healedLocator: string;\n  confidence: number; // 0-100\n  healMethod: string; // e.g., \"Semantic Attribute Matching\"\n}\n\nexport type SelfHealAction = 'accept' | 'reject' | 'modify-accept';\n\nexport interface BasicStepConfig extends BaseStepConfig {\n  type: 'basic';\n  subSteps: SubStep[];\n  selfHealAnalysis?: SelfHealAnalysisData;\n  selfHealed?: boolean; // Indicates if self-healing was applied\n  selfHealDuration?: number; // Duration of self-healing process\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface StepGroupConfig extends BaseStepConfig {\n  type: 'step-group';\n  groupName: string; // e.g., \"Navigation\", \"Checkout flow\"\n  steps: ExecutionStepConfig[]; // nested steps\n}\n\nexport interface LoopStepConfig extends BaseStepConfig {\n  type: 'loop';\n  loopType?: 'for' | 'while';\n  iterations: LoopIteration[];\n  selectedIterationId?: string; // which iteration to display\n  defaultIteration?: 'first' | 'last'; // default iteration to show\n  nestedSteps?: ExecutionStepConfig[]; // steps nested within the loop\n  showViewAllIterations?: boolean;\n  subSteps?: SubStep[];\n  iterationData?: any; // data for each iteration in live execution\n}\n\nexport interface ConditionStepConfig extends BaseStepConfig {\n  type: 'condition';\n  conditionText: string; // e.g., \"if user type is 'Premium'\"\n  branches: ConditionBranch[];\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface FailedStepConfig extends BaseStepConfig {\n  type: 'failed';\n  subSteps: SubStep[];\n  failureDetails: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n}\n\nexport interface AIAgentAction {\n  id: string;\n  description: string;\n  type: 'extract' | 'validate' | 'AI_AGENT_ACTION' | 'TYPE' | 'CLICK';\n  status: StepStatus;\n  confidence?: number; // 0-100\n  duration: number;\n}\n\nexport interface AIAgentStepConfig extends BaseStepConfig {\n  type: 'ai-agent';\n  prompt: string;\n  optimizedRun?: boolean;\n  actionCount?: number;\n  actions: AIAgentAction[];\n  selectedTab?: 'action-trace' | 'planner-timeline' | 'ai-reasoning';\n}\n\nexport interface ApiAssertion {\n  id: string;\n  description: string;\n  status: 'passed' | 'failed';\n  expected: string;\n  actual: string;\n}\n\nexport interface ApiStepConfig extends BaseStepConfig {\n  type: 'api';\n  method: string; // GET, POST, PUT, DELETE, etc.\n  endpoint: string;\n  statusCode: number;\n  responseTime: number; // in milliseconds\n  requestBody?: any;\n  responseBody?: any;\n  requestHeaders?: any;\n  responseHeaders?: any;\n  assertions?: ApiAssertion[];\n  initialActions?: SubStep[]; // Actions before the API call\n}\n\nexport interface FileDownloadStepConfig extends BaseStepConfig {\n  type: 'file-download';\n  fileName: string;\n  fileType: string; // PDF, CSV, etc.\n  fileSize?: string; // e.g., \"2.4 MB\"\n  filePath?: string;\n  downloaded: boolean;\n}\n\nexport interface ExtractedField {\n  label: string;\n  value: string;\n  confidence: number; // 0-100\n}\n\nexport interface VerificationCheck {\n  id: string;\n  description: string;\n  status: 'pass' | 'fail';\n  expected: string;\n  actual: string;\n}\n\nexport interface DocumentVerificationStepConfig extends BaseStepConfig {\n  type: 'document-verification';\n  documentScreenshot?: string; // URL or base64\n  extractedFields: ExtractedField[];\n  verificationChecks: VerificationCheck[];\n}\n\nexport interface LiveSubStep {\n  id: string;\n  text: string; // Text from socket\n  isRunning: boolean; // true = show loader, false = show icon\n  duration?: number; // in seconds\n}\n\nexport interface LiveExecutionStepConfig extends BaseStepConfig {\n  type: 'live-execution';\n  subSteps: LiveSubStep[];\n}\n\nexport interface PrerequisiteItem {\n  id: string;\n  title: string; // e.g., \"Prerequisite 1: Login flow\"\n  status: StepStatus;\n  duration: number; // in seconds\n  stepConfig?: ExecutionStepConfig; // Optional: full step config to view details\n}\n\nexport type ExecutionStepConfig =\n  | BasicStepConfig\n  | StepGroupConfig\n  | LoopStepConfig\n  | ConditionStepConfig\n  | FailedStepConfig\n  | AIAgentStepConfig\n  | ApiStepConfig\n  | FileDownloadStepConfig\n  | DocumentVerificationStepConfig\n  | LiveExecutionStepConfig;\n"]}
5
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution-step.models.js","sourceRoot":"","sources":["../../../../../src/lib/execution-screen/execution-step.models.ts"],"names":[],"mappings":"AAAA;;GAEG","sourcesContent":["/**\n * Models and interfaces for execution screen step components\n */\n\nexport type StepStatus = 'success' | 'failed' | 'pending' | 'running' | 'SUCCESS' | 'FAILED' | 'PENDING' | 'RUNNING';\n\nexport interface TimingBreakdown {\n  app: number; // in seconds\n  tool: number; // in seconds\n}\n\nexport interface SubStep {\n  id: string;\n  description: string;\n  status: StepStatus;\n  duration: number; // in seconds\n  timestamp?: string; // optional timestamp like \"00:12\"\n}\n\nexport interface LoopIteration {\n  id: string;\n  label: string; // e.g., \"Iter 2 (production)\"\n  status: StepStatus;\n  subSteps?: SubStep[];\n}\n\nexport interface ConditionBranch {\n  type: 'if' | 'else' | 'else if';\n  label: string; // e.g., \"Premium user\" or \"Free user\"\n  executed: boolean;\n  subSteps?: SubStep[];\n  branchStepId?: string | number; // ID of the branch step to load children from\n  isIfBranch?: boolean; // Whether this is the main IF branch\n  branchStep: any;\n}\n\nexport interface ScreenshotData {\n  baseline?: string; // URL or base64\n  current?: string; // URL or base64\n  difference?: string; // URL or base64\n}\n\nexport interface LogEntry {\n  level: 'info' | 'warning' | 'error';\n  message: string;\n  timestamp?: string;\n}\n\nexport interface FailureDetails {\n  testStepId: string;\n  expected: string;\n  actual: string;\n  failedAction?: string; // The action that failed\n  aiFixApplied?: boolean;\n  aiFixMessage?: string;\n  screenshots?: ScreenshotData;\n  logs?: LogEntry[];\n}\n\nexport interface BaseStepConfig {\n  id: string;\n  testStepResultId: string;\n  stepNumber: string; // e.g., \"1\", \"2.1\", \"3.2.1\"\n  title: string;\n  status: StepStatus;\n  duration: number; // total duration in seconds\n  timingBreakdown?: TimingBreakdown;\n  expanded?: boolean; // whether the step is expanded by default\n  failureDetails?: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n  // Additional properties for live execution and runtime use\n  selectedIterationId?: string; // which iteration to display\n}\n\nexport interface SelfHealAnalysisData {\n  originalLocator: string;\n  healedLocator: string;\n  confidence: number; // 0-100\n  healMethod: string; // e.g., \"Semantic Attribute Matching\"\n}\n\nexport type SelfHealAction = 'accept' | 'reject' | 'modify-accept';\n\nexport interface BasicStepConfig extends BaseStepConfig {\n  type: 'basic';\n  subSteps: SubStep[];\n  selfHealAnalysis?: SelfHealAnalysisData;\n  selfHealed?: boolean; // Indicates if self-healing was applied\n  selfHealDuration?: number; // Duration of self-healing process\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface StepGroupConfig extends BaseStepConfig {\n  type: 'step-group';\n  groupName: string; // e.g., \"Navigation\", \"Checkout flow\"\n  steps: ExecutionStepConfig[]; // nested steps\n}\n\nexport interface LoopStepConfig extends BaseStepConfig {\n  type: 'loop';\n  loopType?: 'for' | 'while';\n  iterations: LoopIteration[];\n  selectedIterationId?: string; // which iteration to display\n  defaultIteration?: 'first' | 'last'; // default iteration to show\n  nestedSteps?: ExecutionStepConfig[]; // steps nested within the loop\n  showViewAllIterations?: boolean;\n  subSteps?: SubStep[];\n  iterationData?: any; // data for each iteration in live execution\n}\n\nexport interface ConditionStepConfig extends BaseStepConfig {\n  type: 'condition';\n  conditionText: string; // e.g., \"if user type is 'Premium'\"\n  branches: ConditionBranch[];\n  nestedSteps?: ExecutionStepConfig[]; // nested child steps for recursive rendering\n}\n\nexport interface FailedStepConfig extends BaseStepConfig {\n  type: 'failed';\n  subSteps: SubStep[];\n  failureDetails: FailureDetails;\n  reasoning?: string[];\n  confidence?: string;\n}\n\nexport interface AIAgentAction {\n  id: string;\n  description: string;\n  type: 'extract' | 'validate' | 'AI_AGENT_ACTION' | 'TYPE' | 'CLICK';\n  status: StepStatus;\n  confidence?: number; // 0-100\n  duration: number;\n}\n\nexport interface AIAgentStepConfig extends BaseStepConfig {\n  type: 'ai-agent';\n  prompt: string;\n  optimizedRun?: boolean;\n  actionCount?: number;\n  actions: AIAgentAction[];\n  selectedTab?: 'action-trace' | 'planner-timeline' | 'ai-reasoning';\n}\n\nexport interface ApiAssertion {\n  id: string;\n  description: string;\n  status: 'passed' | 'failed';\n  expected: string;\n  actual: string;\n}\n\nexport interface ApiStepConfig extends BaseStepConfig {\n  type: 'api';\n  method: string; // GET, POST, PUT, DELETE, etc.\n  endpoint: string;\n  statusCode: number;\n  responseTime: number; // in milliseconds\n  requestBody?: any;\n  responseBody?: any;\n  requestHeaders?: any;\n  responseHeaders?: any;\n  assertions?: ApiAssertion[];\n  initialActions?: SubStep[]; // Actions before the API call\n}\n\nexport interface FileDownloadStepConfig extends BaseStepConfig {\n  type: 'file-download';\n  fileName: string;\n  fileType: string; // PDF, CSV, etc.\n  fileSize?: string; // e.g., \"2.4 MB\"\n  filePath?: string;\n  downloaded: boolean;\n}\n\nexport interface ExtractedField {\n  label: string;\n  value: string;\n  confidence: number; // 0-100\n}\n\nexport interface VerificationCheck {\n  id: string;\n  description: string;\n  status: 'pass' | 'fail';\n  expected: string;\n  actual: string;\n}\n\nexport interface DocumentVerificationStepConfig extends BaseStepConfig {\n  type: 'document-verification';\n  documentScreenshot?: string; // URL or base64\n  extractedFields: ExtractedField[];\n  verificationChecks: VerificationCheck[];\n}\n\nexport interface LiveSubStep {\n  id: string;\n  text: string; // Text from socket\n  isRunning: boolean; // true = show loader, false = show icon\n  duration?: number; // in seconds\n}\n\nexport interface LiveExecutionStepConfig extends BaseStepConfig {\n  type: 'live-execution';\n  subSteps: LiveSubStep[];\n}\n\nexport interface DbQueryResult {\n  data: any[];\n  query: string;\n  executedAt?: string;\n  duration?: number; // in seconds\n  [key: string]: any; // For dynamic keys like \"Demo_user\"\n}\n\nexport interface DbAssertionResult {\n  variableName: string;\n  jsonPath: string;\n  expectedValue: any;\n  actualValue: any;\n  verificationType: string;\n  expectedType: string;\n  passed: boolean;\n}\n\nexport interface DbTestResult {\n  success: boolean;\n  results: { [key: string]: DbQueryResult };\n  assertionResults: DbAssertionResult[];\n  allAssertionsPassed: boolean;\n}\n\nexport interface DbVerificationStepConfig extends BaseStepConfig {\n  type: 'db-verification';\n  dbTestResult: DbTestResult;\n  dbConfig?: {\n    name: string; // Environment name\n    dbType: string; // Database type (MySQL, SQLServer, etc.)\n  };\n}\n\nexport interface PrerequisiteItem {\n  id: string;\n  title: string; // e.g., \"Prerequisite 1: Login flow\"\n  status: StepStatus;\n  duration: number; // in seconds\n  stepConfig?: ExecutionStepConfig; // Optional: full step config to view details\n}\n\nexport type ExecutionStepConfig =\n  | BasicStepConfig\n  | StepGroupConfig\n  | LoopStepConfig\n  | ConditionStepConfig\n  | FailedStepConfig\n  | AIAgentStepConfig\n  | ApiStepConfig\n  | FileDownloadStepConfig\n  | DocumentVerificationStepConfig\n  | LiveExecutionStepConfig\n  | DbVerificationStepConfig;\n"]}