@codeyam/codeyam-cli 0.1.7 → 0.1.9

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 (576) hide show
  1. package/analyzer-template/.build-info.json +8 -8
  2. package/analyzer-template/log.txt +3 -3
  3. package/analyzer-template/package.json +9 -9
  4. package/analyzer-template/packages/ai/package.json +1 -1
  5. package/analyzer-template/packages/ai/src/lib/astScopes/astScopeAnalyzer.ts +34 -3
  6. package/analyzer-template/packages/ai/src/lib/completionCall.ts +14 -2
  7. package/analyzer-template/packages/ai/src/lib/dataStructure/ScopeDataStructure.ts +27 -0
  8. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.ts +62 -0
  9. package/analyzer-template/packages/ai/src/lib/generateEntityScenarioData.ts +78 -2
  10. package/analyzer-template/packages/ai/src/lib/generateExecutionFlows.ts +0 -33
  11. package/analyzer-template/packages/analyze/src/lib/ProjectAnalyzer.ts +19 -7
  12. package/analyzer-template/packages/analyze/src/lib/asts/index.ts +7 -2
  13. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities.ts +9 -1
  14. package/analyzer-template/packages/analyze/src/lib/files/analyze/dependencyResolver.ts +0 -6
  15. package/analyzer-template/packages/analyze/src/lib/files/analyze/findOrCreateEntity.ts +12 -0
  16. package/analyzer-template/packages/analyze/src/lib/files/scenarios/TransformationTracer.ts +65 -28
  17. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateDataStructure.ts +83 -0
  18. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.ts +0 -98
  19. package/analyzer-template/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.ts +23 -4
  20. package/analyzer-template/packages/aws/package.json +2 -2
  21. package/analyzer-template/packages/database/index.ts +1 -0
  22. package/analyzer-template/packages/database/package.json +3 -3
  23. package/analyzer-template/packages/database/src/lib/kysely/db.ts +8 -0
  24. package/analyzer-template/packages/database/src/lib/kysely/tables/editorScenariosTable.ts +93 -0
  25. package/analyzer-template/packages/database/src/lib/loadCommits.ts +31 -20
  26. package/analyzer-template/packages/database/src/lib/loadEntities.ts +0 -6
  27. package/analyzer-template/packages/database/src/lib/loadReadyToBeCapturedAnalyses.ts +0 -5
  28. package/analyzer-template/packages/database/src/lib/updateCommitMetadata.ts +94 -143
  29. package/analyzer-template/packages/database/src/lib/updateFreshAnalysisStatus.ts +58 -42
  30. package/analyzer-template/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.ts +81 -65
  31. package/analyzer-template/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.ts +29 -1
  32. package/analyzer-template/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.ts +33 -5
  33. package/analyzer-template/packages/github/dist/database/index.d.ts +1 -0
  34. package/analyzer-template/packages/github/dist/database/index.d.ts.map +1 -1
  35. package/analyzer-template/packages/github/dist/database/index.js +1 -0
  36. package/analyzer-template/packages/github/dist/database/index.js.map +1 -1
  37. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts +2 -0
  38. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts.map +1 -1
  39. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js +5 -0
  40. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js.map +1 -1
  41. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts +25 -0
  42. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts.map +1 -0
  43. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js +76 -0
  44. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -0
  45. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/scenariosTable.d.ts +5 -0
  46. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/scenariosTable.d.ts.map +1 -1
  47. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.d.ts.map +1 -1
  48. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.js +23 -13
  49. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.js.map +1 -1
  50. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.d.ts.map +1 -1
  51. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.js +0 -6
  52. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.js.map +1 -1
  53. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.d.ts.map +1 -1
  54. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.js +1 -4
  55. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.js.map +1 -1
  56. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.d.ts.map +1 -1
  57. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.js +76 -90
  58. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.js.map +1 -1
  59. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.d.ts.map +1 -1
  60. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.js +41 -30
  61. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.js.map +1 -1
  62. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.d.ts.map +1 -1
  63. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.js +68 -57
  64. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.js.map +1 -1
  65. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.d.ts.map +1 -1
  66. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js +29 -1
  67. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js.map +1 -1
  68. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.d.ts.map +1 -1
  69. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js +33 -5
  70. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js.map +1 -1
  71. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.d.ts +2 -0
  72. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.d.ts.map +1 -1
  73. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.js +2 -0
  74. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.js.map +1 -1
  75. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts +1 -0
  76. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  77. package/analyzer-template/packages/github/dist/types/src/types/Scenario.d.ts +10 -0
  78. package/analyzer-template/packages/github/dist/types/src/types/Scenario.d.ts.map +1 -1
  79. package/analyzer-template/packages/github/package.json +1 -1
  80. package/analyzer-template/packages/types/src/enums/ProjectFramework.ts +2 -0
  81. package/analyzer-template/packages/types/src/types/ProjectMetadata.ts +1 -0
  82. package/analyzer-template/packages/types/src/types/Scenario.ts +10 -0
  83. package/analyzer-template/packages/ui-components/package.json +1 -1
  84. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.d.ts +2 -0
  85. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.d.ts.map +1 -1
  86. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.js +2 -0
  87. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.js.map +1 -1
  88. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts +1 -0
  89. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  90. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.d.ts +10 -0
  91. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.d.ts.map +1 -1
  92. package/analyzer-template/playwright/captureFromUrl.ts +89 -82
  93. package/analyzer-template/project/constructMockCode.ts +136 -43
  94. package/analyzer-template/project/reconcileMockDataKeys.ts +19 -14
  95. package/analyzer-template/project/start.ts +3 -0
  96. package/analyzer-template/project/startScenarioCapture.ts +9 -0
  97. package/analyzer-template/project/writeClientLogRoute.ts +125 -0
  98. package/analyzer-template/project/writeMockDataTsx.ts +17 -0
  99. package/analyzer-template/project/writeScenarioComponents.ts +36 -7
  100. package/analyzer-template/tsconfig.json +13 -1
  101. package/background/src/lib/virtualized/project/constructMockCode.js +115 -34
  102. package/background/src/lib/virtualized/project/constructMockCode.js.map +1 -1
  103. package/background/src/lib/virtualized/project/reconcileMockDataKeys.js +17 -11
  104. package/background/src/lib/virtualized/project/reconcileMockDataKeys.js.map +1 -1
  105. package/background/src/lib/virtualized/project/start.js +2 -0
  106. package/background/src/lib/virtualized/project/start.js.map +1 -1
  107. package/background/src/lib/virtualized/project/startScenarioCapture.js +5 -0
  108. package/background/src/lib/virtualized/project/startScenarioCapture.js.map +1 -1
  109. package/background/src/lib/virtualized/project/writeClientLogRoute.js +110 -0
  110. package/background/src/lib/virtualized/project/writeClientLogRoute.js.map +1 -0
  111. package/background/src/lib/virtualized/project/writeMockDataTsx.js +12 -0
  112. package/background/src/lib/virtualized/project/writeMockDataTsx.js.map +1 -1
  113. package/background/src/lib/virtualized/project/writeScenarioComponents.js +29 -7
  114. package/background/src/lib/virtualized/project/writeScenarioComponents.js.map +1 -1
  115. package/codeyam-cli/scripts/apply-setup.js +208 -11
  116. package/codeyam-cli/scripts/apply-setup.js.map +1 -1
  117. package/codeyam-cli/src/__tests__/memory-scripts/filter-session.test.js +196 -0
  118. package/codeyam-cli/src/__tests__/memory-scripts/filter-session.test.js.map +1 -0
  119. package/codeyam-cli/src/__tests__/memory-scripts/read-json-field.test.js +114 -0
  120. package/codeyam-cli/src/__tests__/memory-scripts/read-json-field.test.js.map +1 -0
  121. package/codeyam-cli/src/__tests__/memory-scripts/ripgrep-fallback.test.js +149 -0
  122. package/codeyam-cli/src/__tests__/memory-scripts/ripgrep-fallback.test.js.map +1 -0
  123. package/codeyam-cli/src/cli.js +2 -0
  124. package/codeyam-cli/src/cli.js.map +1 -1
  125. package/codeyam-cli/src/commands/__tests__/editor.stepDispatch.test.js +45 -0
  126. package/codeyam-cli/src/commands/__tests__/editor.stepDispatch.test.js.map +1 -0
  127. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js +101 -47
  128. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js.map +1 -1
  129. package/codeyam-cli/src/commands/analyze.js +17 -7
  130. package/codeyam-cli/src/commands/analyze.js.map +1 -1
  131. package/codeyam-cli/src/commands/default.js +14 -2
  132. package/codeyam-cli/src/commands/default.js.map +1 -1
  133. package/codeyam-cli/src/commands/editor.js +3215 -0
  134. package/codeyam-cli/src/commands/editor.js.map +1 -0
  135. package/codeyam-cli/src/commands/init.js +107 -45
  136. package/codeyam-cli/src/commands/init.js.map +1 -1
  137. package/codeyam-cli/src/data/techStacks.js +77 -0
  138. package/codeyam-cli/src/data/techStacks.js.map +1 -0
  139. package/codeyam-cli/src/utils/__tests__/analyzerFinalization.test.js +144 -0
  140. package/codeyam-cli/src/utils/__tests__/analyzerFinalization.test.js.map +1 -0
  141. package/codeyam-cli/src/utils/__tests__/backgroundServer.test.js +46 -0
  142. package/codeyam-cli/src/utils/__tests__/backgroundServer.test.js.map +1 -0
  143. package/codeyam-cli/src/utils/__tests__/devServerState.test.js +134 -0
  144. package/codeyam-cli/src/utils/__tests__/devServerState.test.js.map +1 -0
  145. package/codeyam-cli/src/utils/__tests__/editorApi.test.js +127 -0
  146. package/codeyam-cli/src/utils/__tests__/editorApi.test.js.map +1 -0
  147. package/codeyam-cli/src/utils/__tests__/editorAudit.test.js +855 -0
  148. package/codeyam-cli/src/utils/__tests__/editorAudit.test.js.map +1 -0
  149. package/codeyam-cli/src/utils/__tests__/editorCapture.test.js +93 -0
  150. package/codeyam-cli/src/utils/__tests__/editorCapture.test.js.map +1 -0
  151. package/codeyam-cli/src/utils/__tests__/editorDevServer.test.js +304 -0
  152. package/codeyam-cli/src/utils/__tests__/editorDevServer.test.js.map +1 -0
  153. package/codeyam-cli/src/utils/__tests__/editorEntityChangeStatus.test.js +121 -0
  154. package/codeyam-cli/src/utils/__tests__/editorEntityChangeStatus.test.js.map +1 -0
  155. package/codeyam-cli/src/utils/__tests__/editorImageVerifier.test.js +294 -0
  156. package/codeyam-cli/src/utils/__tests__/editorImageVerifier.test.js.map +1 -0
  157. package/codeyam-cli/src/utils/__tests__/editorJournal.test.js +542 -0
  158. package/codeyam-cli/src/utils/__tests__/editorJournal.test.js.map +1 -0
  159. package/codeyam-cli/src/utils/__tests__/editorLoaderHelpers.test.js +520 -0
  160. package/codeyam-cli/src/utils/__tests__/editorLoaderHelpers.test.js.map +1 -0
  161. package/codeyam-cli/src/utils/__tests__/editorMockState.test.js +270 -0
  162. package/codeyam-cli/src/utils/__tests__/editorMockState.test.js.map +1 -0
  163. package/codeyam-cli/src/utils/__tests__/editorPreloadHelpers.test.js +217 -0
  164. package/codeyam-cli/src/utils/__tests__/editorPreloadHelpers.test.js.map +1 -0
  165. package/codeyam-cli/src/utils/__tests__/editorPreview.test.js +339 -0
  166. package/codeyam-cli/src/utils/__tests__/editorPreview.test.js.map +1 -0
  167. package/codeyam-cli/src/utils/__tests__/editorProxySession.test.js +153 -0
  168. package/codeyam-cli/src/utils/__tests__/editorProxySession.test.js.map +1 -0
  169. package/codeyam-cli/src/utils/__tests__/editorScenarioLookup.test.js +139 -0
  170. package/codeyam-cli/src/utils/__tests__/editorScenarioLookup.test.js.map +1 -0
  171. package/codeyam-cli/src/utils/__tests__/editorScenarioSwitch.test.js +221 -0
  172. package/codeyam-cli/src/utils/__tests__/editorScenarioSwitch.test.js.map +1 -0
  173. package/codeyam-cli/src/utils/__tests__/editorScenarios.test.js +855 -0
  174. package/codeyam-cli/src/utils/__tests__/editorScenarios.test.js.map +1 -0
  175. package/codeyam-cli/src/utils/__tests__/editorSeedAdapter.test.js +213 -0
  176. package/codeyam-cli/src/utils/__tests__/editorSeedAdapter.test.js.map +1 -0
  177. package/codeyam-cli/src/utils/__tests__/entityChangeStatus.test.js +1742 -0
  178. package/codeyam-cli/src/utils/__tests__/entityChangeStatus.test.js.map +1 -0
  179. package/codeyam-cli/src/utils/__tests__/git.editor.test.js +134 -0
  180. package/codeyam-cli/src/utils/__tests__/git.editor.test.js.map +1 -0
  181. package/codeyam-cli/src/utils/__tests__/journalCaptureStabilization.test.js +107 -0
  182. package/codeyam-cli/src/utils/__tests__/journalCaptureStabilization.test.js.map +1 -0
  183. package/codeyam-cli/src/utils/__tests__/parseRegisterArg.test.js +101 -0
  184. package/codeyam-cli/src/utils/__tests__/parseRegisterArg.test.js.map +1 -0
  185. package/codeyam-cli/src/utils/__tests__/pathIgnoring.test.js +9 -0
  186. package/codeyam-cli/src/utils/__tests__/pathIgnoring.test.js.map +1 -1
  187. package/codeyam-cli/src/utils/__tests__/project.test.js +65 -0
  188. package/codeyam-cli/src/utils/__tests__/project.test.js.map +1 -0
  189. package/codeyam-cli/src/utils/__tests__/scenarioCoverage.test.js +227 -0
  190. package/codeyam-cli/src/utils/__tests__/scenarioCoverage.test.js.map +1 -0
  191. package/codeyam-cli/src/utils/__tests__/scenarioMarkers.test.js +121 -0
  192. package/codeyam-cli/src/utils/__tests__/scenarioMarkers.test.js.map +1 -0
  193. package/codeyam-cli/src/utils/__tests__/scenariosManifest.test.js +300 -0
  194. package/codeyam-cli/src/utils/__tests__/scenariosManifest.test.js.map +1 -0
  195. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js +50 -4
  196. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js.map +1 -1
  197. package/codeyam-cli/src/utils/__tests__/templateConsistency.test.js +51 -0
  198. package/codeyam-cli/src/utils/__tests__/templateConsistency.test.js.map +1 -0
  199. package/codeyam-cli/src/utils/__tests__/webappDetection.test.js +142 -0
  200. package/codeyam-cli/src/utils/__tests__/webappDetection.test.js.map +1 -0
  201. package/codeyam-cli/src/utils/analyzer.js +9 -0
  202. package/codeyam-cli/src/utils/analyzer.js.map +1 -1
  203. package/codeyam-cli/src/utils/analyzerFinalization.js +96 -0
  204. package/codeyam-cli/src/utils/analyzerFinalization.js.map +1 -0
  205. package/codeyam-cli/src/utils/backgroundServer.js +104 -12
  206. package/codeyam-cli/src/utils/backgroundServer.js.map +1 -1
  207. package/codeyam-cli/src/utils/buildFlags.js +4 -0
  208. package/codeyam-cli/src/utils/buildFlags.js.map +1 -0
  209. package/codeyam-cli/src/utils/database.js +37 -2
  210. package/codeyam-cli/src/utils/database.js.map +1 -1
  211. package/codeyam-cli/src/utils/devModeEvents.js +40 -0
  212. package/codeyam-cli/src/utils/devModeEvents.js.map +1 -0
  213. package/codeyam-cli/src/utils/devServerState.js +71 -0
  214. package/codeyam-cli/src/utils/devServerState.js.map +1 -0
  215. package/codeyam-cli/src/utils/editorApi.js +73 -0
  216. package/codeyam-cli/src/utils/editorApi.js.map +1 -0
  217. package/codeyam-cli/src/utils/editorAudit.js +176 -0
  218. package/codeyam-cli/src/utils/editorAudit.js.map +1 -0
  219. package/codeyam-cli/src/utils/editorCapture.js +102 -0
  220. package/codeyam-cli/src/utils/editorCapture.js.map +1 -0
  221. package/codeyam-cli/src/utils/editorDevServer.js +197 -0
  222. package/codeyam-cli/src/utils/editorDevServer.js.map +1 -0
  223. package/codeyam-cli/src/utils/editorEntityChangeStatus.js +44 -0
  224. package/codeyam-cli/src/utils/editorEntityChangeStatus.js.map +1 -0
  225. package/codeyam-cli/src/utils/editorImageVerifier.js +155 -0
  226. package/codeyam-cli/src/utils/editorImageVerifier.js.map +1 -0
  227. package/codeyam-cli/src/utils/editorJournal.js +225 -0
  228. package/codeyam-cli/src/utils/editorJournal.js.map +1 -0
  229. package/codeyam-cli/src/utils/editorLoaderHelpers.js +113 -0
  230. package/codeyam-cli/src/utils/editorLoaderHelpers.js.map +1 -0
  231. package/codeyam-cli/src/utils/editorMockState.js +248 -0
  232. package/codeyam-cli/src/utils/editorMockState.js.map +1 -0
  233. package/codeyam-cli/src/utils/editorPreloadHelpers.js +135 -0
  234. package/codeyam-cli/src/utils/editorPreloadHelpers.js.map +1 -0
  235. package/codeyam-cli/src/utils/editorPreview.js +132 -0
  236. package/codeyam-cli/src/utils/editorPreview.js.map +1 -0
  237. package/codeyam-cli/src/utils/editorScenarioSwitch.js +112 -0
  238. package/codeyam-cli/src/utils/editorScenarioSwitch.js.map +1 -0
  239. package/codeyam-cli/src/utils/editorScenarios.js +332 -0
  240. package/codeyam-cli/src/utils/editorScenarios.js.map +1 -0
  241. package/codeyam-cli/src/utils/editorSeedAdapter.js +173 -0
  242. package/codeyam-cli/src/utils/editorSeedAdapter.js.map +1 -0
  243. package/codeyam-cli/src/utils/entityChangeStatus.js +349 -0
  244. package/codeyam-cli/src/utils/entityChangeStatus.js.map +1 -0
  245. package/codeyam-cli/src/utils/entityChangeStatus.server.js +158 -0
  246. package/codeyam-cli/src/utils/entityChangeStatus.server.js.map +1 -0
  247. package/codeyam-cli/src/utils/fileMetadata.js +5 -0
  248. package/codeyam-cli/src/utils/fileMetadata.js.map +1 -1
  249. package/codeyam-cli/src/utils/fileWatcher.js +25 -9
  250. package/codeyam-cli/src/utils/fileWatcher.js.map +1 -1
  251. package/codeyam-cli/src/utils/git.js +103 -0
  252. package/codeyam-cli/src/utils/git.js.map +1 -1
  253. package/codeyam-cli/src/utils/install-skills.js +55 -13
  254. package/codeyam-cli/src/utils/install-skills.js.map +1 -1
  255. package/codeyam-cli/src/utils/interactiveSyncWatcher.js +126 -0
  256. package/codeyam-cli/src/utils/interactiveSyncWatcher.js.map +1 -0
  257. package/codeyam-cli/src/utils/parseRegisterArg.js +31 -0
  258. package/codeyam-cli/src/utils/parseRegisterArg.js.map +1 -0
  259. package/codeyam-cli/src/utils/pathIgnoring.js +19 -7
  260. package/codeyam-cli/src/utils/pathIgnoring.js.map +1 -1
  261. package/codeyam-cli/src/utils/project.js +15 -5
  262. package/codeyam-cli/src/utils/project.js.map +1 -1
  263. package/codeyam-cli/src/utils/queue/__tests__/heartbeat.test.js +11 -11
  264. package/codeyam-cli/src/utils/queue/__tests__/heartbeat.test.js.map +1 -1
  265. package/codeyam-cli/src/utils/queue/__tests__/manager.test.js +22 -0
  266. package/codeyam-cli/src/utils/queue/__tests__/manager.test.js.map +1 -1
  267. package/codeyam-cli/src/utils/queue/heartbeat.js +13 -5
  268. package/codeyam-cli/src/utils/queue/heartbeat.js.map +1 -1
  269. package/codeyam-cli/src/utils/queue/job.js +70 -1
  270. package/codeyam-cli/src/utils/queue/job.js.map +1 -1
  271. package/codeyam-cli/src/utils/queue/manager.js +7 -6
  272. package/codeyam-cli/src/utils/queue/manager.js.map +1 -1
  273. package/codeyam-cli/src/utils/scenarioCoverage.js +75 -0
  274. package/codeyam-cli/src/utils/scenarioCoverage.js.map +1 -0
  275. package/codeyam-cli/src/utils/scenarioMarkers.js +134 -0
  276. package/codeyam-cli/src/utils/scenarioMarkers.js.map +1 -0
  277. package/codeyam-cli/src/utils/scenariosManifest.js +159 -0
  278. package/codeyam-cli/src/utils/scenariosManifest.js.map +1 -0
  279. package/codeyam-cli/src/utils/serverState.js +57 -2
  280. package/codeyam-cli/src/utils/serverState.js.map +1 -1
  281. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js +82 -11
  282. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js.map +1 -1
  283. package/codeyam-cli/src/utils/simulationGateMiddleware.js +8 -1
  284. package/codeyam-cli/src/utils/simulationGateMiddleware.js.map +1 -1
  285. package/codeyam-cli/src/utils/slugUtils.js +25 -0
  286. package/codeyam-cli/src/utils/slugUtils.js.map +1 -0
  287. package/codeyam-cli/src/utils/syncMocksMiddleware.js +2 -2
  288. package/codeyam-cli/src/utils/syncMocksMiddleware.js.map +1 -1
  289. package/codeyam-cli/src/utils/testRunner.js +158 -0
  290. package/codeyam-cli/src/utils/testRunner.js.map +1 -0
  291. package/codeyam-cli/src/utils/webappDetection.js +35 -2
  292. package/codeyam-cli/src/utils/webappDetection.js.map +1 -1
  293. package/codeyam-cli/src/webserver/__tests__/clientErrors.test.js +40 -0
  294. package/codeyam-cli/src/webserver/__tests__/clientErrors.test.js.map +1 -0
  295. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js +567 -0
  296. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js.map +1 -0
  297. package/codeyam-cli/src/webserver/app/lib/clientErrors.js +65 -0
  298. package/codeyam-cli/src/webserver/app/lib/clientErrors.js.map +1 -0
  299. package/codeyam-cli/src/webserver/app/lib/database.js +41 -27
  300. package/codeyam-cli/src/webserver/app/lib/database.js.map +1 -1
  301. package/codeyam-cli/src/webserver/app/lib/dbNotifier.js.map +1 -1
  302. package/codeyam-cli/src/webserver/app/lib/git.js +397 -0
  303. package/codeyam-cli/src/webserver/app/lib/git.js.map +1 -0
  304. package/codeyam-cli/src/webserver/backgroundServer.js +108 -18
  305. package/codeyam-cli/src/webserver/backgroundServer.js.map +1 -1
  306. package/codeyam-cli/src/webserver/build/client/assets/{CopyButton-CtmbP4Gl.js → CopyButton-BPXZwM4t.js} +1 -1
  307. package/codeyam-cli/src/webserver/build/client/assets/{EntityItem-DlMph_Hm.js → EntityItem-BcgbViKV.js} +3 -3
  308. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeBadge-B-0PjGOU.js → EntityTypeBadge-g3saevPb.js} +1 -1
  309. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeIcon-DN9eiJAO.js → EntityTypeIcon-CQIG2qda.js} +9 -9
  310. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-Bu6c6aDe.js +1 -0
  311. package/codeyam-cli/src/webserver/build/client/assets/{InteractivePreview-rE_fI2h2.js → InteractivePreview-DYFW3lDD.js} +3 -3
  312. package/codeyam-cli/src/webserver/build/client/assets/{LibraryFunctionPreview-CnatsCw2.js → LibraryFunctionPreview-DLeucoVX.js} +1 -1
  313. package/codeyam-cli/src/webserver/build/client/assets/{LoadingDots-CSP6DZrh.js → LoadingDots-BU_OAEMP.js} +1 -1
  314. package/codeyam-cli/src/webserver/build/client/assets/{LogViewer-CMK8Q7yk.js → LogViewer-ceAyBX-H.js} +1 -1
  315. package/codeyam-cli/src/webserver/build/client/assets/{ReportIssueModal-TCV_HBjy.js → ReportIssueModal-BzHcG7SE.js} +3 -3
  316. package/codeyam-cli/src/webserver/build/client/assets/{SafeScreenshot-CG2uh31y.js → SafeScreenshot-BED4B6sP.js} +1 -1
  317. package/codeyam-cli/src/webserver/build/client/assets/{ScenarioViewer-CU_TDYd8.js → ScenarioViewer-Bd-hxofb.js} +3 -3
  318. package/codeyam-cli/src/webserver/build/client/assets/Spinner-Bb5uFQ5V.js +34 -0
  319. package/codeyam-cli/src/webserver/build/client/assets/{TruncatedFilePath-D7IoaWUW.js → TruncatedFilePath-C8OKAR5x.js} +1 -1
  320. package/codeyam-cli/src/webserver/build/client/assets/ViewportInspectBar-oAf2Kqsf.js +1 -0
  321. package/codeyam-cli/src/webserver/build/client/assets/{_index-B8z7mjR-.js → _index-DLxKhri3.js} +3 -3
  322. package/codeyam-cli/src/webserver/build/client/assets/{activity.(_tab)-DZu78RI1.js → activity.(_tab)-BcY3q6nt.js} +6 -6
  323. package/codeyam-cli/src/webserver/build/client/assets/addon-canvas-DpzMmAy5.js +1 -0
  324. package/codeyam-cli/src/webserver/build/client/assets/addon-fit-YJmn1quW.js +12 -0
  325. package/codeyam-cli/src/webserver/build/client/assets/addon-web-links-Duc5hnl7.js +1 -0
  326. package/codeyam-cli/src/webserver/build/client/assets/addon-webgl-DI8QOUvO.js +58 -0
  327. package/codeyam-cli/src/webserver/build/client/assets/{agent-transcripts-Dm5RS9il.js → agent-transcripts-Bni3iiUj.js} +5 -5
  328. package/codeyam-cli/src/webserver/build/client/assets/api.dev-mode-events-l0sNRNKZ.js +1 -0
  329. package/codeyam-cli/src/webserver/build/client/assets/api.editor-audit-l0sNRNKZ.js +1 -0
  330. package/codeyam-cli/src/webserver/build/client/assets/api.editor-capture-scenario-l0sNRNKZ.js +1 -0
  331. package/codeyam-cli/src/webserver/build/client/assets/api.editor-client-errors-l0sNRNKZ.js +1 -0
  332. package/codeyam-cli/src/webserver/build/client/assets/api.editor-commit-l0sNRNKZ.js +1 -0
  333. package/codeyam-cli/src/webserver/build/client/assets/api.editor-dev-server-l0sNRNKZ.js +1 -0
  334. package/codeyam-cli/src/webserver/build/client/assets/api.editor-entity-status-l0sNRNKZ.js +1 -0
  335. package/codeyam-cli/src/webserver/build/client/assets/api.editor-file-diff-l0sNRNKZ.js +1 -0
  336. package/codeyam-cli/src/webserver/build/client/assets/api.editor-file-l0sNRNKZ.js +1 -0
  337. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-entry-l0sNRNKZ.js +1 -0
  338. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-image._-l0sNRNKZ.js +1 -0
  339. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-l0sNRNKZ.js +1 -0
  340. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-screenshot-l0sNRNKZ.js +1 -0
  341. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-update-l0sNRNKZ.js +1 -0
  342. package/codeyam-cli/src/webserver/build/client/assets/api.editor-load-commit-l0sNRNKZ.js +1 -0
  343. package/codeyam-cli/src/webserver/build/client/assets/api.editor-project-info-l0sNRNKZ.js +1 -0
  344. package/codeyam-cli/src/webserver/build/client/assets/api.editor-refresh-l0sNRNKZ.js +1 -0
  345. package/codeyam-cli/src/webserver/build/client/assets/api.editor-register-scenario-l0sNRNKZ.js +1 -0
  346. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-coverage-l0sNRNKZ.js +1 -0
  347. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-data-l0sNRNKZ.js +1 -0
  348. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-image._-l0sNRNKZ.js +1 -0
  349. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenarios-l0sNRNKZ.js +1 -0
  350. package/codeyam-cli/src/webserver/build/client/assets/api.editor-switch-scenario-l0sNRNKZ.js +1 -0
  351. package/codeyam-cli/src/webserver/build/client/assets/api.editor-test-results-l0sNRNKZ.js +1 -0
  352. package/codeyam-cli/src/webserver/build/client/assets/{book-open-Bp5FLkd4.js → book-open-BYOypzCa.js} +2 -2
  353. package/codeyam-cli/src/webserver/build/client/assets/{chevron-down-DQJA9f4o.js → chevron-down-C_Pmso5S.js} +2 -2
  354. package/codeyam-cli/src/webserver/build/client/assets/{chunk-JZWAC4HX-7VptmeIr.js → chunk-JZWAC4HX-C4pqxYJB.js} +1 -1
  355. package/codeyam-cli/src/webserver/build/client/assets/{circle-check-B6C4LY9o.js → circle-check-BVMi9VA5.js} +2 -2
  356. package/codeyam-cli/src/webserver/build/client/assets/{copy-6nzYCu0G.js → copy-n2FB0_Sw.js} +3 -3
  357. package/codeyam-cli/src/webserver/build/client/assets/createLucideIcon-CC6AbExI.js +41 -0
  358. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-BsDh6TSF.js +1 -0
  359. package/codeyam-cli/src/webserver/build/client/assets/editor-PBc_6L9R.js +10 -0
  360. package/codeyam-cli/src/webserver/build/client/assets/editorPreview-4FzHlcNn.js +41 -0
  361. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha._-C6PQhwY5.js → entity._sha._-BsDXNp45.js} +9 -9
  362. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.dev-BgAqUtTZ.js +6 -0
  363. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-Bmshgrij.js +6 -0
  364. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-p9hhkjJM.js +6 -0
  365. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha_.edit._scenarioId-C7ysA4Jq.js → entity._sha_.edit._scenarioId-BMvVHNXU.js} +2 -2
  366. package/codeyam-cli/src/webserver/build/client/assets/{entry.client-CU6EUArK.js → entry.client-DTvKq3TY.js} +1 -1
  367. package/codeyam-cli/src/webserver/build/client/assets/{fileTableUtils-EWpfFU4X.js → fileTableUtils-cPo8LiG3.js} +1 -1
  368. package/codeyam-cli/src/webserver/build/client/assets/{files-CrxAoWIL.js → files-BZrlFE1F.js} +1 -1
  369. package/codeyam-cli/src/webserver/build/client/assets/git-DdZcvjGh.js +1 -0
  370. package/codeyam-cli/src/webserver/build/client/assets/globals-B8vTTNy2.css +1 -0
  371. package/codeyam-cli/src/webserver/build/client/assets/{index-7-1FmlHo.js → index-10oVnAAH.js} +1 -1
  372. package/codeyam-cli/src/webserver/build/client/assets/{index-DuYcwYp_.js → index-BcvgDzbZ.js} +1 -1
  373. package/codeyam-cli/src/webserver/build/client/assets/index-yHOVb4rc.js +15 -0
  374. package/codeyam-cli/src/webserver/build/client/assets/{labs-CPPVOSWB.js → labs-Zk7ryIM1.js} +1 -1
  375. package/codeyam-cli/src/webserver/build/client/assets/{loader-circle-BnDcD54R.js → loader-circle-DaAZ_H2w.js} +2 -2
  376. package/codeyam-cli/src/webserver/build/client/assets/manifest-65850841.js +1 -0
  377. package/codeyam-cli/src/webserver/build/client/assets/memory-9gnxSZlb.js +101 -0
  378. package/codeyam-cli/src/webserver/build/client/assets/{pause-DhQX2g22.js → pause-f5-1lKBt.js} +3 -3
  379. package/codeyam-cli/src/webserver/build/client/assets/root-BwX8YgFb.js +67 -0
  380. package/codeyam-cli/src/webserver/build/client/assets/{search-DborVoKD.js → search-Di64LWVb.js} +2 -2
  381. package/codeyam-cli/src/webserver/build/client/assets/{settings-BWunYSXt.js → settings-0OrEMU6J.js} +1 -1
  382. package/codeyam-cli/src/webserver/build/client/assets/{simulations-BtrtCYJg.js → simulations-DWT-CvLy.js} +1 -1
  383. package/codeyam-cli/src/webserver/build/client/assets/{terminal-Bs4NC-VZ.js → terminal-Br7MOqts.js} +3 -3
  384. package/codeyam-cli/src/webserver/build/client/assets/{triangle-alert-DTf3Jojp.js → triangle-alert-BLdiCuG-.js} +2 -2
  385. package/codeyam-cli/src/webserver/build/client/assets/useCustomSizes-BE43Hjti.js +1 -0
  386. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-C14nCb1q.js +2 -0
  387. package/codeyam-cli/src/webserver/build/client/assets/{useReportContext-BsQb6rFd.js → useReportContext-O-jkvSPx.js} +1 -1
  388. package/codeyam-cli/src/webserver/build/client/assets/{useToast-BOur3mUv.js → useToast-9FIWuYfK.js} +1 -1
  389. package/codeyam-cli/src/webserver/build/client/assets/xterm-BqvuqXEL.js +27 -0
  390. package/codeyam-cli/src/webserver/build/server/assets/index-DEEQf4pi.js +1 -0
  391. package/codeyam-cli/src/webserver/build/server/assets/init-CkWmyFY2.js +10 -0
  392. package/codeyam-cli/src/webserver/build/server/assets/server-build-BHi-9O8W.js +439 -0
  393. package/codeyam-cli/src/webserver/build/server/index.js +1 -1
  394. package/codeyam-cli/src/webserver/build-info.json +5 -5
  395. package/codeyam-cli/src/webserver/devServer.js +39 -5
  396. package/codeyam-cli/src/webserver/devServer.js.map +1 -1
  397. package/codeyam-cli/src/webserver/editorProxy.js +877 -0
  398. package/codeyam-cli/src/webserver/editorProxy.js.map +1 -0
  399. package/codeyam-cli/src/webserver/scripts/codeyam-preload.mjs +414 -0
  400. package/codeyam-cli/src/webserver/scripts/journalCapture.ts +230 -0
  401. package/codeyam-cli/src/webserver/server.js +258 -1
  402. package/codeyam-cli/src/webserver/server.js.map +1 -1
  403. package/codeyam-cli/src/webserver/terminalServer.js +726 -0
  404. package/codeyam-cli/src/webserver/terminalServer.js.map +1 -0
  405. package/codeyam-cli/templates/chrome-extension-react/EXTENSION_SETUP.md +75 -0
  406. package/codeyam-cli/templates/chrome-extension-react/README.md +46 -0
  407. package/codeyam-cli/templates/chrome-extension-react/gitignore +15 -0
  408. package/codeyam-cli/templates/chrome-extension-react/index.html +12 -0
  409. package/codeyam-cli/templates/chrome-extension-react/package.json +27 -0
  410. package/codeyam-cli/templates/chrome-extension-react/popup.html +12 -0
  411. package/codeyam-cli/templates/chrome-extension-react/public/manifest.json +15 -0
  412. package/codeyam-cli/templates/chrome-extension-react/src/background/service-worker.ts +7 -0
  413. package/codeyam-cli/templates/chrome-extension-react/src/globals.css +6 -0
  414. package/codeyam-cli/templates/chrome-extension-react/src/lib/storage.ts +37 -0
  415. package/codeyam-cli/templates/chrome-extension-react/src/popup/App.tsx +12 -0
  416. package/codeyam-cli/templates/chrome-extension-react/src/popup/main.tsx +10 -0
  417. package/codeyam-cli/templates/chrome-extension-react/tsconfig.json +24 -0
  418. package/codeyam-cli/templates/chrome-extension-react/vite.config.ts +41 -0
  419. package/codeyam-cli/templates/codeyam-editor-claude.md +147 -0
  420. package/codeyam-cli/templates/editor-step-hook.py +236 -0
  421. package/codeyam-cli/templates/expo-react-native/MOBILE_SETUP.md +89 -0
  422. package/codeyam-cli/templates/expo-react-native/README.md +41 -0
  423. package/codeyam-cli/templates/expo-react-native/app/(tabs)/_layout.tsx +33 -0
  424. package/codeyam-cli/templates/expo-react-native/app/(tabs)/index.tsx +12 -0
  425. package/codeyam-cli/templates/expo-react-native/app/(tabs)/settings.tsx +12 -0
  426. package/codeyam-cli/templates/expo-react-native/app/_layout.tsx +12 -0
  427. package/codeyam-cli/templates/expo-react-native/app.json +18 -0
  428. package/codeyam-cli/templates/expo-react-native/babel.config.js +9 -0
  429. package/codeyam-cli/templates/expo-react-native/gitignore +12 -0
  430. package/codeyam-cli/templates/expo-react-native/global.css +3 -0
  431. package/codeyam-cli/templates/expo-react-native/lib/storage.ts +32 -0
  432. package/codeyam-cli/templates/expo-react-native/metro.config.js +6 -0
  433. package/codeyam-cli/templates/expo-react-native/nativewind-env.d.ts +1 -0
  434. package/codeyam-cli/templates/expo-react-native/package.json +38 -0
  435. package/codeyam-cli/templates/expo-react-native/tailwind.config.js +10 -0
  436. package/codeyam-cli/templates/expo-react-native/tsconfig.json +10 -0
  437. package/codeyam-cli/templates/isolation-route/next-app.tsx.template +80 -0
  438. package/codeyam-cli/templates/isolation-route/next-pages.tsx.template +79 -0
  439. package/codeyam-cli/templates/isolation-route/vite-react.tsx.template +78 -0
  440. package/codeyam-cli/templates/msw/browser-setup.ts.template +47 -0
  441. package/codeyam-cli/templates/msw/handler-router.ts.template +47 -0
  442. package/codeyam-cli/templates/msw/server-setup.ts.template +52 -0
  443. package/codeyam-cli/templates/nextjs-prisma-sqlite/AUTH_PATTERNS.md +308 -0
  444. package/codeyam-cli/templates/nextjs-prisma-sqlite/AUTH_UPGRADE.md +304 -0
  445. package/codeyam-cli/templates/nextjs-prisma-sqlite/DATABASE.md +126 -0
  446. package/codeyam-cli/templates/nextjs-prisma-sqlite/FEATURE_PATTERNS.md +37 -0
  447. package/codeyam-cli/templates/nextjs-prisma-sqlite/README.md +53 -0
  448. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/api/todos/route.ts +17 -0
  449. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/codeyam-isolate/layout.tsx +12 -0
  450. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/globals.css +26 -0
  451. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/layout.tsx +34 -0
  452. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/lib/prisma.ts +24 -0
  453. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/page.tsx +10 -0
  454. package/codeyam-cli/templates/nextjs-prisma-sqlite/env +4 -0
  455. package/codeyam-cli/templates/nextjs-prisma-sqlite/eslint.config.mjs +11 -0
  456. package/codeyam-cli/templates/nextjs-prisma-sqlite/gitignore +64 -0
  457. package/codeyam-cli/templates/nextjs-prisma-sqlite/next.config.ts +14 -0
  458. package/codeyam-cli/templates/nextjs-prisma-sqlite/package.json +39 -0
  459. package/codeyam-cli/templates/nextjs-prisma-sqlite/postcss.config.mjs +7 -0
  460. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma/schema.prisma +27 -0
  461. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma/seed.ts +40 -0
  462. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma.config.ts +12 -0
  463. package/codeyam-cli/templates/nextjs-prisma-sqlite/seed-adapter.ts +92 -0
  464. package/codeyam-cli/templates/nextjs-prisma-sqlite/tsconfig.json +34 -0
  465. package/codeyam-cli/templates/nextjs-prisma-sqlite/vitest.config.ts +13 -0
  466. package/codeyam-cli/templates/nextjs-prisma-supabase/README.md +52 -0
  467. package/codeyam-cli/templates/nextjs-prisma-supabase/SUPABASE_SETUP.md +104 -0
  468. package/codeyam-cli/templates/nextjs-prisma-supabase/app/api/todos/route.ts +17 -0
  469. package/codeyam-cli/templates/nextjs-prisma-supabase/app/globals.css +26 -0
  470. package/codeyam-cli/templates/nextjs-prisma-supabase/app/layout.tsx +34 -0
  471. package/codeyam-cli/templates/nextjs-prisma-supabase/app/lib/prisma.ts +20 -0
  472. package/codeyam-cli/templates/nextjs-prisma-supabase/app/lib/supabase.ts +12 -0
  473. package/codeyam-cli/templates/nextjs-prisma-supabase/app/page.tsx +10 -0
  474. package/codeyam-cli/templates/nextjs-prisma-supabase/env +9 -0
  475. package/codeyam-cli/templates/nextjs-prisma-supabase/eslint.config.mjs +11 -0
  476. package/codeyam-cli/templates/nextjs-prisma-supabase/gitignore +40 -0
  477. package/codeyam-cli/templates/nextjs-prisma-supabase/next.config.ts +11 -0
  478. package/codeyam-cli/templates/nextjs-prisma-supabase/package.json +37 -0
  479. package/codeyam-cli/templates/nextjs-prisma-supabase/postcss.config.mjs +7 -0
  480. package/codeyam-cli/templates/nextjs-prisma-supabase/prisma/schema.prisma +27 -0
  481. package/codeyam-cli/templates/nextjs-prisma-supabase/prisma/seed.ts +39 -0
  482. package/codeyam-cli/templates/nextjs-prisma-supabase/prisma.config.ts +12 -0
  483. package/codeyam-cli/templates/nextjs-prisma-supabase/tsconfig.json +34 -0
  484. package/codeyam-cli/templates/skills/codeyam-dev-mode/SKILL.md +237 -0
  485. package/codeyam-cli/templates/skills/codeyam-editor/SKILL.md +148 -0
  486. package/codeyam-cli/templates/{codeyam-memory.md → skills/codeyam-memory/SKILL.md} +215 -0
  487. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/deprecated-prompt.md +100 -0
  488. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/detect-deprecated-patterns.mjs +139 -0
  489. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/find-exports.mjs +52 -0
  490. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/misleading-api-prompt.md +117 -0
  491. package/codeyam-cli/templates/skills/codeyam-memory/scripts/lib/read-json-field.mjs +61 -0
  492. package/codeyam-cli/templates/skills/codeyam-memory/scripts/lib/ripgrep-fallback.mjs +155 -0
  493. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/analyze-prompt.md +46 -0
  494. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/cleanup.mjs +13 -0
  495. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/filter-session.mjs +95 -0
  496. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/preprocess.mjs +160 -0
  497. package/package.json +17 -10
  498. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js +22 -4
  499. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js.map +1 -1
  500. package/packages/ai/src/lib/completionCall.js +10 -2
  501. package/packages/ai/src/lib/completionCall.js.map +1 -1
  502. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js +21 -0
  503. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js.map +1 -1
  504. package/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.js +54 -0
  505. package/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.js.map +1 -0
  506. package/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.js +34 -0
  507. package/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.js.map +1 -0
  508. package/packages/ai/src/lib/generateEntityScenarioData.js +57 -2
  509. package/packages/ai/src/lib/generateEntityScenarioData.js.map +1 -1
  510. package/packages/ai/src/lib/generateExecutionFlows.js +0 -11
  511. package/packages/ai/src/lib/generateExecutionFlows.js.map +1 -1
  512. package/packages/analyze/src/lib/ProjectAnalyzer.js +13 -4
  513. package/packages/analyze/src/lib/ProjectAnalyzer.js.map +1 -1
  514. package/packages/analyze/src/lib/asts/index.js +4 -2
  515. package/packages/analyze/src/lib/asts/index.js.map +1 -1
  516. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js +8 -1
  517. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js.map +1 -1
  518. package/packages/analyze/src/lib/files/analyze/dependencyResolver.js +0 -5
  519. package/packages/analyze/src/lib/files/analyze/dependencyResolver.js.map +1 -1
  520. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js +9 -0
  521. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js.map +1 -1
  522. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js +54 -27
  523. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js.map +1 -1
  524. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js +65 -0
  525. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js.map +1 -1
  526. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js +0 -40
  527. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js.map +1 -1
  528. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js +18 -4
  529. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js.map +1 -1
  530. package/packages/database/index.js +1 -0
  531. package/packages/database/index.js.map +1 -1
  532. package/packages/database/src/lib/kysely/db.js +5 -0
  533. package/packages/database/src/lib/kysely/db.js.map +1 -1
  534. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js +76 -0
  535. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -0
  536. package/packages/database/src/lib/loadCommits.js +23 -13
  537. package/packages/database/src/lib/loadCommits.js.map +1 -1
  538. package/packages/database/src/lib/loadEntities.js +0 -6
  539. package/packages/database/src/lib/loadEntities.js.map +1 -1
  540. package/packages/database/src/lib/loadReadyToBeCapturedAnalyses.js +1 -4
  541. package/packages/database/src/lib/loadReadyToBeCapturedAnalyses.js.map +1 -1
  542. package/packages/database/src/lib/updateCommitMetadata.js +76 -90
  543. package/packages/database/src/lib/updateCommitMetadata.js.map +1 -1
  544. package/packages/database/src/lib/updateFreshAnalysisStatus.js +41 -30
  545. package/packages/database/src/lib/updateFreshAnalysisStatus.js.map +1 -1
  546. package/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.js +68 -57
  547. package/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.js.map +1 -1
  548. package/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js +29 -1
  549. package/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js.map +1 -1
  550. package/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js +33 -5
  551. package/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js.map +1 -1
  552. package/packages/types/src/enums/ProjectFramework.js +2 -0
  553. package/packages/types/src/enums/ProjectFramework.js.map +1 -1
  554. package/scripts/npm-post-install.cjs +34 -0
  555. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-C1rIyZdV.js +0 -34
  556. package/codeyam-cli/src/webserver/build/client/assets/createLucideIcon-D-QUFOwe.js +0 -21
  557. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-DmzSmblj.js +0 -1
  558. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-DVTcUnur.js +0 -6
  559. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-BVgNO76F.js +0 -6
  560. package/codeyam-cli/src/webserver/build/client/assets/git-BldHtKeW.js +0 -15
  561. package/codeyam-cli/src/webserver/build/client/assets/globals-CLmFdUae.css +0 -1
  562. package/codeyam-cli/src/webserver/build/client/assets/manifest-717e346a.js +0 -1
  563. package/codeyam-cli/src/webserver/build/client/assets/memory-0wMU4KXe.js +0 -93
  564. package/codeyam-cli/src/webserver/build/client/assets/root-DqfSDjyQ.js +0 -62
  565. package/codeyam-cli/src/webserver/build/client/assets/useCustomSizes-D_bDZyDU.js +0 -1
  566. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-DZp6rrQD.js +0 -2
  567. package/codeyam-cli/src/webserver/build/server/assets/index-B8jmgmn2.js +0 -1
  568. package/codeyam-cli/src/webserver/build/server/assets/server-build-9OU4lmvL.js +0 -285
  569. package/scripts/finalize-analyzer.cjs +0 -13
  570. /package/codeyam-cli/templates/{codeyam-diagnose.md → commands/codeyam-diagnose.md} +0 -0
  571. /package/codeyam-cli/templates/{codeyam-debug.md → skills/codeyam-debug/SKILL.md} +0 -0
  572. /package/codeyam-cli/templates/{codeyam-new-rule.md → skills/codeyam-new-rule/SKILL.md} +0 -0
  573. /package/codeyam-cli/templates/{codeyam-setup.md → skills/codeyam-setup/SKILL.md} +0 -0
  574. /package/codeyam-cli/templates/{codeyam-sim.md → skills/codeyam-sim/SKILL.md} +0 -0
  575. /package/codeyam-cli/templates/{codeyam-test.md → skills/codeyam-test/SKILL.md} +0 -0
  576. /package/codeyam-cli/templates/{codeyam-verify.md → skills/codeyam-verify/SKILL.md} +0 -0
