create-appraisejs 0.1.7 → 0.1.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +45 -45
- package/dist/cli.js +2 -2
- package/dist/cli.js.map +1 -1
- package/dist/copy-template.d.ts.map +1 -1
- package/dist/copy-template.js.map +1 -1
- package/package.json +69 -67
- package/templates/default/.vscode/settings.json +5 -0
- package/templates/default/appraisejs.config.json +1 -1
- package/templates/default/components.json +24 -24
- package/templates/default/cucumber.mjs +16 -0
- package/templates/default/eslint.config.mjs +15 -15
- package/templates/default/next.config.ts +13 -7
- package/templates/default/package-lock.json +13732 -14321
- package/templates/default/package.json +11 -9
- package/templates/default/postcss.config.mjs +8 -8
- package/templates/default/prisma/migrations/20251104113456_add_type_for_template_step_groups/migration.sql +16 -16
- package/templates/default/prisma/migrations/20251104170946_add_tags_to_test_suite_and_test_case/migration.sql +27 -27
- package/templates/default/prisma/migrations/20251112190024_add_cascade_delete_to_test_run_test_case/migration.sql +17 -17
- package/templates/default/prisma/migrations/20251113181100_add_test_run_log/migration.sql +12 -12
- package/templates/default/prisma/migrations/20251119191838_add_tag_type/migration.sql +28 -28
- package/templates/default/prisma/migrations/20251121164059_add_conflict_resolution/migration.sql +12 -12
- package/templates/default/prisma/migrations/20251223183400_add_report_model_to_db_schema/migration.sql +10 -10
- package/templates/default/prisma/migrations/20251223183637_add_report_test_case_entity_for_storing_test_results_for_individual_test_cases/migration.sql +10 -10
- package/templates/default/prisma/migrations/20251224083549_add_comprehensive_report_storage/migration.sql +108 -108
- package/templates/default/prisma/migrations/20251229194422_migrate_duration_to_string/migration.sql +55 -55
- package/templates/default/prisma/migrations/20251230124637_add_unique_constraint_to_test_run_name/migration.sql +27 -27
- package/templates/default/prisma/migrations/20260115094436_add_dashboard_metrics/migration.sql +59 -59
- package/templates/default/prisma/migrations/20260127172022_add_cascade_delete_to_step_parameters/migration.sql +34 -34
- package/templates/default/prisma/schema.prisma +554 -554
- package/templates/default/scripts/regenerate-features.ts +94 -94
- package/templates/default/scripts/setup-env.ts +19 -19
- package/templates/default/scripts/sync-all.ts +341 -341
- package/templates/default/scripts/sync-appraise-base-template.ts +52 -2
- package/templates/default/scripts/sync-environments.ts +323 -323
- package/templates/default/scripts/sync-locator-groups.ts +413 -413
- package/templates/default/scripts/sync-locators.ts +402 -402
- package/templates/default/scripts/sync-modules.ts +349 -349
- package/templates/default/scripts/sync-tags.ts +292 -292
- package/templates/default/scripts/sync-template-step-groups.ts +399 -399
- package/templates/default/scripts/sync-template-steps.ts +806 -806
- package/templates/default/scripts/sync-test-cases.ts +905 -905
- package/templates/default/scripts/sync-test-suites.ts +411 -411
- package/templates/default/src/actions/conflict/conflict.action.ts +33 -33
- package/templates/default/src/actions/dashboard/dashboard-actions.ts +240 -240
- package/templates/default/src/actions/environments/environment-actions.ts +205 -205
- package/templates/default/src/actions/locator/locator-actions.ts +547 -547
- package/templates/default/src/actions/locator-groups/locator-group-actions.ts +344 -344
- package/templates/default/src/actions/modules/module-actions.ts +133 -133
- package/templates/default/src/actions/reports/report-actions.ts +613 -613
- package/templates/default/src/actions/review/review-actions.ts +147 -147
- package/templates/default/src/actions/tags/tag-actions.ts +104 -104
- package/templates/default/src/actions/template-step/template-step-actions.ts +332 -332
- package/templates/default/src/actions/template-step-group/template-step-group-actions.ts +278 -278
- package/templates/default/src/actions/template-test-case/template-test-case-actions.ts +238 -238
- package/templates/default/src/actions/test-case/test-case-actions.ts +419 -419
- package/templates/default/src/actions/test-run/test-run-actions.ts +1185 -1185
- package/templates/default/src/actions/test-suite/test-suite-actions.ts +253 -253
- package/templates/default/src/actions/user/user-actions.ts +13 -13
- package/templates/default/src/app/(base)/environments/create/page.tsx +28 -28
- package/templates/default/src/app/(base)/environments/environment-form.tsx +219 -219
- package/templates/default/src/app/(base)/environments/environment-table-columns.tsx +96 -96
- package/templates/default/src/app/(base)/environments/environment-table.tsx +24 -24
- package/templates/default/src/app/(base)/environments/modify/[id]/page.tsx +46 -46
- package/templates/default/src/app/(base)/environments/page.tsx +59 -59
- package/templates/default/src/app/(base)/layout.tsx +10 -10
- package/templates/default/src/app/(base)/locator-groups/create/page.tsx +44 -44
- package/templates/default/src/app/(base)/locator-groups/locator-group-form.tsx +215 -215
- package/templates/default/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -77
- package/templates/default/src/app/(base)/locator-groups/locator-group-table.tsx +28 -28
- package/templates/default/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -46
- package/templates/default/src/app/(base)/locator-groups/page.tsx +61 -61
- package/templates/default/src/app/(base)/locators/create/page.tsx +38 -38
- package/templates/default/src/app/(base)/locators/locator-form.tsx +163 -163
- package/templates/default/src/app/(base)/locators/locator-table-columns.tsx +73 -90
- package/templates/default/src/app/(base)/locators/locator-table.tsx +28 -28
- package/templates/default/src/app/(base)/locators/modify/[id]/page.tsx +45 -45
- package/templates/default/src/app/(base)/locators/page.tsx +65 -65
- package/templates/default/src/app/(base)/locators/sync-locators-button.tsx +66 -66
- package/templates/default/src/app/(base)/modules/create/page.tsx +34 -34
- package/templates/default/src/app/(base)/modules/modify/[id]/page.tsx +46 -46
- package/templates/default/src/app/(base)/modules/module-form.tsx +126 -126
- package/templates/default/src/app/(base)/modules/module-table-columns.tsx +85 -85
- package/templates/default/src/app/(base)/modules/module-table.tsx +24 -24
- package/templates/default/src/app/(base)/modules/page.tsx +59 -59
- package/templates/default/src/app/(base)/reports/[id]/page.tsx +517 -517
- package/templates/default/src/app/(base)/reports/duration-chart.tsx +33 -33
- package/templates/default/src/app/(base)/reports/feature-chart.tsx +78 -78
- package/templates/default/src/app/(base)/reports/overview-chart.tsx +46 -46
- package/templates/default/src/app/(base)/reports/page.tsx +98 -98
- package/templates/default/src/app/(base)/reports/report-metric-card.tsx +16 -16
- package/templates/default/src/app/(base)/reports/report-table-columns.tsx +189 -189
- package/templates/default/src/app/(base)/reports/report-table.tsx +72 -72
- package/templates/default/src/app/(base)/reports/report-view-table-columns.tsx +131 -131
- package/templates/default/src/app/(base)/reports/report-view-table.tsx +82 -82
- package/templates/default/src/app/(base)/reports/test-cases/page.tsx +42 -42
- package/templates/default/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +115 -115
- package/templates/default/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -27
- package/templates/default/src/app/(base)/reports/test-suites/page.tsx +42 -42
- package/templates/default/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +79 -79
- package/templates/default/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +27 -27
- package/templates/default/src/app/(base)/reports/view-logs-button.tsx +60 -60
- package/templates/default/src/app/(base)/reviews/create/page.tsx +26 -26
- package/templates/default/src/app/(base)/reviews/created-reviews-table.tsx +15 -15
- package/templates/default/src/app/(base)/reviews/modify/[id]/page.tsx +26 -26
- package/templates/default/src/app/(base)/reviews/page.tsx +26 -26
- package/templates/default/src/app/(base)/reviews/review/[id]/page.tsx +26 -26
- package/templates/default/src/app/(base)/reviews/review-form.tsx +11 -11
- package/templates/default/src/app/(base)/reviews/review-table-by-creator-columns.tsx +9 -9
- package/templates/default/src/app/(base)/reviews/review-table-by-reviewer-columns.tsx +9 -9
- package/templates/default/src/app/(base)/reviews/reviewer-reviews-table.tsx +15 -15
- package/templates/default/src/app/(base)/tags/create/page.tsx +39 -39
- package/templates/default/src/app/(base)/tags/modify/[id]/page.tsx +50 -50
- package/templates/default/src/app/(base)/tags/page.tsx +58 -58
- package/templates/default/src/app/(base)/tags/tag-form.tsx +147 -147
- package/templates/default/src/app/(base)/tags/tag-table-columns.tsx +63 -63
- package/templates/default/src/app/(base)/tags/tag-table.tsx +29 -29
- package/templates/default/src/app/(base)/template-step-groups/create/page.tsx +28 -28
- package/templates/default/src/app/(base)/template-step-groups/modify/[id]/page.tsx +45 -45
- package/templates/default/src/app/(base)/template-step-groups/page.tsx +60 -60
- package/templates/default/src/app/(base)/template-step-groups/template-step-group-form.tsx +167 -167
- package/templates/default/src/app/(base)/template-step-groups/template-step-group-table-columns.tsx +89 -89
- package/templates/default/src/app/(base)/template-step-groups/template-step-group-table.tsx +32 -32
- package/templates/default/src/app/(base)/template-steps/create/page.tsx +37 -37
- package/templates/default/src/app/(base)/template-steps/modify/[id]/page.tsx +49 -49
- package/templates/default/src/app/(base)/template-steps/page.tsx +59 -59
- package/templates/default/src/app/(base)/template-steps/paramChip.tsx +213 -213
- package/templates/default/src/app/(base)/template-steps/template-step-form.tsx +384 -384
- package/templates/default/src/app/(base)/template-steps/template-step-table-columns.tsx +158 -158
- package/templates/default/src/app/(base)/template-steps/template-step-table.tsx +24 -24
- package/templates/default/src/app/(base)/template-test-cases/create/page.tsx +56 -56
- package/templates/default/src/app/(base)/template-test-cases/modify/[id]/page.tsx +89 -89
- package/templates/default/src/app/(base)/template-test-cases/page.tsx +58 -58
- package/templates/default/src/app/(base)/template-test-cases/template-test-case-flow.tsx +84 -84
- package/templates/default/src/app/(base)/template-test-cases/template-test-case-form.tsx +262 -262
- package/templates/default/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -76
- package/templates/default/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -32
- package/templates/default/src/app/(base)/test-cases/create/page.tsx +76 -76
- package/templates/default/src/app/(base)/test-cases/create-from-template/generate/[id]/page.tsx +96 -96
- package/templates/default/src/app/(base)/test-cases/create-from-template/page.tsx +38 -38
- package/templates/default/src/app/(base)/test-cases/create-from-template/template-selection-form.tsx +73 -73
- package/templates/default/src/app/(base)/test-cases/modify/[id]/page.tsx +106 -106
- package/templates/default/src/app/(base)/test-cases/page.tsx +60 -60
- package/templates/default/src/app/(base)/test-cases/test-case-flow.tsx +82 -82
- package/templates/default/src/app/(base)/test-cases/test-case-form.tsx +395 -395
- package/templates/default/src/app/(base)/test-cases/test-case-table-columns.tsx +90 -90
- package/templates/default/src/app/(base)/test-cases/test-case-table.tsx +35 -35
- package/templates/default/src/app/(base)/test-runs/[id]/page.tsx +56 -56
- package/templates/default/src/app/(base)/test-runs/create/page.tsx +47 -47
- package/templates/default/src/app/(base)/test-runs/page.tsx +60 -60
- package/templates/default/src/app/(base)/test-runs/test-run-form.tsx +508 -512
- package/templates/default/src/app/(base)/test-runs/test-run-table-columns.tsx +229 -229
- package/templates/default/src/app/(base)/test-runs/test-run-table.tsx +127 -127
- package/templates/default/src/app/(base)/test-suites/create/page.tsx +45 -45
- package/templates/default/src/app/(base)/test-suites/modify/[id]/page.tsx +55 -55
- package/templates/default/src/app/(base)/test-suites/page.tsx +82 -82
- package/templates/default/src/app/(base)/test-suites/test-suite-form.tsx +269 -269
- package/templates/default/src/app/(base)/test-suites/test-suite-table-columns.tsx +97 -97
- package/templates/default/src/app/(base)/test-suites/test-suite-table.tsx +29 -29
- package/templates/default/src/app/(dashboard-components)/app-drawer.tsx +187 -187
- package/templates/default/src/app/(dashboard-components)/data-card-grid.tsx +12 -12
- package/templates/default/src/app/(dashboard-components)/data-card.tsx +26 -26
- package/templates/default/src/app/(dashboard-components)/execution-health-panel.tsx +56 -56
- package/templates/default/src/app/(dashboard-components)/ongoing-test-runs-card.tsx +87 -87
- package/templates/default/src/app/(dashboard-components)/quick-actions-drawer.tsx +44 -44
- package/templates/default/src/app/api/test-runs/[runId]/download/route.ts +133 -133
- package/templates/default/src/app/api/test-runs/[runId]/logs/route.ts +420 -420
- package/templates/default/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.ts +146 -146
- package/templates/default/src/app/globals.css +147 -147
- package/templates/default/src/app/layout.tsx +171 -171
- package/templates/default/src/app/page.tsx +64 -64
- package/templates/default/src/assets/icons/empty-tube.tsx +23 -23
- package/templates/default/src/assets/icons/tube-plus.tsx +29 -29
- package/templates/default/src/components/base-node.tsx +21 -21
- package/templates/default/src/components/chart/pie-chart.tsx +73 -73
- package/templates/default/src/components/data-extraction/locator-inspector.tsx +460 -460
- package/templates/default/src/components/data-state/empty-state.tsx +40 -40
- package/templates/default/src/components/data-visualization/info-card.tsx +70 -70
- package/templates/default/src/components/data-visualization/info-grid.tsx +22 -22
- package/templates/default/src/components/devtools/providers.tsx +19 -13
- package/templates/default/src/components/diagram/button-edge.tsx +54 -54
- package/templates/default/src/components/diagram/dynamic-parameters.tsx +438 -438
- package/templates/default/src/components/diagram/edit-header-option.tsx +36 -36
- package/templates/default/src/components/diagram/flow-diagram.tsx +470 -470
- package/templates/default/src/components/diagram/node-form.tsx +262 -262
- package/templates/default/src/components/diagram/options-header-node.tsx +57 -57
- package/templates/default/src/components/diagram/template-step-combobox.tsx +155 -155
- package/templates/default/src/components/form/error-message.tsx +7 -7
- package/templates/default/src/components/kokonutui/smooth-tab.tsx +453 -453
- package/templates/default/src/components/loading-skeleton/data-table/data-table-skeleton.tsx +30 -30
- package/templates/default/src/components/loading-skeleton/form/button-skeleton.tsx +8 -8
- package/templates/default/src/components/loading-skeleton/form/icon-button-skeleton.tsx +8 -8
- package/templates/default/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -8
- package/templates/default/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -14
- package/templates/default/src/components/logo.tsx +15 -15
- package/templates/default/src/components/navigation/command-badge.tsx +34 -34
- package/templates/default/src/components/navigation/command-chain-input.tsx +51 -51
- package/templates/default/src/components/navigation/entity-search-command.tsx +116 -116
- package/templates/default/src/components/navigation/nav-card.tsx +31 -31
- package/templates/default/src/components/navigation/nav-command.tsx +508 -508
- package/templates/default/src/components/navigation/nav-link.tsx +60 -60
- package/templates/default/src/components/navigation/nav-menu-card-deck.tsx +112 -112
- package/templates/default/src/components/node-header.tsx +159 -159
- package/templates/default/src/components/reports/test-case-logs-modal.tsx +253 -253
- package/templates/default/src/components/table/table-actions.tsx +172 -172
- package/templates/default/src/components/test-run/download-logs-button.tsx +99 -99
- package/templates/default/src/components/test-run/log-viewer.tsx +445 -445
- package/templates/default/src/components/test-run/test-run-details.tsx +611 -611
- package/templates/default/src/components/test-run/test-run-header.tsx +149 -149
- package/templates/default/src/components/test-run/view-report-button.tsx +102 -102
- package/templates/default/src/components/theme/mode-toggle.tsx +54 -54
- package/templates/default/src/components/theme/theme-provider.tsx +8 -8
- package/templates/default/src/components/typography/page-header-subtitle.tsx +7 -7
- package/templates/default/src/components/typography/page-header.tsx +7 -7
- package/templates/default/src/components/ui/alert-dialog.tsx +106 -106
- package/templates/default/src/components/ui/alert.tsx +43 -43
- package/templates/default/src/components/ui/avatar.tsx +40 -40
- package/templates/default/src/components/ui/badge.tsx +29 -29
- package/templates/default/src/components/ui/button.tsx +47 -47
- package/templates/default/src/components/ui/calendar.tsx +158 -158
- package/templates/default/src/components/ui/card.tsx +43 -43
- package/templates/default/src/components/ui/checkbox.tsx +28 -28
- package/templates/default/src/components/ui/command.tsx +135 -135
- package/templates/default/src/components/ui/data-table-column-header.tsx +61 -61
- package/templates/default/src/components/ui/data-table-pagination.tsx +87 -87
- package/templates/default/src/components/ui/data-table-view-options.tsx +50 -50
- package/templates/default/src/components/ui/data-table.tsx +267 -267
- package/templates/default/src/components/ui/dialog.tsx +97 -97
- package/templates/default/src/components/ui/dropdown-menu.tsx +182 -182
- package/templates/default/src/components/ui/input.tsx +22 -22
- package/templates/default/src/components/ui/kbd.tsx +28 -28
- package/templates/default/src/components/ui/label.tsx +19 -19
- package/templates/default/src/components/ui/loading.tsx +12 -12
- package/templates/default/src/components/ui/multi-select-with-preview.tsx +116 -116
- package/templates/default/src/components/ui/multi-select.tsx +142 -142
- package/templates/default/src/components/ui/navigation-menu.tsx +120 -120
- package/templates/default/src/components/ui/popover.tsx +33 -33
- package/templates/default/src/components/ui/progress.tsx +25 -25
- package/templates/default/src/components/ui/radio-group.tsx +44 -44
- package/templates/default/src/components/ui/scroll-area.tsx +40 -40
- package/templates/default/src/components/ui/select.tsx +151 -144
- package/templates/default/src/components/ui/separator.tsx +22 -22
- package/templates/default/src/components/ui/skeleton.tsx +7 -7
- package/templates/default/src/components/ui/table.tsx +76 -76
- package/templates/default/src/components/ui/tabs.tsx +55 -55
- package/templates/default/src/components/ui/textarea.tsx +21 -21
- package/templates/default/src/components/ui/toast.tsx +113 -113
- package/templates/default/src/components/ui/toaster.tsx +26 -26
- package/templates/default/src/components/user-prompt/delete-prompt.tsx +87 -87
- package/templates/default/src/config/db-config.ts +10 -10
- package/templates/default/src/constants/form-opts/diagram/node-form.ts +30 -30
- package/templates/default/src/constants/form-opts/environment-form-opts.ts +24 -24
- package/templates/default/src/constants/form-opts/locator-form-opts.ts +20 -20
- package/templates/default/src/constants/form-opts/locator-group-form-opts.ts +28 -28
- package/templates/default/src/constants/form-opts/module-form-opts.ts +21 -21
- package/templates/default/src/constants/form-opts/review-form-opts.ts +23 -23
- package/templates/default/src/constants/form-opts/tag-form-opts.ts +42 -42
- package/templates/default/src/constants/form-opts/template-selection-form-opts.ts +16 -16
- package/templates/default/src/constants/form-opts/template-step-group-form-opts.ts +24 -24
- package/templates/default/src/constants/form-opts/template-test-case-form-opts.ts +39 -39
- package/templates/default/src/constants/form-opts/template-test-step-form-opts.ts +36 -36
- package/templates/default/src/constants/form-opts/test-case-form-opts.ts +43 -43
- package/templates/default/src/constants/form-opts/test-run-form-opts.ts +31 -31
- package/templates/default/src/constants/form-opts/test-suite-form-opts.ts +24 -24
- package/templates/default/src/hooks/use-toast.ts +187 -187
- package/templates/default/src/lib/bidirectional-sync.ts +432 -432
- package/templates/default/src/lib/database-sync.ts +531 -531
- package/templates/default/src/lib/environment-file-utils.ts +221 -221
- package/templates/default/src/lib/feature-file-generator.ts +411 -411
- package/templates/default/src/lib/gherkin-parser.ts +259 -259
- package/templates/default/src/lib/locator-group-file-utils.ts +370 -370
- package/templates/default/src/lib/metrics/metric-calculator.ts +613 -613
- package/templates/default/src/lib/module-hierarchy-builder.ts +205 -205
- package/templates/default/src/lib/path-helpers/module-path.ts +71 -71
- package/templates/default/src/lib/test-case-utils.ts +6 -6
- package/templates/default/src/lib/test-run/log-formatter.ts +83 -83
- package/templates/default/src/lib/test-run/process-manager.ts +191 -191
- package/templates/default/src/lib/test-run/report-parser.ts +316 -316
- package/templates/default/src/lib/test-run/test-run-executor.ts +144 -144
- package/templates/default/src/lib/test-run/winston-logger.ts +95 -95
- package/templates/default/src/lib/transformers/gherkin-converter.ts +42 -42
- package/templates/default/src/lib/transformers/key-to-icon-transformer.tsx +95 -95
- package/templates/default/src/lib/transformers/template-test-case-converter.ts +160 -160
- package/templates/default/src/lib/utils/node-param-validation.ts +81 -81
- package/templates/default/src/lib/utils/template-step-file-generator.ts +167 -167
- package/templates/default/src/lib/utils/template-step-file-manager-intelligent.ts +723 -723
- package/templates/default/src/lib/utils/template-step-file-manager.ts +166 -166
- package/templates/default/src/lib/utils.ts +31 -31
- package/templates/default/src/tests/config/executor/world.ts +41 -41
- package/templates/default/src/tests/executor.ts +80 -80
- package/templates/default/src/tests/hooks/hooks.ts +99 -99
- package/templates/default/src/tests/mapping/locator-map.json +1 -1
- package/templates/default/src/tests/steps/actions/click.step.ts +62 -62
- package/templates/default/src/tests/steps/actions/navigation.step.ts +73 -72
- package/templates/default/src/tests/steps/validations/active_state_assertion.step.ts +34 -34
- package/templates/default/src/tests/steps/validations/navigation_assertion.step.ts +24 -23
- package/templates/default/src/tests/steps/validations/text_assertion.step.ts +111 -111
- package/templates/default/src/tests/steps/validations/visibility_assertion.step.ts +30 -30
- package/templates/default/src/tests/support/parameter-types.ts +12 -12
- package/templates/default/src/tests/utils/cache.util.ts +260 -260
- package/templates/default/src/tests/utils/cli.util.ts +177 -177
- package/templates/default/src/tests/utils/environment.util.ts +65 -65
- package/templates/default/src/tests/utils/locator.util.ts +248 -248
- package/templates/default/src/tests/utils/random-data.util.ts +44 -44
- package/templates/default/src/tests/utils/spawner.util.ts +617 -617
- package/templates/default/src/types/diagram/diagram.ts +34 -34
- package/templates/default/src/types/diagram/template-step.ts +11 -11
- package/templates/default/src/types/executor/browser.type.ts +1 -1
- package/templates/default/src/types/form/actionHandler.ts +6 -6
- package/templates/default/src/types/locator/locator.type.ts +11 -11
- package/templates/default/src/types/step/step.type.ts +1 -1
- package/templates/default/src/types/table/data-table.ts +6 -6
- package/templates/default/tailwind.config.ts +62 -62
- package/templates/default/.env +0 -2
- package/templates/default/next-env.d.ts +0 -6
- package/templates/default/prisma/prisma/dev.db +0 -0
- package/templates/default/src/tests/config/environments/environments.json +0 -14
|
@@ -1,614 +1,614 @@
|
|
|
1
|
-
'use server'
|
|
2
|
-
|
|
3
|
-
import { ActionResponse } from '@/types/form/actionHandler'
|
|
4
|
-
import prisma from '@/config/db-config'
|
|
5
|
-
import { parseCucumberReport, getStepStatusEnum, getStepKeywordEnum } from '@/lib/test-run/report-parser'
|
|
6
|
-
import { TagType, Prisma } from '@prisma/client'
|
|
7
|
-
import { existsSync } from 'fs'
|
|
8
|
-
import { updateTestSuiteMetrics } from '@/lib/metrics/metric-calculator'
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* Type for report with all relations from getAllReportsAction
|
|
12
|
-
*/
|
|
13
|
-
type ReportWithRelations = Prisma.ReportGetPayload<{
|
|
14
|
-
include: {
|
|
15
|
-
testRun: {
|
|
16
|
-
include: {
|
|
17
|
-
environment: true
|
|
18
|
-
tags: true
|
|
19
|
-
}
|
|
20
|
-
}
|
|
21
|
-
testCases: {
|
|
22
|
-
include: {
|
|
23
|
-
testRunTestCase: {
|
|
24
|
-
include: {
|
|
25
|
-
testCase: {
|
|
26
|
-
include: {
|
|
27
|
-
tags: true
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
reportScenario: {
|
|
33
|
-
include: {
|
|
34
|
-
reportFeature: true
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
}
|
|
39
|
-
}
|
|
40
|
-
}>
|
|
41
|
-
|
|
42
|
-
/**
|
|
43
|
-
* Type for report with all relations from getReportByIdAction
|
|
44
|
-
*/
|
|
45
|
-
type ReportDetailWithRelations = Prisma.ReportGetPayload<{
|
|
46
|
-
include: {
|
|
47
|
-
testRun: {
|
|
48
|
-
include: {
|
|
49
|
-
environment: true
|
|
50
|
-
tags: true
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
features: {
|
|
54
|
-
include: {
|
|
55
|
-
tags: true
|
|
56
|
-
scenarios: {
|
|
57
|
-
include: {
|
|
58
|
-
tags: true
|
|
59
|
-
steps: true
|
|
60
|
-
hooks: true
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
testCases: {
|
|
66
|
-
include: {
|
|
67
|
-
testRunTestCase: {
|
|
68
|
-
include: {
|
|
69
|
-
testCase: {
|
|
70
|
-
include: {
|
|
71
|
-
tags: true
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
reportScenario: {
|
|
77
|
-
include: {
|
|
78
|
-
reportFeature: true
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
}>
|
|
85
|
-
|
|
86
|
-
/**
|
|
87
|
-
* Stores a cucumber.json report in the database
|
|
88
|
-
* Parses the report file and creates all related records (features, scenarios, steps, hooks, tags)
|
|
89
|
-
*
|
|
90
|
-
* @param testRunId - The test run ID (runId, not id)
|
|
91
|
-
* @param reportPath - Path to the cucumber.json file
|
|
92
|
-
* @returns ActionResponse indicating success or failure
|
|
93
|
-
*/
|
|
94
|
-
export async function storeReportFromFile(testRunId: string, reportPath: string): Promise<ActionResponse> {
|
|
95
|
-
try {
|
|
96
|
-
// Check if file exists
|
|
97
|
-
if (!existsSync(reportPath)) {
|
|
98
|
-
console.warn(`[ReportActions] Report file not found at ${reportPath} for testRunId: ${testRunId}`)
|
|
99
|
-
return {
|
|
100
|
-
status: 404,
|
|
101
|
-
error: `Report file not found at ${reportPath}`,
|
|
102
|
-
}
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
// Find the test run
|
|
106
|
-
const testRun = await prisma.testRun.findUnique({
|
|
107
|
-
where: { runId: testRunId },
|
|
108
|
-
include: {
|
|
109
|
-
testCases: {
|
|
110
|
-
include: {
|
|
111
|
-
testCase: {
|
|
112
|
-
include: {
|
|
113
|
-
tags: true,
|
|
114
|
-
},
|
|
115
|
-
},
|
|
116
|
-
},
|
|
117
|
-
},
|
|
118
|
-
},
|
|
119
|
-
})
|
|
120
|
-
|
|
121
|
-
if (!testRun) {
|
|
122
|
-
return {
|
|
123
|
-
status: 404,
|
|
124
|
-
error: `Test run not found for runId: ${testRunId}`,
|
|
125
|
-
}
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
// Parse the report
|
|
129
|
-
const parsedReport = await parseCucumberReport(reportPath)
|
|
130
|
-
|
|
131
|
-
// Create Report record
|
|
132
|
-
const report = await prisma.report.create({
|
|
133
|
-
data: {
|
|
134
|
-
name: `Test Run Report - ${testRun.name}`,
|
|
135
|
-
description: `Report for test run: ${testRun.name}`,
|
|
136
|
-
reportPath,
|
|
137
|
-
testRunId: testRun.id,
|
|
138
|
-
},
|
|
139
|
-
})
|
|
140
|
-
|
|
141
|
-
// Track test case IDs that were matched and executed
|
|
142
|
-
const executedTestCaseIds = new Set<string>()
|
|
143
|
-
|
|
144
|
-
// Process each feature
|
|
145
|
-
for (const feature of parsedReport.features) {
|
|
146
|
-
// Create ReportFeature
|
|
147
|
-
const reportFeature = await prisma.reportFeature.create({
|
|
148
|
-
data: {
|
|
149
|
-
reportId: report.id,
|
|
150
|
-
name: feature.name,
|
|
151
|
-
description: feature.description,
|
|
152
|
-
uri: feature.uri,
|
|
153
|
-
line: feature.line,
|
|
154
|
-
keyword: feature.keyword,
|
|
155
|
-
},
|
|
156
|
-
})
|
|
157
|
-
|
|
158
|
-
// Create feature tags
|
|
159
|
-
for (const tag of feature.tags) {
|
|
160
|
-
await prisma.reportFeatureTag.create({
|
|
161
|
-
data: {
|
|
162
|
-
reportFeatureId: reportFeature.id,
|
|
163
|
-
tagName: tag.name,
|
|
164
|
-
line: tag.line,
|
|
165
|
-
},
|
|
166
|
-
})
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
// Process each scenario
|
|
170
|
-
for (const scenario of feature.scenarios) {
|
|
171
|
-
// Create ReportScenario
|
|
172
|
-
const reportScenario = await prisma.reportScenario.create({
|
|
173
|
-
data: {
|
|
174
|
-
reportFeatureId: reportFeature.id,
|
|
175
|
-
name: scenario.name,
|
|
176
|
-
description: scenario.description,
|
|
177
|
-
line: scenario.line,
|
|
178
|
-
keyword: scenario.keyword,
|
|
179
|
-
type: scenario.type,
|
|
180
|
-
cucumberId: scenario.cucumberId,
|
|
181
|
-
},
|
|
182
|
-
})
|
|
183
|
-
|
|
184
|
-
// Create scenario tags
|
|
185
|
-
for (const tag of scenario.tags) {
|
|
186
|
-
await prisma.reportScenarioTag.create({
|
|
187
|
-
data: {
|
|
188
|
-
reportScenarioId: reportScenario.id,
|
|
189
|
-
tagName: tag.name,
|
|
190
|
-
line: tag.line,
|
|
191
|
-
},
|
|
192
|
-
})
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
// Create steps
|
|
196
|
-
for (const step of scenario.steps) {
|
|
197
|
-
await prisma.reportStep.create({
|
|
198
|
-
data: {
|
|
199
|
-
reportScenarioId: reportScenario.id,
|
|
200
|
-
keyword: getStepKeywordEnum(step.keyword),
|
|
201
|
-
line: step.line,
|
|
202
|
-
name: step.name,
|
|
203
|
-
matchLocation: step.matchLocation,
|
|
204
|
-
status: getStepStatusEnum(step.status),
|
|
205
|
-
duration: String(step.duration),
|
|
206
|
-
errorMessage: step.errorMessage,
|
|
207
|
-
errorTrace: step.errorTrace,
|
|
208
|
-
hidden: step.hidden,
|
|
209
|
-
order: step.order,
|
|
210
|
-
},
|
|
211
|
-
})
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
// Create hooks
|
|
215
|
-
for (const hook of scenario.hooks) {
|
|
216
|
-
await prisma.reportHook.create({
|
|
217
|
-
data: {
|
|
218
|
-
reportScenarioId: reportScenario.id,
|
|
219
|
-
keyword: getStepKeywordEnum(hook.keyword),
|
|
220
|
-
status: getStepStatusEnum(hook.status),
|
|
221
|
-
duration: String(hook.duration),
|
|
222
|
-
errorMessage: hook.errorMessage,
|
|
223
|
-
errorTrace: hook.errorTrace,
|
|
224
|
-
hidden: hook.hidden,
|
|
225
|
-
},
|
|
226
|
-
})
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
// Try to match scenario to test case
|
|
230
|
-
// Method 1: Extract test case title from scenario name (format: "[Title] Description")
|
|
231
|
-
let matchedTestCase: (typeof testRun.testCases)[0] | undefined
|
|
232
|
-
const bracketMatch = scenario.name.match(/^\[([^\]]+)\]/)
|
|
233
|
-
if (bracketMatch) {
|
|
234
|
-
const testCaseTitle = bracketMatch[1].trim()
|
|
235
|
-
matchedTestCase = testRun.testCases.find(trtc => trtc.testCase.title === testCaseTitle)
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
// Method 2: Match by identifier tags if method 1 didn't work
|
|
239
|
-
if (!matchedTestCase) {
|
|
240
|
-
const scenarioTagNames = scenario.tags.map(tag => tag.name.toLowerCase())
|
|
241
|
-
matchedTestCase = testRun.testCases.find(trtc => {
|
|
242
|
-
const identifierTags = trtc.testCase.tags.filter(tag => tag.type === TagType.IDENTIFIER)
|
|
243
|
-
return identifierTags.some(tag => {
|
|
244
|
-
const tagExpression = tag.tagExpression.toLowerCase()
|
|
245
|
-
return scenarioTagNames.some(scenarioTag => tagExpression.includes(scenarioTag))
|
|
246
|
-
})
|
|
247
|
-
})
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
// If we found a matching test case, create ReportTestCase link
|
|
251
|
-
if (matchedTestCase) {
|
|
252
|
-
// Track this test case as executed
|
|
253
|
-
executedTestCaseIds.add(matchedTestCase.testCaseId)
|
|
254
|
-
|
|
255
|
-
// Calculate scenario duration
|
|
256
|
-
const scenarioDuration =
|
|
257
|
-
scenario.steps.reduce((total, step) => total + step.duration, 0) +
|
|
258
|
-
scenario.hooks.reduce((total, hook) => total + hook.duration, 0)
|
|
259
|
-
|
|
260
|
-
await prisma.reportTestCase.create({
|
|
261
|
-
data: {
|
|
262
|
-
reportId: report.id,
|
|
263
|
-
testCaseId: matchedTestCase.testCaseId,
|
|
264
|
-
testRunTestCaseId: matchedTestCase.id,
|
|
265
|
-
reportScenarioId: reportScenario.id,
|
|
266
|
-
duration: String(scenarioDuration),
|
|
267
|
-
},
|
|
268
|
-
})
|
|
269
|
-
}
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
|
|
273
|
-
// Update test suite metrics for all test suites that had test cases executed
|
|
274
|
-
// This serves as a backup to ensure metrics are updated even if updateTestSuitesForTestRun missed some
|
|
275
|
-
if (executedTestCaseIds.size > 0) {
|
|
276
|
-
try {
|
|
277
|
-
const executedAt = testRun.completedAt || testRun.startedAt || new Date()
|
|
278
|
-
|
|
279
|
-
// Get all test cases that were executed and their associated test suites
|
|
280
|
-
const testCases = await prisma.testCase.findMany({
|
|
281
|
-
where: {
|
|
282
|
-
id: {
|
|
283
|
-
in: Array.from(executedTestCaseIds),
|
|
284
|
-
},
|
|
285
|
-
},
|
|
286
|
-
include: {
|
|
287
|
-
TestSuite: true,
|
|
288
|
-
},
|
|
289
|
-
})
|
|
290
|
-
|
|
291
|
-
// Extract unique test suite IDs
|
|
292
|
-
const testSuiteIds = new Set<string>()
|
|
293
|
-
testCases.forEach(testCase => {
|
|
294
|
-
if (testCase.TestSuite) {
|
|
295
|
-
testCase.TestSuite.forEach(suite => {
|
|
296
|
-
testSuiteIds.add(suite.id)
|
|
297
|
-
})
|
|
298
|
-
}
|
|
299
|
-
})
|
|
300
|
-
|
|
301
|
-
// Update lastExecutedAt for each suite
|
|
302
|
-
for (const suiteId of testSuiteIds) {
|
|
303
|
-
await updateTestSuiteMetrics(suiteId, executedAt)
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
if (testSuiteIds.size > 0) {
|
|
307
|
-
console.log(
|
|
308
|
-
`[ReportActions] Updated test suite metrics for ${testSuiteIds.size} test suite(s) based on executed test cases`,
|
|
309
|
-
)
|
|
310
|
-
}
|
|
311
|
-
} catch (error) {
|
|
312
|
-
console.error(
|
|
313
|
-
`[ReportActions] Error updating test suite metrics after storing report: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
314
|
-
)
|
|
315
|
-
// Don't fail the report storage if metrics update fails
|
|
316
|
-
}
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
return {
|
|
320
|
-
status: 200,
|
|
321
|
-
message: 'Report stored successfully',
|
|
322
|
-
data: { reportId: report.id },
|
|
323
|
-
}
|
|
324
|
-
} catch (error) {
|
|
325
|
-
console.error(`[ReportActions] Error storing report from file ${reportPath}:`, error)
|
|
326
|
-
return {
|
|
327
|
-
status: 500,
|
|
328
|
-
error: `Failed to store report: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
329
|
-
}
|
|
330
|
-
}
|
|
331
|
-
}
|
|
332
|
-
|
|
333
|
-
/**
|
|
334
|
-
* Gets all reports from the database
|
|
335
|
-
*/
|
|
336
|
-
export const getAllReportsAction = async (): Promise<ActionResponse> => {
|
|
337
|
-
try {
|
|
338
|
-
const reports = await prisma.report.findMany({
|
|
339
|
-
include: {
|
|
340
|
-
testRun: {
|
|
341
|
-
include: {
|
|
342
|
-
environment: true,
|
|
343
|
-
tags: true,
|
|
344
|
-
},
|
|
345
|
-
},
|
|
346
|
-
testCases: {
|
|
347
|
-
include: {
|
|
348
|
-
testRunTestCase: {
|
|
349
|
-
include: {
|
|
350
|
-
testCase: {
|
|
351
|
-
include: {
|
|
352
|
-
tags: true,
|
|
353
|
-
},
|
|
354
|
-
},
|
|
355
|
-
},
|
|
356
|
-
},
|
|
357
|
-
reportScenario: {
|
|
358
|
-
include: {
|
|
359
|
-
reportFeature: true,
|
|
360
|
-
},
|
|
361
|
-
},
|
|
362
|
-
},
|
|
363
|
-
},
|
|
364
|
-
},
|
|
365
|
-
orderBy: {
|
|
366
|
-
createdAt: 'desc',
|
|
367
|
-
},
|
|
368
|
-
})
|
|
369
|
-
|
|
370
|
-
return {
|
|
371
|
-
status: 200,
|
|
372
|
-
data: reports as ReportWithRelations[],
|
|
373
|
-
}
|
|
374
|
-
} catch (error) {
|
|
375
|
-
console.error('[ReportActions] Error fetching all reports:', error)
|
|
376
|
-
return {
|
|
377
|
-
status: 500,
|
|
378
|
-
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
379
|
-
}
|
|
380
|
-
}
|
|
381
|
-
}
|
|
382
|
-
|
|
383
|
-
/**
|
|
384
|
-
* Gets a report by ID with all related data
|
|
385
|
-
*/
|
|
386
|
-
export const getReportByIdAction = async (reportId: string): Promise<ActionResponse> => {
|
|
387
|
-
try {
|
|
388
|
-
const report = await prisma.report.findUnique({
|
|
389
|
-
where: { id: reportId },
|
|
390
|
-
include: {
|
|
391
|
-
testRun: {
|
|
392
|
-
include: {
|
|
393
|
-
environment: true,
|
|
394
|
-
tags: true,
|
|
395
|
-
},
|
|
396
|
-
},
|
|
397
|
-
features: {
|
|
398
|
-
include: {
|
|
399
|
-
tags: true,
|
|
400
|
-
scenarios: {
|
|
401
|
-
include: {
|
|
402
|
-
tags: true,
|
|
403
|
-
steps: {
|
|
404
|
-
orderBy: {
|
|
405
|
-
order: 'asc',
|
|
406
|
-
},
|
|
407
|
-
},
|
|
408
|
-
hooks: true,
|
|
409
|
-
},
|
|
410
|
-
},
|
|
411
|
-
},
|
|
412
|
-
},
|
|
413
|
-
testCases: {
|
|
414
|
-
include: {
|
|
415
|
-
testRunTestCase: {
|
|
416
|
-
include: {
|
|
417
|
-
testCase: {
|
|
418
|
-
include: {
|
|
419
|
-
tags: true,
|
|
420
|
-
},
|
|
421
|
-
},
|
|
422
|
-
},
|
|
423
|
-
},
|
|
424
|
-
reportScenario: {
|
|
425
|
-
include: {
|
|
426
|
-
reportFeature: true,
|
|
427
|
-
tags: true,
|
|
428
|
-
steps: {
|
|
429
|
-
orderBy: {
|
|
430
|
-
order: 'asc',
|
|
431
|
-
},
|
|
432
|
-
},
|
|
433
|
-
hooks: true,
|
|
434
|
-
},
|
|
435
|
-
},
|
|
436
|
-
},
|
|
437
|
-
},
|
|
438
|
-
},
|
|
439
|
-
})
|
|
440
|
-
|
|
441
|
-
if (!report) {
|
|
442
|
-
return {
|
|
443
|
-
status: 404,
|
|
444
|
-
error: 'Report not found',
|
|
445
|
-
}
|
|
446
|
-
}
|
|
447
|
-
|
|
448
|
-
return {
|
|
449
|
-
status: 200,
|
|
450
|
-
data: report as ReportDetailWithRelations,
|
|
451
|
-
}
|
|
452
|
-
} catch (error) {
|
|
453
|
-
console.error(`[ReportActions] Error fetching report ${reportId}:`, error)
|
|
454
|
-
return {
|
|
455
|
-
status: 500,
|
|
456
|
-
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
457
|
-
}
|
|
458
|
-
}
|
|
459
|
-
}
|
|
460
|
-
|
|
461
|
-
/**
|
|
462
|
-
* Gets a report by test run ID
|
|
463
|
-
*/
|
|
464
|
-
export const getReportByTestRunIdAction = async (testRunId: string): Promise<ActionResponse> => {
|
|
465
|
-
try {
|
|
466
|
-
// Find test run by runId
|
|
467
|
-
const testRun = await prisma.testRun.findUnique({
|
|
468
|
-
where: { runId: testRunId },
|
|
469
|
-
select: { id: true },
|
|
470
|
-
})
|
|
471
|
-
|
|
472
|
-
if (!testRun) {
|
|
473
|
-
return {
|
|
474
|
-
status: 404,
|
|
475
|
-
error: 'Test run not found',
|
|
476
|
-
}
|
|
477
|
-
}
|
|
478
|
-
|
|
479
|
-
// Find report by testRun.id
|
|
480
|
-
const report = await prisma.report.findFirst({
|
|
481
|
-
where: { testRunId: testRun.id },
|
|
482
|
-
include: {
|
|
483
|
-
testRun: {
|
|
484
|
-
include: {
|
|
485
|
-
environment: true,
|
|
486
|
-
tags: true,
|
|
487
|
-
},
|
|
488
|
-
},
|
|
489
|
-
features: {
|
|
490
|
-
include: {
|
|
491
|
-
tags: true,
|
|
492
|
-
scenarios: {
|
|
493
|
-
include: {
|
|
494
|
-
tags: true,
|
|
495
|
-
steps: {
|
|
496
|
-
orderBy: {
|
|
497
|
-
order: 'asc',
|
|
498
|
-
},
|
|
499
|
-
},
|
|
500
|
-
hooks: true,
|
|
501
|
-
},
|
|
502
|
-
},
|
|
503
|
-
},
|
|
504
|
-
},
|
|
505
|
-
testCases: {
|
|
506
|
-
include: {
|
|
507
|
-
testRunTestCase: {
|
|
508
|
-
include: {
|
|
509
|
-
testCase: {
|
|
510
|
-
include: {
|
|
511
|
-
tags: true,
|
|
512
|
-
},
|
|
513
|
-
},
|
|
514
|
-
},
|
|
515
|
-
},
|
|
516
|
-
reportScenario: {
|
|
517
|
-
include: {
|
|
518
|
-
reportFeature: true,
|
|
519
|
-
},
|
|
520
|
-
},
|
|
521
|
-
},
|
|
522
|
-
},
|
|
523
|
-
},
|
|
524
|
-
})
|
|
525
|
-
|
|
526
|
-
if (!report) {
|
|
527
|
-
return {
|
|
528
|
-
status: 404,
|
|
529
|
-
error: 'Report not found for this test run',
|
|
530
|
-
}
|
|
531
|
-
}
|
|
532
|
-
|
|
533
|
-
return {
|
|
534
|
-
status: 200,
|
|
535
|
-
data: report as ReportDetailWithRelations,
|
|
536
|
-
}
|
|
537
|
-
} catch (error) {
|
|
538
|
-
console.error(`[ReportActions] Error fetching report for testRunId ${testRunId}:`, error)
|
|
539
|
-
return {
|
|
540
|
-
status: 500,
|
|
541
|
-
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
542
|
-
}
|
|
543
|
-
}
|
|
544
|
-
}
|
|
545
|
-
|
|
546
|
-
/**
|
|
547
|
-
* Gets the test case metrics for a report
|
|
548
|
-
*/
|
|
549
|
-
export const getAllTestCaseMetricsAction = async (filter: string): Promise<ActionResponse> => {
|
|
550
|
-
try {
|
|
551
|
-
let testCaseMetrics = await prisma.testCaseMetrics.findMany({
|
|
552
|
-
include: {
|
|
553
|
-
testCase: {
|
|
554
|
-
include: {
|
|
555
|
-
tags: true,
|
|
556
|
-
},
|
|
557
|
-
},
|
|
558
|
-
},
|
|
559
|
-
})
|
|
560
|
-
if (filter === 'repeatedlyFailing') {
|
|
561
|
-
testCaseMetrics = testCaseMetrics.filter(tc => tc.isRepeatedlyFailing)
|
|
562
|
-
} else if (filter === 'flaky') {
|
|
563
|
-
testCaseMetrics = testCaseMetrics.filter(tc => tc.isFlaky)
|
|
564
|
-
}
|
|
565
|
-
return {
|
|
566
|
-
status: 200,
|
|
567
|
-
data: testCaseMetrics,
|
|
568
|
-
}
|
|
569
|
-
} catch (error) {
|
|
570
|
-
console.error(`[ReportActions] Error fetching all test case metrics:`, error)
|
|
571
|
-
return {
|
|
572
|
-
status: 500,
|
|
573
|
-
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
574
|
-
}
|
|
575
|
-
}
|
|
576
|
-
}
|
|
577
|
-
|
|
578
|
-
/**
|
|
579
|
-
* Gets the test suite metrics for a report
|
|
580
|
-
*/
|
|
581
|
-
export const getAllTestSuiteMetricsAction = async (filter: string): Promise<ActionResponse> => {
|
|
582
|
-
try {
|
|
583
|
-
let testSuiteMetrics = await prisma.testSuiteMetrics.findMany({
|
|
584
|
-
include: {
|
|
585
|
-
testSuite: {
|
|
586
|
-
include: {
|
|
587
|
-
tags: true,
|
|
588
|
-
testCases: true,
|
|
589
|
-
},
|
|
590
|
-
},
|
|
591
|
-
},
|
|
592
|
-
})
|
|
593
|
-
if (filter === 'notExecutedRecently') {
|
|
594
|
-
// Calculate the date threshold for recent runs (7 days ago)
|
|
595
|
-
const recentPeriodDate = new Date()
|
|
596
|
-
recentPeriodDate.setDate(recentPeriodDate.getDate() - 7)
|
|
597
|
-
|
|
598
|
-
// Filter for suites that have never been executed OR were executed more than 7 days ago
|
|
599
|
-
testSuiteMetrics = testSuiteMetrics.filter(
|
|
600
|
-
ts => ts.lastExecutedAt === null || ts.lastExecutedAt < recentPeriodDate,
|
|
601
|
-
)
|
|
602
|
-
}
|
|
603
|
-
return {
|
|
604
|
-
status: 200,
|
|
605
|
-
data: testSuiteMetrics,
|
|
606
|
-
}
|
|
607
|
-
} catch (error) {
|
|
608
|
-
console.error(`[ReportActions] Error fetching all test suite metrics:`, error)
|
|
609
|
-
return {
|
|
610
|
-
status: 500,
|
|
611
|
-
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
612
|
-
}
|
|
613
|
-
}
|
|
1
|
+
'use server'
|
|
2
|
+
|
|
3
|
+
import { ActionResponse } from '@/types/form/actionHandler'
|
|
4
|
+
import prisma from '@/config/db-config'
|
|
5
|
+
import { parseCucumberReport, getStepStatusEnum, getStepKeywordEnum } from '@/lib/test-run/report-parser'
|
|
6
|
+
import { TagType, Prisma } from '@prisma/client'
|
|
7
|
+
import { existsSync } from 'fs'
|
|
8
|
+
import { updateTestSuiteMetrics } from '@/lib/metrics/metric-calculator'
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Type for report with all relations from getAllReportsAction
|
|
12
|
+
*/
|
|
13
|
+
type ReportWithRelations = Prisma.ReportGetPayload<{
|
|
14
|
+
include: {
|
|
15
|
+
testRun: {
|
|
16
|
+
include: {
|
|
17
|
+
environment: true
|
|
18
|
+
tags: true
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
testCases: {
|
|
22
|
+
include: {
|
|
23
|
+
testRunTestCase: {
|
|
24
|
+
include: {
|
|
25
|
+
testCase: {
|
|
26
|
+
include: {
|
|
27
|
+
tags: true
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
reportScenario: {
|
|
33
|
+
include: {
|
|
34
|
+
reportFeature: true
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}>
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Type for report with all relations from getReportByIdAction
|
|
44
|
+
*/
|
|
45
|
+
type ReportDetailWithRelations = Prisma.ReportGetPayload<{
|
|
46
|
+
include: {
|
|
47
|
+
testRun: {
|
|
48
|
+
include: {
|
|
49
|
+
environment: true
|
|
50
|
+
tags: true
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
features: {
|
|
54
|
+
include: {
|
|
55
|
+
tags: true
|
|
56
|
+
scenarios: {
|
|
57
|
+
include: {
|
|
58
|
+
tags: true
|
|
59
|
+
steps: true
|
|
60
|
+
hooks: true
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
testCases: {
|
|
66
|
+
include: {
|
|
67
|
+
testRunTestCase: {
|
|
68
|
+
include: {
|
|
69
|
+
testCase: {
|
|
70
|
+
include: {
|
|
71
|
+
tags: true
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
reportScenario: {
|
|
77
|
+
include: {
|
|
78
|
+
reportFeature: true
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}>
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Stores a cucumber.json report in the database
|
|
88
|
+
* Parses the report file and creates all related records (features, scenarios, steps, hooks, tags)
|
|
89
|
+
*
|
|
90
|
+
* @param testRunId - The test run ID (runId, not id)
|
|
91
|
+
* @param reportPath - Path to the cucumber.json file
|
|
92
|
+
* @returns ActionResponse indicating success or failure
|
|
93
|
+
*/
|
|
94
|
+
export async function storeReportFromFile(testRunId: string, reportPath: string): Promise<ActionResponse> {
|
|
95
|
+
try {
|
|
96
|
+
// Check if file exists
|
|
97
|
+
if (!existsSync(reportPath)) {
|
|
98
|
+
console.warn(`[ReportActions] Report file not found at ${reportPath} for testRunId: ${testRunId}`)
|
|
99
|
+
return {
|
|
100
|
+
status: 404,
|
|
101
|
+
error: `Report file not found at ${reportPath}`,
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
// Find the test run
|
|
106
|
+
const testRun = await prisma.testRun.findUnique({
|
|
107
|
+
where: { runId: testRunId },
|
|
108
|
+
include: {
|
|
109
|
+
testCases: {
|
|
110
|
+
include: {
|
|
111
|
+
testCase: {
|
|
112
|
+
include: {
|
|
113
|
+
tags: true,
|
|
114
|
+
},
|
|
115
|
+
},
|
|
116
|
+
},
|
|
117
|
+
},
|
|
118
|
+
},
|
|
119
|
+
})
|
|
120
|
+
|
|
121
|
+
if (!testRun) {
|
|
122
|
+
return {
|
|
123
|
+
status: 404,
|
|
124
|
+
error: `Test run not found for runId: ${testRunId}`,
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// Parse the report
|
|
129
|
+
const parsedReport = await parseCucumberReport(reportPath)
|
|
130
|
+
|
|
131
|
+
// Create Report record
|
|
132
|
+
const report = await prisma.report.create({
|
|
133
|
+
data: {
|
|
134
|
+
name: `Test Run Report - ${testRun.name}`,
|
|
135
|
+
description: `Report for test run: ${testRun.name}`,
|
|
136
|
+
reportPath,
|
|
137
|
+
testRunId: testRun.id,
|
|
138
|
+
},
|
|
139
|
+
})
|
|
140
|
+
|
|
141
|
+
// Track test case IDs that were matched and executed
|
|
142
|
+
const executedTestCaseIds = new Set<string>()
|
|
143
|
+
|
|
144
|
+
// Process each feature
|
|
145
|
+
for (const feature of parsedReport.features) {
|
|
146
|
+
// Create ReportFeature
|
|
147
|
+
const reportFeature = await prisma.reportFeature.create({
|
|
148
|
+
data: {
|
|
149
|
+
reportId: report.id,
|
|
150
|
+
name: feature.name,
|
|
151
|
+
description: feature.description,
|
|
152
|
+
uri: feature.uri,
|
|
153
|
+
line: feature.line,
|
|
154
|
+
keyword: feature.keyword,
|
|
155
|
+
},
|
|
156
|
+
})
|
|
157
|
+
|
|
158
|
+
// Create feature tags
|
|
159
|
+
for (const tag of feature.tags) {
|
|
160
|
+
await prisma.reportFeatureTag.create({
|
|
161
|
+
data: {
|
|
162
|
+
reportFeatureId: reportFeature.id,
|
|
163
|
+
tagName: tag.name,
|
|
164
|
+
line: tag.line,
|
|
165
|
+
},
|
|
166
|
+
})
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
// Process each scenario
|
|
170
|
+
for (const scenario of feature.scenarios) {
|
|
171
|
+
// Create ReportScenario
|
|
172
|
+
const reportScenario = await prisma.reportScenario.create({
|
|
173
|
+
data: {
|
|
174
|
+
reportFeatureId: reportFeature.id,
|
|
175
|
+
name: scenario.name,
|
|
176
|
+
description: scenario.description,
|
|
177
|
+
line: scenario.line,
|
|
178
|
+
keyword: scenario.keyword,
|
|
179
|
+
type: scenario.type,
|
|
180
|
+
cucumberId: scenario.cucumberId,
|
|
181
|
+
},
|
|
182
|
+
})
|
|
183
|
+
|
|
184
|
+
// Create scenario tags
|
|
185
|
+
for (const tag of scenario.tags) {
|
|
186
|
+
await prisma.reportScenarioTag.create({
|
|
187
|
+
data: {
|
|
188
|
+
reportScenarioId: reportScenario.id,
|
|
189
|
+
tagName: tag.name,
|
|
190
|
+
line: tag.line,
|
|
191
|
+
},
|
|
192
|
+
})
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
// Create steps
|
|
196
|
+
for (const step of scenario.steps) {
|
|
197
|
+
await prisma.reportStep.create({
|
|
198
|
+
data: {
|
|
199
|
+
reportScenarioId: reportScenario.id,
|
|
200
|
+
keyword: getStepKeywordEnum(step.keyword),
|
|
201
|
+
line: step.line,
|
|
202
|
+
name: step.name,
|
|
203
|
+
matchLocation: step.matchLocation,
|
|
204
|
+
status: getStepStatusEnum(step.status),
|
|
205
|
+
duration: String(step.duration),
|
|
206
|
+
errorMessage: step.errorMessage,
|
|
207
|
+
errorTrace: step.errorTrace,
|
|
208
|
+
hidden: step.hidden,
|
|
209
|
+
order: step.order,
|
|
210
|
+
},
|
|
211
|
+
})
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
// Create hooks
|
|
215
|
+
for (const hook of scenario.hooks) {
|
|
216
|
+
await prisma.reportHook.create({
|
|
217
|
+
data: {
|
|
218
|
+
reportScenarioId: reportScenario.id,
|
|
219
|
+
keyword: getStepKeywordEnum(hook.keyword),
|
|
220
|
+
status: getStepStatusEnum(hook.status),
|
|
221
|
+
duration: String(hook.duration),
|
|
222
|
+
errorMessage: hook.errorMessage,
|
|
223
|
+
errorTrace: hook.errorTrace,
|
|
224
|
+
hidden: hook.hidden,
|
|
225
|
+
},
|
|
226
|
+
})
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
// Try to match scenario to test case
|
|
230
|
+
// Method 1: Extract test case title from scenario name (format: "[Title] Description")
|
|
231
|
+
let matchedTestCase: (typeof testRun.testCases)[0] | undefined
|
|
232
|
+
const bracketMatch = scenario.name.match(/^\[([^\]]+)\]/)
|
|
233
|
+
if (bracketMatch) {
|
|
234
|
+
const testCaseTitle = bracketMatch[1].trim()
|
|
235
|
+
matchedTestCase = testRun.testCases.find(trtc => trtc.testCase.title === testCaseTitle)
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
// Method 2: Match by identifier tags if method 1 didn't work
|
|
239
|
+
if (!matchedTestCase) {
|
|
240
|
+
const scenarioTagNames = scenario.tags.map(tag => tag.name.toLowerCase())
|
|
241
|
+
matchedTestCase = testRun.testCases.find(trtc => {
|
|
242
|
+
const identifierTags = trtc.testCase.tags.filter(tag => tag.type === TagType.IDENTIFIER)
|
|
243
|
+
return identifierTags.some(tag => {
|
|
244
|
+
const tagExpression = tag.tagExpression.toLowerCase()
|
|
245
|
+
return scenarioTagNames.some(scenarioTag => tagExpression.includes(scenarioTag))
|
|
246
|
+
})
|
|
247
|
+
})
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
// If we found a matching test case, create ReportTestCase link
|
|
251
|
+
if (matchedTestCase) {
|
|
252
|
+
// Track this test case as executed
|
|
253
|
+
executedTestCaseIds.add(matchedTestCase.testCaseId)
|
|
254
|
+
|
|
255
|
+
// Calculate scenario duration
|
|
256
|
+
const scenarioDuration =
|
|
257
|
+
scenario.steps.reduce((total, step) => total + step.duration, 0) +
|
|
258
|
+
scenario.hooks.reduce((total, hook) => total + hook.duration, 0)
|
|
259
|
+
|
|
260
|
+
await prisma.reportTestCase.create({
|
|
261
|
+
data: {
|
|
262
|
+
reportId: report.id,
|
|
263
|
+
testCaseId: matchedTestCase.testCaseId,
|
|
264
|
+
testRunTestCaseId: matchedTestCase.id,
|
|
265
|
+
reportScenarioId: reportScenario.id,
|
|
266
|
+
duration: String(scenarioDuration),
|
|
267
|
+
},
|
|
268
|
+
})
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
// Update test suite metrics for all test suites that had test cases executed
|
|
274
|
+
// This serves as a backup to ensure metrics are updated even if updateTestSuitesForTestRun missed some
|
|
275
|
+
if (executedTestCaseIds.size > 0) {
|
|
276
|
+
try {
|
|
277
|
+
const executedAt = testRun.completedAt || testRun.startedAt || new Date()
|
|
278
|
+
|
|
279
|
+
// Get all test cases that were executed and their associated test suites
|
|
280
|
+
const testCases = await prisma.testCase.findMany({
|
|
281
|
+
where: {
|
|
282
|
+
id: {
|
|
283
|
+
in: Array.from(executedTestCaseIds),
|
|
284
|
+
},
|
|
285
|
+
},
|
|
286
|
+
include: {
|
|
287
|
+
TestSuite: true,
|
|
288
|
+
},
|
|
289
|
+
})
|
|
290
|
+
|
|
291
|
+
// Extract unique test suite IDs
|
|
292
|
+
const testSuiteIds = new Set<string>()
|
|
293
|
+
testCases.forEach(testCase => {
|
|
294
|
+
if (testCase.TestSuite) {
|
|
295
|
+
testCase.TestSuite.forEach(suite => {
|
|
296
|
+
testSuiteIds.add(suite.id)
|
|
297
|
+
})
|
|
298
|
+
}
|
|
299
|
+
})
|
|
300
|
+
|
|
301
|
+
// Update lastExecutedAt for each suite
|
|
302
|
+
for (const suiteId of testSuiteIds) {
|
|
303
|
+
await updateTestSuiteMetrics(suiteId, executedAt)
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
if (testSuiteIds.size > 0) {
|
|
307
|
+
console.log(
|
|
308
|
+
`[ReportActions] Updated test suite metrics for ${testSuiteIds.size} test suite(s) based on executed test cases`,
|
|
309
|
+
)
|
|
310
|
+
}
|
|
311
|
+
} catch (error) {
|
|
312
|
+
console.error(
|
|
313
|
+
`[ReportActions] Error updating test suite metrics after storing report: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
314
|
+
)
|
|
315
|
+
// Don't fail the report storage if metrics update fails
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
return {
|
|
320
|
+
status: 200,
|
|
321
|
+
message: 'Report stored successfully',
|
|
322
|
+
data: { reportId: report.id },
|
|
323
|
+
}
|
|
324
|
+
} catch (error) {
|
|
325
|
+
console.error(`[ReportActions] Error storing report from file ${reportPath}:`, error)
|
|
326
|
+
return {
|
|
327
|
+
status: 500,
|
|
328
|
+
error: `Failed to store report: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
/**
|
|
334
|
+
* Gets all reports from the database
|
|
335
|
+
*/
|
|
336
|
+
export const getAllReportsAction = async (): Promise<ActionResponse> => {
|
|
337
|
+
try {
|
|
338
|
+
const reports = await prisma.report.findMany({
|
|
339
|
+
include: {
|
|
340
|
+
testRun: {
|
|
341
|
+
include: {
|
|
342
|
+
environment: true,
|
|
343
|
+
tags: true,
|
|
344
|
+
},
|
|
345
|
+
},
|
|
346
|
+
testCases: {
|
|
347
|
+
include: {
|
|
348
|
+
testRunTestCase: {
|
|
349
|
+
include: {
|
|
350
|
+
testCase: {
|
|
351
|
+
include: {
|
|
352
|
+
tags: true,
|
|
353
|
+
},
|
|
354
|
+
},
|
|
355
|
+
},
|
|
356
|
+
},
|
|
357
|
+
reportScenario: {
|
|
358
|
+
include: {
|
|
359
|
+
reportFeature: true,
|
|
360
|
+
},
|
|
361
|
+
},
|
|
362
|
+
},
|
|
363
|
+
},
|
|
364
|
+
},
|
|
365
|
+
orderBy: {
|
|
366
|
+
createdAt: 'desc',
|
|
367
|
+
},
|
|
368
|
+
})
|
|
369
|
+
|
|
370
|
+
return {
|
|
371
|
+
status: 200,
|
|
372
|
+
data: reports as ReportWithRelations[],
|
|
373
|
+
}
|
|
374
|
+
} catch (error) {
|
|
375
|
+
console.error('[ReportActions] Error fetching all reports:', error)
|
|
376
|
+
return {
|
|
377
|
+
status: 500,
|
|
378
|
+
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
/**
|
|
384
|
+
* Gets a report by ID with all related data
|
|
385
|
+
*/
|
|
386
|
+
export const getReportByIdAction = async (reportId: string): Promise<ActionResponse> => {
|
|
387
|
+
try {
|
|
388
|
+
const report = await prisma.report.findUnique({
|
|
389
|
+
where: { id: reportId },
|
|
390
|
+
include: {
|
|
391
|
+
testRun: {
|
|
392
|
+
include: {
|
|
393
|
+
environment: true,
|
|
394
|
+
tags: true,
|
|
395
|
+
},
|
|
396
|
+
},
|
|
397
|
+
features: {
|
|
398
|
+
include: {
|
|
399
|
+
tags: true,
|
|
400
|
+
scenarios: {
|
|
401
|
+
include: {
|
|
402
|
+
tags: true,
|
|
403
|
+
steps: {
|
|
404
|
+
orderBy: {
|
|
405
|
+
order: 'asc',
|
|
406
|
+
},
|
|
407
|
+
},
|
|
408
|
+
hooks: true,
|
|
409
|
+
},
|
|
410
|
+
},
|
|
411
|
+
},
|
|
412
|
+
},
|
|
413
|
+
testCases: {
|
|
414
|
+
include: {
|
|
415
|
+
testRunTestCase: {
|
|
416
|
+
include: {
|
|
417
|
+
testCase: {
|
|
418
|
+
include: {
|
|
419
|
+
tags: true,
|
|
420
|
+
},
|
|
421
|
+
},
|
|
422
|
+
},
|
|
423
|
+
},
|
|
424
|
+
reportScenario: {
|
|
425
|
+
include: {
|
|
426
|
+
reportFeature: true,
|
|
427
|
+
tags: true,
|
|
428
|
+
steps: {
|
|
429
|
+
orderBy: {
|
|
430
|
+
order: 'asc',
|
|
431
|
+
},
|
|
432
|
+
},
|
|
433
|
+
hooks: true,
|
|
434
|
+
},
|
|
435
|
+
},
|
|
436
|
+
},
|
|
437
|
+
},
|
|
438
|
+
},
|
|
439
|
+
})
|
|
440
|
+
|
|
441
|
+
if (!report) {
|
|
442
|
+
return {
|
|
443
|
+
status: 404,
|
|
444
|
+
error: 'Report not found',
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
return {
|
|
449
|
+
status: 200,
|
|
450
|
+
data: report as ReportDetailWithRelations,
|
|
451
|
+
}
|
|
452
|
+
} catch (error) {
|
|
453
|
+
console.error(`[ReportActions] Error fetching report ${reportId}:`, error)
|
|
454
|
+
return {
|
|
455
|
+
status: 500,
|
|
456
|
+
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
/**
|
|
462
|
+
* Gets a report by test run ID
|
|
463
|
+
*/
|
|
464
|
+
export const getReportByTestRunIdAction = async (testRunId: string): Promise<ActionResponse> => {
|
|
465
|
+
try {
|
|
466
|
+
// Find test run by runId
|
|
467
|
+
const testRun = await prisma.testRun.findUnique({
|
|
468
|
+
where: { runId: testRunId },
|
|
469
|
+
select: { id: true },
|
|
470
|
+
})
|
|
471
|
+
|
|
472
|
+
if (!testRun) {
|
|
473
|
+
return {
|
|
474
|
+
status: 404,
|
|
475
|
+
error: 'Test run not found',
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
|
|
479
|
+
// Find report by testRun.id
|
|
480
|
+
const report = await prisma.report.findFirst({
|
|
481
|
+
where: { testRunId: testRun.id },
|
|
482
|
+
include: {
|
|
483
|
+
testRun: {
|
|
484
|
+
include: {
|
|
485
|
+
environment: true,
|
|
486
|
+
tags: true,
|
|
487
|
+
},
|
|
488
|
+
},
|
|
489
|
+
features: {
|
|
490
|
+
include: {
|
|
491
|
+
tags: true,
|
|
492
|
+
scenarios: {
|
|
493
|
+
include: {
|
|
494
|
+
tags: true,
|
|
495
|
+
steps: {
|
|
496
|
+
orderBy: {
|
|
497
|
+
order: 'asc',
|
|
498
|
+
},
|
|
499
|
+
},
|
|
500
|
+
hooks: true,
|
|
501
|
+
},
|
|
502
|
+
},
|
|
503
|
+
},
|
|
504
|
+
},
|
|
505
|
+
testCases: {
|
|
506
|
+
include: {
|
|
507
|
+
testRunTestCase: {
|
|
508
|
+
include: {
|
|
509
|
+
testCase: {
|
|
510
|
+
include: {
|
|
511
|
+
tags: true,
|
|
512
|
+
},
|
|
513
|
+
},
|
|
514
|
+
},
|
|
515
|
+
},
|
|
516
|
+
reportScenario: {
|
|
517
|
+
include: {
|
|
518
|
+
reportFeature: true,
|
|
519
|
+
},
|
|
520
|
+
},
|
|
521
|
+
},
|
|
522
|
+
},
|
|
523
|
+
},
|
|
524
|
+
})
|
|
525
|
+
|
|
526
|
+
if (!report) {
|
|
527
|
+
return {
|
|
528
|
+
status: 404,
|
|
529
|
+
error: 'Report not found for this test run',
|
|
530
|
+
}
|
|
531
|
+
}
|
|
532
|
+
|
|
533
|
+
return {
|
|
534
|
+
status: 200,
|
|
535
|
+
data: report as ReportDetailWithRelations,
|
|
536
|
+
}
|
|
537
|
+
} catch (error) {
|
|
538
|
+
console.error(`[ReportActions] Error fetching report for testRunId ${testRunId}:`, error)
|
|
539
|
+
return {
|
|
540
|
+
status: 500,
|
|
541
|
+
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
542
|
+
}
|
|
543
|
+
}
|
|
544
|
+
}
|
|
545
|
+
|
|
546
|
+
/**
|
|
547
|
+
* Gets the test case metrics for a report
|
|
548
|
+
*/
|
|
549
|
+
export const getAllTestCaseMetricsAction = async (filter: string): Promise<ActionResponse> => {
|
|
550
|
+
try {
|
|
551
|
+
let testCaseMetrics = await prisma.testCaseMetrics.findMany({
|
|
552
|
+
include: {
|
|
553
|
+
testCase: {
|
|
554
|
+
include: {
|
|
555
|
+
tags: true,
|
|
556
|
+
},
|
|
557
|
+
},
|
|
558
|
+
},
|
|
559
|
+
})
|
|
560
|
+
if (filter === 'repeatedlyFailing') {
|
|
561
|
+
testCaseMetrics = testCaseMetrics.filter(tc => tc.isRepeatedlyFailing)
|
|
562
|
+
} else if (filter === 'flaky') {
|
|
563
|
+
testCaseMetrics = testCaseMetrics.filter(tc => tc.isFlaky)
|
|
564
|
+
}
|
|
565
|
+
return {
|
|
566
|
+
status: 200,
|
|
567
|
+
data: testCaseMetrics,
|
|
568
|
+
}
|
|
569
|
+
} catch (error) {
|
|
570
|
+
console.error(`[ReportActions] Error fetching all test case metrics:`, error)
|
|
571
|
+
return {
|
|
572
|
+
status: 500,
|
|
573
|
+
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
574
|
+
}
|
|
575
|
+
}
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
/**
|
|
579
|
+
* Gets the test suite metrics for a report
|
|
580
|
+
*/
|
|
581
|
+
export const getAllTestSuiteMetricsAction = async (filter: string): Promise<ActionResponse> => {
|
|
582
|
+
try {
|
|
583
|
+
let testSuiteMetrics = await prisma.testSuiteMetrics.findMany({
|
|
584
|
+
include: {
|
|
585
|
+
testSuite: {
|
|
586
|
+
include: {
|
|
587
|
+
tags: true,
|
|
588
|
+
testCases: true,
|
|
589
|
+
},
|
|
590
|
+
},
|
|
591
|
+
},
|
|
592
|
+
})
|
|
593
|
+
if (filter === 'notExecutedRecently') {
|
|
594
|
+
// Calculate the date threshold for recent runs (7 days ago)
|
|
595
|
+
const recentPeriodDate = new Date()
|
|
596
|
+
recentPeriodDate.setDate(recentPeriodDate.getDate() - 7)
|
|
597
|
+
|
|
598
|
+
// Filter for suites that have never been executed OR were executed more than 7 days ago
|
|
599
|
+
testSuiteMetrics = testSuiteMetrics.filter(
|
|
600
|
+
ts => ts.lastExecutedAt === null || ts.lastExecutedAt < recentPeriodDate,
|
|
601
|
+
)
|
|
602
|
+
}
|
|
603
|
+
return {
|
|
604
|
+
status: 200,
|
|
605
|
+
data: testSuiteMetrics,
|
|
606
|
+
}
|
|
607
|
+
} catch (error) {
|
|
608
|
+
console.error(`[ReportActions] Error fetching all test suite metrics:`, error)
|
|
609
|
+
return {
|
|
610
|
+
status: 500,
|
|
611
|
+
error: `Server error occurred: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
612
|
+
}
|
|
613
|
+
}
|
|
614
614
|
}
|