create-appraisejs 0.1.7 → 0.1.8-alpha

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 (316) hide show
  1. package/README.md +45 -45
  2. package/dist/cli.js +2 -2
  3. package/dist/cli.js.map +1 -1
  4. package/dist/copy-template.d.ts.map +1 -1
  5. package/dist/copy-template.js.map +1 -1
  6. package/package.json +69 -67
  7. package/templates/default/.vscode/settings.json +5 -0
  8. package/templates/default/appraisejs.config.json +1 -1
  9. package/templates/default/components.json +24 -24
  10. package/templates/default/cucumber.mjs +16 -0
  11. package/templates/default/eslint.config.mjs +15 -15
  12. package/templates/default/next.config.ts +13 -7
  13. package/templates/default/package-lock.json +13732 -14321
  14. package/templates/default/package.json +11 -9
  15. package/templates/default/postcss.config.mjs +8 -8
  16. package/templates/default/prisma/migrations/20251104113456_add_type_for_template_step_groups/migration.sql +16 -16
  17. package/templates/default/prisma/migrations/20251104170946_add_tags_to_test_suite_and_test_case/migration.sql +27 -27
  18. package/templates/default/prisma/migrations/20251112190024_add_cascade_delete_to_test_run_test_case/migration.sql +17 -17
  19. package/templates/default/prisma/migrations/20251113181100_add_test_run_log/migration.sql +12 -12
  20. package/templates/default/prisma/migrations/20251119191838_add_tag_type/migration.sql +28 -28
  21. package/templates/default/prisma/migrations/20251121164059_add_conflict_resolution/migration.sql +12 -12
  22. package/templates/default/prisma/migrations/20251223183400_add_report_model_to_db_schema/migration.sql +10 -10
  23. package/templates/default/prisma/migrations/20251223183637_add_report_test_case_entity_for_storing_test_results_for_individual_test_cases/migration.sql +10 -10
  24. package/templates/default/prisma/migrations/20251224083549_add_comprehensive_report_storage/migration.sql +108 -108
  25. package/templates/default/prisma/migrations/20251229194422_migrate_duration_to_string/migration.sql +55 -55
  26. package/templates/default/prisma/migrations/20251230124637_add_unique_constraint_to_test_run_name/migration.sql +27 -27
  27. package/templates/default/prisma/migrations/20260115094436_add_dashboard_metrics/migration.sql +59 -59
  28. package/templates/default/prisma/migrations/20260127172022_add_cascade_delete_to_step_parameters/migration.sql +34 -34
  29. package/templates/default/prisma/schema.prisma +554 -554
  30. package/templates/default/scripts/regenerate-features.ts +94 -94
  31. package/templates/default/scripts/setup-env.ts +19 -19
  32. package/templates/default/scripts/sync-all.ts +341 -341
  33. package/templates/default/scripts/sync-appraise-base-template.ts +52 -2
  34. package/templates/default/scripts/sync-environments.ts +323 -323
  35. package/templates/default/scripts/sync-locator-groups.ts +413 -413
  36. package/templates/default/scripts/sync-locators.ts +402 -402
  37. package/templates/default/scripts/sync-modules.ts +349 -349
  38. package/templates/default/scripts/sync-tags.ts +292 -292
  39. package/templates/default/scripts/sync-template-step-groups.ts +399 -399
  40. package/templates/default/scripts/sync-template-steps.ts +806 -806
  41. package/templates/default/scripts/sync-test-cases.ts +905 -905
  42. package/templates/default/scripts/sync-test-suites.ts +411 -411
  43. package/templates/default/src/actions/conflict/conflict.action.ts +33 -33
  44. package/templates/default/src/actions/dashboard/dashboard-actions.ts +240 -240
  45. package/templates/default/src/actions/environments/environment-actions.ts +205 -205
  46. package/templates/default/src/actions/locator/locator-actions.ts +547 -547
  47. package/templates/default/src/actions/locator-groups/locator-group-actions.ts +344 -344
  48. package/templates/default/src/actions/modules/module-actions.ts +133 -133
  49. package/templates/default/src/actions/reports/report-actions.ts +613 -613
  50. package/templates/default/src/actions/review/review-actions.ts +147 -147
  51. package/templates/default/src/actions/tags/tag-actions.ts +104 -104
  52. package/templates/default/src/actions/template-step/template-step-actions.ts +332 -332
  53. package/templates/default/src/actions/template-step-group/template-step-group-actions.ts +278 -278
  54. package/templates/default/src/actions/template-test-case/template-test-case-actions.ts +238 -238
  55. package/templates/default/src/actions/test-case/test-case-actions.ts +419 -419
  56. package/templates/default/src/actions/test-run/test-run-actions.ts +1185 -1185
  57. package/templates/default/src/actions/test-suite/test-suite-actions.ts +253 -253
  58. package/templates/default/src/actions/user/user-actions.ts +13 -13
  59. package/templates/default/src/app/(base)/environments/create/page.tsx +28 -28
  60. package/templates/default/src/app/(base)/environments/environment-form.tsx +219 -219
  61. package/templates/default/src/app/(base)/environments/environment-table-columns.tsx +96 -96
  62. package/templates/default/src/app/(base)/environments/environment-table.tsx +24 -24
  63. package/templates/default/src/app/(base)/environments/modify/[id]/page.tsx +46 -46
  64. package/templates/default/src/app/(base)/environments/page.tsx +59 -59
  65. package/templates/default/src/app/(base)/layout.tsx +10 -10
  66. package/templates/default/src/app/(base)/locator-groups/create/page.tsx +44 -44
  67. package/templates/default/src/app/(base)/locator-groups/locator-group-form.tsx +215 -215
  68. package/templates/default/src/app/(base)/locator-groups/locator-group-table-columns.tsx +77 -77
  69. package/templates/default/src/app/(base)/locator-groups/locator-group-table.tsx +28 -28
  70. package/templates/default/src/app/(base)/locator-groups/modify/[id]/page.tsx +46 -46
  71. package/templates/default/src/app/(base)/locator-groups/page.tsx +61 -61
  72. package/templates/default/src/app/(base)/locators/create/page.tsx +38 -38
  73. package/templates/default/src/app/(base)/locators/locator-form.tsx +163 -163
  74. package/templates/default/src/app/(base)/locators/locator-table-columns.tsx +73 -90
  75. package/templates/default/src/app/(base)/locators/locator-table.tsx +28 -28
  76. package/templates/default/src/app/(base)/locators/modify/[id]/page.tsx +45 -45
  77. package/templates/default/src/app/(base)/locators/page.tsx +65 -65
  78. package/templates/default/src/app/(base)/locators/sync-locators-button.tsx +66 -66
  79. package/templates/default/src/app/(base)/modules/create/page.tsx +34 -34
  80. package/templates/default/src/app/(base)/modules/modify/[id]/page.tsx +46 -46
  81. package/templates/default/src/app/(base)/modules/module-form.tsx +126 -126
  82. package/templates/default/src/app/(base)/modules/module-table-columns.tsx +85 -85
  83. package/templates/default/src/app/(base)/modules/module-table.tsx +24 -24
  84. package/templates/default/src/app/(base)/modules/page.tsx +59 -59
  85. package/templates/default/src/app/(base)/reports/[id]/page.tsx +517 -517
  86. package/templates/default/src/app/(base)/reports/duration-chart.tsx +33 -33
  87. package/templates/default/src/app/(base)/reports/feature-chart.tsx +78 -78
  88. package/templates/default/src/app/(base)/reports/overview-chart.tsx +46 -46
  89. package/templates/default/src/app/(base)/reports/page.tsx +98 -98
  90. package/templates/default/src/app/(base)/reports/report-metric-card.tsx +16 -16
  91. package/templates/default/src/app/(base)/reports/report-table-columns.tsx +189 -189
  92. package/templates/default/src/app/(base)/reports/report-table.tsx +72 -72
  93. package/templates/default/src/app/(base)/reports/report-view-table-columns.tsx +131 -131
  94. package/templates/default/src/app/(base)/reports/report-view-table.tsx +82 -82
  95. package/templates/default/src/app/(base)/reports/test-cases/page.tsx +42 -42
  96. package/templates/default/src/app/(base)/reports/test-cases/test-cases-metric-table-columns.tsx +115 -115
  97. package/templates/default/src/app/(base)/reports/test-cases/test-cases-metric-table.tsx +27 -27
  98. package/templates/default/src/app/(base)/reports/test-suites/page.tsx +42 -42
  99. package/templates/default/src/app/(base)/reports/test-suites/test-suites-metric-table-columns.tsx +79 -79
  100. package/templates/default/src/app/(base)/reports/test-suites/test-suites-metric-table.tsx +27 -27
  101. package/templates/default/src/app/(base)/reports/view-logs-button.tsx +60 -60
  102. package/templates/default/src/app/(base)/reviews/create/page.tsx +26 -26
  103. package/templates/default/src/app/(base)/reviews/created-reviews-table.tsx +15 -15
  104. package/templates/default/src/app/(base)/reviews/modify/[id]/page.tsx +26 -26
  105. package/templates/default/src/app/(base)/reviews/page.tsx +26 -26
  106. package/templates/default/src/app/(base)/reviews/review/[id]/page.tsx +26 -26
  107. package/templates/default/src/app/(base)/reviews/review-form.tsx +11 -11
  108. package/templates/default/src/app/(base)/reviews/review-table-by-creator-columns.tsx +9 -9
  109. package/templates/default/src/app/(base)/reviews/review-table-by-reviewer-columns.tsx +9 -9
  110. package/templates/default/src/app/(base)/reviews/reviewer-reviews-table.tsx +15 -15
  111. package/templates/default/src/app/(base)/tags/create/page.tsx +39 -39
  112. package/templates/default/src/app/(base)/tags/modify/[id]/page.tsx +50 -50
  113. package/templates/default/src/app/(base)/tags/page.tsx +58 -58
  114. package/templates/default/src/app/(base)/tags/tag-form.tsx +147 -147
  115. package/templates/default/src/app/(base)/tags/tag-table-columns.tsx +63 -63
  116. package/templates/default/src/app/(base)/tags/tag-table.tsx +29 -29
  117. package/templates/default/src/app/(base)/template-step-groups/create/page.tsx +28 -28
  118. package/templates/default/src/app/(base)/template-step-groups/modify/[id]/page.tsx +45 -45
  119. package/templates/default/src/app/(base)/template-step-groups/page.tsx +60 -60
  120. package/templates/default/src/app/(base)/template-step-groups/template-step-group-form.tsx +167 -167
  121. package/templates/default/src/app/(base)/template-step-groups/template-step-group-table-columns.tsx +89 -89
  122. package/templates/default/src/app/(base)/template-step-groups/template-step-group-table.tsx +32 -32
  123. package/templates/default/src/app/(base)/template-steps/create/page.tsx +37 -37
  124. package/templates/default/src/app/(base)/template-steps/modify/[id]/page.tsx +49 -49
  125. package/templates/default/src/app/(base)/template-steps/page.tsx +59 -59
  126. package/templates/default/src/app/(base)/template-steps/paramChip.tsx +213 -213
  127. package/templates/default/src/app/(base)/template-steps/template-step-form.tsx +384 -384
  128. package/templates/default/src/app/(base)/template-steps/template-step-table-columns.tsx +158 -158
  129. package/templates/default/src/app/(base)/template-steps/template-step-table.tsx +24 -24
  130. package/templates/default/src/app/(base)/template-test-cases/create/page.tsx +56 -56
  131. package/templates/default/src/app/(base)/template-test-cases/modify/[id]/page.tsx +89 -89
  132. package/templates/default/src/app/(base)/template-test-cases/page.tsx +58 -58
  133. package/templates/default/src/app/(base)/template-test-cases/template-test-case-flow.tsx +84 -84
  134. package/templates/default/src/app/(base)/template-test-cases/template-test-case-form.tsx +262 -262
  135. package/templates/default/src/app/(base)/template-test-cases/template-test-case-table-columns.tsx +76 -76
  136. package/templates/default/src/app/(base)/template-test-cases/template-test-case-table.tsx +32 -32
  137. package/templates/default/src/app/(base)/test-cases/create/page.tsx +76 -76
  138. package/templates/default/src/app/(base)/test-cases/create-from-template/generate/[id]/page.tsx +96 -96
  139. package/templates/default/src/app/(base)/test-cases/create-from-template/page.tsx +38 -38
  140. package/templates/default/src/app/(base)/test-cases/create-from-template/template-selection-form.tsx +73 -73
  141. package/templates/default/src/app/(base)/test-cases/modify/[id]/page.tsx +106 -106
  142. package/templates/default/src/app/(base)/test-cases/page.tsx +60 -60
  143. package/templates/default/src/app/(base)/test-cases/test-case-flow.tsx +82 -82
  144. package/templates/default/src/app/(base)/test-cases/test-case-form.tsx +395 -395
  145. package/templates/default/src/app/(base)/test-cases/test-case-table-columns.tsx +90 -90
  146. package/templates/default/src/app/(base)/test-cases/test-case-table.tsx +35 -35
  147. package/templates/default/src/app/(base)/test-runs/[id]/page.tsx +56 -56
  148. package/templates/default/src/app/(base)/test-runs/create/page.tsx +47 -47
  149. package/templates/default/src/app/(base)/test-runs/page.tsx +60 -60
  150. package/templates/default/src/app/(base)/test-runs/test-run-form.tsx +508 -512
  151. package/templates/default/src/app/(base)/test-runs/test-run-table-columns.tsx +229 -229
  152. package/templates/default/src/app/(base)/test-runs/test-run-table.tsx +127 -127
  153. package/templates/default/src/app/(base)/test-suites/create/page.tsx +45 -45
  154. package/templates/default/src/app/(base)/test-suites/modify/[id]/page.tsx +55 -55
  155. package/templates/default/src/app/(base)/test-suites/page.tsx +82 -82
  156. package/templates/default/src/app/(base)/test-suites/test-suite-form.tsx +269 -269
  157. package/templates/default/src/app/(base)/test-suites/test-suite-table-columns.tsx +97 -97
  158. package/templates/default/src/app/(base)/test-suites/test-suite-table.tsx +29 -29
  159. package/templates/default/src/app/(dashboard-components)/app-drawer.tsx +187 -187
  160. package/templates/default/src/app/(dashboard-components)/data-card-grid.tsx +12 -12
  161. package/templates/default/src/app/(dashboard-components)/data-card.tsx +26 -26
  162. package/templates/default/src/app/(dashboard-components)/execution-health-panel.tsx +56 -56
  163. package/templates/default/src/app/(dashboard-components)/ongoing-test-runs-card.tsx +87 -87
  164. package/templates/default/src/app/(dashboard-components)/quick-actions-drawer.tsx +44 -44
  165. package/templates/default/src/app/api/test-runs/[runId]/download/route.ts +133 -133
  166. package/templates/default/src/app/api/test-runs/[runId]/logs/route.ts +420 -420
  167. package/templates/default/src/app/api/test-runs/[runId]/trace/[testCaseId]/route.ts +146 -146
  168. package/templates/default/src/app/globals.css +147 -147
  169. package/templates/default/src/app/layout.tsx +171 -171
  170. package/templates/default/src/app/page.tsx +64 -64
  171. package/templates/default/src/assets/icons/empty-tube.tsx +23 -23
  172. package/templates/default/src/assets/icons/tube-plus.tsx +29 -29
  173. package/templates/default/src/components/base-node.tsx +21 -21
  174. package/templates/default/src/components/chart/pie-chart.tsx +73 -73
  175. package/templates/default/src/components/data-extraction/locator-inspector.tsx +460 -460
  176. package/templates/default/src/components/data-state/empty-state.tsx +40 -40
  177. package/templates/default/src/components/data-visualization/info-card.tsx +70 -70
  178. package/templates/default/src/components/data-visualization/info-grid.tsx +22 -22
  179. package/templates/default/src/components/devtools/providers.tsx +19 -13
  180. package/templates/default/src/components/diagram/button-edge.tsx +54 -54
  181. package/templates/default/src/components/diagram/dynamic-parameters.tsx +438 -438
  182. package/templates/default/src/components/diagram/edit-header-option.tsx +36 -36
  183. package/templates/default/src/components/diagram/flow-diagram.tsx +470 -470
  184. package/templates/default/src/components/diagram/node-form.tsx +262 -262
  185. package/templates/default/src/components/diagram/options-header-node.tsx +57 -57
  186. package/templates/default/src/components/diagram/template-step-combobox.tsx +155 -155
  187. package/templates/default/src/components/form/error-message.tsx +7 -7
  188. package/templates/default/src/components/kokonutui/smooth-tab.tsx +453 -453
  189. package/templates/default/src/components/loading-skeleton/data-table/data-table-skeleton.tsx +30 -30
  190. package/templates/default/src/components/loading-skeleton/form/button-skeleton.tsx +8 -8
  191. package/templates/default/src/components/loading-skeleton/form/icon-button-skeleton.tsx +8 -8
  192. package/templates/default/src/components/loading-skeleton/form/text-input-skeleton.tsx +8 -8
  193. package/templates/default/src/components/loading-skeleton/visualization/table-skeleton.tsx +14 -14
  194. package/templates/default/src/components/logo.tsx +15 -15
  195. package/templates/default/src/components/navigation/command-badge.tsx +34 -34
  196. package/templates/default/src/components/navigation/command-chain-input.tsx +51 -51
  197. package/templates/default/src/components/navigation/entity-search-command.tsx +116 -116
  198. package/templates/default/src/components/navigation/nav-card.tsx +31 -31
  199. package/templates/default/src/components/navigation/nav-command.tsx +508 -508
  200. package/templates/default/src/components/navigation/nav-link.tsx +60 -60
  201. package/templates/default/src/components/navigation/nav-menu-card-deck.tsx +112 -112
  202. package/templates/default/src/components/node-header.tsx +159 -159
  203. package/templates/default/src/components/reports/test-case-logs-modal.tsx +253 -253
  204. package/templates/default/src/components/table/table-actions.tsx +172 -172
  205. package/templates/default/src/components/test-run/download-logs-button.tsx +99 -99
  206. package/templates/default/src/components/test-run/log-viewer.tsx +445 -445
  207. package/templates/default/src/components/test-run/test-run-details.tsx +611 -611
  208. package/templates/default/src/components/test-run/test-run-header.tsx +149 -149
  209. package/templates/default/src/components/test-run/view-report-button.tsx +102 -102
  210. package/templates/default/src/components/theme/mode-toggle.tsx +54 -54
  211. package/templates/default/src/components/theme/theme-provider.tsx +8 -8
  212. package/templates/default/src/components/typography/page-header-subtitle.tsx +7 -7
  213. package/templates/default/src/components/typography/page-header.tsx +7 -7
  214. package/templates/default/src/components/ui/alert-dialog.tsx +106 -106
  215. package/templates/default/src/components/ui/alert.tsx +43 -43
  216. package/templates/default/src/components/ui/avatar.tsx +40 -40
  217. package/templates/default/src/components/ui/badge.tsx +29 -29
  218. package/templates/default/src/components/ui/button.tsx +47 -47
  219. package/templates/default/src/components/ui/calendar.tsx +158 -158
  220. package/templates/default/src/components/ui/card.tsx +43 -43
  221. package/templates/default/src/components/ui/checkbox.tsx +28 -28
  222. package/templates/default/src/components/ui/command.tsx +135 -135
  223. package/templates/default/src/components/ui/data-table-column-header.tsx +61 -61
  224. package/templates/default/src/components/ui/data-table-pagination.tsx +87 -87
  225. package/templates/default/src/components/ui/data-table-view-options.tsx +50 -50
  226. package/templates/default/src/components/ui/data-table.tsx +267 -267
  227. package/templates/default/src/components/ui/dialog.tsx +97 -97
  228. package/templates/default/src/components/ui/dropdown-menu.tsx +182 -182
  229. package/templates/default/src/components/ui/input.tsx +22 -22
  230. package/templates/default/src/components/ui/kbd.tsx +28 -28
  231. package/templates/default/src/components/ui/label.tsx +19 -19
  232. package/templates/default/src/components/ui/loading.tsx +12 -12
  233. package/templates/default/src/components/ui/multi-select-with-preview.tsx +116 -116
  234. package/templates/default/src/components/ui/multi-select.tsx +142 -142
  235. package/templates/default/src/components/ui/navigation-menu.tsx +120 -120
  236. package/templates/default/src/components/ui/popover.tsx +33 -33
  237. package/templates/default/src/components/ui/progress.tsx +25 -25
  238. package/templates/default/src/components/ui/radio-group.tsx +44 -44
  239. package/templates/default/src/components/ui/scroll-area.tsx +40 -40
  240. package/templates/default/src/components/ui/select.tsx +151 -144
  241. package/templates/default/src/components/ui/separator.tsx +22 -22
  242. package/templates/default/src/components/ui/skeleton.tsx +7 -7
  243. package/templates/default/src/components/ui/table.tsx +76 -76
  244. package/templates/default/src/components/ui/tabs.tsx +55 -55
  245. package/templates/default/src/components/ui/textarea.tsx +21 -21
  246. package/templates/default/src/components/ui/toast.tsx +113 -113
  247. package/templates/default/src/components/ui/toaster.tsx +26 -26
  248. package/templates/default/src/components/user-prompt/delete-prompt.tsx +87 -87
  249. package/templates/default/src/config/db-config.ts +10 -10
  250. package/templates/default/src/constants/form-opts/diagram/node-form.ts +30 -30
  251. package/templates/default/src/constants/form-opts/environment-form-opts.ts +24 -24
  252. package/templates/default/src/constants/form-opts/locator-form-opts.ts +20 -20
  253. package/templates/default/src/constants/form-opts/locator-group-form-opts.ts +28 -28
  254. package/templates/default/src/constants/form-opts/module-form-opts.ts +21 -21
  255. package/templates/default/src/constants/form-opts/review-form-opts.ts +23 -23
  256. package/templates/default/src/constants/form-opts/tag-form-opts.ts +42 -42
  257. package/templates/default/src/constants/form-opts/template-selection-form-opts.ts +16 -16
  258. package/templates/default/src/constants/form-opts/template-step-group-form-opts.ts +24 -24
  259. package/templates/default/src/constants/form-opts/template-test-case-form-opts.ts +39 -39
  260. package/templates/default/src/constants/form-opts/template-test-step-form-opts.ts +36 -36
  261. package/templates/default/src/constants/form-opts/test-case-form-opts.ts +43 -43
  262. package/templates/default/src/constants/form-opts/test-run-form-opts.ts +31 -31
  263. package/templates/default/src/constants/form-opts/test-suite-form-opts.ts +24 -24
  264. package/templates/default/src/hooks/use-toast.ts +187 -187
  265. package/templates/default/src/lib/bidirectional-sync.ts +432 -432
  266. package/templates/default/src/lib/database-sync.ts +531 -531
  267. package/templates/default/src/lib/environment-file-utils.ts +221 -221
  268. package/templates/default/src/lib/feature-file-generator.ts +411 -411
  269. package/templates/default/src/lib/gherkin-parser.ts +259 -259
  270. package/templates/default/src/lib/locator-group-file-utils.ts +370 -370
  271. package/templates/default/src/lib/metrics/metric-calculator.ts +613 -613
  272. package/templates/default/src/lib/module-hierarchy-builder.ts +205 -205
  273. package/templates/default/src/lib/path-helpers/module-path.ts +71 -71
  274. package/templates/default/src/lib/test-case-utils.ts +6 -6
  275. package/templates/default/src/lib/test-run/log-formatter.ts +83 -83
  276. package/templates/default/src/lib/test-run/process-manager.ts +191 -191
  277. package/templates/default/src/lib/test-run/report-parser.ts +316 -316
  278. package/templates/default/src/lib/test-run/test-run-executor.ts +144 -144
  279. package/templates/default/src/lib/test-run/winston-logger.ts +95 -95
  280. package/templates/default/src/lib/transformers/gherkin-converter.ts +42 -42
  281. package/templates/default/src/lib/transformers/key-to-icon-transformer.tsx +95 -95
  282. package/templates/default/src/lib/transformers/template-test-case-converter.ts +160 -160
  283. package/templates/default/src/lib/utils/node-param-validation.ts +81 -81
  284. package/templates/default/src/lib/utils/template-step-file-generator.ts +167 -167
  285. package/templates/default/src/lib/utils/template-step-file-manager-intelligent.ts +723 -723
  286. package/templates/default/src/lib/utils/template-step-file-manager.ts +166 -166
  287. package/templates/default/src/lib/utils.ts +31 -31
  288. package/templates/default/src/tests/config/executor/world.ts +41 -41
  289. package/templates/default/src/tests/executor.ts +80 -80
  290. package/templates/default/src/tests/hooks/hooks.ts +99 -99
  291. package/templates/default/src/tests/mapping/locator-map.json +1 -1
  292. package/templates/default/src/tests/steps/actions/click.step.ts +62 -62
  293. package/templates/default/src/tests/steps/actions/navigation.step.ts +73 -72
  294. package/templates/default/src/tests/steps/validations/active_state_assertion.step.ts +34 -34
  295. package/templates/default/src/tests/steps/validations/navigation_assertion.step.ts +24 -23
  296. package/templates/default/src/tests/steps/validations/text_assertion.step.ts +111 -111
  297. package/templates/default/src/tests/steps/validations/visibility_assertion.step.ts +30 -30
  298. package/templates/default/src/tests/support/parameter-types.ts +12 -12
  299. package/templates/default/src/tests/utils/cache.util.ts +260 -260
  300. package/templates/default/src/tests/utils/cli.util.ts +177 -177
  301. package/templates/default/src/tests/utils/environment.util.ts +65 -65
  302. package/templates/default/src/tests/utils/locator.util.ts +248 -248
  303. package/templates/default/src/tests/utils/random-data.util.ts +44 -44
  304. package/templates/default/src/tests/utils/spawner.util.ts +617 -617
  305. package/templates/default/src/types/diagram/diagram.ts +34 -34
  306. package/templates/default/src/types/diagram/template-step.ts +11 -11
  307. package/templates/default/src/types/executor/browser.type.ts +1 -1
  308. package/templates/default/src/types/form/actionHandler.ts +6 -6
  309. package/templates/default/src/types/locator/locator.type.ts +11 -11
  310. package/templates/default/src/types/step/step.type.ts +1 -1
  311. package/templates/default/src/types/table/data-table.ts +6 -6
  312. package/templates/default/tailwind.config.ts +62 -62
  313. package/templates/default/.env +0 -2
  314. package/templates/default/next-env.d.ts +0 -6
  315. package/templates/default/prisma/prisma/dev.db +0 -0
  316. package/templates/default/src/tests/config/environments/environments.json +0 -14
