create-appraisejs 0.3.1-alpha.0 → 0.3.1-alpha.1

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 (503) hide show
  1. package/README.md +17 -17
  2. package/dist/cli.e2e.test.d.ts +2 -2
  3. package/dist/cli.e2e.test.js +74 -72
  4. package/dist/config.d.ts.map +1 -1
  5. package/dist/config.js.map +1 -1
  6. package/dist/config.test.d.ts +2 -2
  7. package/dist/config.test.js +64 -64
  8. package/dist/copy-template.test.d.ts +2 -2
  9. package/dist/copy-template.test.js +70 -70
  10. package/dist/download-repo.test.d.ts +2 -2
  11. package/dist/download-repo.test.js +15 -13
  12. package/dist/install.test.d.ts +2 -2
  13. package/dist/install.test.js +115 -114
  14. package/dist/prepare-template-utils.d.ts.map +1 -1
  15. package/dist/prepare-template-utils.js.map +1 -1
  16. package/dist/prompts.d.ts.map +1 -1
  17. package/dist/prompts.js +2 -2
  18. package/dist/prompts.js.map +1 -1
  19. package/dist/prompts.test.d.ts +2 -2
  20. package/dist/prompts.test.js +54 -54
  21. package/dist/sync-templates-utils.d.ts.map +1 -1
  22. package/dist/sync-templates-utils.js +1 -5
  23. package/dist/sync-templates-utils.js.map +1 -1
  24. package/package.json +1 -1
  25. package/templates/blank/.editorconfig +11 -0
  26. package/templates/blank/.gitattributes +21 -0
  27. package/templates/blank/.gitconfig.appraise +14 -0
  28. package/templates/blank/.prettierrc +13 -0
  29. package/templates/blank/.vscode/settings.json +2 -0
  30. package/templates/blank/README.md +12 -12
  31. package/templates/blank/components.json +24 -24
  32. package/templates/blank/cucumber.mjs +16 -16
  33. package/templates/blank/e2e/README.md +56 -0
  34. package/templates/blank/e2e/apply-migrations.mjs +76 -0
  35. package/templates/blank/e2e/appraise-smoke.spec.ts +78 -0
  36. package/templates/blank/e2e/authoring.spec.ts +53 -0
  37. package/templates/blank/e2e/crud-configuration.spec.ts +138 -0
  38. package/templates/blank/e2e/crud-tests.spec.ts +76 -0
  39. package/templates/blank/e2e/helpers/forms.ts +186 -0
  40. package/templates/blank/e2e/helpers/navigation.ts +27 -0
  41. package/templates/blank/e2e/helpers/table.ts +45 -0
  42. package/templates/blank/e2e/helpers/test-data.ts +549 -0
  43. package/templates/blank/e2e/helpers/ui.ts +41 -0
  44. package/templates/blank/e2e/navigation.spec.ts +101 -0
  45. package/templates/blank/e2e/runs-and-reports.spec.ts +82 -0
  46. package/templates/blank/e2e/settings-sync.spec.ts +39 -0
  47. package/templates/blank/e2e/start-server.mjs +61 -0
  48. package/templates/blank/eslint.config.mjs +4 -4
  49. package/templates/blank/gitignore +0 -3
  50. package/templates/blank/next-env.d.ts +1 -1
  51. package/templates/blank/package-lock.json +58 -102
  52. package/templates/blank/package.json +10 -9
  53. package/templates/blank/packages/cucumber-runtime/package.json +13 -13
  54. package/templates/blank/packages/cucumber-runtime/src/cache.util.ts +93 -93
  55. package/templates/blank/packages/cucumber-runtime/src/cli.ts +77 -68
  56. package/templates/blank/packages/cucumber-runtime/src/environment.util.ts +21 -21
  57. package/templates/blank/packages/cucumber-runtime/src/executor.ts +32 -32
  58. package/templates/blank/packages/cucumber-runtime/src/index.ts +17 -17
  59. package/templates/blank/packages/cucumber-runtime/src/locator.util.ts +234 -234
  60. package/templates/blank/packages/cucumber-runtime/src/parameter-types.ts +7 -7
  61. package/templates/blank/packages/cucumber-runtime/src/random-data.util.ts +35 -35
  62. package/templates/blank/packages/cucumber-runtime/src/types.ts +13 -13
  63. package/templates/blank/packages/cucumber-runtime/src/world.ts +44 -44
  64. package/templates/blank/packages/cucumber-runtime/tsconfig.json +11 -11
  65. package/templates/blank/packages/locator-picker-companion/dist/cli.js +1 -1
  66. package/templates/blank/packages/locator-picker-companion/dist/index.d.ts +3 -3
  67. package/templates/blank/packages/locator-picker-companion/dist/index.js +3 -3
  68. package/templates/blank/packages/locator-picker-companion/dist/injected-picker-script.js +13 -3
  69. package/templates/blank/packages/locator-picker-companion/dist/selector-generator.d.ts +6 -3
  70. package/templates/blank/packages/locator-picker-companion/dist/selector-generator.js +233 -229
  71. package/templates/blank/packages/locator-picker-companion/src/cli.ts +1 -7
  72. package/templates/blank/packages/locator-picker-companion/src/injected-picker-script.ts +26 -5
  73. package/templates/blank/packages/locator-picker-companion/src/launcher.ts +1 -3
  74. package/templates/blank/packages/locator-picker-companion/src/session-file.ts +4 -15
  75. package/templates/blank/packages/locator-picker-companion/src/types.ts +1 -8
  76. package/templates/blank/playwright.config.ts +46 -0
  77. package/templates/blank/postcss.config.mjs +8 -8
  78. package/templates/blank/prisma/dev.db +0 -0
  79. package/templates/blank/scripts/configure-git-line-endings.mjs +91 -0
  80. package/templates/blank/scripts/install-template-step.ts +5 -1
  81. package/templates/blank/scripts/lib/step-matcher.test.ts +3 -3
  82. package/templates/blank/scripts/lib/step-matcher.ts +5 -1
  83. package/templates/blank/scripts/lib/template-step-installer.test.ts +4 -2
  84. package/templates/blank/scripts/lib/template-step-installer.ts +16 -4
  85. package/templates/blank/scripts/lib/template-step-registry.test.ts +1 -1
  86. package/templates/blank/scripts/lib/template-step-registry.ts +4 -1
  87. package/templates/blank/scripts/run-fallow-commit.mjs +4 -6
  88. package/templates/blank/scripts/sync-all.ts +7 -6
  89. package/templates/blank/scripts/sync-appraise-base-template.ts +84 -77
  90. package/templates/blank/scripts/sync-environments.ts +47 -44
  91. package/templates/blank/scripts/sync-locator-groups.ts +67 -76
  92. package/templates/blank/scripts/sync-locators.ts +50 -53
  93. package/templates/blank/scripts/sync-modules.ts +42 -42
  94. package/templates/blank/scripts/sync-tags.ts +51 -51
  95. package/templates/blank/scripts/sync-template-step-groups.ts +34 -32
  96. package/templates/blank/scripts/sync-template-steps.ts +97 -97
  97. package/templates/blank/scripts/sync-test-cases.ts +65 -74
  98. package/templates/blank/scripts/sync-test-suites.ts +85 -80
  99. package/templates/blank/src/actions/server-action-boundary.test.ts +1 -7
  100. package/templates/blank/src/actions/settings/sync-actions.ts +3 -1
  101. package/templates/blank/src/actions/tags/tag-actions.ts +1 -7
  102. package/templates/blank/src/actions/template-test-case/template-test-case-actions.ts +3 -1
  103. package/templates/blank/src/app/(base)/environments/create/page.tsx +28 -28
  104. package/templates/blank/src/app/(base)/environments/environment-form.tsx +3 -2
  105. package/templates/blank/src/app/(base)/environments/environment-helpers.ts +4 -1
  106. package/templates/blank/src/app/(base)/layout.tsx +10 -10
  107. package/templates/blank/src/app/(base)/locator-groups/locator-group-form.tsx +5 -3
  108. package/templates/blank/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -77
  109. package/templates/blank/src/app/(base)/locator-groups/locator-group-table.tsx +28 -28
  110. package/templates/blank/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -46
  111. package/templates/blank/src/app/(base)/locator-groups/page.tsx +9 -19
  112. package/templates/blank/src/app/(base)/locators/create/create-locator-workspace-helpers.test.ts +3 -6
  113. package/templates/blank/src/app/(base)/locators/create/create-locator-workspace-response.ts +22 -5
  114. package/templates/blank/src/app/(base)/locators/create/create-locator-workspace-state.ts +5 -1
  115. package/templates/blank/src/app/(base)/locators/create/use-locator-workspace.ts +7 -2
  116. package/templates/blank/src/app/(base)/locators/locator-file-sync.ts +1 -5
  117. package/templates/blank/src/app/(base)/locators/locator-helpers.ts +2 -12
  118. package/templates/blank/src/app/(base)/locators/locator-table-columns.tsx +59 -60
  119. package/templates/blank/src/app/(base)/modules/module-form.tsx +3 -1
  120. package/templates/blank/src/app/(base)/reports/overview-chart.tsx +49 -49
  121. package/templates/blank/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +114 -115
  122. package/templates/blank/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -27
  123. package/templates/blank/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +80 -79
  124. package/templates/blank/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +29 -27
  125. package/templates/blank/src/app/(base)/settings/settings-sync-panel-helpers.test.tsx +4 -6
  126. package/templates/blank/src/app/(base)/settings/settings-sync-panel.tsx +22 -23
  127. package/templates/blank/src/app/(base)/tags/tag-form.test.tsx +2 -10
  128. package/templates/blank/src/app/(base)/tags/tag-form.tsx +5 -3
  129. package/templates/blank/src/app/(base)/tags/tag-table-columns.tsx +63 -63
  130. package/templates/blank/src/app/(base)/tags/tag-table.tsx +29 -29
  131. package/templates/blank/src/app/(base)/template-step-groups/create/page.tsx +28 -28
  132. package/templates/blank/src/app/(base)/template-step-groups/template-step-group-form.tsx +3 -1
  133. package/templates/blank/src/app/(base)/template-step-groups/template-step-group-helpers.ts +4 -1
  134. package/templates/blank/src/app/(base)/template-steps/paramChip.tsx +6 -10
  135. package/templates/blank/src/app/(base)/template-steps/template-step-form-helpers.ts +4 -2
  136. package/templates/blank/src/app/(base)/template-steps/template-step-helpers.test.ts +19 -1
  137. package/templates/blank/src/app/(base)/template-steps/template-step-helpers.ts +1 -5
  138. package/templates/blank/src/app/(base)/template-steps/template-step-row-guards.ts +1 -5
  139. package/templates/blank/src/app/(base)/template-steps/template-step-types.ts +1 -5
  140. package/templates/blank/src/app/(base)/template-test-cases/create-template-test-case-page-data.test.ts +4 -1
  141. package/templates/blank/src/app/(base)/template-test-cases/create-template-test-case-page-data.ts +1 -2
  142. package/templates/blank/src/app/(base)/template-test-cases/modify/[id]/page.tsx +1 -0
  143. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-flow.test.tsx +0 -1
  144. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-form.tsx +3 -1
  145. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -76
  146. package/templates/blank/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -32
  147. package/templates/blank/src/app/(base)/test-cases/create/page.tsx +1 -5
  148. package/templates/blank/src/app/(base)/test-cases/create-from-template/create-from-template-selection-helpers.ts +1 -4
  149. package/templates/blank/src/app/(base)/test-cases/create-from-template/page.test.tsx +2 -2
  150. package/templates/blank/src/app/(base)/test-cases/inline-tag-creation-dialog.tsx +1 -7
  151. package/templates/blank/src/app/(base)/test-cases/inline-test-suite-creation-dialog.tsx +1 -7
  152. package/templates/blank/src/app/(base)/test-cases/modify/[id]/page.tsx +1 -5
  153. package/templates/blank/src/app/(base)/test-cases/test-case-form.tsx +63 -78
  154. package/templates/blank/src/app/(base)/test-cases/test-case-route-helpers.test.ts +2 -7
  155. package/templates/blank/src/app/(base)/test-suites/editable-test-suite-helpers.ts +1 -6
  156. package/templates/blank/src/app/(base)/test-suites/test-suite-form.test.tsx +2 -10
  157. package/templates/blank/src/app/(dashboard-components)/data-card-grid.tsx +10 -10
  158. package/templates/blank/src/app/api/test-runs/[runId]/download/route.ts +1 -4
  159. package/templates/blank/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.test.ts +18 -9
  160. package/templates/blank/src/assets/icons/empty-tube.tsx +23 -23
  161. package/templates/blank/src/assets/icons/tube-plus.tsx +29 -29
  162. package/templates/blank/src/components/data-visualization/info-card.tsx +56 -70
  163. package/templates/blank/src/components/data-visualization/info-grid.tsx +22 -22
  164. package/templates/blank/src/components/diagram/add-node-prompt-node.tsx +1 -1
  165. package/templates/blank/src/components/diagram/dynamic-parameters-helpers.ts +4 -3
  166. package/templates/blank/src/components/diagram/dynamic-parameters.test.tsx +4 -1
  167. package/templates/blank/src/components/diagram/dynamic-parameters.tsx +0 -1
  168. package/templates/blank/src/components/diagram/flow-diagram-block-dialog.test.tsx +26 -0
  169. package/templates/blank/src/components/diagram/flow-diagram-block-dialog.tsx +3 -1
  170. package/templates/blank/src/components/diagram/flow-diagram-connection-guards.ts +24 -6
  171. package/templates/blank/src/components/diagram/flow-diagram-helpers.test.ts +27 -0
  172. package/templates/blank/src/components/diagram/flow-diagram-node-search.tsx +65 -65
  173. package/templates/blank/src/components/diagram/flow-diagram-toolbar.tsx +1 -3
  174. package/templates/blank/src/components/diagram/flow-diagram-view.tsx +1 -7
  175. package/templates/blank/src/components/diagram/flow-node-data-helpers.ts +1 -1
  176. package/templates/blank/src/components/diagram/node-form-fields-content.tsx +9 -2
  177. package/templates/blank/src/components/diagram/node-form-helpers.test.ts +3 -1
  178. package/templates/blank/src/components/diagram/node-form-helpers.ts +1 -4
  179. package/templates/blank/src/components/diagram/node-form-template-step-selection.ts +1 -4
  180. package/templates/blank/src/components/diagram/node-form.test.tsx +25 -0
  181. package/templates/blank/src/components/diagram/node-form.tsx +2 -12
  182. package/templates/blank/src/components/diagram/options-header-node.test.tsx +4 -1
  183. package/templates/blank/src/components/diagram/options-header-node.tsx +140 -130
  184. package/templates/blank/src/components/diagram/use-flow-diagram.ts +1 -2
  185. package/templates/blank/src/components/form/error-message.tsx +7 -7
  186. package/templates/blank/src/components/loading-skeleton/form/button-skeleton.tsx +8 -8
  187. package/templates/blank/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -8
  188. package/templates/blank/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -14
  189. package/templates/blank/src/components/navigation/entity-search-command.tsx +3 -9
  190. package/templates/blank/src/components/navigation/nav-link.tsx +60 -60
  191. package/templates/blank/src/components/test-case/test-case-form-helpers.test.ts +1 -5
  192. package/templates/blank/src/components/test-case/test-case-form-helpers.ts +1 -5
  193. package/templates/blank/src/components/test-case/test-case-picker-helpers.ts +8 -7
  194. package/templates/blank/src/components/test-case/test-case-picker.tsx +2 -6
  195. package/templates/blank/src/components/test-run/log-viewer.test.tsx +25 -0
  196. package/templates/blank/src/components/test-run/log-viewer.tsx +2 -2
  197. package/templates/blank/src/components/test-run/test-run-details-guards.ts +10 -1
  198. package/templates/blank/src/components/test-run/test-run-details.test.tsx +9 -19
  199. package/templates/blank/src/components/test-run/use-log-viewer.ts +2 -8
  200. package/templates/blank/src/components/test-run/use-test-run-header.ts +1 -4
  201. package/templates/blank/src/components/test-suite/test-suite-picker.tsx +1 -1
  202. package/templates/blank/src/components/theme/theme-provider.tsx +7 -1
  203. package/templates/blank/src/components/typography/page-header-subtitle.tsx +7 -7
  204. package/templates/blank/src/components/typography/page-header.tsx +7 -7
  205. package/templates/blank/src/components/ui/chart.tsx +2 -8
  206. package/templates/blank/src/components/ui/command.tsx +2 -5
  207. package/templates/blank/src/components/ui/data-table-column-header.tsx +3 -3
  208. package/templates/blank/src/components/ui/data-table.test.tsx +1 -8
  209. package/templates/blank/src/components/ui/dialog.tsx +1 -4
  210. package/templates/blank/src/components/ui/select.tsx +2 -2
  211. package/templates/blank/src/components/ui/separator.tsx +5 -1
  212. package/templates/blank/src/components/ui/skeleton.tsx +7 -7
  213. package/templates/blank/src/components/ui/table.tsx +1 -1
  214. package/templates/blank/src/components/ui/toaster.tsx +26 -26
  215. package/templates/blank/src/components/ui/tooltip.tsx +1 -1
  216. package/templates/blank/src/constants/form-opts/locator-group-form-opts.ts +1 -1
  217. package/templates/blank/src/lib/locator-group-file-utils.ts +5 -1
  218. package/templates/blank/src/lib/locator-picker/session-manager.ts +1 -2
  219. package/templates/blank/src/lib/metrics/metric-calculator.test.ts +105 -6
  220. package/templates/blank/src/lib/metrics/metric-calculator.ts +39 -40
  221. package/templates/blank/src/lib/sync/sync-executor.ts +1 -4
  222. package/templates/blank/src/lib/sync/sync-pending-counts.test.ts +201 -7
  223. package/templates/blank/src/lib/sync/sync-pending-counts.ts +81 -13
  224. package/templates/blank/src/lib/template-automation-paths.ts +1 -1
  225. package/templates/blank/src/lib/template-sync-utils.d.ts +14 -7
  226. package/templates/blank/src/lib/test-case-utils.ts +6 -6
  227. package/templates/blank/src/lib/test-run/log-formatter.ts +82 -83
  228. package/templates/blank/src/lib/test-run/report-parser.ts +352 -352
  229. package/templates/blank/src/lib/transformers/gherkin-converter.ts +42 -42
  230. package/templates/blank/src/lib/transformers/key-to-icon-transformer.tsx +95 -95
  231. package/templates/blank/src/lib/utils/node-param-validation.ts +81 -81
  232. package/templates/blank/src/lib/utils/template-step-file-generator.ts +5 -3
  233. package/templates/blank/src/lib/utils/template-step-file-manager-intelligent.ts +11 -1
  234. package/templates/blank/src/services/dashboard/dashboard-service.test.ts +28 -1
  235. package/templates/blank/src/services/dashboard/dashboard-service.ts +2 -0
  236. package/templates/blank/src/services/locator/locator-service.test.ts +1 -3
  237. package/templates/blank/src/services/locator/locator-service.ts +3 -1
  238. package/templates/blank/src/services/locator/locator-sync-utils.test.ts +1 -4
  239. package/templates/blank/src/services/module/module-service.ts +1 -4
  240. package/templates/blank/src/services/report/report-service.test.ts +70 -6
  241. package/templates/blank/src/services/report/report-service.ts +52 -9
  242. package/templates/blank/src/services/shared/errors.ts +2 -4
  243. package/templates/blank/src/services/template-step/template-step-service.ts +3 -1
  244. package/templates/blank/src/services/template-step-group/template-step-group-service.ts +3 -1
  245. package/templates/blank/src/services/test-case/test-case-service.ts +0 -5
  246. package/templates/blank/src/services/test-run/test-run-service.test.ts +3 -6
  247. package/templates/blank/src/services/test-run/test-run-service.ts +12 -7
  248. package/templates/blank/src/services/test-suite/test-suite-service.test.ts +1 -5
  249. package/templates/blank/src/services/test-suite/test-suite-service.ts +2 -4
  250. package/templates/blank/src/types/form/actionHandler.ts +1 -5
  251. package/templates/blank/tsconfig.json +8 -8
  252. package/templates/default/packages/locator-picker-companion/dist/cli.d.ts +1 -1
  253. package/templates/default/packages/locator-picker-companion/dist/cli.js +313 -279
  254. package/templates/default/packages/locator-picker-companion/dist/index.d.ts +3 -3
  255. package/templates/default/packages/locator-picker-companion/dist/index.js +3 -3
  256. package/templates/default/packages/locator-picker-companion/dist/injected-picker-script.d.ts +1 -1
  257. package/templates/default/packages/locator-picker-companion/dist/injected-picker-script.js +526 -481
  258. package/templates/default/packages/locator-picker-companion/dist/launcher.d.ts +13 -10
  259. package/templates/default/packages/locator-picker-companion/dist/launcher.js +47 -48
  260. package/templates/default/packages/locator-picker-companion/dist/selector-generator.d.ts +6 -3
  261. package/templates/default/packages/locator-picker-companion/dist/selector-generator.js +233 -229
  262. package/templates/default/packages/locator-picker-companion/dist/session-file.d.ts +30 -22
  263. package/templates/default/packages/locator-picker-companion/dist/session-file.js +116 -104
  264. package/templates/default/packages/locator-picker-companion/dist/types.d.ts +25 -25
  265. package/templates/default/packages/locator-picker-companion/dist/types.js +1 -1
  266. package/templates/starter/.editorconfig +11 -0
  267. package/templates/starter/.gitattributes +21 -0
  268. package/templates/starter/.gitconfig.appraise +14 -0
  269. package/templates/starter/.prettierrc +13 -0
  270. package/templates/starter/.vscode/settings.json +2 -0
  271. package/templates/starter/README.md +12 -12
  272. package/templates/starter/automation/steps/actions/click.step.ts +56 -58
  273. package/templates/starter/automation/steps/actions/hover.step.ts +25 -27
  274. package/templates/starter/automation/steps/actions/input.step.ts +85 -108
  275. package/templates/starter/automation/steps/actions/navigation.step.ts +67 -70
  276. package/templates/starter/automation/steps/actions/random_data.step.ts +149 -143
  277. package/templates/starter/automation/steps/actions/store.step.ts +69 -1
  278. package/templates/starter/automation/steps/actions/wait.step.ts +84 -1
  279. package/templates/starter/automation/steps/validations/active_state_assertion.step.ts +28 -30
  280. package/templates/starter/automation/steps/validations/navigation_assertion.step.ts +20 -22
  281. package/templates/starter/automation/steps/validations/text_assertion.step.ts +67 -107
  282. package/templates/starter/automation/steps/validations/visibility_assertion.step.ts +24 -26
  283. package/templates/starter/components.json +24 -24
  284. package/templates/starter/cucumber.mjs +16 -16
  285. package/templates/starter/e2e/README.md +56 -0
  286. package/templates/starter/e2e/apply-migrations.mjs +76 -0
  287. package/templates/starter/e2e/appraise-smoke.spec.ts +78 -0
  288. package/templates/starter/e2e/authoring.spec.ts +53 -0
  289. package/templates/starter/e2e/crud-configuration.spec.ts +138 -0
  290. package/templates/starter/e2e/crud-tests.spec.ts +76 -0
  291. package/templates/starter/e2e/helpers/forms.ts +186 -0
  292. package/templates/starter/e2e/helpers/navigation.ts +27 -0
  293. package/templates/starter/e2e/helpers/table.ts +45 -0
  294. package/templates/starter/e2e/helpers/test-data.ts +549 -0
  295. package/templates/starter/e2e/helpers/ui.ts +41 -0
  296. package/templates/starter/e2e/navigation.spec.ts +101 -0
  297. package/templates/starter/e2e/runs-and-reports.spec.ts +82 -0
  298. package/templates/starter/e2e/settings-sync.spec.ts +39 -0
  299. package/templates/starter/e2e/start-server.mjs +61 -0
  300. package/templates/starter/eslint.config.mjs +4 -4
  301. package/templates/starter/gitignore +0 -3
  302. package/templates/starter/next-env.d.ts +1 -1
  303. package/templates/starter/package-lock.json +58 -102
  304. package/templates/starter/package.json +10 -9
  305. package/templates/starter/packages/cucumber-runtime/package.json +13 -13
  306. package/templates/starter/packages/cucumber-runtime/src/cache.util.ts +93 -93
  307. package/templates/starter/packages/cucumber-runtime/src/cli.ts +77 -68
  308. package/templates/starter/packages/cucumber-runtime/src/environment.util.ts +21 -21
  309. package/templates/starter/packages/cucumber-runtime/src/executor.ts +32 -32
  310. package/templates/starter/packages/cucumber-runtime/src/index.ts +17 -17
  311. package/templates/starter/packages/cucumber-runtime/src/locator.util.ts +234 -234
  312. package/templates/starter/packages/cucumber-runtime/src/parameter-types.ts +7 -7
  313. package/templates/starter/packages/cucumber-runtime/src/random-data.util.ts +35 -35
  314. package/templates/starter/packages/cucumber-runtime/src/types.ts +13 -13
  315. package/templates/starter/packages/cucumber-runtime/src/world.ts +44 -44
  316. package/templates/starter/packages/cucumber-runtime/tsconfig.json +11 -11
  317. package/templates/starter/packages/locator-picker-companion/dist/cli.js +1 -1
  318. package/templates/starter/packages/locator-picker-companion/dist/index.d.ts +3 -3
  319. package/templates/starter/packages/locator-picker-companion/dist/index.js +3 -3
  320. package/templates/starter/packages/locator-picker-companion/dist/injected-picker-script.js +13 -3
  321. package/templates/starter/packages/locator-picker-companion/dist/selector-generator.d.ts +6 -3
  322. package/templates/starter/packages/locator-picker-companion/dist/selector-generator.js +233 -229
  323. package/templates/starter/packages/locator-picker-companion/src/cli.ts +1 -7
  324. package/templates/starter/packages/locator-picker-companion/src/injected-picker-script.ts +26 -5
  325. package/templates/starter/packages/locator-picker-companion/src/launcher.ts +1 -3
  326. package/templates/starter/packages/locator-picker-companion/src/session-file.ts +4 -15
  327. package/templates/starter/packages/locator-picker-companion/src/types.ts +1 -8
  328. package/templates/starter/playwright.config.ts +46 -0
  329. package/templates/starter/postcss.config.mjs +8 -8
  330. package/templates/starter/prisma/dev.db +0 -0
  331. package/templates/starter/scripts/configure-git-line-endings.mjs +91 -0
  332. package/templates/starter/scripts/install-template-step.ts +5 -1
  333. package/templates/starter/scripts/lib/step-matcher.test.ts +3 -3
  334. package/templates/starter/scripts/lib/step-matcher.ts +5 -1
  335. package/templates/starter/scripts/lib/template-step-installer.test.ts +4 -2
  336. package/templates/starter/scripts/lib/template-step-installer.ts +16 -4
  337. package/templates/starter/scripts/lib/template-step-registry.test.ts +1 -1
  338. package/templates/starter/scripts/lib/template-step-registry.ts +4 -1
  339. package/templates/starter/scripts/run-fallow-commit.mjs +4 -6
  340. package/templates/starter/scripts/sync-all.ts +7 -6
  341. package/templates/starter/scripts/sync-appraise-base-template.ts +84 -77
  342. package/templates/starter/scripts/sync-environments.ts +47 -44
  343. package/templates/starter/scripts/sync-locator-groups.ts +67 -76
  344. package/templates/starter/scripts/sync-locators.ts +50 -53
  345. package/templates/starter/scripts/sync-modules.ts +42 -42
  346. package/templates/starter/scripts/sync-tags.ts +51 -51
  347. package/templates/starter/scripts/sync-template-step-groups.ts +34 -32
  348. package/templates/starter/scripts/sync-template-steps.ts +97 -97
  349. package/templates/starter/scripts/sync-test-cases.ts +65 -74
  350. package/templates/starter/scripts/sync-test-suites.ts +85 -80
  351. package/templates/starter/src/actions/server-action-boundary.test.ts +1 -7
  352. package/templates/starter/src/actions/settings/sync-actions.ts +3 -1
  353. package/templates/starter/src/actions/tags/tag-actions.ts +1 -7
  354. package/templates/starter/src/actions/template-test-case/template-test-case-actions.ts +3 -1
  355. package/templates/starter/src/app/(base)/environments/create/page.tsx +28 -28
  356. package/templates/starter/src/app/(base)/environments/environment-form.tsx +3 -2
  357. package/templates/starter/src/app/(base)/environments/environment-helpers.ts +4 -1
  358. package/templates/starter/src/app/(base)/layout.tsx +10 -10
  359. package/templates/starter/src/app/(base)/locator-groups/locator-group-form.tsx +5 -3
  360. package/templates/starter/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -77
  361. package/templates/starter/src/app/(base)/locator-groups/locator-group-table.tsx +28 -28
  362. package/templates/starter/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -46
  363. package/templates/starter/src/app/(base)/locator-groups/page.tsx +9 -19
  364. package/templates/starter/src/app/(base)/locators/create/create-locator-workspace-helpers.test.ts +3 -6
  365. package/templates/starter/src/app/(base)/locators/create/create-locator-workspace-response.ts +22 -5
  366. package/templates/starter/src/app/(base)/locators/create/create-locator-workspace-state.ts +5 -1
  367. package/templates/starter/src/app/(base)/locators/create/use-locator-workspace.ts +7 -2
  368. package/templates/starter/src/app/(base)/locators/locator-file-sync.ts +1 -5
  369. package/templates/starter/src/app/(base)/locators/locator-helpers.ts +2 -12
  370. package/templates/starter/src/app/(base)/locators/locator-table-columns.tsx +59 -60
  371. package/templates/starter/src/app/(base)/modules/module-form.tsx +3 -1
  372. package/templates/starter/src/app/(base)/reports/overview-chart.tsx +49 -49
  373. package/templates/starter/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +114 -115
  374. package/templates/starter/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -27
  375. package/templates/starter/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +80 -79
  376. package/templates/starter/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +29 -27
  377. package/templates/starter/src/app/(base)/settings/settings-sync-panel-helpers.test.tsx +4 -6
  378. package/templates/starter/src/app/(base)/settings/settings-sync-panel.tsx +22 -23
  379. package/templates/starter/src/app/(base)/tags/tag-form.test.tsx +2 -10
  380. package/templates/starter/src/app/(base)/tags/tag-form.tsx +5 -3
  381. package/templates/starter/src/app/(base)/tags/tag-table-columns.tsx +63 -63
  382. package/templates/starter/src/app/(base)/tags/tag-table.tsx +29 -29
  383. package/templates/starter/src/app/(base)/template-step-groups/create/page.tsx +28 -28
  384. package/templates/starter/src/app/(base)/template-step-groups/template-step-group-form.tsx +3 -1
  385. package/templates/starter/src/app/(base)/template-step-groups/template-step-group-helpers.ts +4 -1
  386. package/templates/starter/src/app/(base)/template-steps/paramChip.tsx +6 -10
  387. package/templates/starter/src/app/(base)/template-steps/template-step-form-helpers.ts +4 -2
  388. package/templates/starter/src/app/(base)/template-steps/template-step-helpers.test.ts +19 -1
  389. package/templates/starter/src/app/(base)/template-steps/template-step-helpers.ts +1 -5
  390. package/templates/starter/src/app/(base)/template-steps/template-step-row-guards.ts +1 -5
  391. package/templates/starter/src/app/(base)/template-steps/template-step-types.ts +1 -5
  392. package/templates/starter/src/app/(base)/template-test-cases/create-template-test-case-page-data.test.ts +4 -1
  393. package/templates/starter/src/app/(base)/template-test-cases/create-template-test-case-page-data.ts +1 -2
  394. package/templates/starter/src/app/(base)/template-test-cases/modify/[id]/page.tsx +1 -0
  395. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-flow.test.tsx +0 -1
  396. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-form.tsx +3 -1
  397. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -76
  398. package/templates/starter/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -32
  399. package/templates/starter/src/app/(base)/test-cases/create/page.tsx +1 -5
  400. package/templates/starter/src/app/(base)/test-cases/create-from-template/create-from-template-selection-helpers.ts +1 -4
  401. package/templates/starter/src/app/(base)/test-cases/create-from-template/page.test.tsx +2 -2
  402. package/templates/starter/src/app/(base)/test-cases/inline-tag-creation-dialog.tsx +1 -7
  403. package/templates/starter/src/app/(base)/test-cases/inline-test-suite-creation-dialog.tsx +1 -7
  404. package/templates/starter/src/app/(base)/test-cases/modify/[id]/page.tsx +1 -5
  405. package/templates/starter/src/app/(base)/test-cases/test-case-form.tsx +63 -78
  406. package/templates/starter/src/app/(base)/test-cases/test-case-route-helpers.test.ts +2 -7
  407. package/templates/starter/src/app/(base)/test-suites/editable-test-suite-helpers.ts +1 -6
  408. package/templates/starter/src/app/(base)/test-suites/test-suite-form.test.tsx +2 -10
  409. package/templates/starter/src/app/(dashboard-components)/data-card-grid.tsx +10 -10
  410. package/templates/starter/src/app/api/test-runs/[runId]/download/route.ts +1 -4
  411. package/templates/starter/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.test.ts +18 -9
  412. package/templates/starter/src/assets/icons/empty-tube.tsx +23 -23
  413. package/templates/starter/src/assets/icons/tube-plus.tsx +29 -29
  414. package/templates/starter/src/components/data-visualization/info-card.tsx +56 -70
  415. package/templates/starter/src/components/data-visualization/info-grid.tsx +22 -22
  416. package/templates/starter/src/components/diagram/add-node-prompt-node.tsx +1 -1
  417. package/templates/starter/src/components/diagram/dynamic-parameters-helpers.ts +4 -3
  418. package/templates/starter/src/components/diagram/dynamic-parameters.test.tsx +4 -1
  419. package/templates/starter/src/components/diagram/dynamic-parameters.tsx +0 -1
  420. package/templates/starter/src/components/diagram/flow-diagram-block-dialog.test.tsx +26 -0
  421. package/templates/starter/src/components/diagram/flow-diagram-block-dialog.tsx +3 -1
  422. package/templates/starter/src/components/diagram/flow-diagram-connection-guards.ts +24 -6
  423. package/templates/starter/src/components/diagram/flow-diagram-helpers.test.ts +27 -0
  424. package/templates/starter/src/components/diagram/flow-diagram-node-search.tsx +65 -65
  425. package/templates/starter/src/components/diagram/flow-diagram-toolbar.tsx +1 -3
  426. package/templates/starter/src/components/diagram/flow-diagram-view.tsx +1 -7
  427. package/templates/starter/src/components/diagram/flow-node-data-helpers.ts +1 -1
  428. package/templates/starter/src/components/diagram/node-form-fields-content.tsx +9 -2
  429. package/templates/starter/src/components/diagram/node-form-helpers.test.ts +3 -1
  430. package/templates/starter/src/components/diagram/node-form-helpers.ts +1 -4
  431. package/templates/starter/src/components/diagram/node-form-template-step-selection.ts +1 -4
  432. package/templates/starter/src/components/diagram/node-form.test.tsx +25 -0
  433. package/templates/starter/src/components/diagram/node-form.tsx +2 -12
  434. package/templates/starter/src/components/diagram/options-header-node.test.tsx +4 -1
  435. package/templates/starter/src/components/diagram/options-header-node.tsx +140 -130
  436. package/templates/starter/src/components/diagram/use-flow-diagram.ts +1 -2
  437. package/templates/starter/src/components/form/error-message.tsx +7 -7
  438. package/templates/starter/src/components/loading-skeleton/form/button-skeleton.tsx +8 -8
  439. package/templates/starter/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -8
  440. package/templates/starter/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -14
  441. package/templates/starter/src/components/navigation/entity-search-command.tsx +3 -9
  442. package/templates/starter/src/components/navigation/nav-link.tsx +60 -60
  443. package/templates/starter/src/components/test-case/test-case-form-helpers.test.ts +1 -5
  444. package/templates/starter/src/components/test-case/test-case-form-helpers.ts +1 -5
  445. package/templates/starter/src/components/test-case/test-case-picker-helpers.ts +8 -7
  446. package/templates/starter/src/components/test-case/test-case-picker.tsx +2 -6
  447. package/templates/starter/src/components/test-run/log-viewer.test.tsx +25 -0
  448. package/templates/starter/src/components/test-run/log-viewer.tsx +2 -2
  449. package/templates/starter/src/components/test-run/test-run-details-guards.ts +10 -1
  450. package/templates/starter/src/components/test-run/test-run-details.test.tsx +9 -19
  451. package/templates/starter/src/components/test-run/use-log-viewer.ts +2 -8
  452. package/templates/starter/src/components/test-run/use-test-run-header.ts +1 -4
  453. package/templates/starter/src/components/test-suite/test-suite-picker.tsx +1 -1
  454. package/templates/starter/src/components/theme/theme-provider.tsx +7 -1
  455. package/templates/starter/src/components/typography/page-header-subtitle.tsx +7 -7
  456. package/templates/starter/src/components/typography/page-header.tsx +7 -7
  457. package/templates/starter/src/components/ui/chart.tsx +2 -8
  458. package/templates/starter/src/components/ui/command.tsx +2 -5
  459. package/templates/starter/src/components/ui/data-table-column-header.tsx +3 -3
  460. package/templates/starter/src/components/ui/data-table.test.tsx +1 -8
  461. package/templates/starter/src/components/ui/dialog.tsx +1 -4
  462. package/templates/starter/src/components/ui/select.tsx +2 -2
  463. package/templates/starter/src/components/ui/separator.tsx +5 -1
  464. package/templates/starter/src/components/ui/skeleton.tsx +7 -7
  465. package/templates/starter/src/components/ui/table.tsx +1 -1
  466. package/templates/starter/src/components/ui/toaster.tsx +26 -26
  467. package/templates/starter/src/components/ui/tooltip.tsx +1 -1
  468. package/templates/starter/src/constants/form-opts/locator-group-form-opts.ts +1 -1
  469. package/templates/starter/src/lib/locator-group-file-utils.ts +5 -1
  470. package/templates/starter/src/lib/locator-picker/session-manager.ts +1 -2
  471. package/templates/starter/src/lib/metrics/metric-calculator.test.ts +105 -6
  472. package/templates/starter/src/lib/metrics/metric-calculator.ts +39 -40
  473. package/templates/starter/src/lib/sync/sync-executor.ts +1 -4
  474. package/templates/starter/src/lib/sync/sync-pending-counts.test.ts +201 -7
  475. package/templates/starter/src/lib/sync/sync-pending-counts.ts +81 -13
  476. package/templates/starter/src/lib/template-automation-paths.ts +1 -1
  477. package/templates/starter/src/lib/template-sync-utils.d.ts +14 -7
  478. package/templates/starter/src/lib/test-case-utils.ts +6 -6
  479. package/templates/starter/src/lib/test-run/log-formatter.ts +82 -83
  480. package/templates/starter/src/lib/test-run/report-parser.ts +352 -352
  481. package/templates/starter/src/lib/transformers/gherkin-converter.ts +42 -42
  482. package/templates/starter/src/lib/transformers/key-to-icon-transformer.tsx +95 -95
  483. package/templates/starter/src/lib/utils/node-param-validation.ts +81 -81
  484. package/templates/starter/src/lib/utils/template-step-file-generator.ts +5 -3
  485. package/templates/starter/src/lib/utils/template-step-file-manager-intelligent.ts +11 -1
  486. package/templates/starter/src/services/dashboard/dashboard-service.test.ts +28 -1
  487. package/templates/starter/src/services/dashboard/dashboard-service.ts +2 -0
  488. package/templates/starter/src/services/locator/locator-service.test.ts +1 -3
  489. package/templates/starter/src/services/locator/locator-service.ts +3 -1
  490. package/templates/starter/src/services/locator/locator-sync-utils.test.ts +1 -4
  491. package/templates/starter/src/services/module/module-service.ts +1 -4
  492. package/templates/starter/src/services/report/report-service.test.ts +70 -6
  493. package/templates/starter/src/services/report/report-service.ts +52 -9
  494. package/templates/starter/src/services/shared/errors.ts +2 -4
  495. package/templates/starter/src/services/template-step/template-step-service.ts +3 -1
  496. package/templates/starter/src/services/template-step-group/template-step-group-service.ts +3 -1
  497. package/templates/starter/src/services/test-case/test-case-service.ts +0 -5
  498. package/templates/starter/src/services/test-run/test-run-service.test.ts +3 -6
  499. package/templates/starter/src/services/test-run/test-run-service.ts +12 -7
  500. package/templates/starter/src/services/test-suite/test-suite-service.test.ts +1 -5
  501. package/templates/starter/src/services/test-suite/test-suite-service.ts +2 -4
  502. package/templates/starter/src/types/form/actionHandler.ts +1 -5
  503. package/templates/starter/tsconfig.json +8 -8
