create-appraise 0.1.0

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.
Files changed (330) hide show
  1. package/README.md +52 -0
  2. package/package.json +63 -0
  3. package/templates/default/.env.example +2 -0
  4. package/templates/default/README.md +51 -0
  5. package/templates/default/appraise.config.json +4 -0
  6. package/templates/default/components.json +24 -0
  7. package/templates/default/eslint.config.mjs +15 -0
  8. package/templates/default/next-env.d.ts +6 -0
  9. package/templates/default/next.config.ts +7 -0
  10. package/templates/default/package-lock.json +14321 -0
  11. package/templates/default/package.json +124 -0
  12. package/templates/default/postcss.config.mjs +8 -0
  13. package/templates/default/prisma/migrations/20251026202316_migrate_back_to_sqlite/migration.sql +257 -0
  14. package/templates/default/prisma/migrations/20251104113456_add_type_for_template_step_groups/migration.sql +16 -0
  15. package/templates/default/prisma/migrations/20251104170946_add_tags_to_test_suite_and_test_case/migration.sql +27 -0
  16. package/templates/default/prisma/migrations/20251112190024_add_cascade_delete_to_test_run_test_case/migration.sql +17 -0
  17. package/templates/default/prisma/migrations/20251113181100_add_test_run_log/migration.sql +12 -0
  18. package/templates/default/prisma/migrations/20251119191838_add_tag_type/migration.sql +28 -0
  19. package/templates/default/prisma/migrations/20251121164059_add_conflict_resolution/migration.sql +12 -0
  20. package/templates/default/prisma/migrations/20251130190737_add_trace_path_to_test_run_test_case/migration.sql +2 -0
  21. package/templates/default/prisma/migrations/20251213074835_add_log_path_to_test_run/migration.sql +2 -0
  22. package/templates/default/prisma/migrations/20251213183952_add_name_property_for_the_test_run_entities/migration.sql +30 -0
  23. package/templates/default/prisma/migrations/20251223183400_add_report_model_to_db_schema/migration.sql +10 -0
  24. package/templates/default/prisma/migrations/20251223183637_add_report_test_case_entity_for_storing_test_results_for_individual_test_cases/migration.sql +10 -0
  25. package/templates/default/prisma/migrations/20251224083549_add_comprehensive_report_storage/migration.sql +108 -0
  26. package/templates/default/prisma/migrations/20251229194422_migrate_duration_to_string/migration.sql +55 -0
  27. package/templates/default/prisma/migrations/20251230124637_add_unique_constraint_to_test_run_name/migration.sql +27 -0
  28. package/templates/default/prisma/migrations/20260115094436_add_dashboard_metrics/migration.sql +59 -0
  29. package/templates/default/prisma/migrations/20260127172022_add_cascade_delete_to_step_parameters/migration.sql +34 -0
  30. package/templates/default/prisma/migrations/migration_lock.toml +3 -0
  31. package/templates/default/prisma/schema.prisma +554 -0
  32. package/templates/default/public/favicon.ico +0 -0
  33. package/templates/default/public/file.svg +1 -0
  34. package/templates/default/public/globe.svg +1 -0
  35. package/templates/default/public/next.svg +1 -0
  36. package/templates/default/public/vercel.svg +1 -0
  37. package/templates/default/public/window.svg +1 -0
  38. package/templates/default/scripts/regenerate-features.ts +94 -0
  39. package/templates/default/scripts/setup-env.ts +19 -0
  40. package/templates/default/scripts/sync-all.ts +341 -0
  41. package/templates/default/scripts/sync-environments.ts +323 -0
  42. package/templates/default/scripts/sync-locator-groups.ts +413 -0
  43. package/templates/default/scripts/sync-locators.ts +402 -0
  44. package/templates/default/scripts/sync-modules.ts +349 -0
  45. package/templates/default/scripts/sync-tags.ts +292 -0
  46. package/templates/default/scripts/sync-template-step-groups.ts +399 -0
  47. package/templates/default/scripts/sync-template-steps.ts +806 -0
  48. package/templates/default/scripts/sync-test-cases.ts +905 -0
  49. package/templates/default/scripts/sync-test-suites.ts +411 -0
  50. package/templates/default/src/actions/conflict/conflict.action.ts +33 -0
  51. package/templates/default/src/actions/dashboard/dashboard-actions.ts +241 -0
  52. package/templates/default/src/actions/environments/environment-actions.ts +205 -0
  53. package/templates/default/src/actions/locator/locator-actions.ts +547 -0
  54. package/templates/default/src/actions/locator-groups/locator-group-actions.ts +344 -0
  55. package/templates/default/src/actions/modules/module-actions.ts +133 -0
  56. package/templates/default/src/actions/reports/report-actions.ts +614 -0
  57. package/templates/default/src/actions/review/review-actions.ts +147 -0
  58. package/templates/default/src/actions/tags/tag-actions.ts +104 -0
  59. package/templates/default/src/actions/template-step/template-step-actions.ts +332 -0
  60. package/templates/default/src/actions/template-step-group/template-step-group-actions.ts +278 -0
  61. package/templates/default/src/actions/template-test-case/template-test-case-actions.ts +238 -0
  62. package/templates/default/src/actions/test-case/test-case-actions.ts +419 -0
  63. package/templates/default/src/actions/test-run/test-run-actions.ts +1185 -0
  64. package/templates/default/src/actions/test-suite/test-suite-actions.ts +253 -0
  65. package/templates/default/src/actions/user/user-actions.ts +13 -0
  66. package/templates/default/src/app/(base)/environments/create/page.tsx +28 -0
  67. package/templates/default/src/app/(base)/environments/environment-form.tsx +219 -0
  68. package/templates/default/src/app/(base)/environments/environment-table-columns.tsx +96 -0
  69. package/templates/default/src/app/(base)/environments/environment-table.tsx +24 -0
  70. package/templates/default/src/app/(base)/environments/modify/[id]/page.tsx +46 -0
  71. package/templates/default/src/app/(base)/environments/page.tsx +59 -0
  72. package/templates/default/src/app/(base)/layout.tsx +10 -0
  73. package/templates/default/src/app/(base)/locator-groups/create/page.tsx +44 -0
  74. package/templates/default/src/app/(base)/locator-groups/locator-group-form.tsx +215 -0
  75. package/templates/default/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -0
  76. package/templates/default/src/app/(base)/locator-groups/locator-group-table.tsx +28 -0
  77. package/templates/default/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -0
  78. package/templates/default/src/app/(base)/locator-groups/page.tsx +61 -0
  79. package/templates/default/src/app/(base)/locators/create/page.tsx +38 -0
  80. package/templates/default/src/app/(base)/locators/locator-form.tsx +163 -0
  81. package/templates/default/src/app/(base)/locators/locator-table-columns.tsx +90 -0
  82. package/templates/default/src/app/(base)/locators/locator-table.tsx +28 -0
  83. package/templates/default/src/app/(base)/locators/modify/[id]/page.tsx +45 -0
  84. package/templates/default/src/app/(base)/locators/page.tsx +65 -0
  85. package/templates/default/src/app/(base)/locators/sync-locators-button.tsx +66 -0
  86. package/templates/default/src/app/(base)/modules/create/page.tsx +34 -0
  87. package/templates/default/src/app/(base)/modules/modify/[id]/page.tsx +46 -0
  88. package/templates/default/src/app/(base)/modules/module-form.tsx +126 -0
  89. package/templates/default/src/app/(base)/modules/module-table-columns.tsx +85 -0
  90. package/templates/default/src/app/(base)/modules/module-table.tsx +24 -0
  91. package/templates/default/src/app/(base)/modules/page.tsx +59 -0
  92. package/templates/default/src/app/(base)/reports/[id]/page.tsx +517 -0
  93. package/templates/default/src/app/(base)/reports/duration-chart.tsx +33 -0
  94. package/templates/default/src/app/(base)/reports/feature-chart.tsx +78 -0
  95. package/templates/default/src/app/(base)/reports/overview-chart.tsx +46 -0
  96. package/templates/default/src/app/(base)/reports/page.tsx +98 -0
  97. package/templates/default/src/app/(base)/reports/report-metric-card.tsx +16 -0
  98. package/templates/default/src/app/(base)/reports/report-table-columns.tsx +189 -0
  99. package/templates/default/src/app/(base)/reports/report-table.tsx +72 -0
  100. package/templates/default/src/app/(base)/reports/report-view-table-columns.tsx +131 -0
  101. package/templates/default/src/app/(base)/reports/report-view-table.tsx +82 -0
  102. package/templates/default/src/app/(base)/reports/test-cases/page.tsx +42 -0
  103. package/templates/default/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +115 -0
  104. package/templates/default/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -0
  105. package/templates/default/src/app/(base)/reports/test-suites/page.tsx +42 -0
  106. package/templates/default/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +79 -0
  107. package/templates/default/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +27 -0
  108. package/templates/default/src/app/(base)/reports/view-logs-button.tsx +60 -0
  109. package/templates/default/src/app/(base)/reviews/create/page.tsx +26 -0
  110. package/templates/default/src/app/(base)/reviews/created-reviews-table.tsx +15 -0
  111. package/templates/default/src/app/(base)/reviews/modify/[id]/page.tsx +26 -0
  112. package/templates/default/src/app/(base)/reviews/page.tsx +26 -0
  113. package/templates/default/src/app/(base)/reviews/review/[id]/page.tsx +26 -0
  114. package/templates/default/src/app/(base)/reviews/review-form.tsx +11 -0
  115. package/templates/default/src/app/(base)/reviews/review-table-by-creator-columns.tsx +9 -0
  116. package/templates/default/src/app/(base)/reviews/review-table-by-reviewer-columns.tsx +9 -0
  117. package/templates/default/src/app/(base)/reviews/reviewer-reviews-table.tsx +15 -0
  118. package/templates/default/src/app/(base)/tags/create/page.tsx +39 -0
  119. package/templates/default/src/app/(base)/tags/modify/[id]/page.tsx +50 -0
  120. package/templates/default/src/app/(base)/tags/page.tsx +58 -0
  121. package/templates/default/src/app/(base)/tags/tag-form.tsx +147 -0
  122. package/templates/default/src/app/(base)/tags/tag-table-columns.tsx +63 -0
  123. package/templates/default/src/app/(base)/tags/tag-table.tsx +29 -0
  124. package/templates/default/src/app/(base)/template-step-groups/create/page.tsx +28 -0
  125. package/templates/default/src/app/(base)/template-step-groups/modify/[id]/page.tsx +45 -0
  126. package/templates/default/src/app/(base)/template-step-groups/page.tsx +60 -0
  127. package/templates/default/src/app/(base)/template-step-groups/template-step-group-form.tsx +167 -0
  128. package/templates/default/src/app/(base)/template-step-groups/template-step-group-table-columns.tsx +89 -0
  129. package/templates/default/src/app/(base)/template-step-groups/template-step-group-table.tsx +32 -0
  130. package/templates/default/src/app/(base)/template-steps/create/page.tsx +37 -0
  131. package/templates/default/src/app/(base)/template-steps/modify/[id]/page.tsx +49 -0
  132. package/templates/default/src/app/(base)/template-steps/page.tsx +59 -0
  133. package/templates/default/src/app/(base)/template-steps/paramChip.tsx +213 -0
  134. package/templates/default/src/app/(base)/template-steps/template-step-form.tsx +384 -0
  135. package/templates/default/src/app/(base)/template-steps/template-step-table-columns.tsx +158 -0
  136. package/templates/default/src/app/(base)/template-steps/template-step-table.tsx +24 -0
  137. package/templates/default/src/app/(base)/template-test-cases/create/page.tsx +56 -0
  138. package/templates/default/src/app/(base)/template-test-cases/modify/[id]/page.tsx +89 -0
  139. package/templates/default/src/app/(base)/template-test-cases/page.tsx +58 -0
  140. package/templates/default/src/app/(base)/template-test-cases/template-test-case-flow.tsx +84 -0
  141. package/templates/default/src/app/(base)/template-test-cases/template-test-case-form.tsx +262 -0
  142. package/templates/default/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -0
  143. package/templates/default/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -0
  144. package/templates/default/src/app/(base)/test-cases/create/page.tsx +76 -0
  145. package/templates/default/src/app/(base)/test-cases/create-from-template/generate/[id]/page.tsx +96 -0
  146. package/templates/default/src/app/(base)/test-cases/create-from-template/page.tsx +38 -0
  147. package/templates/default/src/app/(base)/test-cases/create-from-template/template-selection-form.tsx +73 -0
  148. package/templates/default/src/app/(base)/test-cases/modify/[id]/page.tsx +106 -0
  149. package/templates/default/src/app/(base)/test-cases/page.tsx +60 -0
  150. package/templates/default/src/app/(base)/test-cases/test-case-flow.tsx +82 -0
  151. package/templates/default/src/app/(base)/test-cases/test-case-form.tsx +395 -0
  152. package/templates/default/src/app/(base)/test-cases/test-case-table-columns.tsx +90 -0
  153. package/templates/default/src/app/(base)/test-cases/test-case-table.tsx +35 -0
  154. package/templates/default/src/app/(base)/test-runs/[id]/page.tsx +56 -0
  155. package/templates/default/src/app/(base)/test-runs/create/page.tsx +47 -0
  156. package/templates/default/src/app/(base)/test-runs/page.tsx +60 -0
  157. package/templates/default/src/app/(base)/test-runs/test-run-form.tsx +512 -0
  158. package/templates/default/src/app/(base)/test-runs/test-run-table-columns.tsx +229 -0
  159. package/templates/default/src/app/(base)/test-runs/test-run-table.tsx +127 -0
  160. package/templates/default/src/app/(base)/test-suites/create/page.tsx +45 -0
  161. package/templates/default/src/app/(base)/test-suites/modify/[id]/page.tsx +55 -0
  162. package/templates/default/src/app/(base)/test-suites/page.tsx +82 -0
  163. package/templates/default/src/app/(base)/test-suites/test-suite-form.tsx +269 -0
  164. package/templates/default/src/app/(base)/test-suites/test-suite-table-columns.tsx +97 -0
  165. package/templates/default/src/app/(base)/test-suites/test-suite-table.tsx +29 -0
  166. package/templates/default/src/app/(dashboard-components)/app-drawer.tsx +187 -0
  167. package/templates/default/src/app/(dashboard-components)/data-card-grid.tsx +13 -0
  168. package/templates/default/src/app/(dashboard-components)/data-card.tsx +27 -0
  169. package/templates/default/src/app/(dashboard-components)/execution-health-panel.tsx +57 -0
  170. package/templates/default/src/app/(dashboard-components)/ongoing-test-runs-card.tsx +87 -0
  171. package/templates/default/src/app/(dashboard-components)/quick-actions-drawer.tsx +45 -0
  172. package/templates/default/src/app/api/test-runs/[runId]/download/route.ts +133 -0
  173. package/templates/default/src/app/api/test-runs/[runId]/logs/route.ts +420 -0
  174. package/templates/default/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.ts +146 -0
  175. package/templates/default/src/app/favicon.ico +0 -0
  176. package/templates/default/src/app/globals.css +147 -0
  177. package/templates/default/src/app/layout.tsx +171 -0
  178. package/templates/default/src/app/page.tsx +64 -0
  179. package/templates/default/src/assets/icons/empty-tube.tsx +23 -0
  180. package/templates/default/src/assets/icons/tube-plus.tsx +29 -0
  181. package/templates/default/src/components/base-node.tsx +21 -0
  182. package/templates/default/src/components/chart/pie-chart.tsx +73 -0
  183. package/templates/default/src/components/data-extraction/locator-inspector.tsx +460 -0
  184. package/templates/default/src/components/data-state/empty-state.tsx +40 -0
  185. package/templates/default/src/components/data-visualization/info-card.tsx +70 -0
  186. package/templates/default/src/components/data-visualization/info-grid.tsx +22 -0
  187. package/templates/default/src/components/devtools/providers.tsx +13 -0
  188. package/templates/default/src/components/diagram/button-edge.tsx +54 -0
  189. package/templates/default/src/components/diagram/dynamic-parameters.tsx +438 -0
  190. package/templates/default/src/components/diagram/edit-header-option.tsx +36 -0
  191. package/templates/default/src/components/diagram/flow-diagram.tsx +470 -0
  192. package/templates/default/src/components/diagram/node-form.tsx +262 -0
  193. package/templates/default/src/components/diagram/options-header-node.tsx +57 -0
  194. package/templates/default/src/components/diagram/template-step-combobox.tsx +155 -0
  195. package/templates/default/src/components/form/error-message.tsx +7 -0
  196. package/templates/default/src/components/kokonutui/smooth-tab.tsx +453 -0
  197. package/templates/default/src/components/loading-skeleton/data-table/data-table-skeleton.tsx +30 -0
  198. package/templates/default/src/components/loading-skeleton/form/button-skeleton.tsx +8 -0
  199. package/templates/default/src/components/loading-skeleton/form/icon-button-skeleton.tsx +8 -0
  200. package/templates/default/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -0
  201. package/templates/default/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -0
  202. package/templates/default/src/components/logo.tsx +15 -0
  203. package/templates/default/src/components/navigation/command-badge.tsx +34 -0
  204. package/templates/default/src/components/navigation/command-chain-input.tsx +51 -0
  205. package/templates/default/src/components/navigation/entity-search-command.tsx +116 -0
  206. package/templates/default/src/components/navigation/nav-card.tsx +31 -0
  207. package/templates/default/src/components/navigation/nav-command.tsx +508 -0
  208. package/templates/default/src/components/navigation/nav-link.tsx +60 -0
  209. package/templates/default/src/components/navigation/nav-menu-card-deck.tsx +112 -0
  210. package/templates/default/src/components/node-header.tsx +159 -0
  211. package/templates/default/src/components/reports/test-case-logs-modal.tsx +253 -0
  212. package/templates/default/src/components/table/table-actions.tsx +172 -0
  213. package/templates/default/src/components/test-run/download-logs-button.tsx +99 -0
  214. package/templates/default/src/components/test-run/log-viewer.tsx +445 -0
  215. package/templates/default/src/components/test-run/test-run-details.tsx +611 -0
  216. package/templates/default/src/components/test-run/test-run-header.tsx +149 -0
  217. package/templates/default/src/components/test-run/view-report-button.tsx +102 -0
  218. package/templates/default/src/components/theme/mode-toggle.tsx +54 -0
  219. package/templates/default/src/components/theme/theme-provider.tsx +8 -0
  220. package/templates/default/src/components/typography/page-header-subtitle.tsx +7 -0
  221. package/templates/default/src/components/typography/page-header.tsx +7 -0
  222. package/templates/default/src/components/ui/alert-dialog.tsx +106 -0
  223. package/templates/default/src/components/ui/alert.tsx +43 -0
  224. package/templates/default/src/components/ui/avatar.tsx +40 -0
  225. package/templates/default/src/components/ui/badge.tsx +29 -0
  226. package/templates/default/src/components/ui/button.tsx +47 -0
  227. package/templates/default/src/components/ui/calendar.tsx +158 -0
  228. package/templates/default/src/components/ui/card.tsx +43 -0
  229. package/templates/default/src/components/ui/chart.tsx +369 -0
  230. package/templates/default/src/components/ui/checkbox.tsx +28 -0
  231. package/templates/default/src/components/ui/command.tsx +135 -0
  232. package/templates/default/src/components/ui/data-table-column-header.tsx +61 -0
  233. package/templates/default/src/components/ui/data-table-pagination.tsx +87 -0
  234. package/templates/default/src/components/ui/data-table-view-options.tsx +50 -0
  235. package/templates/default/src/components/ui/data-table.tsx +267 -0
  236. package/templates/default/src/components/ui/dialog.tsx +97 -0
  237. package/templates/default/src/components/ui/dropdown-menu.tsx +182 -0
  238. package/templates/default/src/components/ui/empty.tsx +104 -0
  239. package/templates/default/src/components/ui/input.tsx +22 -0
  240. package/templates/default/src/components/ui/kbd.tsx +28 -0
  241. package/templates/default/src/components/ui/label.tsx +19 -0
  242. package/templates/default/src/components/ui/loading.tsx +12 -0
  243. package/templates/default/src/components/ui/multi-select-with-preview.tsx +116 -0
  244. package/templates/default/src/components/ui/multi-select.tsx +142 -0
  245. package/templates/default/src/components/ui/navigation-menu.tsx +120 -0
  246. package/templates/default/src/components/ui/popover.tsx +33 -0
  247. package/templates/default/src/components/ui/progress.tsx +25 -0
  248. package/templates/default/src/components/ui/radio-group.tsx +44 -0
  249. package/templates/default/src/components/ui/scroll-area.tsx +40 -0
  250. package/templates/default/src/components/ui/select.tsx +144 -0
  251. package/templates/default/src/components/ui/separator.tsx +22 -0
  252. package/templates/default/src/components/ui/skeleton.tsx +7 -0
  253. package/templates/default/src/components/ui/table.tsx +76 -0
  254. package/templates/default/src/components/ui/tabs.tsx +55 -0
  255. package/templates/default/src/components/ui/textarea.tsx +21 -0
  256. package/templates/default/src/components/ui/toast.tsx +113 -0
  257. package/templates/default/src/components/ui/toaster.tsx +26 -0
  258. package/templates/default/src/components/ui/tooltip.tsx +32 -0
  259. package/templates/default/src/components/user-prompt/delete-prompt.tsx +87 -0
  260. package/templates/default/src/config/db-config.ts +10 -0
  261. package/templates/default/src/constants/form-opts/diagram/node-form.ts +30 -0
  262. package/templates/default/src/constants/form-opts/environment-form-opts.ts +24 -0
  263. package/templates/default/src/constants/form-opts/locator-form-opts.ts +20 -0
  264. package/templates/default/src/constants/form-opts/locator-group-form-opts.ts +28 -0
  265. package/templates/default/src/constants/form-opts/module-form-opts.ts +21 -0
  266. package/templates/default/src/constants/form-opts/review-form-opts.ts +23 -0
  267. package/templates/default/src/constants/form-opts/tag-form-opts.ts +42 -0
  268. package/templates/default/src/constants/form-opts/template-selection-form-opts.ts +16 -0
  269. package/templates/default/src/constants/form-opts/template-step-group-form-opts.ts +24 -0
  270. package/templates/default/src/constants/form-opts/template-test-case-form-opts.ts +39 -0
  271. package/templates/default/src/constants/form-opts/template-test-step-form-opts.ts +36 -0
  272. package/templates/default/src/constants/form-opts/test-case-form-opts.ts +43 -0
  273. package/templates/default/src/constants/form-opts/test-run-form-opts.ts +31 -0
  274. package/templates/default/src/constants/form-opts/test-suite-form-opts.ts +24 -0
  275. package/templates/default/src/hooks/use-toast.ts +187 -0
  276. package/templates/default/src/lib/bidirectional-sync.ts +432 -0
  277. package/templates/default/src/lib/database-sync.ts +531 -0
  278. package/templates/default/src/lib/environment-file-utils.ts +221 -0
  279. package/templates/default/src/lib/feature-file-generator.ts +411 -0
  280. package/templates/default/src/lib/gherkin-parser.ts +259 -0
  281. package/templates/default/src/lib/locator-group-file-utils.ts +370 -0
  282. package/templates/default/src/lib/metrics/metric-calculator.ts +613 -0
  283. package/templates/default/src/lib/module-hierarchy-builder.ts +205 -0
  284. package/templates/default/src/lib/path-helpers/module-path.ts +71 -0
  285. package/templates/default/src/lib/test-case-utils.ts +6 -0
  286. package/templates/default/src/lib/test-run/log-formatter.ts +83 -0
  287. package/templates/default/src/lib/test-run/process-manager.ts +191 -0
  288. package/templates/default/src/lib/test-run/report-parser.ts +316 -0
  289. package/templates/default/src/lib/test-run/test-run-executor.ts +144 -0
  290. package/templates/default/src/lib/test-run/winston-logger.ts +95 -0
  291. package/templates/default/src/lib/transformers/gherkin-converter.ts +42 -0
  292. package/templates/default/src/lib/transformers/key-to-icon-transformer.tsx +95 -0
  293. package/templates/default/src/lib/transformers/template-test-case-converter.ts +160 -0
  294. package/templates/default/src/lib/utils/node-param-validation.ts +81 -0
  295. package/templates/default/src/lib/utils/template-step-file-generator.ts +167 -0
  296. package/templates/default/src/lib/utils/template-step-file-manager-intelligent.ts +723 -0
  297. package/templates/default/src/lib/utils/template-step-file-manager.ts +166 -0
  298. package/templates/default/src/lib/utils.ts +31 -0
  299. package/templates/default/src/tests/config/environments/environments.json +14 -0
  300. package/templates/default/src/tests/config/executor/world.ts +41 -0
  301. package/templates/default/src/tests/executor.ts +80 -0
  302. package/templates/default/src/tests/hooks/hooks.ts +99 -0
  303. package/templates/default/src/tests/mapping/locator-map.json +1 -0
  304. package/templates/default/src/tests/steps/actions/click.step.ts +62 -0
  305. package/templates/default/src/tests/steps/actions/hover.step.ts +31 -0
  306. package/templates/default/src/tests/steps/actions/input.step.ts +149 -0
  307. package/templates/default/src/tests/steps/actions/navigation.step.ts +72 -0
  308. package/templates/default/src/tests/steps/actions/random_data.step.ts +146 -0
  309. package/templates/default/src/tests/steps/actions/store.step.ts +90 -0
  310. package/templates/default/src/tests/steps/actions/wait.step.ts +107 -0
  311. package/templates/default/src/tests/steps/validations/active_state_assertion.step.ts +34 -0
  312. package/templates/default/src/tests/steps/validations/navigation_assertion.step.ts +23 -0
  313. package/templates/default/src/tests/steps/validations/text_assertion.step.ts +111 -0
  314. package/templates/default/src/tests/steps/validations/visibility_assertion.step.ts +30 -0
  315. package/templates/default/src/tests/support/parameter-types.ts +12 -0
  316. package/templates/default/src/tests/utils/cache.util.ts +260 -0
  317. package/templates/default/src/tests/utils/cli.util.ts +177 -0
  318. package/templates/default/src/tests/utils/environment.util.ts +65 -0
  319. package/templates/default/src/tests/utils/locator.util.ts +248 -0
  320. package/templates/default/src/tests/utils/random-data.util.ts +45 -0
  321. package/templates/default/src/tests/utils/spawner.util.ts +617 -0
  322. package/templates/default/src/types/diagram/diagram.ts +34 -0
  323. package/templates/default/src/types/diagram/template-step.ts +11 -0
  324. package/templates/default/src/types/executor/browser.type.ts +1 -0
  325. package/templates/default/src/types/form/actionHandler.ts +6 -0
  326. package/templates/default/src/types/locator/locator.type.ts +11 -0
  327. package/templates/default/src/types/step/step.type.ts +1 -0
  328. package/templates/default/src/types/table/data-table.ts +6 -0
  329. package/templates/default/tailwind.config.ts +62 -0
  330. package/templates/default/tsconfig.json +28 -0