@@ -0,0 +1,855 @@
1
+ import { isComponent, classifyGlossaryEntries, computeAudit, filterGlossaryByChangeStatus, resolveAuditSessionScope, } from "../editorAudit.js";
2
+ describe('editorAudit', () => {
3
+ describe('isComponent', () => {
4
+ it('should return true for JSX.Element return type', () => {
5
+ expect(isComponent({ returnType: 'JSX.Element' })).toBe(true);
6
+ });
7
+ it('should return true for React.ReactElement return type', () => {
8
+ expect(isComponent({ returnType: 'React.ReactElement' })).toBe(true);
9
+ });
10
+ it('should return true for ReactNode return type', () => {
11
+ expect(isComponent({ returnType: 'ReactNode' })).toBe(true);
12
+ });
13
+ it('should return true for Element return type', () => {
14
+ expect(isComponent({ returnType: 'Element' })).toBe(true);
15
+ });
16
+ it('should return true when JSX appears anywhere in the return type', () => {
17
+ expect(isComponent({ returnType: 'Promise<JSX.Element>' })).toBe(true);
18
+ });
19
+ it('should return false for non-component return types', () => {
20
+ expect(isComponent({ returnType: 'number' })).toBe(false);
21
+ expect(isComponent({ returnType: 'string' })).toBe(false);
22
+ expect(isComponent({ returnType: 'void' })).toBe(false);
23
+ });
24
+ it('should return false when returnType is undefined', () => {
25
+ expect(isComponent({})).toBe(false);
26
+ });
27
+ it('should return false when returnType is empty string', () => {
28
+ expect(isComponent({ returnType: '' })).toBe(false);
29
+ });
30
+ });
31
+ describe('classifyGlossaryEntries', () => {
32
+ it('should classify entries with JSX return types as components', () => {
33
+ const entries = [
34
+ {
35
+ name: 'DrinkCard',
36
+ filePath: 'app/components/DrinkCard.tsx',
37
+ returnType: 'JSX.Element',
38
+ },
39
+ {
40
+ name: 'calculatePrice',
41
+ filePath: 'app/lib/pricing.ts',
42
+ returnType: 'number',
43
+ },
44
+ ];
45
+ const result = classifyGlossaryEntries(entries);
46
+ expect(result.components).toHaveLength(1);
47
+ expect(result.components[0].name).toBe('DrinkCard');
48
+ expect(result.functions).toHaveLength(1);
49
+ expect(result.functions[0].name).toBe('calculatePrice');
50
+ });
51
+ it('should return empty arrays for empty input', () => {
52
+ const result = classifyGlossaryEntries([]);
53
+ expect(result.components).toEqual([]);
54
+ expect(result.functions).toEqual([]);
55
+ });
56
+ it('should treat entries without returnType as functions', () => {
57
+ const entries = [{ name: 'helper', filePath: 'app/lib/helper.ts' }];
58
+ const result = classifyGlossaryEntries(entries);
59
+ expect(result.components).toHaveLength(0);
60
+ expect(result.functions).toHaveLength(1);
61
+ });
62
+ it('should handle all components', () => {
63
+ const entries = [
64
+ {
65
+ name: 'Header',
66
+ filePath: 'app/components/Header.tsx',
67
+ returnType: 'ReactNode',
68
+ },
69
+ {
70
+ name: 'Footer',
71
+ filePath: 'app/components/Footer.tsx',
72
+ returnType: 'JSX.Element',
73
+ },
74
+ ];
75
+ const result = classifyGlossaryEntries(entries);
76
+ expect(result.components).toHaveLength(2);
77
+ expect(result.functions).toHaveLength(0);
78
+ });
79
+ it('should handle all functions', () => {
80
+ const entries = [
81
+ { name: 'add', filePath: 'app/lib/math.ts', returnType: 'number' },
82
+ { name: 'format', filePath: 'app/lib/format.ts', returnType: 'string' },
83
+ ];
84
+ const result = classifyGlossaryEntries(entries);
85
+ expect(result.components).toHaveLength(0);
86
+ expect(result.functions).toHaveLength(2);
87
+ });
88
+ });
89
+ describe('computeAudit', () => {
90
+ it('should mark components with scenarios as ok', () => {
91
+ const result = computeAudit({
92
+ components: [
93
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
94
+ ],
95
+ functions: [],
96
+ scenarioCounts: { DrinkCard: 2 },
97
+ testFileExistence: {},
98
+ });
99
+ expect(result.components[0].status).toBe('ok');
100
+ expect(result.components[0].scenarioCount).toBe(2);
101
+ expect(result.summary.componentsOk).toBe(1);
102
+ expect(result.summary.componentsMissing).toBe(0);
103
+ });
104
+ it('should mark components without scenarios as missing', () => {
105
+ const result = computeAudit({
106
+ components: [
107
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
108
+ ],
109
+ functions: [],
110
+ scenarioCounts: {},
111
+ testFileExistence: {},
112
+ });
113
+ expect(result.components[0].status).toBe('missing');
114
+ expect(result.components[0].scenarioCount).toBe(0);
115
+ expect(result.summary.componentsMissing).toBe(1);
116
+ });
117
+ it('should mark functions with existing test files as ok', () => {
118
+ const result = computeAudit({
119
+ components: [],
120
+ functions: [
121
+ {
122
+ name: 'calculatePrice',
123
+ filePath: 'app/lib/pricing.ts',
124
+ testFile: 'app/lib/pricing.test.ts',
125
+ },
126
+ ],
127
+ scenarioCounts: {},
128
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
129
+ });
130
+ expect(result.functions[0].status).toBe('ok');
131
+ expect(result.functions[0].testFileExists).toBe(true);
132
+ expect(result.summary.functionsOk).toBe(1);
133
+ expect(result.summary.functionsMissing).toBe(0);
134
+ });
135
+ it('should mark functions with missing test files as missing', () => {
136
+ const result = computeAudit({
137
+ components: [],
138
+ functions: [
139
+ {
140
+ name: 'calculatePrice',
141
+ filePath: 'app/lib/pricing.ts',
142
+ testFile: 'app/lib/pricing.test.ts',
143
+ },
144
+ ],
145
+ scenarioCounts: {},
146
+ testFileExistence: { 'app/lib/pricing.test.ts': false },
147
+ });
148
+ expect(result.functions[0].status).toBe('missing');
149
+ expect(result.functions[0].testFileExists).toBe(false);
150
+ expect(result.summary.functionsMissing).toBe(1);
151
+ });
152
+ it('should mark functions without testFile field as missing', () => {
153
+ const result = computeAudit({
154
+ components: [],
155
+ functions: [{ name: 'calculatePrice', filePath: 'app/lib/pricing.ts' }],
156
+ scenarioCounts: {},
157
+ testFileExistence: {},
158
+ });
159
+ expect(result.functions[0].status).toBe('missing');
160
+ expect(result.functions[0].testFile).toBeUndefined();
161
+ expect(result.functions[0].testFileExists).toBe(false);
162
+ });
163
+ it('should set allPassing to true when everything is ok', () => {
164
+ const result = computeAudit({
165
+ components: [
166
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
167
+ ],
168
+ functions: [
169
+ {
170
+ name: 'calculatePrice',
171
+ filePath: 'app/lib/pricing.ts',
172
+ testFile: 'app/lib/pricing.test.ts',
173
+ },
174
+ ],
175
+ scenarioCounts: { DrinkCard: 1 },
176
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
177
+ });
178
+ expect(result.summary.allPassing).toBe(true);
179
+ });
180
+ it('should set allPassing to false when any component is missing', () => {
181
+ const result = computeAudit({
182
+ components: [
183
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
184
+ ],
185
+ functions: [],
186
+ scenarioCounts: {},
187
+ testFileExistence: {},
188
+ });
189
+ expect(result.summary.allPassing).toBe(false);
190
+ });
191
+ it('should set allPassing to false when any function is missing', () => {
192
+ const result = computeAudit({
193
+ components: [],
194
+ functions: [{ name: 'calculatePrice', filePath: 'app/lib/pricing.ts' }],
195
+ scenarioCounts: {},
196
+ testFileExistence: {},
197
+ });
198
+ expect(result.summary.allPassing).toBe(false);
199
+ });
200
+ it('should set allPassing to true for empty inputs', () => {
201
+ const result = computeAudit({
202
+ components: [],
203
+ functions: [],
204
+ scenarioCounts: {},
205
+ testFileExistence: {},
206
+ });
207
+ expect(result.summary.allPassing).toBe(true);
208
+ expect(result.summary.totalComponents).toBe(0);
209
+ expect(result.summary.totalFunctions).toBe(0);
210
+ });
211
+ it('should compute correct summary totals', () => {
212
+ const result = computeAudit({
213
+ components: [
214
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
215
+ { name: 'Header', filePath: 'app/components/Header.tsx' },
216
+ { name: 'Footer', filePath: 'app/components/Footer.tsx' },
217
+ ],
218
+ functions: [
219
+ {
220
+ name: 'calculatePrice',
221
+ filePath: 'app/lib/pricing.ts',
222
+ testFile: 'app/lib/pricing.test.ts',
223
+ },
224
+ {
225
+ name: 'formatDate',
226
+ filePath: 'app/lib/format.ts',
227
+ testFile: 'app/lib/format.test.ts',
228
+ },
229
+ ],
230
+ scenarioCounts: { DrinkCard: 3, Header: 1 },
231
+ testFileExistence: {
232
+ 'app/lib/pricing.test.ts': true,
233
+ 'app/lib/format.test.ts': false,
234
+ },
235
+ });
236
+ expect(result.summary.totalComponents).toBe(3);
237
+ expect(result.summary.componentsOk).toBe(2);
238
+ expect(result.summary.componentsMissing).toBe(1);
239
+ expect(result.summary.totalFunctions).toBe(2);
240
+ expect(result.summary.functionsOk).toBe(1);
241
+ expect(result.summary.functionsMissing).toBe(1);
242
+ expect(result.summary.allPassing).toBe(false);
243
+ });
244
+ // ── Test results integration ────────────────────────────────────
245
+ it('should mark function as ok when tests pass and describe matches entity name', () => {
246
+ const result = computeAudit({
247
+ components: [],
248
+ functions: [
249
+ {
250
+ name: 'calculatePrice',
251
+ filePath: 'app/lib/pricing.ts',
252
+ testFile: 'app/lib/pricing.test.ts',
253
+ },
254
+ ],
255
+ scenarioCounts: {},
256
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
257
+ testResults: {
258
+ 'app/lib/pricing.test.ts': {
259
+ passing: true,
260
+ hasEntityNameDescribe: true,
261
+ },
262
+ },
263
+ });
264
+ expect(result.functions[0].status).toBe('ok');
265
+ expect(result.functions[0].testsPassing).toBe(true);
266
+ expect(result.functions[0].testsVisibleInUi).toBe(true);
267
+ });
268
+ it('should mark function as failing when tests fail', () => {
269
+ const result = computeAudit({
270
+ components: [],
271
+ functions: [
272
+ {
273
+ name: 'calculatePrice',
274
+ filePath: 'app/lib/pricing.ts',
275
+ testFile: 'app/lib/pricing.test.ts',
276
+ },
277
+ ],
278
+ scenarioCounts: {},
279
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
280
+ testResults: {
281
+ 'app/lib/pricing.test.ts': {
282
+ passing: false,
283
+ hasEntityNameDescribe: true,
284
+ },
285
+ },
286
+ });
287
+ expect(result.functions[0].status).toBe('failing');
288
+ expect(result.functions[0].testsPassing).toBe(false);
289
+ expect(result.functions[0].testsVisibleInUi).toBe(true);
290
+ });
291
+ it('should mark function as name_mismatch when tests pass but no describe matches', () => {
292
+ const result = computeAudit({
293
+ components: [],
294
+ functions: [
295
+ {
296
+ name: 'useDrinks',
297
+ filePath: 'app/hooks/useDrinks.ts',
298
+ testFile: 'app/hooks/useDrinks.test.ts',
299
+ },
300
+ ],
301
+ scenarioCounts: {},
302
+ testFileExistence: { 'app/hooks/useDrinks.test.ts': true },
303
+ testResults: {
304
+ 'app/hooks/useDrinks.test.ts': {
305
+ passing: true,
306
+ hasEntityNameDescribe: false,
307
+ },
308
+ },
309
+ });
310
+ expect(result.functions[0].status).toBe('name_mismatch');
311
+ expect(result.functions[0].testsPassing).toBe(true);
312
+ expect(result.functions[0].testsVisibleInUi).toBe(false);
313
+ });
314
+ it('should still mark function as missing when test file does not exist even with testResults', () => {
315
+ const result = computeAudit({
316
+ components: [],
317
+ functions: [
318
+ {
319
+ name: 'calculatePrice',
320
+ filePath: 'app/lib/pricing.ts',
321
+ testFile: 'app/lib/pricing.test.ts',
322
+ },
323
+ ],
324
+ scenarioCounts: {},
325
+ testFileExistence: { 'app/lib/pricing.test.ts': false },
326
+ testResults: {},
327
+ });
328
+ expect(result.functions[0].status).toBe('missing');
329
+ expect(result.functions[0].testsPassing).toBeUndefined();
330
+ expect(result.functions[0].testsVisibleInUi).toBeUndefined();
331
+ });
332
+ it('should include functionsFailing and functionsNameMismatch in summary', () => {
333
+ const result = computeAudit({
334
+ components: [],
335
+ functions: [
336
+ {
337
+ name: 'fnOk',
338
+ filePath: 'a.ts',
339
+ testFile: 'a.test.ts',
340
+ },
341
+ {
342
+ name: 'fnFailing',
343
+ filePath: 'b.ts',
344
+ testFile: 'b.test.ts',
345
+ },
346
+ {
347
+ name: 'fnMismatch',
348
+ filePath: 'c.ts',
349
+ testFile: 'c.test.ts',
350
+ },
351
+ {
352
+ name: 'fnMissing',
353
+ filePath: 'd.ts',
354
+ testFile: 'd.test.ts',
355
+ },
356
+ ],
357
+ scenarioCounts: {},
358
+ testFileExistence: {
359
+ 'a.test.ts': true,
360
+ 'b.test.ts': true,
361
+ 'c.test.ts': true,
362
+ 'd.test.ts': false,
363
+ },
364
+ testResults: {
365
+ 'a.test.ts': { passing: true, hasEntityNameDescribe: true },
366
+ 'b.test.ts': { passing: false, hasEntityNameDescribe: true },
367
+ 'c.test.ts': { passing: true, hasEntityNameDescribe: false },
368
+ },
369
+ });
370
+ expect(result.summary.functionsOk).toBe(1);
371
+ expect(result.summary.functionsFailing).toBe(1);
372
+ expect(result.summary.functionsNameMismatch).toBe(1);
373
+ expect(result.summary.functionsMissing).toBe(1);
374
+ expect(result.summary.allPassing).toBe(false);
375
+ });
376
+ it('should set allPassing to true only when all functions are ok', () => {
377
+ const result = computeAudit({
378
+ components: [
379
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
380
+ ],
381
+ functions: [
382
+ {
383
+ name: 'calculatePrice',
384
+ filePath: 'app/lib/pricing.ts',
385
+ testFile: 'app/lib/pricing.test.ts',
386
+ },
387
+ ],
388
+ scenarioCounts: { DrinkCard: 1 },
389
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
390
+ testResults: {
391
+ 'app/lib/pricing.test.ts': {
392
+ passing: true,
393
+ hasEntityNameDescribe: true,
394
+ },
395
+ },
396
+ });
397
+ expect(result.summary.allPassing).toBe(true);
398
+ });
399
+ it('should set allPassing to false when any function is failing', () => {
400
+ const result = computeAudit({
401
+ components: [],
402
+ functions: [
403
+ {
404
+ name: 'calculatePrice',
405
+ filePath: 'app/lib/pricing.ts',
406
+ testFile: 'app/lib/pricing.test.ts',
407
+ },
408
+ ],
409
+ scenarioCounts: {},
410
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
411
+ testResults: {
412
+ 'app/lib/pricing.test.ts': {
413
+ passing: false,
414
+ hasEntityNameDescribe: true,
415
+ },
416
+ },
417
+ });
418
+ expect(result.summary.allPassing).toBe(false);
419
+ });
420
+ it('should set allPassing to false when any function has name_mismatch', () => {
421
+ const result = computeAudit({
422
+ components: [],
423
+ functions: [
424
+ {
425
+ name: 'useDrinks',
426
+ filePath: 'app/hooks/useDrinks.ts',
427
+ testFile: 'app/hooks/useDrinks.test.ts',
428
+ },
429
+ ],
430
+ scenarioCounts: {},
431
+ testFileExistence: { 'app/hooks/useDrinks.test.ts': true },
432
+ testResults: {
433
+ 'app/hooks/useDrinks.test.ts': {
434
+ passing: true,
435
+ hasEntityNameDescribe: false,
436
+ },
437
+ },
438
+ });
439
+ expect(result.summary.allPassing).toBe(false);
440
+ });
441
+ it('should fall back to file-existence-only behavior when testResults is not provided', () => {
442
+ // Backward compat: no testResults → existing behavior
443
+ const result = computeAudit({
444
+ components: [],
445
+ functions: [
446
+ {
447
+ name: 'calculatePrice',
448
+ filePath: 'app/lib/pricing.ts',
449
+ testFile: 'app/lib/pricing.test.ts',
450
+ },
451
+ ],
452
+ scenarioCounts: {},
453
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
454
+ });
455
+ expect(result.functions[0].status).toBe('ok');
456
+ expect(result.functions[0].testsPassing).toBeUndefined();
457
+ expect(result.functions[0].testsVisibleInUi).toBeUndefined();
458
+ });
459
+ it('should fall back to file-existence-only when testResults is empty for a given file', () => {
460
+ const result = computeAudit({
461
+ components: [],
462
+ functions: [
463
+ {
464
+ name: 'calculatePrice',
465
+ filePath: 'app/lib/pricing.ts',
466
+ testFile: 'app/lib/pricing.test.ts',
467
+ },
468
+ ],
469
+ scenarioCounts: {},
470
+ testFileExistence: { 'app/lib/pricing.test.ts': true },
471
+ testResults: {},
472
+ });
473
+ // No test result for this file → fall back to existence check
474
+ expect(result.functions[0].status).toBe('ok');
475
+ expect(result.functions[0].testsPassing).toBeUndefined();
476
+ expect(result.functions[0].testsVisibleInUi).toBeUndefined();
477
+ });
478
+ // ── Client errors integration ─────────────────────────────────────
479
+ it('should mark component as has_errors when clientErrors reports errors', () => {
480
+ const result = computeAudit({
481
+ components: [
482
+ { name: 'DrinkInfo', filePath: 'app/components/DrinkInfo.tsx' },
483
+ ],
484
+ functions: [],
485
+ scenarioCounts: { DrinkInfo: 3 },
486
+ testFileExistence: {},
487
+ clientErrors: {
488
+ DrinkInfo: ['Cannot read properties of undefined'],
489
+ },
490
+ });
491
+ expect(result.components[0].status).toBe('has_errors');
492
+ expect(result.components[0].clientErrors).toEqual([
493
+ 'Cannot read properties of undefined',
494
+ ]);
495
+ expect(result.summary.componentsWithErrors).toBe(1);
496
+ expect(result.summary.allPassing).toBe(false);
497
+ });
498
+ it('should not count client errors for components with no scenarios', () => {
499
+ // If a component has no scenarios AND errors, it's still "missing"
500
+ // (errors come from scenarios, so if there are none, errors are stale)
501
+ const result = computeAudit({
502
+ components: [
503
+ { name: 'DrinkInfo', filePath: 'app/components/DrinkInfo.tsx' },
504
+ ],
505
+ functions: [],
506
+ scenarioCounts: {},
507
+ testFileExistence: {},
508
+ clientErrors: {
509
+ DrinkInfo: ['Cannot read properties of undefined'],
510
+ },
511
+ });
512
+ expect(result.components[0].status).toBe('missing');
513
+ });
514
+ it('should set allPassing false when any component has errors', () => {
515
+ const result = computeAudit({
516
+ components: [
517
+ { name: 'DrinkCard', filePath: 'app/components/DrinkCard.tsx' },
518
+ { name: 'DrinkInfo', filePath: 'app/components/DrinkInfo.tsx' },
519
+ ],
520
+ functions: [],
521
+ scenarioCounts: { DrinkCard: 2, DrinkInfo: 3 },
522
+ testFileExistence: {},
523
+ clientErrors: {
524
+ DrinkInfo: ['TypeError: something broke'],
525
+ },
526
+ });
527
+ expect(result.summary.allPassing).toBe(false);
528
+ expect(result.summary.componentsOk).toBe(1);
529
+ expect(result.summary.componentsWithErrors).toBe(1);
530
+ });
531
+ });
532
+ // ── filterGlossaryByChangeStatus ──────────────────────────────────
533
+ describe('filterGlossaryByChangeStatus', () => {
534
+ const allEntries = [
535
+ // Feature 1 components (unchanged)
536
+ {
537
+ name: 'StarRating',
538
+ filePath: 'app/components/StarRating.tsx',
539
+ returnType: 'JSX.Element',
540
+ },
541
+ {
542
+ name: 'CategoryBadge',
543
+ filePath: 'app/components/CategoryBadge.tsx',
544
+ returnType: 'JSX.Element',
545
+ },
546
+ {
547
+ name: 'DrinkCard',
548
+ filePath: 'app/components/DrinkCard.tsx',
549
+ returnType: 'JSX.Element',
550
+ },
551
+ // Feature 2 components (new)
552
+ {
553
+ name: 'HeroImage',
554
+ filePath: 'app/components/HeroImage.tsx',
555
+ returnType: 'JSX.Element',
556
+ },
557
+ {
558
+ name: 'ReviewCard',
559
+ filePath: 'app/components/ReviewCard.tsx',
560
+ returnType: 'JSX.Element',
561
+ },
562
+ // Feature 1 function (unchanged)
563
+ {
564
+ name: 'computeAvgRating',
565
+ filePath: 'app/lib/ratings.ts',
566
+ returnType: 'number',
567
+ testFile: 'app/lib/ratings.test.ts',
568
+ },
569
+ // Feature 2 function (new)
570
+ {
571
+ name: 'formatReviewCount',
572
+ filePath: 'app/lib/formatting.ts',
573
+ returnType: 'string',
574
+ testFile: 'app/lib/formatting.test.ts',
575
+ },
576
+ ];
577
+ it('should only include entries whose files are new, edited, or impacted', () => {
578
+ const entityChangeStatus = {
579
+ HeroImage: { status: 'new' },
580
+ ReviewCard: { status: 'new' },
581
+ formatReviewCount: { status: 'new' },
582
+ // DrinkCard edited because it now links to detail page
583
+ DrinkCard: { status: 'edited' },
584
+ // StarRating impacted because DrinkInfo uses it
585
+ StarRating: {
586
+ status: 'impacted',
587
+ impactedBy: [
588
+ {
589
+ name: 'DrinkInfo',
590
+ filePath: 'app/components/DrinkInfo.tsx',
591
+ changeType: 'new',
592
+ },
593
+ ],
594
+ },
595
+ };
596
+ const filtered = filterGlossaryByChangeStatus(allEntries, entityChangeStatus);
597
+ const names = filtered.map((e) => e.name).sort();
598
+ expect(names).toEqual([
599
+ 'DrinkCard',
600
+ 'HeroImage',
601
+ 'ReviewCard',
602
+ 'StarRating',
603
+ 'formatReviewCount',
604
+ ]);
605
+ });
606
+ it('should return all entries when entityChangeStatus is empty', () => {
607
+ // When there's no change status data (e.g., no git changes), fall back to auditing everything
608
+ const filtered = filterGlossaryByChangeStatus(allEntries, {});
609
+ expect(filtered).toEqual(allEntries);
610
+ });
611
+ it('should return all entries when entityChangeStatus is undefined', () => {
612
+ const filtered = filterGlossaryByChangeStatus(allEntries, undefined);
613
+ expect(filtered).toEqual(allEntries);
614
+ });
615
+ it('should exclude unchanged entries that have no status', () => {
616
+ const entityChangeStatus = {
617
+ HeroImage: { status: 'new' },
618
+ };
619
+ const filtered = filterGlossaryByChangeStatus(allEntries, entityChangeStatus);
620
+ expect(filtered).toHaveLength(1);
621
+ expect(filtered[0].name).toBe('HeroImage');
622
+ });
623
+ it('should match entries by filePath when name does not match any status key', () => {
624
+ // Sometimes entity names in entityChangeStatus are derived from file paths
625
+ // and may not exactly match glossary names. Fall back to filePath matching.
626
+ const entityChangeStatus = {
627
+ // Entity name derived differently, but filePath matches
628
+ 'app/components/DrinkCard.tsx': { status: 'edited' },
629
+ };
630
+ const filtered = filterGlossaryByChangeStatus(allEntries, entityChangeStatus);
631
+ expect(filtered.map((e) => e.name)).toContain('DrinkCard');
632
+ });
633
+ });
634
+ // ── resolveAuditSessionScope ──────────────────────────────────────
635
+ describe('resolveAuditSessionScope', () => {
636
+ it('should scope to session when entityChangeStatus has entries and featureStartedAt exists', () => {
637
+ const result = resolveAuditSessionScope({
638
+ featureStartedAt: '2026-03-12T14:01:31.291Z',
639
+ entityChangeStatus: {
640
+ ArticleCard: { status: 'new' },
641
+ LibraryHeader: { status: 'new' },
642
+ },
643
+ });
644
+ expect(result.featureStartedAt).toBe('2026-03-12T14:01:31.291Z');
645
+ expect(result.entityChangeStatus).toEqual({
646
+ ArticleCard: { status: 'new' },
647
+ LibraryHeader: { status: 'new' },
648
+ });
649
+ });
650
+ it('should NOT scope scenario counts to session when entityChangeStatus is empty', () => {
651
+ // This is the testapp bug: entityChangeStatus computation returned empty,
652
+ // so the glossary filter falls back to "audit all components", but the
653
+ // scenario count query still filters by featureStartedAt — causing
654
+ // Feature 1 components to appear as "no scenarios" even though they have them.
655
+ const result = resolveAuditSessionScope({
656
+ featureStartedAt: '2026-03-12T14:01:31.291Z',
657
+ entityChangeStatus: undefined,
658
+ });
659
+ // When we can't scope the glossary, we shouldn't scope scenario counts either
660
+ expect(result.featureStartedAt).toBeNull();
661
+ });
662
+ it('should NOT scope scenario counts to session when entityChangeStatus is an empty object', () => {
663
+ const result = resolveAuditSessionScope({
664
+ featureStartedAt: '2026-03-12T14:01:31.291Z',
665
+ entityChangeStatus: {},
666
+ });
667
+ expect(result.featureStartedAt).toBeNull();
668
+ });
669
+ it('should return null featureStartedAt when none was provided', () => {
670
+ const result = resolveAuditSessionScope({
671
+ featureStartedAt: null,
672
+ entityChangeStatus: { Header: { status: 'new' } },
673
+ });
674
+ expect(result.featureStartedAt).toBeNull();
675
+ });
676
+ it('should pass through entityChangeStatus unchanged when it has entries', () => {
677
+ const ecs = {
678
+ ArticleCard: { status: 'new' },
679
+ Header: { status: 'edited' },
680
+ };
681
+ const result = resolveAuditSessionScope({
682
+ featureStartedAt: '2026-03-12T14:01:31.291Z',
683
+ entityChangeStatus: ecs,
684
+ });
685
+ expect(result.entityChangeStatus).toBe(ecs);
686
+ });
687
+ describe('multi-feature audit scenario (testapp reproduction)', () => {
688
+ // Reproduces the exact testapp bug:
689
+ // - Feature 1 built 8 components with scenarios at 13:28-13:30
690
+ // - Feature 2 starts at 14:01, builds 4 new components with scenarios at 14:14-14:16
691
+ // - entityChangeStatus computation returns empty (fails/returns {})
692
+ // - Glossary includes ALL 12 components
693
+ // - Scenario counts only include Feature 2 scenarios (after 14:01)
694
+ // - Result: Feature 1's 8 components reported as "no scenarios"
695
+ const allComponents = [
696
+ // Feature 1 components
697
+ {
698
+ name: 'Header',
699
+ filePath: 'src/components/Header.tsx',
700
+ returnType: 'JSX.Element',
701
+ },
702
+ {
703
+ name: 'Logo',
704
+ filePath: 'src/components/Logo.tsx',
705
+ returnType: 'JSX.Element',
706
+ },
707
+ {
708
+ name: 'TabBar',
709
+ filePath: 'src/components/TabBar.tsx',
710
+ returnType: 'JSX.Element',
711
+ },
712
+ {
713
+ name: 'ArticlePreview',
714
+ filePath: 'src/components/ArticlePreview.tsx',
715
+ returnType: 'JSX.Element',
716
+ },
717
+ {
718
+ name: 'SaveButton',
719
+ filePath: 'src/components/SaveButton.tsx',
720
+ returnType: 'JSX.Element',
721
+ },
722
+ {
723
+ name: 'StatusBanner',
724
+ filePath: 'src/components/StatusBanner.tsx',
725
+ returnType: 'JSX.Element',
726
+ },
727
+ {
728
+ name: 'EmptyState',
729
+ filePath: 'src/components/EmptyState.tsx',
730
+ returnType: 'JSX.Element',
731
+ },
732
+ {
733
+ name: 'ArticleRow',
734
+ filePath: 'src/components/ArticleRow.tsx',
735
+ returnType: 'JSX.Element',
736
+ },
737
+ // Feature 2 components
738
+ {
739
+ name: 'ArticleCard',
740
+ filePath: 'src/components/ArticleCard.tsx',
741
+ returnType: 'JSX.Element',
742
+ },
743
+ {
744
+ name: 'LibraryHeader',
745
+ filePath: 'src/components/LibraryHeader.tsx',
746
+ returnType: 'JSX.Element',
747
+ },
748
+ {
749
+ name: 'ArticleCardGrid',
750
+ filePath: 'src/components/ArticleCardGrid.tsx',
751
+ returnType: 'JSX.Element',
752
+ },
753
+ {
754
+ name: 'OpenLibraryButton',
755
+ filePath: 'src/components/OpenLibraryButton.tsx',
756
+ returnType: 'JSX.Element',
757
+ },
758
+ ];
759
+ // All scenarios that exist in the DB (both features)
760
+ const allScenarioCounts = {
761
+ Header: 2,
762
+ Logo: 1,
763
+ TabBar: 3,
764
+ ArticlePreview: 4,
765
+ SaveButton: 2,
766
+ StatusBanner: 2,
767
+ EmptyState: 2,
768
+ ArticleRow: 3,
769
+ ArticleCard: 4,
770
+ LibraryHeader: 3,
771
+ ArticleCardGrid: 2,
772
+ OpenLibraryButton: 1,
773
+ };
774
+ // Only Feature 2 scenarios (created after featureStartedAt)
775
+ const sessionScopedCounts = {
776
+ ArticleCard: 4,
777
+ LibraryHeader: 3,
778
+ ArticleCardGrid: 2,
779
+ OpenLibraryButton: 1,
780
+ };
781
+ it('should pass audit when entityChangeStatus is empty and all scenarios are counted', () => {
782
+ // With the fix: resolveAuditSessionScope nullifies featureStartedAt
783
+ // when entityChangeStatus is empty, so ALL scenarios are counted
784
+ const scope = resolveAuditSessionScope({
785
+ featureStartedAt: '2026-03-12T14:01:31.291Z',
786
+ entityChangeStatus: undefined,
787
+ });
788
+ // Glossary filter: returns all (no change status to filter by)
789
+ const filtered = filterGlossaryByChangeStatus(allComponents, scope.entityChangeStatus);
790
+ expect(filtered).toHaveLength(12);
791
+ // Since featureStartedAt is now null, the route would query ALL scenarios
792
+ // (not session-scoped), giving us allScenarioCounts
793
+ const countsToUse = scope.featureStartedAt
794
+ ? sessionScopedCounts
795
+ : allScenarioCounts;
796
+ const { components } = classifyGlossaryEntries(filtered);
797
+ const result = computeAudit({
798
+ components,
799
+ functions: [],
800
+ scenarioCounts: countsToUse,
801
+ testFileExistence: {},
802
+ });
803
+ // Every component should have scenarios
804
+ expect(result.summary.componentsMissing).toBe(0);
805
+ expect(result.summary.componentsOk).toBe(12);
806
+ expect(result.summary.allPassing).toBe(true);
807
+ });
808
+ it('demonstrates the bug: session-scoped counts with unscoped glossary fails Feature 1', () => {
809
+ // WITHOUT the fix: entityChangeStatus is empty so all 12 components are audited,
810
+ // but scenario counts are session-scoped so Feature 1 components get 0
811
+ const filtered = filterGlossaryByChangeStatus(allComponents, undefined);
812
+ expect(filtered).toHaveLength(12);
813
+ const { components } = classifyGlossaryEntries(filtered);
814
+ const result = computeAudit({
815
+ components,
816
+ functions: [],
817
+ scenarioCounts: sessionScopedCounts, // BUG: only Feature 2 scenarios
818
+ testFileExistence: {},
819
+ });
820
+ // This is the broken behavior — 8 components incorrectly flagged
821
+ expect(result.summary.componentsMissing).toBe(8);
822
+ expect(result.summary.allPassing).toBe(false);
823
+ });
824
+ it('should correctly scope audit when entityChangeStatus is available', () => {
825
+ // When entityChangeStatus works, only Feature 2 components are audited
826
+ // and only Feature 2 scenarios are counted — both filters agree
827
+ const ecs = {
828
+ ArticleCard: { status: 'new' },
829
+ LibraryHeader: { status: 'new' },
830
+ ArticleCardGrid: { status: 'new' },
831
+ OpenLibraryButton: { status: 'new' },
832
+ };
833
+ const scope = resolveAuditSessionScope({
834
+ featureStartedAt: '2026-03-12T14:01:31.291Z',
835
+ entityChangeStatus: ecs,
836
+ });
837
+ // featureStartedAt preserved — session scoping is valid
838
+ expect(scope.featureStartedAt).toBe('2026-03-12T14:01:31.291Z');
839
+ const filtered = filterGlossaryByChangeStatus(allComponents, scope.entityChangeStatus);
840
+ expect(filtered).toHaveLength(4); // Only Feature 2 components
841
+ const { components } = classifyGlossaryEntries(filtered);
842
+ const result = computeAudit({
843
+ components,
844
+ functions: [],
845
+ scenarioCounts: sessionScopedCounts,
846
+ testFileExistence: {},
847
+ });
848
+ expect(result.summary.componentsMissing).toBe(0);
849
+ expect(result.summary.componentsOk).toBe(4);
850
+ expect(result.summary.allPassing).toBe(true);
851
+ });
852
+ });
853
+ });
854
+ });
855
+ //# sourceMappingURL=editorAudit.test.js.map