@@ -1,60 +1,60 @@
1
- 'use client'
2
-
3
- import { usePathname } from 'next/navigation'
4
- import Link from 'next/link'
5
- import { useMemo } from 'react'
6
- import { cn } from '@/lib/utils'
7
-
8
- const NavLink = ({
9
- children,
10
- href,
11
- icon,
12
- className,
13
- isIconOnly,
14
- }: {
15
- children?: React.ReactNode
16
- href: string
17
- icon?: React.ReactNode
18
- className?: React.HTMLAttributes<HTMLAnchorElement>['className']
19
- isIconOnly?: boolean
20
- }) => {
21
- const pathname = usePathname()
22
-
23
- // Use useMemo to prevent unnecessary re-renders and ensure stable active state
24
- const isActive = useMemo(() => {
25
- // Exact match for root paths
26
- if (href === '/' && pathname === '/') {
27
- return true
28
- }
29
- // For other paths, check if the current path starts with the href
30
- // This handles nested routes like /test-cases/create, /test-cases/modify, etc.
31
- return href !== '/' && pathname.startsWith(href)
32
- }, [pathname, href])
33
-
34
- return (
35
- <Link
36
- href={href}
37
- className={cn(
38
- 'group inline-flex h-9 w-max items-center justify-center rounded-md bg-background px-4 py-2 text-sm font-medium transition-colors',
39
- 'hover:bg-accent hover:text-accent-foreground',
40
- 'focus:bg-accent focus:text-accent-foreground',
41
- 'focus-visible:outline-1 focus-visible:ring-[3px] focus-visible:ring-ring/50',
42
- 'disabled:pointer-events-none disabled:opacity-50',
43
- 'outline-none',
44
- isActive && 'bg-accent text-accent-foreground',
45
- className,
46
- )}
47
- >
48
- {isIconOnly ? (
49
- icon
50
- ) : (
51
- <div className="flex items-center gap-1">
52
- {icon}
53
- {children}
54
- </div>
55
- )}
56
- </Link>
57
- )
58
- }
59
-
60
- export default NavLink
1
+ 'use client'
2
+
3
+ import { usePathname } from 'next/navigation'
4
+ import Link from 'next/link'
5
+ import { useMemo } from 'react'
6
+ import { cn } from '@/lib/utils'
7
+
8
+ const NavLink = ({
9
+ children,
10
+ href,
11
+ icon,
12
+ className,
13
+ isIconOnly,
14
+ }: {
15
+ children?: React.ReactNode
16
+ href: string
17
+ icon?: React.ReactNode
18
+ className?: React.HTMLAttributes<HTMLAnchorElement>['className']
19
+ isIconOnly?: boolean
20
+ }) => {
21
+ const pathname = usePathname()
22
+
23
+ // Use useMemo to prevent unnecessary re-renders and ensure stable active state
24
+ const isActive = useMemo(() => {
25
+ // Exact match for root paths
26
+ if (href === '/' && pathname === '/') {
27
+ return true
28
+ }
29
+ // For other paths, check if the current path starts with the href
30
+ // This handles nested routes like /test-cases/create, /test-cases/modify, etc.
31
+ return href !== '/' && pathname.startsWith(href)
32
+ }, [pathname, href])
33
+
34
+ return (
35
+ <Link
36
+ href={href}
37
+ className={cn(
38
+ 'group inline-flex h-9 w-max items-center justify-center rounded-md bg-background px-4 py-2 text-sm font-medium transition-colors',
39
+ 'hover:bg-accent hover:text-accent-foreground',
40
+ 'focus:bg-accent focus:text-accent-foreground',
41
+ 'focus-visible:ring-ring/50 focus-visible:outline-1 focus-visible:ring-[3px]',
42
+ 'disabled:pointer-events-none disabled:opacity-50',
43
+ 'outline-none',
44
+ isActive && 'bg-accent text-accent-foreground',
45
+ className,
46
+ )}
47
+ >
48
+ {isIconOnly ? (
49
+ icon
50
+ ) : (
51
+ <div className="flex items-center gap-1">
52
+ {icon}
53
+ {children}
54
+ </div>
55
+ )}
56
+ </Link>
57
+ )
58
+ }
59
+
60
+ export default NavLink
@@ -1,11 +1,7 @@
1
1
  import { StepParameterType, TemplateStepIcon } from '@prisma/client'