@@ -0,0 +1,269 @@
1
+ 'use client'
2
+
3
+ import { Button } from '@/components/ui/button'
4
+ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@/components/ui/card'
5
+ import { Input } from '@/components/ui/input'
6
+ import { Label } from '@/components/ui/label'
7
+ import MultiSelectWithPreview from '@/components/ui/multi-select-with-preview'
8
+ import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select'
9
+ import { Textarea } from '@/components/ui/textarea'
10
+ import { formOpts, TestSuite } from '@/constants/form-opts/test-suite-form-opts'
11
+ import { toast } from '@/hooks/use-toast'
12
+ import { ActionResponse } from '@/types/form/actionHandler'
13
+ import { Module, TestCase, Tag } from '@prisma/client'
14
+ import { useForm } from '@tanstack/react-form'
15
+ import { Info, Save } from 'lucide-react'
16
+ import { useRouter } from 'next/navigation'
17
+ import { z } from 'zod'
18
+
19
+ export const TestSuiteForm = ({
20
+ defaultValues,
21
+ successTitle,
22
+ successMessage,
23
+ id,
24
+ onSubmitAction,
25
+ testCases,
26
+ moduleList,
27
+ tags,
28
+ }: {
29
+ defaultValues?: TestSuite
30
+ successTitle: string
31
+ successMessage: string
32
+ id?: string
33
+ onSubmitAction: (_prev: unknown, value: TestSuite, id?: string) => Promise<ActionResponse>
34
+ testCases: TestCase[]
35
+ moduleList: Module[]
36
+ tags: Tag[]
37
+ }) => {
38
+ const router = useRouter()
39
+ const form = useForm({
40
+ defaultValues: defaultValues ?? formOpts?.defaultValues,
41
+ validators: formOpts?.validators,
42
+ onSubmit: async ({ value }) => {
43
+ const res = await onSubmitAction(undefined, value, id)
44
+ if (res.status === 200) {
45
+ toast({
46
+ title: successTitle,
47
+ description: successMessage,
48
+ })
49
+ router.push('/test-suites')
50
+ }
51
+ if (res.status === 400) {
52
+ toast({
53
+ title: 'Error',
54
+ description: res.error,
55
+ variant: 'destructive',
56
+ })
57
+ }
58
+ if (res.status === 500) {
59
+ toast({
60
+ title: 'Error',
61
+ description: res.error,
62
+ variant: 'destructive',
63
+ })
64
+ }
65
+ },
66
+ })
67
+ return (
68
+ <div className="flex justify-between gap-20 overflow-x-hidden">
69
+ <Card className="w-2/3 border-gray-700 bg-gray-500/10">
70
+ <CardHeader>
71
+ <CardTitle className="text-xl font-bold text-primary">Test Suite Details</CardTitle>
72
+ <CardDescription>Enter the details for your test suite</CardDescription>
73
+ </CardHeader>
74
+ <CardContent>
75
+ <form
76
+ onSubmit={e => {
77
+ e.preventDefault()
78
+ e.stopPropagation()
79
+ form.handleSubmit()
80
+ }}
81
+ >
82
+ <form.Field
83
+ name="name"
84
+ validators={{
85
+ onChange: z.string().min(3, { message: 'Name must be at least 3 characters' }),
86
+ }}
87
+ >
88
+ {field => {
89
+ return (
90
+ <div className="mb-6 flex flex-col gap-2">
91
+ <Label htmlFor={field.name} className="font-bold">
92
+ Name
93
+ </Label>
94
+ <Input
95
+ className="w-full"
96
+ id={field.name}
97
+ name={field.name}
98
+ value={field.state.value}
99
+ onChange={e => field.handleChange(e.target.value)}
100
+ placeholder="Enter name for your test suite"
101
+ />
102
+ {field.state.meta.errors.map((error, index) => (
103
+ <p key={index} className="text-xs text-pink-500">
104
+ {typeof error === 'string' ? error : error?.message || String(error)}
105
+ </p>
106
+ ))}
107
+ </div>
108
+ )
109
+ }}
110
+ </form.Field>
111
+ <form.Field name="description">
112
+ {field => {
113
+ return (
114
+ <div className="mb-6 flex flex-col gap-2">
115
+ <Label htmlFor={field.name} className="font-bold">
116
+ Description
117
+ </Label>
118
+ <Textarea
119
+ className="h-24 w-full bg-background"
120
+ id={field.name}
121
+ name={field.name}
122
+ value={field.state.value}
123
+ onChange={e => field.handleChange(e.target.value)}
124
+ placeholder="Enter description for your test suite"
125
+ />
126
+ {field.state.meta.errors.map((error, index) => (
127
+ <p key={index} className="text-xs text-pink-500">
128
+ {typeof error === 'string' ? error : error?.message || String(error)}
129
+ </p>
130
+ ))}
131
+ </div>
132
+ )
133
+ }}
134
+ </form.Field>
135
+ <form.Field name="testCases">
136
+ {field => {
137
+ const testCasesOptions = testCases.map(testCase => ({
138
+ value: testCase.id,
139
+ label: testCase.title,
140
+ }))
141
+ return (
142
+ <div className="mb-6 flex flex-col gap-2">
143
+ <Label htmlFor={field.name} className="font-bold">
144
+ Test Cases
145
+ </Label>
146
+ <MultiSelectWithPreview
147
+ id={field.name}
148
+ className="w-full"
149
+ options={testCasesOptions}
150
+ onSelectChange={value => {
151
+ field.handleChange(value)
152
+ }}
153
+ defaultSelectedValues={field.state.value}
154
+ placeholder="Select test case(s)"
155
+ emptyMessage="No test case(s) found"
156
+ selectedLabel="Selected test case(s)"
157
+ searchPlaceholder="Search test cases..."
158
+ />
159
+ </div>
160
+ )
161
+ }}
162
+ </form.Field>
163
+ <form.Field name="moduleId">
164
+ {field => {
165
+ return (
166
+ <div className="mb-6 flex flex-col gap-2">
167
+ <Label htmlFor={field.name} className="font-bold">
168
+ Module
169
+ </Label>
170
+ <Select value={field.state.value} onValueChange={value => field.handleChange(value)}>
171
+ <SelectTrigger className="w-full bg-background">
172
+ <SelectValue placeholder="Select a module" />
173
+ </SelectTrigger>
174
+ <SelectContent className="w-full">
175
+ {moduleList.map(module => (
176
+ <SelectItem key={module.id} value={module.id}>
177
+ {module.name}
178
+ </SelectItem>
179
+ ))}
180
+ </SelectContent>
181
+ </Select>
182
+ </div>
183
+ )
184
+ }}
185
+ </form.Field>
186
+ <form.Field name="tagIds">
187
+ {field => {
188
+ const tagsOptions = tags.map(tag => ({
189
+ value: tag.id,
190
+ label: tag.name,
191
+ }))
192
+ return (
193
+ <div className="mb-6 flex flex-col gap-2">
194
+ <Label htmlFor={field.name} className="font-bold">
195
+ Tags
196
+ </Label>
197
+ <MultiSelectWithPreview
198
+ id={field.name}
199
+ options={tagsOptions}
200
+ onSelectChange={value => {
201
+ field.handleChange(value)
202
+ }}
203
+ defaultSelectedValues={field.state.value || []}
204
+ placeholder="Select tag(s)"
205
+ emptyMessage="No tag(s) found"
206
+ selectedLabel="Selected tag(s)"
207
+ searchPlaceholder="Search tags..."
208
+ />
209
+ </div>
210
+ )
211
+ }}
212
+ </form.Field>
213
+ <form.Subscribe selector={formState => [formState.canSubmit, formState.isSubmitting]}>
214
+ {([canSubmit, isSubmitting]) => (
215
+ <Button type="submit" disabled={!canSubmit} className="hover:bg-emerald-500">
216
+ <Save className="h-4 w-4" />
217
+ <span className="font-bold">{isSubmitting ? '...' : 'Save'}</span>
218
+ </Button>
219
+ )}
220
+ </form.Subscribe>
221
+ </form>
222
+ </CardContent>
223
+ </Card>
224
+ <div className="lg:w-1/3">
225
+ <Card className="border-gray-700 bg-gray-500/10">
226
+ <CardHeader className="mb-2">
227
+ <CardTitle className="flex items-center gap-2 text-xl text-primary">
228
+ <Info className="h-5 w-5" />
229
+ <span className="font-bold">Quick Tips</span>
230
+ </CardTitle>
231
+ </CardHeader>
232
+ <CardContent className="flex flex-col gap-6">
233
+ <div className="flex items-start gap-4">
234
+ <span className="flex h-6 w-6 items-center justify-center rounded-full bg-emerald-500/10 text-emerald-500">
235
+ 1
236
+ </span>
237
+ <div className="flex flex-col gap-1">
238
+ <span className="text-base font-bold">Choose a descriptive name</span>
239
+ <span className="text-sm text-muted-foreground">
240
+ Use clear, specific names that indicate the purpose
241
+ </span>
242
+ </div>
243
+ </div>
244
+ <div className="flex items-start gap-4">
245
+ <span className="flex h-6 w-6 items-center justify-center rounded-full bg-emerald-500/10 text-emerald-500">
246
+ 2
247
+ </span>
248
+ <div className="flex flex-col gap-1">
249
+ <span className="text-base font-bold">Group related tests</span>
250
+ <span className="text-sm text-muted-foreground">
251
+ Organize tests that validate the same feature together
252
+ </span>
253
+ </div>
254
+ </div>
255
+ <div className="flex items-start gap-4">
256
+ <span className="flex h-6 w-6 items-center justify-center rounded-full bg-emerald-500/10 text-emerald-500">
257
+ 3
258
+ </span>
259
+ <div className="flex flex-col gap-1">
260
+ <span className="text-base font-bold">Use meaningful tags</span>
261
+ <span className="text-sm text-muted-foreground">Tags help filter and categorize effectively</span>
262
+ </div>
263
+ </div>
264
+ </CardContent>
265
+ </Card>
266
+ </div>
267
+ </div>
268
+ )
269
+ }
@@ -0,0 +1,97 @@
1
+ 'use client'
2
+
3
+ import { DataTableColumnHeader } from '@/components/ui/data-table-column-header'
4
+ import { TestSuite, Tag, TestCase, Module } from '@prisma/client'
5
+ import { ColumnDef } from '@tanstack/react-table'
6
+
7
+ import { Checkbox } from '@/components/ui/checkbox'
8
+ import TableActions from '@/components/table/table-actions'
9
+ import { deleteTestSuiteAction } from '@/actions/test-suite/test-suite-actions'
10
+ import { formatDateTime } from '@/lib/utils'
11
+ import { Badge } from '@/components/ui/badge'
12
+ export const testSuiteTableCols: ColumnDef<TestSuite & { tags?: Tag[]; module: Module; testCases: TestCase[] }>[] = [
13
+ {
14
+ id: 'select',
15
+ header: ({ table }) => (
16
+ <Checkbox
17
+ checked={table.getIsAllPageRowsSelected() || (table.getIsSomePageRowsSelected() && 'indeterminate')}
18
+ onCheckedChange={value => table.toggleAllPageRowsSelected(!!value)}
19
+ aria-label="Select all"
20
+ className="mr-2"
21
+ />
22
+ ),
23
+ cell: ({ row }) => (
24
+ <Checkbox
25
+ checked={row.getIsSelected()}
26
+ onCheckedChange={value => row.toggleSelected(!!value)}
27
+ aria-label="Select row"
28
+ className="mr-2"
29
+ />
30
+ ),
31
+ enableSorting: false,
32
+ enableHiding: false,
33
+ },
34
+ {
35
+ accessorKey: 'name',
36
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Name" />,
37
+ },
38
+ {
39
+ accessorKey: 'description',
40
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Description" />,
41
+ },
42
+ {
43
+ accessorKey: 'module.name',
44
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Module" />,
45
+ cell: ({ row }) => {
46
+ const testSuite = row.original
47
+ return <Badge variant="outline">{testSuite.module.name}</Badge>
48
+ },
49
+ },
50
+ {
51
+ accessorKey: 'testCases.length',
52
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Total Test Cases" />,
53
+ cell: ({ row }) => {
54
+ const testSuite = row.original
55
+ return <div>{testSuite.testCases.length}</div>
56
+ },
57
+ },
58
+ {
59
+ accessorKey: 'tags',
60
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Tags" />,
61
+ cell: ({ row }) => {
62
+ const testSuite = row.original
63
+ const tags = testSuite.tags || []
64
+ return (
65
+ <div className="flex flex-wrap gap-1">
66
+ {tags.length > 0 ? tags.map(tag => <Badge key={tag.id}>{tag.name}</Badge>) : '-'}
67
+ </div>
68
+ )
69
+ },
70
+ },
71
+ {
72
+ accessorKey: 'createdAt',
73
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Created At" />,
74
+ cell: ({ row }) => {
75
+ return formatDateTime(row.original.createdAt)
76
+ },
77
+ },
78
+ {
79
+ accessorKey: 'updatedAt',
80
+ header: ({ column }) => <DataTableColumnHeader column={column} title="Updated At" />,
81
+ cell: ({ row }) => {
82
+ return formatDateTime(row.original.updatedAt)
83
+ },
84
+ },
85
+ {
86
+ id: 'actions',
87
+ cell: ({ row }) => {
88
+ const testSuite = row.original
89
+ return (
90
+ <TableActions
91
+ modifyLink={`/test-suites/modify/${testSuite.id}`}
92
+ deleteHandler={() => deleteTestSuiteAction([testSuite.id])}
93
+ />
94
+ )
95
+ },
96
+ },
97
+ ]
@@ -0,0 +1,29 @@
1
+ import { DataTable } from '@/components/ui/data-table'
2
+ import React from 'react'
3
+ import { testSuiteTableCols } from './test-suite-table-columns'
4
+ import { getAllTestSuitesAction, deleteTestSuiteAction } from '@/actions/test-suite/test-suite-actions'
5
+ import { Module, TestCase, TestSuite, Tag } from '@prisma/client'
6
+
7
+ const TestSuiteTable = async () => {
8
+ const { data: testSuites, error: testSuitesError } = await getAllTestSuitesAction()
9
+
10
+ if (testSuitesError) {
11
+ return <div>Error: {testSuitesError}</div>
12
+ }
13
+
14
+ return (
15
+ <>
16
+ <DataTable
17
+ columns={testSuiteTableCols}
18
+ data={testSuites as (TestSuite & { tags?: Tag[]; module: Module; testCases: TestCase[] })[]}
19
+ filterColumn="name"
20
+ filterPlaceholder="Filter by name..."
21
+ createLink="/test-suites/create"
22
+ modifyLink="/test-suites/modify"
23
+ deleteAction={deleteTestSuiteAction}
24
+ />
25
+ </>
26
+ )
27
+ }
28
+
29
+ export default TestSuiteTable
@@ -0,0 +1,187 @@
1
+ 'use client'
2
+ import { Button } from '@/components/ui/button'
3
+ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@/components/ui/card'
4
+ import { AlertCircle, AlertTriangle, Clock, XCircle } from 'lucide-react'
5
+ import { DashboardMetrics } from '@prisma/client'
6
+ import { useRouter } from 'next/navigation'
7
+
8
+ type AppDrawerItem = {
9
+ title: string
10
+ icon: React.ReactNode
11
+ color: keyof typeof AppDrawerItemColor
12
+ count: number
13
+ onClick?: () => void
14
+ }
15
+
16
+ export const AppDrawerItemColor = {
17
+ red: {
18
+ buttonColor: 'bg-red-500/20 hover:bg-red-500/25',
19
+ iconColor: 'text-red-500',
20
+ badgeColor: 'bg-red-400 text-red-900',
21
+ },
22
+ green: {
23
+ buttonColor: 'bg-green-500/20 hover:bg-green-500/25',
24
+ iconColor: 'text-green-500',
25
+ badgeColor: 'bg-green-400 text-green-800',
26
+ },
27
+ yellow: {
28
+ buttonColor: 'bg-yellow-500/20 hover:bg-yellow-500/25',
29
+ iconColor: 'text-yellow-500',
30
+ badgeColor: 'bg-yellow-400 text-yellow-800',
31
+ },
32
+ blue: {
33
+ buttonColor: 'bg-blue-500/20 hover:bg-blue-500/25',
34
+ iconColor: 'text-blue-500',
35
+ badgeColor: 'bg-blue-400 text-blue-800',
36
+ },
37
+ emerald: {
38
+ buttonColor: 'bg-emerald-500/20 hover:bg-emerald-500/25',
39
+ iconColor: 'text-emerald-500',
40
+ badgeColor: 'bg-emerald-400 text-emerald-800',
41
+ },
42
+ indigo: {
43
+ buttonColor: 'bg-indigo-500/20 hover:bg-indigo-500/25',
44
+ iconColor: 'text-indigo-500',
45
+ badgeColor: 'bg-indigo-400 text-indigo-800',
46
+ },
47
+ purple: {
48
+ buttonColor: 'bg-purple-500/20 hover:bg-purple-500/25',
49
+ iconColor: 'text-purple-500',
50
+ badgeColor: 'bg-purple-400 text-purple-800',
51
+ },
52
+ pink: {
53
+ buttonColor: 'bg-pink-500/20 hover:bg-pink-500/25',
54
+ iconColor: 'text-pink-500',
55
+ badgeColor: 'bg-pink-400 text-pink-800',
56
+ },
57
+ rose: {
58
+ buttonColor: 'bg-rose-500/20 hover:bg-rose-500/25',
59
+ iconColor: 'text-rose-500',
60
+ badgeColor: 'bg-rose-400 text-rose-800',
61
+ },
62
+ fuchsia: {
63
+ buttonColor: 'bg-fuchsia-500/20 hover:bg-fuchsia-500/25',
64
+ iconColor: 'text-fuchsia-500',
65
+ badgeColor: 'bg-fuchsia-400 text-fuchsia-800',
66
+ },
67
+ violet: {
68
+ buttonColor: 'bg-violet-500/20 hover:bg-violet-500/25',
69
+ iconColor: 'text-violet-500',
70
+ badgeColor: 'bg-violet-400 text-violet-800',
71
+ },
72
+ sky: {
73
+ buttonColor: 'bg-sky-500/20 hover:bg-sky-500/25',
74
+ iconColor: 'text-sky-500',
75
+ badgeColor: 'bg-sky-400 text-sky-800',
76
+ },
77
+ orange: {
78
+ buttonColor: 'bg-orange-500/20 hover:bg-orange-500/25',
79
+ iconColor: 'text-orange-500',
80
+ badgeColor: 'bg-orange-400 text-orange-800',
81
+ },
82
+ gray: {
83
+ buttonColor: 'bg-gray-500/20 hover:bg-gray-500/25',
84
+ iconColor: 'text-gray-500',
85
+ badgeColor: 'bg-gray-400 text-gray-800',
86
+ },
87
+ }
88
+
89
+ export const AppDrawerItem = ({
90
+ title,
91
+ icon,
92
+ colorKey,
93
+ count,
94
+ onClick,
95
+ isActive,
96
+ }: {
97
+ title: string
98
+ icon: React.ReactNode
99
+ colorKey: keyof typeof AppDrawerItemColor
100
+ count: number
101
+ isActive?: boolean
102
+ onClick?: () => void
103
+ }) => {
104
+ const color = !isActive ? AppDrawerItemColor.gray : AppDrawerItemColor[colorKey]
105
+ return (
106
+ <Button
107
+ variant="outline"
108
+ className={`relative flex h-fit w-full flex-col items-center justify-center border-none hover:text-gray-200 ${color.buttonColor} px-2`}
109
+ onClick={onClick}
110
+ disabled={!isActive}
111
+ >
112
+ <div className={`${color.iconColor} [&_svg]:!h-6 [&_svg]:!w-6`}>{icon}</div>
113
+ <div className="text-xs font-medium text-gray-200">{title}</div>
114
+ <div
115
+ className={`absolute right-[-8px] top-[-8px] flex h-4 w-4 items-center justify-center rounded-full ${color.badgeColor} p-2.5 text-xs`}
116
+ >
117
+ {count}
118
+ </div>
119
+ </Button>
120
+ )
121
+ }
122
+
123
+ export default function AppDrawer({ metrics, title, description }: { metrics: DashboardMetrics | null, title: string, description: string }) {
124
+ const router = useRouter()
125
+ const items: AppDrawerItem[] = [
126
+ {
127
+ title: 'Failed Runs',
128
+ icon: <XCircle className="h-4 w-4" />,
129
+ color: 'orange',
130
+ count: metrics?.failedRecentRunsCount ?? 0,
131
+ onClick: () => {
132
+ router.push('/test-runs?filter=recentFailed')
133
+ },
134
+ },
135
+ {
136
+ title: 'Failing Tests',
137
+ icon: <AlertCircle className="h-4 w-4" />,
138
+ color: 'rose',
139
+ count: metrics?.repeatedlyFailingTestsCount ?? 0,
140
+ onClick: () => {
141
+ router.push('/reports/test-cases?filter=repeatedlyFailing')
142
+ },
143
+ },
144
+ {
145
+ title: 'Flaky Tests',
146
+ icon: <AlertTriangle className="h-4 w-4" />,
147
+ color: 'yellow',
148
+ count: metrics?.flakyTestsCount ?? 0,
149
+ onClick: () => {
150
+ router.push('/reports/test-cases?filter=flaky')
151
+ },
152
+ },
153
+ {
154
+ title: 'Unexecuted Suites',
155
+ icon: <Clock className="h-4 w-4" />,
156
+ color: 'blue',
157
+ count: metrics?.suitesNotExecutedRecentlyCount ?? 0,
158
+ onClick: () => {
159
+ router.push('/reports/test-suites?filter=notExecutedRecently')
160
+ },
161
+ },
162
+ ]
163
+
164
+ return (
165
+ <Card id="container" className="w-fit border-gray-600/10 bg-gray-600/10">
166
+ <CardHeader id="header">
167
+ <CardTitle className="text-primary">{title}</CardTitle>
168
+ <CardDescription>{description}</CardDescription>
169
+ </CardHeader>
170
+ <CardContent id="content">
171
+ <div className="grid grid-cols-2 gap-4 gap-y-6">
172
+ {items.map(item => (
173
+ <AppDrawerItem
174
+ key={item.title}
175
+ title={item.title}
176
+ icon={item.icon}
177
+ colorKey={item.color}
178
+ count={item.count}
179
+ onClick={item.onClick}
180
+ isActive={item.count > 0 ? true : false}
181
+ />
182
+ ))}
183
+ </div>
184
+ </CardContent>
185
+ </Card>
186
+ )
187
+ }
@@ -0,0 +1,13 @@
1
+ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
2
+
3
+ export const DataCardGrid = ({ children }: { children: React.ReactNode }) => {
4
+ return <Card className="border-gray-600/10 bg-gray-600/10 w-fit">
5
+ <CardHeader>
6
+ <CardTitle className="text-primary">States</CardTitle>
7
+ <CardDescription>Overview of entity states</CardDescription>
8
+ </CardHeader>
9
+ <CardContent className="grid grid-cols-2 gap-4">
10
+ {children}
11
+ </CardContent>
12
+ </Card>
13
+ }
@@ -0,0 +1,27 @@
1
+ 'use client'
2
+ import { Button } from "@/components/ui/button";
3
+ import { Card, CardContent, CardTitle, CardHeader } from "@/components/ui/card";
4
+ import { ExternalLink } from "lucide-react";
5
+ import { useRouter } from "next/navigation";
6
+ export default function DataCard({ title, value, link }: { title: string, value: number, link: string }) {
7
+ const router = useRouter()
8
+ return (
9
+ <Card className="h-fit border-gray-600/10 bg-gray-600/10 min-w-40">
10
+ <CardHeader className="flex items-center justify-between flex-row p-2">
11
+ <CardTitle className={`text-xs font-normal ${value > 0 ? 'text-primary' : 'text-gray-400'}`}>{title}</CardTitle>
12
+ <div className="flex items-center gap-2">
13
+ <Button variant="outline" className="text-primary hover:text-primary/80 px-2 py-1 bg-inherit border-gray-600/15 hover:bg-emerald-400/10"
14
+ disabled={value === 0}
15
+ onClick={() => router.push(link)}
16
+ size="sm"
17
+ >
18
+ <ExternalLink className="h-4 w-4" />
19
+ </Button>
20
+ </div>
21
+ </CardHeader>
22
+ <CardContent className="h-full px-2 py-1">
23
+ <div className={`text-2xl h-full flex items-center font-bold ${value > 0 ? 'text-primary' : 'text-gray-400'}`}>{value}</div>
24
+ </CardContent>
25
+ </Card>
26
+ )
27
+ }
@@ -0,0 +1,57 @@
1
+ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@/components/ui/card'
2
+ import { ChartConfig } from '@/components/ui/chart'
3
+ import FeatureChart from '../(base)/reports/feature-chart'
4
+ import { TestSuiteExecutionData } from '@/actions/dashboard/dashboard-actions'
5
+
6
+ const colorMap = {
7
+ passed: 'oklch(59.6% 0.145 163.225)',
8
+ failed: 'oklch(59.2% 0.249 0.584)',
9
+ cancelled: 'oklch(55.4% 0.046 257.417)',
10
+ unknown: 'oklch(79.5% 0.184 86.047)',
11
+ }
12
+
13
+ const resultByTestSuiteBarChartConfig = {
14
+ feature: {
15
+ label: 'Test Suite',
16
+ },
17
+ passed: {
18
+ label: 'Passed',
19
+ color: colorMap.passed,
20
+ },
21
+ failed: {
22
+ label: 'Failed',
23
+ color: colorMap.failed,
24
+ },
25
+ cancelled: {
26
+ label: 'Cancelled',
27
+ color: colorMap.cancelled,
28
+ },
29
+ unknown: {
30
+ label: 'Unknown',
31
+ color: colorMap.unknown,
32
+ },
33
+ } satisfies ChartConfig
34
+
35
+ interface ExecutionHealthPanelProps {
36
+ featureData: TestSuiteExecutionData
37
+ }
38
+
39
+ export const ExecutionHealthPanel = ({ featureData }: ExecutionHealthPanelProps) => {
40
+ return (
41
+ <Card className="border-gray-600/10 bg-gray-600/10 w-full h-full">
42
+ <CardHeader>
43
+ <CardTitle className="text-primary">Execution Health</CardTitle>
44
+ <CardDescription>Pass/fail rates by test suite across last 10 test runs</CardDescription>
45
+ </CardHeader>
46
+ <CardContent>
47
+ {featureData.length === 0 ? (
48
+ <div className="flex h-[300px] items-center justify-center text-sm text-muted-foreground">
49
+ No execution data available. Run some test suites to see execution health metrics.
50
+ </div>
51
+ ) : (
52
+ <FeatureChart config={resultByTestSuiteBarChartConfig} data={featureData} />
53
+ )}
54
+ </CardContent>
55
+ </Card>
56
+ )
57
+ }