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,262 @@
1
+ import { Button } from '@/components/ui/button'
2
+ import {
3
+ Dialog,
4
+ DialogClose,
5
+ DialogContent,
6
+ DialogDescription,
7
+ DialogFooter,
8
+ DialogHeader,
9
+ DialogTitle,
10
+ DialogTrigger,
11
+ } from '@/components/ui/dialog'
12
+ import { Input } from '@/components/ui/input'
13
+ import { Label } from '@/components/ui/label'
14
+ import { NodeData } from '@/constants/form-opts/diagram/node-form'
15
+ import { Locator, LocatorGroup, StepParameterType, TemplateStep, TemplateStepIcon } from '@prisma/client'
16
+ import { TemplateStepParameter } from '@prisma/client'
17
+ import React, { useState, useEffect, useRef, useCallback } from 'react'
18
+ import TemplateStepCombobox from './template-step-combobox'
19
+ import { type TemplateStepWithGroup } from '@/types/diagram/template-step'
20
+ import DynamicFormFields, { DynamicFormFieldsRef } from './dynamic-parameters'
21
+ import { generateGherkinStep } from '@/lib/transformers/gherkin-converter'
22
+ import { z } from 'zod'
23
+ import ErrorMessage from '@/components/form/error-message'
24
+ import { format } from 'date-fns'
25
+
26
+ const errorSchema = z.object({
27
+ label: z.string().min(3, { message: 'Label must be at least 3 characters' }),
28
+ templateStepId: z.string().min(1, { message: 'Template step is required' }),
29
+ })
30
+
31
+ const NodeForm = ({
32
+ onSubmitAction,
33
+ initialValues,
34
+ templateSteps,
35
+ templateStepParams,
36
+ showAddNodeDialog,
37
+ locators,
38
+ locatorGroups,
39
+ setShowAddNodeDialog,
40
+ defaultValueInput = false,
41
+ }: {
42
+ onSubmitAction: (values: NodeData) => void
43
+ initialValues: NodeData
44
+ templateSteps: TemplateStep[]
45
+ templateStepParams: TemplateStepParameter[]
46
+ showAddNodeDialog: boolean
47
+ locators: Locator[]
48
+ locatorGroups: LocatorGroup[]
49
+ setShowAddNodeDialog: (show: boolean) => void
50
+ defaultValueInput?: boolean
51
+ }) => {
52
+ const dynamicFormRef = useRef<DynamicFormFieldsRef>(null)
53
+ const [selectedTemplateId, setSelectedTemplateId] = useState(initialValues.templateStepId)
54
+ // states for dynamic form fields
55
+ const [selectedTemplateStep, setSelectedTemplateStep] = useState<TemplateStep | null>(
56
+ templateSteps.find(step => step.id === initialValues.templateStepId) ?? null,
57
+ )
58
+ const [selectedTemplateStepParams, setSelectedTemplateStepParams] = useState<TemplateStepParameter[]>(
59
+ templateStepParams.filter(param => param.templateStepId === initialValues.templateStepId) ?? [],
60
+ )
61
+ const [parameters, setParameters] = useState<
62
+ {
63
+ name: string
64
+ value: string
65
+ type: StepParameterType
66
+ order: number
67
+ }[]
68
+ >(initialValues.parameters ?? [])
69
+ const [gherkinStep, setGherkinStep] = useState<string>(initialValues.gherkinStep ?? '')
70
+ const [errors, setErrors] = useState<z.inferFlattenedErrors<typeof errorSchema>['fieldErrors']>({})
71
+ // Synchronize state with initialValues when they change (defer setState to avoid sync setState in effect)
72
+ useEffect(() => {
73
+ queueMicrotask(() => {
74
+ setSelectedTemplateId(initialValues.templateStepId)
75
+ const step = templateSteps.find(s => s.id === initialValues.templateStepId) ?? null
76
+ setSelectedTemplateStep(step)
77
+ setSelectedTemplateStepParams(
78
+ templateStepParams.filter(param => param.templateStepId === initialValues.templateStepId),
79
+ )
80
+ setParameters(initialValues.parameters ?? [])
81
+ setGherkinStep(initialValues.gherkinStep ?? '')
82
+ })
83
+ }, [
84
+ initialValues.templateStepId,
85
+ initialValues.parameters,
86
+ initialValues.gherkinStep,
87
+ templateSteps,
88
+ templateStepParams,
89
+ ])
90
+
91
+ const handleTemplateStepChange = useCallback(
92
+ (value: string) => {
93
+ setErrors(prev => ({
94
+ ...prev,
95
+ templateStepId: value ? undefined : ['Template step is required'],
96
+ }))
97
+ setSelectedTemplateId(value)
98
+ const step = templateSteps.find(s => s.id === value)
99
+ if (step) {
100
+ setSelectedTemplateStep(step)
101
+ const newParams = templateStepParams.filter(param => param.templateStepId === step.id)
102
+ setSelectedTemplateStepParams(newParams)
103
+ const initialParamsForStep = newParams.map(param => {
104
+ let defaultValue = ''
105
+ switch (param.type) {
106
+ case 'NUMBER':
107
+ defaultValue = '0'
108
+ break
109
+ case 'STRING':
110
+ defaultValue = ''
111
+ break
112
+ case 'LOCATOR':
113
+ defaultValue = ''
114
+ break
115
+ case 'BOOLEAN':
116
+ defaultValue = 'false'
117
+ break
118
+ case 'DATE':
119
+ defaultValue = format(new Date(), 'PPP')
120
+ break
121
+ }
122
+ return {
123
+ name: param.name,
124
+ value: defaultValue,
125
+ type: param.type,
126
+ order: param.order,
127
+ }
128
+ })
129
+ setParameters(initialParamsForStep)
130
+ if (step.signature) {
131
+ const gherkin = generateGherkinStep(step.type, step.signature, initialParamsForStep)
132
+ setGherkinStep(gherkin)
133
+ }
134
+ }
135
+ },
136
+ [
137
+ templateSteps,
138
+ templateStepParams,
139
+ ],
140
+ )
141
+
142
+ const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => {
143
+ e.preventDefault()
144
+
145
+ const isDynamicFormValid = dynamicFormRef.current?.validate()
146
+
147
+ const formData = new FormData(e.currentTarget)
148
+ const formValues = Object.fromEntries(formData.entries())
149
+
150
+ const dataToValidate = {
151
+ label: formValues.label,
152
+ templateStepId: selectedTemplateId,
153
+ }
154
+
155
+ const parsed = errorSchema.safeParse(dataToValidate)
156
+
157
+ if (!parsed.success || !isDynamicFormValid) {
158
+ if (!parsed.success) {
159
+ setErrors(parsed.error.flatten().fieldErrors)
160
+ }
161
+ return
162
+ }
163
+
164
+ setErrors({})
165
+ const nodeData: NodeData = {
166
+ ...formValues,
167
+ parameters: parameters,
168
+ label: formValues.label as string,
169
+ gherkinStep: gherkinStep,
170
+ templateStepId: selectedTemplateId as string,
171
+ }
172
+ onSubmitAction(nodeData)
173
+ }
174
+
175
+ return (
176
+ <Dialog open={showAddNodeDialog} onOpenChange={setShowAddNodeDialog}>
177
+ <DialogTrigger asChild>
178
+ <Button type="button" onClick={e => e.preventDefault()}>
179
+ Add Node
180
+ </Button>
181
+ </DialogTrigger>
182
+ <DialogContent>
183
+ <form onSubmit={handleSubmit}>
184
+ <DialogHeader>
185
+ <DialogTitle>Add Node</DialogTitle>
186
+ </DialogHeader>
187
+ <DialogDescription>Insert a new node to the diagram</DialogDescription>
188
+ <div className="my-4">
189
+ <div className="mb-4 flex flex-col gap-2">
190
+ <Label htmlFor="label">Label</Label>
191
+ <Input
192
+ id="label"
193
+ name="label"
194
+ defaultValue={initialValues.label}
195
+ onChange={e => {
196
+ setErrors(prev => ({
197
+ ...prev,
198
+ label: e.target.value ? undefined : ['Label is required'],
199
+ }))
200
+ }}
201
+ />
202
+ <ErrorMessage message={errors.label?.[0] ?? ''} visible={!!errors.label} />
203
+ </div>
204
+ <div className="mb-4 flex flex-col gap-2">
205
+ <Label htmlFor="templateStepId">Template Step</Label>
206
+ <TemplateStepCombobox
207
+ id="templateStepId"
208
+ value={selectedTemplateId}
209
+ onValueChange={handleTemplateStepChange}
210
+ templateSteps={templateSteps as TemplateStepWithGroup[]}
211
+ placeholder="Select a template step"
212
+ />
213
+ <input type="hidden" name="templateStepId" value={selectedTemplateId} />
214
+ <ErrorMessage message={errors.templateStepId?.[0] ?? ''} visible={!!errors.templateStepId} />
215
+ </div>
216
+ <div className="mb-4 flex flex-col gap-2">
217
+ <DynamicFormFields
218
+ ref={dynamicFormRef}
219
+ templateStepParams={selectedTemplateStepParams}
220
+ locators={locators}
221
+ locatorGroups={locatorGroups}
222
+ initialParameterValues={initialValues.parameters}
223
+ onChange={values => {
224
+ setParameters([...values])
225
+ // Generate gherkin step directly when parameters change
226
+ if (selectedTemplateStep && selectedTemplateStep.signature) {
227
+ const gherkin = generateGherkinStep(
228
+ selectedTemplateStep.type,
229
+ selectedTemplateStep.signature,
230
+ values,
231
+ )
232
+ setGherkinStep(gherkin)
233
+ }
234
+ }}
235
+ defaultValueInput={defaultValueInput}
236
+ />
237
+ </div>
238
+ {selectedTemplateStep && (
239
+ <div className="mb-4 flex flex-col gap-2">
240
+ <Label htmlFor="gherkinStep">Gherkin Step</Label>
241
+ <Input disabled id="gherkinStep" name="gherkinStep" value={gherkinStep} />
242
+ </div>
243
+ )}
244
+ <input
245
+ type="hidden"
246
+ name="icon"
247
+ value={selectedTemplateStep?.icon ? selectedTemplateStep.icon : TemplateStepIcon.MOUSE}
248
+ />
249
+ </div>
250
+ <DialogFooter>
251
+ <DialogClose asChild>
252
+ <Button variant="outline">Cancel</Button>
253
+ </DialogClose>
254
+ <Button type="submit">Save</Button>
255
+ </DialogFooter>
256
+ </form>
257
+ </DialogContent>
258
+ </Dialog>
259
+ )
260
+ }
261
+
262
+ export default NodeForm
@@ -0,0 +1,57 @@
1
+ import { memo } from 'react'
2
+
3
+ import { NodeProps, Handle, Position } from '@xyflow/react'
4
+
5
+ import { BaseNode } from '@/components/base-node'
6
+ import {
7
+ NodeHeader,
8
+ NodeHeaderTitle,
9
+ NodeHeaderActions,
10
+ NodeHeaderIcon,
11
+ NodeHeaderDeleteAction,
12
+ } from '@/components/node-header'
13
+ import { NodeHeaderEditAction } from './edit-header-option'
14
+ import { TemplateStepIcon } from '@prisma/client'
15
+ import { KeyToIconTransformer } from '@/lib/transformers/key-to-icon-transformer'
16
+ import { cn } from '@/lib/utils'
17
+
18
+ interface OptionsHeaderNodeData {
19
+ label: string
20
+ gherkinStep: string
21
+ isFirstNode?: boolean
22
+ icon?: TemplateStepIcon
23
+ isMissingParams?: boolean
24
+ }
25
+
26
+ interface OptionsHeaderNodeProps extends NodeProps {
27
+ onEdit: (nodeId: string) => void
28
+ }
29
+
30
+ const OptionsHeaderNode = memo(({ selected, data, onEdit }: OptionsHeaderNodeProps) => {
31
+ OptionsHeaderNode.displayName = 'OptionsHeaderNode'
32
+
33
+ const { label, gherkinStep, isFirstNode, icon, isMissingParams } = data as unknown as OptionsHeaderNodeData
34
+
35
+ return (
36
+ <BaseNode selected={selected} className={cn('w-52 border-none px-3 py-2', isMissingParams && 'bg-red-700')}>
37
+ {!isFirstNode && <Handle type="target" position={Position.Left} />}
38
+ <NodeHeader className="-mx-3 -mt-2 border-b">
39
+ <NodeHeaderIcon>{KeyToIconTransformer(icon as TemplateStepIcon)}</NodeHeaderIcon>
40
+ <NodeHeaderTitle>{label}</NodeHeaderTitle>
41
+ <NodeHeaderActions>
42
+ <NodeHeaderEditAction
43
+ label="Edit"
44
+ onClick={nodeId => {
45
+ onEdit(nodeId)
46
+ }}
47
+ />
48
+ <NodeHeaderDeleteAction />
49
+ </NodeHeaderActions>
50
+ </NodeHeader>
51
+ <div className="mt-2">{gherkinStep}</div>
52
+ <Handle type="source" position={Position.Right} />
53
+ </BaseNode>
54
+ )
55
+ })
56
+
57
+ export default OptionsHeaderNode
@@ -0,0 +1,155 @@
1
+ 'use client'
2
+
3
+ import * as React from 'react'
4
+ import { ChevronDown } from 'lucide-react'
5
+ import {
6
+ Command,
7
+ CommandEmpty,
8
+ CommandGroup,
9
+ CommandInput,
10
+ CommandItem,
11
+ CommandList,
12
+ } from '@/components/ui/command'
13
+ import { cn } from '@/lib/utils'
14
+ import {
15
+ capitalizeGroupName,
16
+ type TemplateStepWithGroup,
17
+ } from '@/types/diagram/template-step'
18
+
19
+ const GROUP_KEY_OTHER = 'Other'
20
+
21
+ function groupStepsByGroupName(
22
+ steps: TemplateStepWithGroup[],
23
+ ): Map<string, TemplateStepWithGroup[]> {
24
+ const map = new Map<string, TemplateStepWithGroup[]>()
25
+ for (const step of steps) {
26
+ const key = step.templateStepGroup?.name ?? GROUP_KEY_OTHER
27
+ const list = map.get(key) ?? []
28
+ list.push(step)
29
+ map.set(key, list)
30
+ }
31
+ return map
32
+ }
33
+
34
+ export type TemplateStepComboboxProps = {
35
+ value: string
36
+ onValueChange: (value: string) => void
37
+ templateSteps: TemplateStepWithGroup[]
38
+ placeholder?: string
39
+ disabled?: boolean
40
+ id?: string
41
+ }
42
+
43
+ const TemplateStepCombobox = ({
44
+ value,
45
+ onValueChange,
46
+ templateSteps,
47
+ placeholder = 'Select a template step',
48
+ disabled = false,
49
+ id,
50
+ }: TemplateStepComboboxProps) => {
51
+ const [open, setOpen] = React.useState(false)
52
+ const containerRef = React.useRef<HTMLDivElement>(null)
53
+ const searchInputRef = React.useRef<HTMLInputElement>(null)
54
+
55
+ const selectedStep = React.useMemo(
56
+ () => templateSteps.find(s => s.id === value) ?? null,
57
+ [templateSteps, value],
58
+ )
59
+
60
+ const groups = React.useMemo(() => {
61
+ const map = groupStepsByGroupName(templateSteps)
62
+ return Array.from(map.entries()).sort(([a], [b]) =>
63
+ a.localeCompare(b, undefined, { sensitivity: 'base' }),
64
+ )
65
+ }, [templateSteps])
66
+
67
+ const handleSelect = React.useCallback(
68
+ (stepId: string) => {
69
+ onValueChange(stepId)
70
+ setOpen(false)
71
+ },
72
+ [onValueChange],
73
+ )
74
+
75
+ React.useEffect(() => {
76
+ if (!open) return
77
+ // Focus search input when dropdown opens so user can type immediately
78
+ const focusSearch = () => searchInputRef.current?.focus()
79
+ const id = requestAnimationFrame(focusSearch)
80
+ return () => cancelAnimationFrame(id)
81
+ }, [open])
82
+
83
+ React.useEffect(() => {
84
+ if (!open) return
85
+ const handleClickOutside = (e: MouseEvent) => {
86
+ if (containerRef.current && !containerRef.current.contains(e.target as Node)) {
87
+ setOpen(false)
88
+ }
89
+ }
90
+ const handleEscape = (e: KeyboardEvent) => {
91
+ if (e.key === 'Escape') setOpen(false)
92
+ }
93
+ document.addEventListener('mousedown', handleClickOutside)
94
+ document.addEventListener('keydown', handleEscape)
95
+ return () => {
96
+ document.removeEventListener('mousedown', handleClickOutside)
97
+ document.removeEventListener('keydown', handleEscape)
98
+ }
99
+ }, [open])
100
+
101
+ return (
102
+ <div ref={containerRef} className="relative w-full">
103
+ <button
104
+ type="button"
105
+ id={id}
106
+ role="combobox"
107
+ aria-expanded={open}
108
+ aria-controls={open ? `${id}-listbox` : undefined}
109
+ aria-haspopup="listbox"
110
+ aria-label={placeholder}
111
+ disabled={disabled}
112
+ onClick={() => setOpen(prev => !prev)}
113
+ className={cn(
114
+ 'flex h-9 w-full items-center justify-between whitespace-nowrap rounded-md border border-input bg-transparent px-3 py-2 text-sm shadow-sm ring-offset-background',
115
+ 'focus:outline-none focus:ring-1 focus:ring-ring',
116
+ 'disabled:cursor-not-allowed disabled:opacity-50',
117
+ 'data-[placeholder]:text-muted-foreground [&>span]:line-clamp-1',
118
+ )}
119
+ >
120
+ <span className={cn(!selectedStep && 'text-muted-foreground')}>
121
+ {selectedStep ? selectedStep.name : placeholder}
122
+ </span>
123
+ <ChevronDown className="h-4 w-4 shrink-0 opacity-50" />
124
+ </button>
125
+ {open && (
126
+ <div id={`${id}-listbox`} role="listbox" className="absolute left-0 right-0 top-full z-50 mt-1">
127
+ <Command shouldFilter className="rounded-md border border-input bg-popover shadow-md">
128
+ <CommandInput ref={searchInputRef} placeholder="Search template steps…" />
129
+ <CommandList>
130
+ <CommandEmpty>No step found.</CommandEmpty>
131
+ {groups.map(([groupKey, steps]) => (
132
+ <CommandGroup
133
+ key={groupKey}
134
+ heading={capitalizeGroupName(groupKey)}
135
+ >
136
+ {steps.map(step => (
137
+ <CommandItem
138
+ key={step.id}
139
+ value={step.name}
140
+ onSelect={() => handleSelect(step.id)}
141
+ >
142
+ {step.name}
143
+ </CommandItem>
144
+ ))}
145
+ </CommandGroup>
146
+ ))}
147
+ </CommandList>
148
+ </Command>
149
+ </div>
150
+ )}
151
+ </div>
152
+ )
153
+ }
154
+
155
+ export default TemplateStepCombobox
@@ -0,0 +1,7 @@
1
+ import React from 'react'
2
+
3
+ const ErrorMessage = ({ message, visible }: { message: string; visible: boolean }) => {
4
+ return <div className={`text-sm text-pink-600 ${visible ? 'visible' : 'invisible'}`}>{message}</div>
5
+ }
6
+
7
+ export default ErrorMessage