2
2
  import { describe, expect, it } from 'vitest'
3
3
 
4
- import {
5
- buildScenarioPreview,
6
- buildScenarioSteps,
7
- getNodesWithMissingMandatoryParams,
8
- } from './test-case-form-helpers'
4
+ import { buildScenarioPreview, buildScenarioSteps, getNodesWithMissingMandatoryParams } from './test-case-form-helpers'
9
5
 
10
6
  describe('test-case form helpers', () => {
11
7
  it('builds gherkin preview text from ordered scenario steps', () => {
@@ -54,11 +54,7 @@ export function buildScenarioSteps(nodeOrder: ScenarioNodeOrder) {
54
54
  }))
55
55
  }
56
56
 
57
- export function buildScenarioPreview(
58
- title: string,
59
- description: string | undefined,
60
- nodeOrder: ScenarioNodeOrder,
61
- ) {
57
+ export function buildScenarioPreview(title: string, description: string | undefined, nodeOrder: ScenarioNodeOrder) {
62
58
  if (!title) {
63
59
  return ''
64
60
  }
@@ -10,16 +10,14 @@ export function createSelectionState(selectedIds: string[]): RowSelectionState {
10
10
  }
11
11
 
12
12
  export function testCaseMatchesQuery(testCase: TestCasePickerRow, filterValue: unknown): boolean {
13
- const query = String(filterValue ?? '').trim().toLowerCase()
13
+ const query = String(filterValue ?? '')
14
+ .trim()
15
+ .toLowerCase()
14
16
  if (!query) {
15
17
  return true
16
18
  }
17
19
 
18
- const searchableText = [
19
- testCase.title,
20
- testCase.description ?? '',
21
- ...testCase.tags.map(tag => tag.name),
22
- ]
20
+ const searchableText = [testCase.title, testCase.description ?? '', ...testCase.tags.map(tag => tag.name)]
23
21
  .join(' ')
24
22
  .toLowerCase()
25
23
 
@@ -32,7 +30,10 @@ export function getSavedTestCases(testCases: TestCasePickerRow[], selectedIds: s
32
30
  .filter((testCase): testCase is TestCasePickerRow => Boolean(testCase))
33
31
  }
34
32
 
35
- export function getSelectedIdsFromRowSelection(testCases: TestCasePickerRow[], rowSelection: RowSelectionState): string[] {
33
+ export function getSelectedIdsFromRowSelection(
34
+ testCases: TestCasePickerRow[],
35
+ rowSelection: RowSelectionState,
36
+ ): string[] {
36
37
  return testCases.filter(testCase => rowSelection[testCase.id]).map(testCase => testCase.id)
37
38
  }
38
39
 
@@ -187,9 +187,7 @@ function TestCasePicker({
187
187
  <TableRow key={headerGroup.id}>
188
188
  {headerGroup.headers.map(header => (
189
189
  <TableHead key={header.id}>
190
- {header.isPlaceholder
191
- ? null
192
- : flexRender(header.column.columnDef.header, header.getContext())}
190
+ {header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())}
193
191
  </TableHead>
194
192
  ))}
195
193
  </TableRow>
@@ -200,9 +198,7 @@ function TestCasePicker({
200
198
  table.getRowModel().rows.map(row => (
201
199
  <TableRow key={row.id} data-state={row.getIsSelected() && 'selected'}>
202
200
  {row.getVisibleCells().map(cell => (
203
- <TableCell key={cell.id}>
204
- {flexRender(cell.column.columnDef.cell, cell.getContext())}
205
- </TableCell>
201
+ <TableCell key={cell.id}>{flexRender(cell.column.columnDef.cell, cell.getContext())}</TableCell>
206
202
  ))}
207
203
  </TableRow>
208
204
  ))
@@ -77,6 +77,31 @@ describe('LogViewer', () => {
77
77
  expect(screen.getByText('Download logs for run-1')).toBeInTheDocument()
78
78
  })
79
79
 
80
+ it('renders duplicate log entries without dropping repeated messages', async () => {
81
+ const duplicateTimestamp = '2026-05-30T09:22:19.484Z'
82
+ const duplicateMessage =
83
+ 'Scenario completed: [Demo Run With Template] Demo run using a template test case - unknown'
84
+ getTestRunLogsAction.mockResolvedValue({
85
+ status: 200,
86
+ data: [
87
+ {
88
+ type: 'status',
89
+ message: duplicateMessage,
90
+ timestamp: duplicateTimestamp,
91
+ },
92
+ {
93
+ type: 'status',
94
+ message: duplicateMessage,
95
+ timestamp: duplicateTimestamp,
96
+ },
97
+ ],
98
+ })
99
+
100
+ render(<LogViewer testRunId="run-duplicates" status={TestRunStatus.COMPLETED} />)
101
+
102
+ expect(await screen.findAllByText(duplicateMessage)).toHaveLength(2)
103
+ })
104
+
80
105
  it('streams live logs and dispatches the exit event when the run completes', async () => {
81
106
  const emittedEvents: Array<Event> = []
82
107
  const addEventListenerSpy = vi.spyOn(window, 'dispatchEvent')
@@ -110,9 +110,9 @@ export function LogViewer({ testRunId, status, className }: LogViewerProps) {
110
110
  {logs.length === 0 && connectionStatus !== 'connecting' && connectionStatus !== 'loading' ? (
111
111
  <div className="flex items-center justify-center py-8 text-muted-foreground">No logs available</div>
112
112
  ) : null}
113
- {logs.map(log => (
113
+ {logs.map((log, index) => (
114
114
  <div
115
- key={`${log.timestamp.toISOString()}-${log.type}-${log.message}`}
115
+ key={`${log.timestamp.toISOString()}-${log.type}-${log.message}-${index}`}
116
116
  className={cn(
117
117
  'mb-1 flex items-start gap-2 whitespace-pre-wrap break-words',
118
118
  log.type === 'stderr' && 'text-red-400',
@@ -44,7 +44,16 @@ function isTestRunDetailsTestCase(value: unknown): value is TestRunDetailsTestCa
44
44
 
45
45
  function hasTestRunScalarFields(value: UnknownRecord) {
46
46
  return (
47
- hasFields(value, ['id', 'runId', 'status', 'result', 'startedAt', 'completedAt', 'browserEngine', 'testWorkersCount']) &&
47
+ hasFields(value, [
48
+ 'id',
49
+ 'runId',
50
+ 'status',
51
+ 'result',
52
+ 'startedAt',
53
+ 'completedAt',
54
+ 'browserEngine',
55
+ 'testWorkersCount',
56
+ ]) &&
48
57
  value.startedAt instanceof Date &&
49
58
  (value.completedAt instanceof Date || value.completedAt === null)
50
59
  )
@@ -2,30 +2,20 @@
2
2
 
3
3
  import { act, render, screen, waitFor } from '@testing-library/react'
4
4
  import userEvent from '@testing-library/user-event'
5
- import {
6
- TestRunResult,
7
- TestRunStatus,
8
- TestRunTestCaseResult,
9
- TestRunTestCaseStatus,
10
- } from '@prisma/client'
5
+ import { TestRunResult, TestRunStatus, TestRunTestCaseResult, TestRunTestCaseStatus } from '@prisma/client'
11
6
  import { beforeEach, describe, expect, it, vi } from 'vitest'
12
7
 
13
8
  import { TestRunDetails } from './test-run-details'
14
9
  import type { TestRunDetailsData } from './test-run-details-helpers'
15
10
 
16
- const {
17
- getTestRunByIdAction,
18
- spawnTraceViewerAction,
19
- checkTraceViewerStatusAction,
20
- cancelTestRunAction,
21
- toast,
22
- } = vi.hoisted(() => ({
23
- getTestRunByIdAction: vi.fn(),
24
- spawnTraceViewerAction: vi.fn(),
25
- checkTraceViewerStatusAction: vi.fn(),
26
- cancelTestRunAction: vi.fn(),
27
- toast: vi.fn(),
28
- }))
11
+ const { getTestRunByIdAction, spawnTraceViewerAction, checkTraceViewerStatusAction, cancelTestRunAction, toast } =
12
+ vi.hoisted(() => ({
13
+ getTestRunByIdAction: vi.fn(),
14
+ spawnTraceViewerAction: vi.fn(),
15
+ checkTraceViewerStatusAction: vi.fn(),
16
+ cancelTestRunAction: vi.fn(),
17
+ toast: vi.fn(),
18
+ }))
29
19
 
30
20
  vi.mock('@/actions/test-run/test-run-actions', () => ({
31
21
  getTestRunByIdAction,
@@ -128,10 +128,7 @@ export function useLogViewer({ testRunId, status }: UseLogViewerParams) {
128
128
  console.log('[LogViewer] Received log event:', data)
129
129
  }
130
130
 
131
- setLogs(currentLogs => [
132
- ...currentLogs,
133
- createLogMessage(data.type || 'stdout', data.message || ''),
134
- ])
131
+ setLogs(currentLogs => [...currentLogs, createLogMessage(data.type || 'stdout', data.message || '')])
135
132
  } catch (error) {
136
133
  console.error('[LogViewer] Error parsing log event:', error, 'event.data:', event.data)
137
134
  }
@@ -147,10 +144,7 @@ export function useLogViewer({ testRunId, status }: UseLogViewerParams) {
147
144
  }
148
145
 
149
146
  const data = JSON.parse(event.data)
150
- setLogs(currentLogs => [
151
- ...currentLogs,
152
- createLogMessage('status', `Process exited with code ${data.code}`),
153
- ])
147
+ setLogs(currentLogs => [...currentLogs, createLogMessage('status', `Process exited with code ${data.code}`)])
154
148
  } catch (error) {
155
149
  console.error('[LogViewer] Error parsing exit event:', error, 'event.data:', event.data)
156
150
  } finally {
@@ -4,10 +4,7 @@ import { useCallback, useEffect, useState } from 'react'
4
4
 
5
5
  import { getTestRunByIdAction } from '@/actions/test-run/test-run-actions'
6
6
 
7
- import {
8
- getTestRunDetailsData,
9
- type TestRunDetailsData,
10
- } from './test-run-details-helpers'
7
+ import { getTestRunDetailsData, type TestRunDetailsData } from './test-run-details-helpers'
11
8
  import {
12
9
  getTestRunHeaderPollMode,
13
10
  matchesTestRunExitEvent,
@@ -232,4 +232,4 @@ function TestSuitePicker({
232
232
  )
233
233
  }
234
234
 
235
- export default TestSuitePicker
235
+ export default TestSuitePicker
@@ -9,7 +9,13 @@ type ThemeProviderProps = {
9
9
 
10
10
  export function ThemeProvider({ children }: ThemeProviderProps) {
11
11
  return (
12
- <NextThemesProvider attribute="class" defaultTheme="dark" forcedTheme="dark" enableSystem={false} disableTransitionOnChange>
12
+ <NextThemesProvider
13
+ attribute="class"
14
+ defaultTheme="dark"
15
+ forcedTheme="dark"
16
+ enableSystem={false}
17
+ disableTransitionOnChange
18
+ >
13
19
  {children}
14
20
  </NextThemesProvider>
15
21
  )
@@ -1,7 +1,7 @@
1
- import React from 'react'
2
-
3
- const HeaderSubtitle = ({ children }: { children: React.ReactNode }) => {
4
- return <div className="text-sm font-medium text-muted-foreground">{children}</div>
5
- }
6
-
7
- export default HeaderSubtitle
1
+ import React from 'react'
2
+
3
+ const HeaderSubtitle = ({ children }: { children: React.ReactNode }) => {
4
+ return <div className="text-sm font-medium text-muted-foreground">{children}</div>
5
+ }
6
+
7
+ export default HeaderSubtitle
@@ -1,7 +1,7 @@
1
- import { cn } from '@/lib/utils'
2
-
3
- const PageHeader = ({ className, children }: { className?: string; children: React.ReactNode }) => {
4
- return <div className={cn('text-4xl font-bold text-primary', className)}>{children}</div>
5
- }
6
-
7
- export default PageHeader
1
+ import { cn } from '@/lib/utils'
2
+
3
+ const PageHeader = ({ className, children }: { className?: string; children: React.ReactNode }) => {
4
+ return <div className={cn('text-4xl font-bold text-primary', className)}>{children}</div>
5
+ }
6
+
7
+ export default PageHeader
@@ -166,10 +166,7 @@ function ChartTooltipContentActive({
166
166
  ref,
167
167
  }: ChartTooltipContentProps) {
168
168
  const { config } = useChart()
169
- const resolvedPayload = React.useMemo(
170
- () => (payload?.length ? payload : EMPTY_CHART_TOOLTIP_PAYLOAD),
171
- [payload],
172
- )
169
+ const resolvedPayload = React.useMemo(() => (payload?.length ? payload : EMPTY_CHART_TOOLTIP_PAYLOAD), [payload])
173
170
 
174
171
  if (!resolvedPayload.length) {
175
172
  return null
@@ -239,10 +236,7 @@ function ChartTooltipContentActive({
239
236
  )
240
237
  )}
241
238
  <div
242
- className={cn(
243
- 'flex flex-1 justify-between leading-none',
244
- nestLabel ? 'items-end' : 'items-center',
245
- )}
239
+ className={cn('flex flex-1 justify-between leading-none', nestLabel ? 'items-end' : 'items-center')}
246
240
  >
247
241
  <div className="grid gap-1.5">
248
242
  {nestLabel ? (
@@ -56,10 +56,7 @@ CommandInput.displayName = CommandPrimitive.Input.displayName
56
56
 
57
57
  function CommandList({ className, ...props }: React.ComponentProps<typeof CommandPrimitive.List>) {
58
58
  return (
59
- <CommandPrimitive.List
60
- className={cn('max-h-[300px] overflow-y-auto overflow-x-hidden', className)}
61
- {...props}
62
- />
59
+ <CommandPrimitive.List className={cn('max-h-[300px] overflow-y-auto overflow-x-hidden', className)} {...props} />
63
60
  )
64
61
  }
65
62
 
@@ -89,7 +86,7 @@ function CommandItem({ className, ...props }: React.ComponentProps<typeof Comman
89
86
  return (
90
87
  <CommandPrimitive.Item
91
88
  className={cn(
92
- 'relative flex cursor-default select-none items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-none data-[disabled=true]:pointer-events-none data-[selected=true]:bg-accent data-[selected=true]:text-muted-foreground data-[disabled=true]:opacity-50 dark:data-[selected=true]:text-muted-foreground [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0',
89
+ 'relative flex cursor-default select-none items-center gap-2 rounded-sm px-2 py-1.5 text-sm text-popover-foreground outline-none hover:bg-accent hover:text-accent-foreground data-[disabled=true]:pointer-events-none data-[selected=true]:bg-accent data-[selected=true]:text-accent-foreground data-[disabled=true]:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0',
93
90
  className,
94
91
  )}
95
92
  {...props}
@@ -42,16 +42,16 @@ export function DataTableColumnHeader<TData, TValue>({
42
42
  </DropdownMenuTrigger>
43
43
  <DropdownMenuContent align="start">
44
44
  <DropdownMenuItem onClick={() => column.toggleSorting(false)}>
45
- <ArrowUp className="size-3.5 text-muted-foreground/70" />
45
+ <ArrowUp className="text-muted-foreground/70 size-3.5" />
46
46
  Asc
47
47
  </DropdownMenuItem>
48
48
  <DropdownMenuItem onClick={() => column.toggleSorting(true)}>
49
- <ArrowDown className="size-3.5 text-muted-foreground/70" />
49
+ <ArrowDown className="text-muted-foreground/70 size-3.5" />
50
50
  Desc
51
51
  </DropdownMenuItem>
52
52
  <DropdownMenuSeparator />
53
53
  <DropdownMenuItem onClick={() => column.toggleVisibility(false)}>
54
- <EyeOff className="size-3.5 text-muted-foreground/70" />
54
+ <EyeOff className="text-muted-foreground/70 size-3.5" />
55
55
  Hide
56
56
  </DropdownMenuItem>
57
57
  </DropdownMenuContent>
@@ -15,14 +15,7 @@ const { toast } = vi.hoisted(() => ({
15
15
  }))
16
16
 
17
17
  vi.mock('next/link', () => ({
18
- default: ({
19
- href,
20
- children,
21
- ...props
22
- }: {
23
- href: string
24
- children: ReactNode
25
- }) => (
18
+ default: ({ href, children, ...props }: { href: string; children: ReactNode }) => (
26
19
  <a href={href} {...props}>
27
20
  {children}
28
21
  </a>
@@ -61,10 +61,7 @@ DialogFooter.displayName = 'DialogFooter'
61
61
 
62
62
  function DialogTitle({ className, ...props }: React.ComponentProps<typeof DialogPrimitive.Title>) {
63
63
  return (
64
- <DialogPrimitive.Title
65
- className={cn('text-lg font-semibold leading-none tracking-tight', className)}
66
- {...props}
67
- />
64
+ <DialogPrimitive.Title className={cn('text-lg font-semibold leading-none tracking-tight', className)} {...props} />
68
65
  )
69
66
  }
70
67
 
@@ -105,12 +105,12 @@ function SelectItem({ className, children, ...props }: React.ComponentProps<type
105
105
  return (
106
106
  <SelectPrimitive.Item
107
107
  className={cn(
108
- 'focus:text-accent-foreground/5 relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-2 pr-8 text-sm outline-none focus:bg-accent data-[disabled]:pointer-events-none data-[disabled]:opacity-50',
108
+ 'relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-2 pr-8 text-sm text-popover-foreground outline-none hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[highlighted]:bg-accent data-[highlighted]:text-accent-foreground data-[disabled]:opacity-50',
109
109
  className,
110
110
  )}
111
111
  {...props}
112
112
  >
113
- <span className="absolute right-2 flex h-3.5 w-3.5 items-center justify-center">
113
+ <span className="absolute right-2 flex size-3.5 items-center justify-center">
114
114
  <SelectPrimitive.ItemIndicator>
115
115
  <Check className="size-4" />
116
116
  </SelectPrimitive.ItemIndicator>
@@ -15,7 +15,11 @@ function Separator({
15
15
  <SeparatorPrimitive.Root
16
16
  decorative={decorative}
17
17
  orientation={orientation}
18
- className={cn('shrink-0 bg-border', orientation === 'horizontal' ? 'h-[1px] w-full' : 'h-full w-[1px]', className)}
18
+ className={cn(
19
+ 'shrink-0 bg-border',
20
+ orientation === 'horizontal' ? 'h-[1px] w-full' : 'h-full w-[1px]',
21
+ className,
22
+ )}
19
23
  {...props}
20
24
  />
21
25
  )
@@ -1,7 +1,7 @@
1
- import { cn } from '@/lib/utils'
2
-
3
- function Skeleton({ className, ...props }: React.HTMLAttributes<HTMLDivElement>) {
4
- return <div className={cn('animate-pulse rounded-md bg-muted', className)} {...props} />
5
- }
6
-
7
- export { Skeleton }
1
+ import { cn } from '@/lib/utils'
2
+
3
+ function Skeleton({ className, ...props }: React.HTMLAttributes<HTMLDivElement>) {
4
+ return <div className={cn('animate-pulse rounded-md bg-muted', className)} {...props} />
5
+ }
6
+
7
+ export { Skeleton }
@@ -21,7 +21,7 @@ function TableBody({ className, ...props }: React.ComponentProps<'tbody'>) {
21
21
  function TableRow({ className, ...props }: React.ComponentProps<'tr'>) {
22
22
  return (
23
23
  <tr
24
- className={cn('border-b transition-colors hover:bg-muted/50 data-[state=selected]:bg-muted', className)}
24
+ className={cn('hover:bg-muted/50 border-b transition-colors data-[state=selected]:bg-muted', className)}
25
25
  {...props}
26
26
  />
27
27
  )
@@ -1,26 +1,26 @@
1
- 'use client'
2
-
3
- import { useToast } from '@/hooks/use-toast'
4
- import { Toast, ToastClose, ToastDescription, ToastProvider, ToastTitle, ToastViewport } from '@/components/ui/toast'
5
-
6
- export function Toaster() {
7
- const { toasts } = useToast()
8
-
9
- return (
10
- <ToastProvider>
11
- {toasts.map(function ({ id, title, description, action, ...props }) {
12
- return (
13
- <Toast key={id} {...props}>
14
- <div className="grid gap-1">
15
- {title && <ToastTitle>{title}</ToastTitle>}
16
- {description && <ToastDescription>{description}</ToastDescription>}
17
- </div>
18
- {action}
19
- <ToastClose />
20
- </Toast>
21
- )
22
- })}
23
- <ToastViewport />
24
- </ToastProvider>
25
- )
26
- }
1
+ 'use client'
2
+
3
+ import { useToast } from '@/hooks/use-toast'
4
+ import { Toast, ToastClose, ToastDescription, ToastProvider, ToastTitle, ToastViewport } from '@/components/ui/toast'
5
+
6
+ export function Toaster() {
7
+ const { toasts } = useToast()
8
+
9
+ return (
10
+ <ToastProvider>
11
+ {toasts.map(function ({ id, title, description, action, ...props }) {
12
+ return (
13
+ <Toast key={id} {...props}>
14
+ <div className="grid gap-1">
15
+ {title && <ToastTitle>{title}</ToastTitle>}
16
+ {description && <ToastDescription>{description}</ToastDescription>}
17
+ </div>
18
+ {action}
19
+ <ToastClose />
20
+ </Toast>
21
+ )
22
+ })}
23
+ <ToastViewport />
24
+ </ToastProvider>
25
+ )
26
+ }
@@ -21,7 +21,7 @@ function TooltipContent({
21
21
  <TooltipPrimitive.Content
22
22
  sideOffset={sideOffset}
23
23
  className={cn(
24
- 'z-50 overflow-hidden rounded-md bg-primary px-3 py-1.5 text-xs text-primary-foreground animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 origin-[--radix-tooltip-content-transform-origin]',
24
+ 'z-50 origin-[--radix-tooltip-content-transform-origin] overflow-hidden rounded-md bg-primary px-3 py-1.5 text-xs text-primary-foreground animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2',
25
25
  className,
26
26
  )}
27
27
  {...props}
@@ -4,7 +4,7 @@ export const locatorGroupSchema = z.object({
4
4
  name: z.string().min(1, { message: 'Name is required' }),
5
5
  moduleId: z.string().min(1, { message: 'Module is required' }),
6
6
  locators: z.array(z.string()).optional(),
7
- route: z.string().optional(),
7
+ route: z.string().trim().min(1, { message: 'Route is required' }),
8
8
  })
9
9
 
10
10
  export type LocatorGroup = z.infer<typeof locatorGroupSchema>
@@ -144,7 +144,11 @@ async function cleanupEmptyDirectories(filePath: string): Promise<void> {
144
144
  let currentDir = path.dirname(filePath)
145
145
  const locatorsRoot = getAutomationLocatorsDir()
146
146
 
147
- while (currentDir.startsWith(locatorsRoot) && currentDir !== locatorsRoot && currentDir !== path.dirname(currentDir)) {
147
+ while (
148
+ currentDir.startsWith(locatorsRoot) &&
149
+ currentDir !== locatorsRoot &&
150
+ currentDir !== path.dirname(currentDir)
151
+ ) {
148
152
  try {
149
153
  const files = await fs.readdir(currentDir)
150
154
  if (files.length === 0) {