@@ -1,402 +1,402 @@
1
- #!/usr/bin/env tsx
2
-
3
- /**
4
- * Script to synchronize locators from filesystem to database
5
- * Scans locator JSON files to ensure all locators exist in DB
6
- * Filesystem is the source of truth - locators in DB but not in FS will be deleted
7
- * Run this after merging changes to ensure locator sync
8
- *
9
- * Usage: npx tsx scripts/sync-locators.ts
10
- */
11
-
12
- import { promises as fs } from 'fs'
13
- import { join } from 'path'
14
- import { glob } from 'glob'
15
- import prisma from '../src/config/db-config'
16
- import { buildModuleHierarchy } from '../src/lib/module-hierarchy-builder'
17
- import { getLocatorGroupFilePath } from '../src/lib/locator-group-file-utils'
18
-
19
- interface SyncResult {
20
- locatorsScanned: number
21
- locatorsExisting: number
22
- locatorsCreated: number
23
- locatorsDeleted: number
24
- locatorsUpdated: number
25
- locatorGroupsDeleted: number
26
- errors: string[]
27
- createdLocators: Array<{ name: string; group: string }>
28
- deletedLocators: Array<{ name: string; group: string }>
29
- updatedLocators: Array<{ name: string; group: string }>
30
- deletedLocatorGroups: Array<{ name: string; locatorCount: number }>
31
- }
32
-
33
- /**
34
- * Scans locator directory for all JSON files
35
- */
36
- async function scanLocatorFiles(baseDir: string): Promise<string[]> {
37
- const pattern = 'src/tests/locators/**/*.json'
38
- try {
39
- const files = await glob(pattern, {
40
- cwd: baseDir,
41
- })
42
- return files.map(file => join(baseDir, file))
43
- } catch (error) {
44
- throw new Error(`Error scanning locator files: ${error}`)
45
- }
46
- }
47
-
48
- /**
49
- * Extracts module path from locator file path
50
- * Example: src/tests/locators/users/admins/directors/directors.json -> /users/admins/directors
51
- */
52
- function extractModulePathFromLocatorFile(filePath: string, baseDir: string): string {
53
- const testsDir = join(baseDir, 'src', 'tests')
54
- const relativePath = filePath.replace(testsDir, '').replace(/\\/g, '/')
55
- const pathParts = relativePath.split('/').filter(p => p && p !== 'locators')
56
- const moduleParts = pathParts.slice(0, -1) // Remove filename
57
- return moduleParts.length > 0 ? '/' + moduleParts.join('/') : '/'
58
- }
59
-
60
- /**
61
- * Extracts locator group name from file path
62
- * The group name is just the filename without extension
63
- */
64
- function extractLocatorGroupName(filePath: string): string {
65
- const fileName = filePath.split(/[/\\]/).pop() || ''
66
- return fileName.replace('.json', '')
67
- }
68
-
69
- /**
70
- * Reads and parses a locator JSON file
71
- */
72
- async function readLocatorFile(filePath: string): Promise<Record<string, string>> {
73
- try {
74
- await fs.access(filePath)
75
- } catch {
76
- throw new Error(`Locator file not found at ${filePath}`)
77
- }
78
-
79
- try {
80
- const fileContent = await fs.readFile(filePath, 'utf-8')
81
- const jsonContent = JSON.parse(fileContent) as Record<string, string>
82
-
83
- if (!jsonContent || typeof jsonContent !== 'object') {
84
- throw new Error('Invalid JSON structure: expected an object')
85
- }
86
-
87
- return jsonContent
88
- } catch (error) {
89
- if (error instanceof SyntaxError) {
90
- throw new Error(`Invalid JSON in locator file: ${error.message}`)
91
- }
92
- throw error
93
- }
94
- }
95
-
96
- /**
97
- * Finds or creates a LocatorGroup
98
- */
99
- async function findOrCreateLocatorGroup(
100
- groupName: string,
101
- moduleId: string,
102
- _modulePath: string,
103
- ): Promise<string> {
104
- // Try to find existing locator group
105
- const existingGroup = await prisma.locatorGroup.findFirst({
106
- where: {
107
- name: groupName,
108
- moduleId: moduleId,
109
- },
110
- })
111
-
112
- if (existingGroup) {
113
- return existingGroup.id
114
- }
115
-
116
- // Create new locator group
117
- const newGroup = await prisma.locatorGroup.create({
118
- data: {
119
- name: groupName,
120
- route: `/${groupName}`,
121
- moduleId: moduleId,
122
- },
123
- })
124
-
125
- return newGroup.id
126
- }
127
-
128
- /**
129
- * Syncs locators from a single file to database
130
- */
131
- async function syncLocatorsFromFile(
132
- filePath: string,
133
- baseDir: string,
134
- result: SyncResult,
135
- processedGroupIds: Set<string>,
136
- ): Promise<void> {
137
- try {
138
- // Extract module path and group name
139
- const modulePath = extractModulePathFromLocatorFile(filePath, baseDir)
140
- const groupName = extractLocatorGroupName(filePath)
141
-
142
- console.log(`\n šŸ“„ Processing file: ${filePath.replace(baseDir, '')}`)
143
- console.log(` Module path: ${modulePath}`)
144
- console.log(` Group name: ${groupName}`)
145
-
146
- // Build module hierarchy
147
- const moduleId = await buildModuleHierarchy(modulePath)
148
-
149
- // Find or create locator group
150
- const locatorGroupId = await findOrCreateLocatorGroup(groupName, moduleId, modulePath)
151
-
152
- // Track this group as processed (has a file)
153
- processedGroupIds.add(locatorGroupId)
154
-
155
- // Read locators from file
156
- const fileLocators = await readLocatorFile(filePath)
157
- const fileLocatorNames = Object.keys(fileLocators)
158
- result.locatorsScanned += fileLocatorNames.length
159
-
160
- console.log(` Found ${fileLocatorNames.length} locator(s) in file`)
161
-
162
- // Get existing locators from database for this group
163
- const dbLocators = await prisma.locator.findMany({
164
- where: { locatorGroupId: locatorGroupId },
165
- select: { id: true, name: true, value: true },
166
- })
167
-
168
- const dbLocatorMap = new Map(dbLocators.map(loc => [loc.name, loc]))
169
-
170
- // Add or update locators from file
171
- for (const [locatorName, locatorValue] of Object.entries(fileLocators)) {
172
- const existingLocator = dbLocatorMap.get(locatorName)
173
-
174
- if (existingLocator) {
175
- // Check if value changed
176
- if (existingLocator.value !== locatorValue) {
177
- // Update locator value (file takes precedence)
178
- await prisma.locator.update({
179
- where: { id: existingLocator.id },
180
- data: { value: locatorValue },
181
- })
182
- result.locatorsUpdated++
183
- result.updatedLocators.push({ name: locatorName, group: groupName })
184
- console.log(` šŸ”„ Updated locator '${locatorName}'`)
185
- } else {
186
- result.locatorsExisting++
187
- }
188
- } else {
189
- // Create new locator
190
- await prisma.locator.create({
191
- data: {
192
- name: locatorName,
193
- value: locatorValue,
194
- locatorGroupId: locatorGroupId,
195
- },
196
- })
197
- result.locatorsCreated++
198
- result.createdLocators.push({ name: locatorName, group: groupName })
199
- console.log(` āž• Created locator '${locatorName}'`)
200
- }
201
- }
202
-
203
- // Delete locators that exist in DB but not in file (FS is source of truth)
204
- for (const dbLocator of dbLocators) {
205
- if (!fileLocatorNames.includes(dbLocator.name)) {
206
- await prisma.locator.delete({
207
- where: { id: dbLocator.id },
208
- })
209
- result.locatorsDeleted++
210
- result.deletedLocators.push({ name: dbLocator.name, group: groupName })
211
- console.log(` šŸ—‘ļø Deleted locator '${dbLocator.name}' (not in file)`)
212
- }
213
- }
214
- } catch (error) {
215
- const errorMsg = `Error syncing locator file ${filePath}: ${error}`
216
- result.errors.push(errorMsg)
217
- console.error(` āŒ ${errorMsg}`)
218
- }
219
- }
220
-
221
- /**
222
- * Deletes locator groups that don't have corresponding files
223
- */
224
- async function deleteOrphanedLocatorGroups(
225
- processedGroupIds: Set<string>,
226
- baseDir: string,
227
- result: SyncResult,
228
- ): Promise<void> {
229
- console.log('\nšŸ” Checking for orphaned locator groups (no file in filesystem)...')
230
-
231
- try {
232
- // Get all locator groups from database
233
- const allLocatorGroups = await prisma.locatorGroup.findMany({
234
- include: {
235
- locators: {
236
- select: { id: true },
237
- },
238
- },
239
- })
240
-
241
- for (const group of allLocatorGroups) {
242
- // Skip if this group was processed (has a file)
243
- if (processedGroupIds.has(group.id)) {
244
- continue
245
- }
246
-
247
- // Check if file exists for this group
248
- const relativeFilePath = await getLocatorGroupFilePath(group.id)
249
- if (!relativeFilePath) {
250
- // Can't determine file path, skip
251
- continue
252
- }
253
-
254
- const fullPath = join(baseDir, relativeFilePath)
255
-
256
- try {
257
- // Check if file exists
258
- await fs.access(fullPath)
259
- // File exists, so this group is valid
260
- } catch {
261
- // File doesn't exist - delete the locator group (cascade will delete locators)
262
- const locatorCount = group.locators.length
263
- await prisma.locatorGroup.delete({
264
- where: { id: group.id },
265
- })
266
- result.locatorGroupsDeleted++
267
- result.deletedLocatorGroups.push({ name: group.name, locatorCount })
268
- console.log(` šŸ—‘ļø Deleted locator group '${group.name}' (${locatorCount} locator(s) deleted)`)
269
- }
270
- }
271
- } catch (error) {
272
- const errorMsg = `Error deleting orphaned locator groups: ${error}`
273
- result.errors.push(errorMsg)
274
- console.error(` āŒ ${errorMsg}`)
275
- }
276
- }
277
-
278
- /**
279
- * Syncs all locators to database
280
- */
281
- async function syncLocatorsToDatabase(files: string[], baseDir: string): Promise<SyncResult> {
282
- const result: SyncResult = {
283
- locatorsScanned: 0,
284
- locatorsExisting: 0,
285
- locatorsCreated: 0,
286
- locatorsDeleted: 0,
287
- locatorsUpdated: 0,
288
- locatorGroupsDeleted: 0,
289
- errors: [],
290
- createdLocators: [],
291
- deletedLocators: [],
292
- updatedLocators: [],
293
- deletedLocatorGroups: [],
294
- }
295
-
296
- // Track which locator groups have files
297
- const processedGroupIds = new Set<string>()
298
-
299
- // Process all files
300
- for (const filePath of files) {
301
- await syncLocatorsFromFile(filePath, baseDir, result, processedGroupIds)
302
- }
303
-
304
- // Delete locator groups that don't have files
305
- await deleteOrphanedLocatorGroups(processedGroupIds, baseDir, result)
306
-
307
- return result
308
- }
309
-
310
- /**
311
- * Generates and displays sync summary
312
- */
313
- function generateSummary(result: SyncResult): void {
314
- console.log('\nšŸ“Š Sync Summary:')
315
- console.log(` šŸ“ Locators scanned: ${result.locatorsScanned}`)
316
- console.log(` āœ… Locators existing: ${result.locatorsExisting}`)
317
- console.log(` āž• Locators created: ${result.locatorsCreated}`)
318
- console.log(` šŸ”„ Locators updated: ${result.locatorsUpdated}`)
319
- console.log(` šŸ—‘ļø Locators deleted: ${result.locatorsDeleted}`)
320
- console.log(` šŸ—‘ļø Locator groups deleted: ${result.locatorGroupsDeleted}`)
321
- console.log(` āŒ Errors: ${result.errors.length}`)
322
-
323
- if (result.createdLocators.length > 0) {
324
- console.log('\n Created locators:')
325
- result.createdLocators.forEach((loc, index) => {
326
- console.log(` ${index + 1}. ${loc.name} (group: ${loc.group})`)
327
- })
328
- }
329
-
330
- if (result.updatedLocators.length > 0) {
331
- console.log('\n Updated locators:')
332
- result.updatedLocators.forEach((loc, index) => {
333
- console.log(` ${index + 1}. ${loc.name} (group: ${loc.group})`)
334
- })
335
- }
336
-
337
- if (result.deletedLocators.length > 0) {
338
- console.log('\n Deleted locators:')
339
- result.deletedLocators.forEach((loc, index) => {
340
- console.log(` ${index + 1}. ${loc.name} (group: ${loc.group})`)
341
- })
342
- }
343
-
344
- if (result.deletedLocatorGroups.length > 0) {
345
- console.log('\n Deleted locator groups:')
346
- result.deletedLocatorGroups.forEach((group, index) => {
347
- console.log(` ${index + 1}. ${group.name} (${group.locatorCount} locator(s) deleted)`)
348
- })
349
- }
350
-
351
- if (result.errors.length > 0) {
352
- console.log('\n Errors:')
353
- result.errors.forEach((error, index) => {
354
- console.log(` ${index + 1}. ${error}`)
355
- })
356
- }
357
- }
358
-
359
- /**
360
- * Main function
361
- */
362
- async function main() {
363
- try {
364
- console.log('šŸ”„ Starting locators sync...')
365
- console.log('This will scan locator JSON files and sync locators to database.')
366
- console.log('Filesystem is the source of truth - locators in DB but not in FS will be deleted.\n')
367
-
368
- const baseDir = process.cwd()
369
-
370
- // Scan locator files
371
- console.log('šŸ“ Scanning src/tests/locators...')
372
- const files = await scanLocatorFiles(baseDir)
373
- console.log(` Found ${files.length} locator file(s)`)
374
-
375
- if (files.length === 0) {
376
- console.log('\nāš ļø No locator files found. Nothing to sync.')
377
- return
378
- }
379
-
380
- // Sync to database
381
- console.log('\nāœ… Syncing locators to database...')
382
- const result = await syncLocatorsToDatabase(files, baseDir)
383
-
384
- // Generate summary
385
- generateSummary(result)
386
-
387
- if (result.errors.length === 0) {
388
- console.log('\nāœ… Sync completed successfully!')
389
- } else {
390
- console.log('\nāš ļø Sync completed with errors. Please review the errors above.')
391
- process.exit(1)
392
- }
393
- } catch (error) {
394
- console.error('\nāŒ Error during sync:', error)
395
- process.exit(1)
396
- } finally {
397
- await prisma.$disconnect()
398
- }
399
- }
400
-
401
- main()
402
-
1
+ #!/usr/bin/env tsx
2
+
3
+ /**
4
+ * Script to synchronize locators from filesystem to database
5
+ * Scans locator JSON files to ensure all locators exist in DB
6
+ * Filesystem is the source of truth - locators in DB but not in FS will be deleted
7
+ * Run this after merging changes to ensure locator sync
8
+ *
9
+ * Usage: npx tsx scripts/sync-locators.ts
10
+ */
11
+
12
+ import { promises as fs } from 'fs'
13
+ import { join } from 'path'
14
+ import { glob } from 'glob'
15
+ import prisma from '../src/config/db-config'
16
+ import { buildModuleHierarchy } from '../src/lib/module-hierarchy-builder'
17
+ import { getLocatorGroupFilePath } from '../src/lib/locator-group-file-utils'
18
+
19
+ interface SyncResult {
20
+ locatorsScanned: number
21
+ locatorsExisting: number
22
+ locatorsCreated: number
23
+ locatorsDeleted: number
24
+ locatorsUpdated: number
25
+ locatorGroupsDeleted: number
26
+ errors: string[]
27
+ createdLocators: Array<{ name: string; group: string }>
28
+ deletedLocators: Array<{ name: string; group: string }>
29
+ updatedLocators: Array<{ name: string; group: string }>
30
+ deletedLocatorGroups: Array<{ name: string; locatorCount: number }>
31
+ }
32
+
33
+ /**
34
+ * Scans locator directory for all JSON files
35
+ */
36
+ async function scanLocatorFiles(baseDir: string): Promise<string[]> {
37
+ const pattern = 'src/tests/locators/**/*.json'
38
+ try {
39
+ const files = await glob(pattern, {
40
+ cwd: baseDir,
41
+ })
42
+ return files.map(file => join(baseDir, file))
43
+ } catch (error) {
44
+ throw new Error(`Error scanning locator files: ${error}`)
45
+ }
46
+ }
47
+
48
+ /**
49
+ * Extracts module path from locator file path
50
+ * Example: src/tests/locators/users/admins/directors/directors.json -> /users/admins/directors
51
+ */
52
+ function extractModulePathFromLocatorFile(filePath: string, baseDir: string): string {
53
+ const testsDir = join(baseDir, 'src', 'tests')
54
+ const relativePath = filePath.replace(testsDir, '').replace(/\\/g, '/')
55
+ const pathParts = relativePath.split('/').filter(p => p && p !== 'locators')
56
+ const moduleParts = pathParts.slice(0, -1) // Remove filename
57
+ return moduleParts.length > 0 ? '/' + moduleParts.join('/') : '/'
58
+ }
59
+
60
+ /**
61
+ * Extracts locator group name from file path
62
+ * The group name is just the filename without extension
63
+ */
64
+ function extractLocatorGroupName(filePath: string): string {
65
+ const fileName = filePath.split(/[/\\]/).pop() || ''
66
+ return fileName.replace('.json', '')
67
+ }
68
+
69
+ /**
70
+ * Reads and parses a locator JSON file
71
+ */
72
+ async function readLocatorFile(filePath: string): Promise<Record<string, string>> {
73
+ try {
74
+ await fs.access(filePath)
75
+ } catch {
76
+ throw new Error(`Locator file not found at ${filePath}`)
77
+ }
78
+
79
+ try {
80
+ const fileContent = await fs.readFile(filePath, 'utf-8')
81
+ const jsonContent = JSON.parse(fileContent) as Record<string, string>
82
+
83
+ if (!jsonContent || typeof jsonContent !== 'object') {
84
+ throw new Error('Invalid JSON structure: expected an object')
85
+ }
86
+
87
+ return jsonContent
88
+ } catch (error) {
89
+ if (error instanceof SyntaxError) {
90
+ throw new Error(`Invalid JSON in locator file: ${error.message}`)
91
+ }
92
+ throw error
93
+ }
94
+ }
95
+
96
+ /**
97
+ * Finds or creates a LocatorGroup
98
+ */
99
+ async function findOrCreateLocatorGroup(
100
+ groupName: string,
101
+ moduleId: string,
102
+ _modulePath: string,
103
+ ): Promise<string> {
104
+ // Try to find existing locator group
105
+ const existingGroup = await prisma.locatorGroup.findFirst({
106
+ where: {
107
+ name: groupName,
108
+ moduleId: moduleId,
109
+ },
110
+ })
111
+
112
+ if (existingGroup) {
113
+ return existingGroup.id
114
+ }
115
+
116
+ // Create new locator group
117
+ const newGroup = await prisma.locatorGroup.create({
118
+ data: {
119
+ name: groupName,
120
+ route: `/${groupName}`,
121
+ moduleId: moduleId,
122
+ },
123
+ })
124
+
125
+ return newGroup.id
126
+ }
127
+
128
+ /**
129
+ * Syncs locators from a single file to database
130
+ */
131
+ async function syncLocatorsFromFile(
132
+ filePath: string,
133
+ baseDir: string,
134
+ result: SyncResult,
135
+ processedGroupIds: Set<string>,
136
+ ): Promise<void> {
137
+ try {
138
+ // Extract module path and group name
139
+ const modulePath = extractModulePathFromLocatorFile(filePath, baseDir)
140
+ const groupName = extractLocatorGroupName(filePath)
141
+
142
+ console.log(`\n šŸ“„ Processing file: ${filePath.replace(baseDir, '')}`)
143
+ console.log(` Module path: ${modulePath}`)
144
+ console.log(` Group name: ${groupName}`)
145
+
146
+ // Build module hierarchy
147
+ const moduleId = await buildModuleHierarchy(modulePath)
148
+
149
+ // Find or create locator group
150
+ const locatorGroupId = await findOrCreateLocatorGroup(groupName, moduleId, modulePath)
151
+
152
+ // Track this group as processed (has a file)
153
+ processedGroupIds.add(locatorGroupId)
154
+
155
+ // Read locators from file
156
+ const fileLocators = await readLocatorFile(filePath)
157
+ const fileLocatorNames = Object.keys(fileLocators)
158
+ result.locatorsScanned += fileLocatorNames.length
159
+
160
+ console.log(` Found ${fileLocatorNames.length} locator(s) in file`)
161
+
162
+ // Get existing locators from database for this group
163
+ const dbLocators = await prisma.locator.findMany({
164
+ where: { locatorGroupId: locatorGroupId },
165
+ select: { id: true, name: true, value: true },
166
+ })
167
+
168
+ const dbLocatorMap = new Map(dbLocators.map(loc => [loc.name, loc]))
169
+
170
+ // Add or update locators from file
171
+ for (const [locatorName, locatorValue] of Object.entries(fileLocators)) {
172
+ const existingLocator = dbLocatorMap.get(locatorName)
173
+
174
+ if (existingLocator) {
175
+ // Check if value changed
176
+ if (existingLocator.value !== locatorValue) {
177
+ // Update locator value (file takes precedence)
178
+ await prisma.locator.update({
179
+ where: { id: existingLocator.id },
180
+ data: { value: locatorValue },
181
+ })
182
+ result.locatorsUpdated++
183
+ result.updatedLocators.push({ name: locatorName, group: groupName })
184
+ console.log(` šŸ”„ Updated locator '${locatorName}'`)
185
+ } else {
186
+ result.locatorsExisting++
187
+ }
188
+ } else {
189
+ // Create new locator
190
+ await prisma.locator.create({
191
+ data: {
192
+ name: locatorName,
193
+ value: locatorValue,
194
+ locatorGroupId: locatorGroupId,
195
+ },
196
+ })
197
+ result.locatorsCreated++
198
+ result.createdLocators.push({ name: locatorName, group: groupName })
199
+ console.log(` āž• Created locator '${locatorName}'`)
200
+ }
201
+ }
202
+
203
+ // Delete locators that exist in DB but not in file (FS is source of truth)
204
+ for (const dbLocator of dbLocators) {
205
+ if (!fileLocatorNames.includes(dbLocator.name)) {
206
+ await prisma.locator.delete({
207
+ where: { id: dbLocator.id },
208
+ })
209
+ result.locatorsDeleted++
210
+ result.deletedLocators.push({ name: dbLocator.name, group: groupName })
211
+ console.log(` šŸ—‘ļø Deleted locator '${dbLocator.name}' (not in file)`)
212
+ }
213
+ }
214
+ } catch (error) {
215
+ const errorMsg = `Error syncing locator file ${filePath}: ${error}`
216
+ result.errors.push(errorMsg)
217
+ console.error(` āŒ ${errorMsg}`)
218
+ }
219
+ }
220
+
221
+ /**
222
+ * Deletes locator groups that don't have corresponding files
223
+ */
224
+ async function deleteOrphanedLocatorGroups(
225
+ processedGroupIds: Set<string>,
226
+ baseDir: string,
227
+ result: SyncResult,
228
+ ): Promise<void> {
229
+ console.log('\nšŸ” Checking for orphaned locator groups (no file in filesystem)...')
230
+
231
+ try {
232
+ // Get all locator groups from database
233
+ const allLocatorGroups = await prisma.locatorGroup.findMany({
234
+ include: {
235
+ locators: {
236
+ select: { id: true },
237
+ },
238
+ },
239
+ })
240
+
241
+ for (const group of allLocatorGroups) {
242
+ // Skip if this group was processed (has a file)
243
+ if (processedGroupIds.has(group.id)) {
244
+ continue
245
+ }
246
+
247
+ // Check if file exists for this group
248
+ const relativeFilePath = await getLocatorGroupFilePath(group.id)
249
+ if (!relativeFilePath) {
250
+ // Can't determine file path, skip
251
+ continue
252
+ }
253
+
254
+ const fullPath = join(baseDir, relativeFilePath)
255
+
256
+ try {
257
+ // Check if file exists
258
+ await fs.access(fullPath)
259
+ // File exists, so this group is valid
260
+ } catch {
261
+ // File doesn't exist - delete the locator group (cascade will delete locators)
262
+ const locatorCount = group.locators.length
263
+ await prisma.locatorGroup.delete({
264
+ where: { id: group.id },
265
+ })
266
+ result.locatorGroupsDeleted++
267
+ result.deletedLocatorGroups.push({ name: group.name, locatorCount })
268
+ console.log(` šŸ—‘ļø Deleted locator group '${group.name}' (${locatorCount} locator(s) deleted)`)
269
+ }
270
+ }
271
+ } catch (error) {
272
+ const errorMsg = `Error deleting orphaned locator groups: ${error}`
273
+ result.errors.push(errorMsg)
274
+ console.error(` āŒ ${errorMsg}`)
275
+ }
276
+ }
277
+
278
+ /**
279
+ * Syncs all locators to database
280
+ */
281
+ async function syncLocatorsToDatabase(files: string[], baseDir: string): Promise<SyncResult> {
282
+ const result: SyncResult = {
283
+ locatorsScanned: 0,
284
+ locatorsExisting: 0,
285
+ locatorsCreated: 0,
286
+ locatorsDeleted: 0,
287
+ locatorsUpdated: 0,
288
+ locatorGroupsDeleted: 0,
289
+ errors: [],
290
+ createdLocators: [],
291
+ deletedLocators: [],
292
+ updatedLocators: [],
293
+ deletedLocatorGroups: [],
294
+ }
295
+
296
+ // Track which locator groups have files
297
+ const processedGroupIds = new Set<string>()
298
+
299
+ // Process all files
300
+ for (const filePath of files) {
301
+ await syncLocatorsFromFile(filePath, baseDir, result, processedGroupIds)
302
+ }
303
+
304
+ // Delete locator groups that don't have files
305
+ await deleteOrphanedLocatorGroups(processedGroupIds, baseDir, result)
306
+
307
+ return result
308
+ }
309
+
310
+ /**
311
+ * Generates and displays sync summary
312
+ */
313
+ function generateSummary(result: SyncResult): void {
314
+ console.log('\nšŸ“Š Sync Summary:')
315
+ console.log(` šŸ“ Locators scanned: ${result.locatorsScanned}`)
316
+ console.log(` āœ… Locators existing: ${result.locatorsExisting}`)
317
+ console.log(` āž• Locators created: ${result.locatorsCreated}`)
318
+ console.log(` šŸ”„ Locators updated: ${result.locatorsUpdated}`)
319
+ console.log(` šŸ—‘ļø Locators deleted: ${result.locatorsDeleted}`)
320
+ console.log(` šŸ—‘ļø Locator groups deleted: ${result.locatorGroupsDeleted}`)
321
+ console.log(` āŒ Errors: ${result.errors.length}`)
322
+
323
+ if (result.createdLocators.length > 0) {
324
+ console.log('\n Created locators:')
325
+ result.createdLocators.forEach((loc, index) => {
326
+ console.log(` ${index + 1}. ${loc.name} (group: ${loc.group})`)
327
+ })
328
+ }
329
+
330
+ if (result.updatedLocators.length > 0) {
331
+ console.log('\n Updated locators:')
332
+ result.updatedLocators.forEach((loc, index) => {
333
+ console.log(` ${index + 1}. ${loc.name} (group: ${loc.group})`)
334
+ })
335
+ }
336
+
337
+ if (result.deletedLocators.length > 0) {
338
+ console.log('\n Deleted locators:')
339
+ result.deletedLocators.forEach((loc, index) => {
340
+ console.log(` ${index + 1}. ${loc.name} (group: ${loc.group})`)
341
+ })
342
+ }
343
+
344
+ if (result.deletedLocatorGroups.length > 0) {
345
+ console.log('\n Deleted locator groups:')
346
+ result.deletedLocatorGroups.forEach((group, index) => {
347
+ console.log(` ${index + 1}. ${group.name} (${group.locatorCount} locator(s) deleted)`)
348
+ })
349
+ }
350
+
351
+ if (result.errors.length > 0) {
352
+ console.log('\n Errors:')
353
+ result.errors.forEach((error, index) => {
354
+ console.log(` ${index + 1}. ${error}`)
355
+ })
356
+ }
357
+ }
358
+
359
+ /**
360
+ * Main function
361
+ */
362
+ async function main() {
363
+ try {
364
+ console.log('šŸ”„ Starting locators sync...')
365
+ console.log('This will scan locator JSON files and sync locators to database.')
366
+ console.log('Filesystem is the source of truth - locators in DB but not in FS will be deleted.\n')
367
+
368
+ const baseDir = process.cwd()
369
+
370
+ // Scan locator files
371
+ console.log('šŸ“ Scanning src/tests/locators...')
372
+ const files = await scanLocatorFiles(baseDir)
373
+ console.log(` Found ${files.length} locator file(s)`)
374
+
375
+ if (files.length === 0) {
376
+ console.log('\nāš ļø No locator files found. Nothing to sync.')
377
+ return
378
+ }
379
+
380
+ // Sync to database
381
+ console.log('\nāœ… Syncing locators to database...')
382
+ const result = await syncLocatorsToDatabase(files, baseDir)
383
+
384
+ // Generate summary
385
+ generateSummary(result)
386
+
387
+ if (result.errors.length === 0) {
388
+ console.log('\nāœ… Sync completed successfully!')
389
+ } else {
390
+ console.log('\nāš ļø Sync completed with errors. Please review the errors above.')
391
+ process.exit(1)
392
+ }
393
+ } catch (error) {
394
+ console.error('\nāŒ Error during sync:', error)
395
+ process.exit(1)
396
+ } finally {
397
+ await prisma.$disconnect()
398
+ }
399
+ }
400
+
401
+ main()
402
+