@codeyam/codeyam-cli 0.1.0-staging.d0ad4ae → 0.1.0-staging.d57a578

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 (1199) 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 +27 -27
  4. package/analyzer-template/packages/ai/index.ts +21 -5
  5. package/analyzer-template/packages/ai/package.json +3 -3
  6. package/analyzer-template/packages/ai/src/lib/__mocks__/completionCall.ts +122 -0
  7. package/analyzer-template/packages/ai/src/lib/analyzeScope.ts +226 -24
  8. package/analyzer-template/packages/ai/src/lib/astScopes/arrayDerivationDetector.ts +199 -0
  9. package/analyzer-template/packages/ai/src/lib/astScopes/astScopeAnalyzer.ts +217 -13
  10. package/analyzer-template/packages/ai/src/lib/astScopes/conditionalEffectsExtractor.ts +644 -0
  11. package/analyzer-template/packages/ai/src/lib/astScopes/methodSemantics.ts +181 -23
  12. package/analyzer-template/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.ts +10 -17
  13. package/analyzer-template/packages/ai/src/lib/astScopes/patterns/ifStatementHandler.ts +18 -0
  14. package/analyzer-template/packages/ai/src/lib/astScopes/patterns/switchStatementHandler.ts +15 -0
  15. package/analyzer-template/packages/ai/src/lib/astScopes/patterns/variableDeclarationHandler.ts +181 -1
  16. package/analyzer-template/packages/ai/src/lib/astScopes/processExpression.ts +1215 -29
  17. package/analyzer-template/packages/ai/src/lib/astScopes/sharedPatterns.ts +28 -0
  18. package/analyzer-template/packages/ai/src/lib/astScopes/types.ts +265 -6
  19. package/analyzer-template/packages/ai/src/lib/completionCall.ts +247 -66
  20. package/analyzer-template/packages/ai/src/lib/dataStructure/ScopeDataStructure.ts +2020 -334
  21. package/analyzer-template/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.ts +5 -1
  22. package/analyzer-template/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/MuiManager.ts +205 -0
  23. package/analyzer-template/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/ReactFrameworkManager.ts +10 -2
  24. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.ts +16 -3
  25. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.ts +6 -4
  26. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.ts +54 -3
  27. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.ts +129 -20
  28. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.ts +70 -0
  29. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.ts +62 -0
  30. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.ts +140 -14
  31. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.ts +179 -0
  32. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.ts +40 -30
  33. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.ts +393 -90
  34. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/fixNullIdsBySchema.ts +129 -0
  35. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.ts +35 -0
  36. package/analyzer-template/packages/ai/src/lib/dataStructureChunking.ts +183 -0
  37. package/analyzer-template/packages/ai/src/lib/e2eDataTracking.ts +334 -0
  38. package/analyzer-template/packages/ai/src/lib/extractCriticalDataKeys.ts +120 -0
  39. package/analyzer-template/packages/ai/src/lib/generateChangesEntityScenarioData.ts +4 -3
  40. package/analyzer-template/packages/ai/src/lib/generateChangesEntityScenarios.ts +86 -149
  41. package/analyzer-template/packages/ai/src/lib/generateEntityDataStructure.ts +59 -3
  42. package/analyzer-template/packages/ai/src/lib/generateEntityScenarioData.ts +1458 -65
  43. package/analyzer-template/packages/ai/src/lib/generateEntityScenarios.ts +200 -196
  44. package/analyzer-template/packages/ai/src/lib/generateExecutionFlows.ts +677 -0
  45. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromConditionalEffects.ts +528 -0
  46. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromConditionals.ts +2484 -0
  47. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromJsxUsages.ts +239 -0
  48. package/analyzer-template/packages/ai/src/lib/guessScenarioDataFromDescription.ts +5 -5
  49. package/analyzer-template/packages/ai/src/lib/isolateScopes.ts +328 -7
  50. package/analyzer-template/packages/ai/src/lib/mergeJsonTypeDefinitions.ts +5 -0
  51. package/analyzer-template/packages/ai/src/lib/mergeStatements.ts +111 -87
  52. package/analyzer-template/packages/ai/src/lib/promptGenerators/collapseNullableObjects.ts +118 -0
  53. package/analyzer-template/packages/ai/src/lib/promptGenerators/gatherAttributesMap.ts +10 -7
  54. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateChangesEntityScenarioDataGenerator.ts +1 -1
  55. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateChangesEntityScenariosGenerator.ts +28 -170
  56. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateChunkPrompt.ts +82 -0
  57. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateCriticalKeysPrompt.ts +103 -0
  58. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.ts +110 -6
  59. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateEntityScenariosGenerator.ts +14 -89
  60. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateMissingKeysPrompt.ts +58 -0
  61. package/analyzer-template/packages/ai/src/lib/promptGenerators/guessNewScenarioDataFromDescriptionGenerator.ts +11 -11
  62. package/analyzer-template/packages/ai/src/lib/promptGenerators/simplifyKeysForLLM.ts +391 -0
  63. package/analyzer-template/packages/ai/src/lib/resolvePathToControllable.ts +824 -0
  64. package/analyzer-template/packages/ai/src/lib/splitOutsideParentheses.ts +5 -1
  65. package/analyzer-template/packages/ai/src/lib/validateExecutionFlowPaths.ts +531 -0
  66. package/analyzer-template/packages/ai/src/lib/worker/SerializableDataStructure.ts +122 -3
  67. package/analyzer-template/packages/ai/src/lib/worker/analyzeScopeWorker.ts +114 -2
  68. package/analyzer-template/packages/analyze/index.ts +2 -0
  69. package/analyzer-template/packages/analyze/src/lib/FileAnalyzer.ts +65 -59
  70. package/analyzer-template/packages/analyze/src/lib/ProjectAnalyzer.ts +132 -33
  71. package/analyzer-template/packages/analyze/src/lib/analysisContext.ts +44 -4
  72. package/analyzer-template/packages/analyze/src/lib/asts/index.ts +7 -2
  73. package/analyzer-template/packages/analyze/src/lib/asts/nodes/getNodeType.ts +1 -0
  74. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.ts +19 -0
  75. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.ts +19 -0
  76. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllExports.ts +11 -0
  77. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.ts +8 -0
  78. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.ts +49 -1
  79. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.ts +2 -1
  80. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.ts +447 -255
  81. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities.ts +39 -4
  82. package/analyzer-template/packages/analyze/src/lib/files/analyze/findOrCreateEntity.ts +12 -0
  83. package/analyzer-template/packages/analyze/src/lib/files/analyze/gatherEntityMap.ts +4 -2
  84. package/analyzer-template/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.ts +33 -10
  85. package/analyzer-template/packages/analyze/src/lib/files/analyzeChange.ts +14 -14
  86. package/analyzer-template/packages/analyze/src/lib/files/analyzeEntity.ts +4 -4
  87. package/analyzer-template/packages/analyze/src/lib/files/analyzeInitial.ts +11 -12
  88. package/analyzer-template/packages/analyze/src/lib/files/analyzeRemixRoute.ts +4 -5
  89. package/analyzer-template/packages/analyze/src/lib/files/enums/steps.ts +1 -1
  90. package/analyzer-template/packages/analyze/src/lib/files/getImportedExports.ts +14 -12
  91. package/analyzer-template/packages/analyze/src/lib/files/scenarios/TransformationTracer.ts +1352 -0
  92. package/analyzer-template/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.ts +193 -76
  93. package/analyzer-template/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.ts +203 -41
  94. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateChangesScenarios.ts +28 -188
  95. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateDataStructure.ts +355 -23
  96. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.ts +166 -0
  97. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateScenarioData.ts +1 -0
  98. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateScenarios.ts +2 -3
  99. package/analyzer-template/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.ts +845 -72
  100. package/analyzer-template/packages/analyze/src/lib/files/scenarios/mergeValidatedDataStructures.ts +56 -11
  101. package/analyzer-template/packages/analyze/src/lib/files/scenarios/propagateArrayItemSchemas.ts +474 -0
  102. package/analyzer-template/packages/analyze/src/lib/files/setImportedExports.ts +2 -1
  103. package/analyzer-template/packages/analyze/src/lib/index.ts +1 -0
  104. package/analyzer-template/packages/analyze/src/lib/utils/getFileByPath.ts +19 -0
  105. package/analyzer-template/packages/aws/package.json +10 -10
  106. package/analyzer-template/packages/database/index.ts +1 -0
  107. package/analyzer-template/packages/database/package.json +4 -4
  108. package/analyzer-template/packages/database/src/lib/analysisBranchToDb.ts +1 -1
  109. package/analyzer-template/packages/database/src/lib/analysisToDb.ts +1 -1
  110. package/analyzer-template/packages/database/src/lib/branchToDb.ts +1 -1
  111. package/analyzer-template/packages/database/src/lib/commitBranchToDb.ts +1 -1
  112. package/analyzer-template/packages/database/src/lib/commitToDb.ts +1 -1
  113. package/analyzer-template/packages/database/src/lib/fileToDb.ts +1 -1
  114. package/analyzer-template/packages/database/src/lib/kysely/db.ts +22 -1
  115. package/analyzer-template/packages/database/src/lib/kysely/tables/commitsTable.ts +6 -0
  116. package/analyzer-template/packages/database/src/lib/kysely/tables/debugReportsTable.ts +17 -1
  117. package/analyzer-template/packages/database/src/lib/kysely/tables/editorScenariosTable.ts +138 -0
  118. package/analyzer-template/packages/database/src/lib/kysely/tables/labsRequestsTable.ts +52 -0
  119. package/analyzer-template/packages/database/src/lib/loadAnalyses.ts +58 -1
  120. package/analyzer-template/packages/database/src/lib/loadAnalysis.ts +13 -0
  121. package/analyzer-template/packages/database/src/lib/loadBranch.ts +16 -1
  122. package/analyzer-template/packages/database/src/lib/loadCommit.ts +11 -0
  123. package/analyzer-template/packages/database/src/lib/loadCommits.ts +58 -19
  124. package/analyzer-template/packages/database/src/lib/loadEntities.ts +26 -9
  125. package/analyzer-template/packages/database/src/lib/loadEntityBranches.ts +12 -0
  126. package/analyzer-template/packages/database/src/lib/loadReadyToBeCapturedAnalyses.ts +5 -6
  127. package/analyzer-template/packages/database/src/lib/projectToDb.ts +1 -1
  128. package/analyzer-template/packages/database/src/lib/saveFiles.ts +1 -1
  129. package/analyzer-template/packages/database/src/lib/scenarioToDb.ts +1 -1
  130. package/analyzer-template/packages/database/src/lib/updateCommitMetadata.ts +96 -152
  131. package/analyzer-template/packages/database/src/lib/updateFreshAnalysisStatus.ts +58 -42
  132. package/analyzer-template/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.ts +81 -65
  133. package/analyzer-template/packages/database/src/lib/userScenarioToDb.ts +1 -1
  134. package/analyzer-template/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.ts +29 -1
  135. package/analyzer-template/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.ts +33 -5
  136. package/analyzer-template/packages/github/dist/database/index.d.ts +1 -0
  137. package/analyzer-template/packages/github/dist/database/index.d.ts.map +1 -1
  138. package/analyzer-template/packages/github/dist/database/index.js +1 -0
  139. package/analyzer-template/packages/github/dist/database/index.js.map +1 -1
  140. package/analyzer-template/packages/github/dist/database/src/lib/analysisBranchToDb.js +1 -1
  141. package/analyzer-template/packages/github/dist/database/src/lib/analysisBranchToDb.js.map +1 -1
  142. package/analyzer-template/packages/github/dist/database/src/lib/analysisToDb.js +1 -1
  143. package/analyzer-template/packages/github/dist/database/src/lib/analysisToDb.js.map +1 -1
  144. package/analyzer-template/packages/github/dist/database/src/lib/branchToDb.js +1 -1
  145. package/analyzer-template/packages/github/dist/database/src/lib/branchToDb.js.map +1 -1
  146. package/analyzer-template/packages/github/dist/database/src/lib/commitBranchToDb.js +1 -1
  147. package/analyzer-template/packages/github/dist/database/src/lib/commitBranchToDb.js.map +1 -1
  148. package/analyzer-template/packages/github/dist/database/src/lib/commitToDb.js +1 -1
  149. package/analyzer-template/packages/github/dist/database/src/lib/commitToDb.js.map +1 -1
  150. package/analyzer-template/packages/github/dist/database/src/lib/fileToDb.js +1 -1
  151. package/analyzer-template/packages/github/dist/database/src/lib/fileToDb.js.map +1 -1
  152. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts +4 -0
  153. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts.map +1 -1
  154. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js +16 -1
  155. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js.map +1 -1
  156. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/analysesTable.d.ts +1 -18
  157. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/analysesTable.d.ts.map +1 -1
  158. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/commitsTable.d.ts +1 -0
  159. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/commitsTable.d.ts.map +1 -1
  160. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/commitsTable.js +3 -0
  161. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/commitsTable.js.map +1 -1
  162. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/debugReportsTable.d.ts +17 -1
  163. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/debugReportsTable.d.ts.map +1 -1
  164. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/debugReportsTable.js.map +1 -1
  165. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts +27 -0
  166. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts.map +1 -0
  167. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js +121 -0
  168. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -0
  169. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.d.ts +23 -0
  170. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.d.ts.map +1 -0
  171. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.js +35 -0
  172. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.js.map +1 -0
  173. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/scenariosTable.d.ts +6 -6
  174. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/scenariosTable.d.ts.map +1 -1
  175. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalyses.d.ts +2 -0
  176. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalyses.d.ts.map +1 -1
  177. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalyses.js +45 -2
  178. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalyses.js.map +1 -1
  179. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalysis.d.ts.map +1 -1
  180. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalysis.js +8 -0
  181. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalysis.js.map +1 -1
  182. package/analyzer-template/packages/github/dist/database/src/lib/loadBranch.js +11 -1
  183. package/analyzer-template/packages/github/dist/database/src/lib/loadBranch.js.map +1 -1
  184. package/analyzer-template/packages/github/dist/database/src/lib/loadCommit.d.ts.map +1 -1
  185. package/analyzer-template/packages/github/dist/database/src/lib/loadCommit.js +7 -0
  186. package/analyzer-template/packages/github/dist/database/src/lib/loadCommit.js.map +1 -1
  187. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.d.ts +3 -1
  188. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.d.ts.map +1 -1
  189. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.js +45 -14
  190. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.js.map +1 -1
  191. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.d.ts +3 -1
  192. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.d.ts.map +1 -1
  193. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.js +23 -10
  194. package/analyzer-template/packages/github/dist/database/src/lib/loadEntities.js.map +1 -1
  195. package/analyzer-template/packages/github/dist/database/src/lib/loadEntityBranches.d.ts.map +1 -1
  196. package/analyzer-template/packages/github/dist/database/src/lib/loadEntityBranches.js +9 -0
  197. package/analyzer-template/packages/github/dist/database/src/lib/loadEntityBranches.js.map +1 -1
  198. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.d.ts.map +1 -1
  199. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.js +5 -5
  200. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.js.map +1 -1
  201. package/analyzer-template/packages/github/dist/database/src/lib/projectToDb.js +1 -1
  202. package/analyzer-template/packages/github/dist/database/src/lib/projectToDb.js.map +1 -1
  203. package/analyzer-template/packages/github/dist/database/src/lib/saveFiles.js +1 -1
  204. package/analyzer-template/packages/github/dist/database/src/lib/saveFiles.js.map +1 -1
  205. package/analyzer-template/packages/github/dist/database/src/lib/scenarioToDb.js +1 -1
  206. package/analyzer-template/packages/github/dist/database/src/lib/scenarioToDb.js.map +1 -1
  207. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.d.ts +2 -2
  208. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.d.ts.map +1 -1
  209. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.js +76 -89
  210. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.js.map +1 -1
  211. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.d.ts.map +1 -1
  212. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.js +41 -30
  213. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.js.map +1 -1
  214. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.d.ts.map +1 -1
  215. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.js +68 -57
  216. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.js.map +1 -1
  217. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.d.ts.map +1 -1
  218. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js +29 -1
  219. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js.map +1 -1
  220. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.d.ts.map +1 -1
  221. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js +33 -5
  222. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js.map +1 -1
  223. package/analyzer-template/packages/github/dist/types/index.d.ts +3 -4
  224. package/analyzer-template/packages/github/dist/types/index.d.ts.map +1 -1
  225. package/analyzer-template/packages/github/dist/types/index.js +0 -1
  226. package/analyzer-template/packages/github/dist/types/index.js.map +1 -1
  227. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.d.ts +2 -0
  228. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.d.ts.map +1 -1
  229. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.js +2 -0
  230. package/analyzer-template/packages/github/dist/types/src/enums/ProjectFramework.js.map +1 -1
  231. package/analyzer-template/packages/github/dist/types/src/types/Analysis.d.ts +71 -27
  232. package/analyzer-template/packages/github/dist/types/src/types/Analysis.d.ts.map +1 -1
  233. package/analyzer-template/packages/github/dist/types/src/types/Commit.d.ts +2 -0
  234. package/analyzer-template/packages/github/dist/types/src/types/Commit.d.ts.map +1 -1
  235. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts +8 -0
  236. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  237. package/analyzer-template/packages/github/dist/types/src/types/Scenario.d.ts +13 -54
  238. package/analyzer-template/packages/github/dist/types/src/types/Scenario.d.ts.map +1 -1
  239. package/analyzer-template/packages/github/dist/types/src/types/Scenario.js +1 -21
  240. package/analyzer-template/packages/github/dist/types/src/types/Scenario.js.map +1 -1
  241. package/analyzer-template/packages/github/dist/types/src/types/ScenariosDataStructure.d.ts +153 -5
  242. package/analyzer-template/packages/github/dist/types/src/types/ScenariosDataStructure.d.ts.map +1 -1
  243. package/analyzer-template/packages/github/dist/types/src/types/ScopeAnalysis.d.ts +6 -1
  244. package/analyzer-template/packages/github/dist/types/src/types/ScopeAnalysis.d.ts.map +1 -1
  245. package/analyzer-template/packages/github/dist/types/src/types/StatementInfo.d.ts +2 -0
  246. package/analyzer-template/packages/github/dist/types/src/types/StatementInfo.d.ts.map +1 -1
  247. package/analyzer-template/packages/github/dist/utils/src/lib/safeFileName.d.ts +9 -1
  248. package/analyzer-template/packages/github/dist/utils/src/lib/safeFileName.d.ts.map +1 -1
  249. package/analyzer-template/packages/github/dist/utils/src/lib/safeFileName.js +29 -3
  250. package/analyzer-template/packages/github/dist/utils/src/lib/safeFileName.js.map +1 -1
  251. package/analyzer-template/packages/github/package.json +2 -2
  252. package/analyzer-template/packages/types/index.ts +3 -6
  253. package/analyzer-template/packages/types/src/enums/ProjectFramework.ts +2 -0
  254. package/analyzer-template/packages/types/src/types/Analysis.ts +87 -27
  255. package/analyzer-template/packages/types/src/types/Commit.ts +2 -0
  256. package/analyzer-template/packages/types/src/types/ProjectMetadata.ts +8 -0
  257. package/analyzer-template/packages/types/src/types/Scenario.ts +13 -77
  258. package/analyzer-template/packages/types/src/types/ScenariosDataStructure.ts +181 -5
  259. package/analyzer-template/packages/types/src/types/ScopeAnalysis.ts +6 -1
  260. package/analyzer-template/packages/types/src/types/StatementInfo.ts +2 -0
  261. package/analyzer-template/packages/ui-components/package.json +1 -1
  262. package/analyzer-template/packages/utils/dist/types/index.d.ts +3 -4
  263. package/analyzer-template/packages/utils/dist/types/index.d.ts.map +1 -1
  264. package/analyzer-template/packages/utils/dist/types/index.js +0 -1
  265. package/analyzer-template/packages/utils/dist/types/index.js.map +1 -1
  266. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.d.ts +2 -0
  267. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.d.ts.map +1 -1
  268. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.js +2 -0
  269. package/analyzer-template/packages/utils/dist/types/src/enums/ProjectFramework.js.map +1 -1
  270. package/analyzer-template/packages/utils/dist/types/src/types/Analysis.d.ts +71 -27
  271. package/analyzer-template/packages/utils/dist/types/src/types/Analysis.d.ts.map +1 -1
  272. package/analyzer-template/packages/utils/dist/types/src/types/Commit.d.ts +2 -0
  273. package/analyzer-template/packages/utils/dist/types/src/types/Commit.d.ts.map +1 -1
  274. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts +8 -0
  275. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  276. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.d.ts +13 -54
  277. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.d.ts.map +1 -1
  278. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.js +1 -21
  279. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.js.map +1 -1
  280. package/analyzer-template/packages/utils/dist/types/src/types/ScenariosDataStructure.d.ts +153 -5
  281. package/analyzer-template/packages/utils/dist/types/src/types/ScenariosDataStructure.d.ts.map +1 -1
  282. package/analyzer-template/packages/utils/dist/types/src/types/ScopeAnalysis.d.ts +6 -1
  283. package/analyzer-template/packages/utils/dist/types/src/types/ScopeAnalysis.d.ts.map +1 -1
  284. package/analyzer-template/packages/utils/dist/types/src/types/StatementInfo.d.ts +2 -0
  285. package/analyzer-template/packages/utils/dist/types/src/types/StatementInfo.d.ts.map +1 -1
  286. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.d.ts.map +1 -1
  287. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.js +98 -3
  288. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.js.map +1 -1
  289. package/analyzer-template/packages/utils/dist/utils/src/lib/safeFileName.d.ts +9 -1
  290. package/analyzer-template/packages/utils/dist/utils/src/lib/safeFileName.d.ts.map +1 -1
  291. package/analyzer-template/packages/utils/dist/utils/src/lib/safeFileName.js +29 -3
  292. package/analyzer-template/packages/utils/dist/utils/src/lib/safeFileName.js.map +1 -1
  293. package/analyzer-template/packages/utils/src/lib/fs/rsyncCopy.ts +121 -3
  294. package/analyzer-template/packages/utils/src/lib/safeFileName.ts +48 -3
  295. package/analyzer-template/playwright/capture.ts +20 -8
  296. package/analyzer-template/playwright/captureFromUrl.ts +89 -82
  297. package/analyzer-template/playwright/captureStatic.ts +1 -1
  298. package/analyzer-template/playwright/getCodeYamInfo.ts +12 -7
  299. package/analyzer-template/project/analyzeBaselineCommit.ts +9 -0
  300. package/analyzer-template/project/analyzeBranchCommit.ts +4 -0
  301. package/analyzer-template/project/analyzeFileEntities.ts +4 -0
  302. package/analyzer-template/project/analyzeRegularCommit.ts +9 -0
  303. package/analyzer-template/project/captureLibraryFunctionDirect.ts +29 -26
  304. package/analyzer-template/project/constructMockCode.ts +593 -91
  305. package/analyzer-template/project/controller/startController.ts +16 -1
  306. package/analyzer-template/project/createEntitiesAndSortFiles.ts +83 -0
  307. package/analyzer-template/project/executeLibraryFunctionDirect.ts +7 -3
  308. package/analyzer-template/project/loadReadyToBeCaptured.ts +65 -41
  309. package/analyzer-template/project/mocks/analyzeFileMock.ts +8 -7
  310. package/analyzer-template/project/orchestrateCapture/AwsCaptureTaskRunner.ts +12 -4
  311. package/analyzer-template/project/orchestrateCapture/SequentialCaptureTaskRunner.ts +18 -7
  312. package/analyzer-template/project/orchestrateCapture/taskRunner.ts +4 -2
  313. package/analyzer-template/project/orchestrateCapture.ts +75 -7
  314. package/analyzer-template/project/reconcileMockDataKeys.ts +220 -1
  315. package/analyzer-template/project/runAnalysis.ts +6 -0
  316. package/analyzer-template/project/start.ts +49 -12
  317. package/analyzer-template/project/startScenarioCapture.ts +9 -0
  318. package/analyzer-template/project/writeClientLogRoute.ts +125 -0
  319. package/analyzer-template/project/writeMockDataTsx.ts +312 -10
  320. package/analyzer-template/project/writeScenarioComponents.ts +314 -43
  321. package/analyzer-template/project/writeSimpleRoot.ts +21 -11
  322. package/analyzer-template/scripts/comboWorkerLoop.cjs +98 -50
  323. package/analyzer-template/tsconfig.json +14 -1
  324. package/background/src/lib/local/createLocalAnalyzer.js +1 -1
  325. package/background/src/lib/local/createLocalAnalyzer.js.map +1 -1
  326. package/background/src/lib/virtualized/project/analyzeBaselineCommit.js +7 -1
  327. package/background/src/lib/virtualized/project/analyzeBaselineCommit.js.map +1 -1
  328. package/background/src/lib/virtualized/project/analyzeBranchCommit.js +2 -1
  329. package/background/src/lib/virtualized/project/analyzeBranchCommit.js.map +1 -1
  330. package/background/src/lib/virtualized/project/analyzeFileEntities.js +2 -1
  331. package/background/src/lib/virtualized/project/analyzeFileEntities.js.map +1 -1
  332. package/background/src/lib/virtualized/project/analyzeRegularCommit.js +7 -1
  333. package/background/src/lib/virtualized/project/analyzeRegularCommit.js.map +1 -1
  334. package/background/src/lib/virtualized/project/captureLibraryFunctionDirect.js +3 -3
  335. package/background/src/lib/virtualized/project/captureLibraryFunctionDirect.js.map +1 -1
  336. package/background/src/lib/virtualized/project/constructMockCode.js +493 -52
  337. package/background/src/lib/virtualized/project/constructMockCode.js.map +1 -1
  338. package/background/src/lib/virtualized/project/controller/startController.js +11 -1
  339. package/background/src/lib/virtualized/project/controller/startController.js.map +1 -1
  340. package/background/src/lib/virtualized/project/createEntitiesAndSortFiles.js +73 -1
  341. package/background/src/lib/virtualized/project/createEntitiesAndSortFiles.js.map +1 -1
  342. package/background/src/lib/virtualized/project/executeLibraryFunctionDirect.js +6 -3
  343. package/background/src/lib/virtualized/project/executeLibraryFunctionDirect.js.map +1 -1
  344. package/background/src/lib/virtualized/project/loadReadyToBeCaptured.js +19 -8
  345. package/background/src/lib/virtualized/project/loadReadyToBeCaptured.js.map +1 -1
  346. package/background/src/lib/virtualized/project/mocks/analyzeFileMock.js +7 -7
  347. package/background/src/lib/virtualized/project/mocks/analyzeFileMock.js.map +1 -1
  348. package/background/src/lib/virtualized/project/orchestrateCapture/AwsCaptureTaskRunner.js +2 -2
  349. package/background/src/lib/virtualized/project/orchestrateCapture/AwsCaptureTaskRunner.js.map +1 -1
  350. package/background/src/lib/virtualized/project/orchestrateCapture/SequentialCaptureTaskRunner.js +7 -5
  351. package/background/src/lib/virtualized/project/orchestrateCapture/SequentialCaptureTaskRunner.js.map +1 -1
  352. package/background/src/lib/virtualized/project/orchestrateCapture.js +62 -7
  353. package/background/src/lib/virtualized/project/orchestrateCapture.js.map +1 -1
  354. package/background/src/lib/virtualized/project/reconcileMockDataKeys.js +184 -1
  355. package/background/src/lib/virtualized/project/reconcileMockDataKeys.js.map +1 -1
  356. package/background/src/lib/virtualized/project/runAnalysis.js +5 -0
  357. package/background/src/lib/virtualized/project/runAnalysis.js.map +1 -1
  358. package/background/src/lib/virtualized/project/start.js +44 -12
  359. package/background/src/lib/virtualized/project/start.js.map +1 -1
  360. package/background/src/lib/virtualized/project/startScenarioCapture.js +5 -0
  361. package/background/src/lib/virtualized/project/startScenarioCapture.js.map +1 -1
  362. package/background/src/lib/virtualized/project/writeClientLogRoute.js +110 -0
  363. package/background/src/lib/virtualized/project/writeClientLogRoute.js.map +1 -0
  364. package/background/src/lib/virtualized/project/writeMockDataTsx.js +263 -6
  365. package/background/src/lib/virtualized/project/writeMockDataTsx.js.map +1 -1
  366. package/background/src/lib/virtualized/project/writeScenarioComponents.js +237 -41
  367. package/background/src/lib/virtualized/project/writeScenarioComponents.js.map +1 -1
  368. package/background/src/lib/virtualized/project/writeSimpleRoot.js +21 -11
  369. package/background/src/lib/virtualized/project/writeSimpleRoot.js.map +1 -1
  370. package/codeyam-cli/scripts/apply-setup.js +386 -9
  371. package/codeyam-cli/scripts/apply-setup.js.map +1 -1
  372. package/codeyam-cli/src/__tests__/memory-scripts/filter-session.test.js +196 -0
  373. package/codeyam-cli/src/__tests__/memory-scripts/filter-session.test.js.map +1 -0
  374. package/codeyam-cli/src/__tests__/memory-scripts/read-json-field.test.js +114 -0
  375. package/codeyam-cli/src/__tests__/memory-scripts/read-json-field.test.js.map +1 -0
  376. package/codeyam-cli/src/__tests__/memory-scripts/ripgrep-fallback.test.js +149 -0
  377. package/codeyam-cli/src/__tests__/memory-scripts/ripgrep-fallback.test.js.map +1 -0
  378. package/codeyam-cli/src/cli.js +35 -24
  379. package/codeyam-cli/src/cli.js.map +1 -1
  380. package/codeyam-cli/src/codeyam-cli.js +18 -2
  381. package/codeyam-cli/src/codeyam-cli.js.map +1 -1
  382. package/codeyam-cli/src/commands/__tests__/editor.stepDispatch.test.js +45 -0
  383. package/codeyam-cli/src/commands/__tests__/editor.stepDispatch.test.js.map +1 -0
  384. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js +101 -47
  385. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js.map +1 -1
  386. package/codeyam-cli/src/commands/analyze.js +21 -9
  387. package/codeyam-cli/src/commands/analyze.js.map +1 -1
  388. package/codeyam-cli/src/commands/baseline.js +10 -11
  389. package/codeyam-cli/src/commands/baseline.js.map +1 -1
  390. package/codeyam-cli/src/commands/debug.js +37 -23
  391. package/codeyam-cli/src/commands/debug.js.map +1 -1
  392. package/codeyam-cli/src/commands/default.js +43 -35
  393. package/codeyam-cli/src/commands/default.js.map +1 -1
  394. package/codeyam-cli/src/commands/editor.js +3375 -0
  395. package/codeyam-cli/src/commands/editor.js.map +1 -0
  396. package/codeyam-cli/src/commands/init.js +146 -292
  397. package/codeyam-cli/src/commands/init.js.map +1 -1
  398. package/codeyam-cli/src/commands/memory.js +278 -0
  399. package/codeyam-cli/src/commands/memory.js.map +1 -0
  400. package/codeyam-cli/src/commands/recapture.js +31 -18
  401. package/codeyam-cli/src/commands/recapture.js.map +1 -1
  402. package/codeyam-cli/src/commands/report.js +46 -1
  403. package/codeyam-cli/src/commands/report.js.map +1 -1
  404. package/codeyam-cli/src/commands/setup-sandbox.js +2 -0
  405. package/codeyam-cli/src/commands/setup-sandbox.js.map +1 -1
  406. package/codeyam-cli/src/commands/setup-simulations.js +284 -0
  407. package/codeyam-cli/src/commands/setup-simulations.js.map +1 -0
  408. package/codeyam-cli/src/commands/start.js +8 -12
  409. package/codeyam-cli/src/commands/start.js.map +1 -1
  410. package/codeyam-cli/src/commands/test-startup.js +2 -0
  411. package/codeyam-cli/src/commands/test-startup.js.map +1 -1
  412. package/codeyam-cli/src/commands/verify.js +14 -2
  413. package/codeyam-cli/src/commands/verify.js.map +1 -1
  414. package/codeyam-cli/src/data/techStacks.js +77 -0
  415. package/codeyam-cli/src/data/techStacks.js.map +1 -0
  416. package/codeyam-cli/src/utils/__tests__/analyzerFinalization.test.js +173 -0
  417. package/codeyam-cli/src/utils/__tests__/analyzerFinalization.test.js.map +1 -0
  418. package/codeyam-cli/src/utils/__tests__/backgroundServer.test.js +46 -0
  419. package/codeyam-cli/src/utils/__tests__/backgroundServer.test.js.map +1 -0
  420. package/codeyam-cli/src/utils/__tests__/devServerState.test.js +134 -0
  421. package/codeyam-cli/src/utils/__tests__/devServerState.test.js.map +1 -0
  422. package/codeyam-cli/src/utils/__tests__/editorApi.test.js +137 -0
  423. package/codeyam-cli/src/utils/__tests__/editorApi.test.js.map +1 -0
  424. package/codeyam-cli/src/utils/__tests__/editorAudit.test.js +987 -0
  425. package/codeyam-cli/src/utils/__tests__/editorAudit.test.js.map +1 -0
  426. package/codeyam-cli/src/utils/__tests__/editorCapture.test.js +93 -0
  427. package/codeyam-cli/src/utils/__tests__/editorCapture.test.js.map +1 -0
  428. package/codeyam-cli/src/utils/__tests__/editorDevServer.test.js +304 -0
  429. package/codeyam-cli/src/utils/__tests__/editorDevServer.test.js.map +1 -0
  430. package/codeyam-cli/src/utils/__tests__/editorEntityChangeStatus.test.js +121 -0
  431. package/codeyam-cli/src/utils/__tests__/editorEntityChangeStatus.test.js.map +1 -0
  432. package/codeyam-cli/src/utils/__tests__/editorImageVerifier.test.js +294 -0
  433. package/codeyam-cli/src/utils/__tests__/editorImageVerifier.test.js.map +1 -0
  434. package/codeyam-cli/src/utils/__tests__/editorJournal.test.js +542 -0
  435. package/codeyam-cli/src/utils/__tests__/editorJournal.test.js.map +1 -0
  436. package/codeyam-cli/src/utils/__tests__/editorLoaderHelpers.test.js +520 -0
  437. package/codeyam-cli/src/utils/__tests__/editorLoaderHelpers.test.js.map +1 -0
  438. package/codeyam-cli/src/utils/__tests__/editorMockState.test.js +270 -0
  439. package/codeyam-cli/src/utils/__tests__/editorMockState.test.js.map +1 -0
  440. package/codeyam-cli/src/utils/__tests__/editorPreloadHelpers.test.js +217 -0
  441. package/codeyam-cli/src/utils/__tests__/editorPreloadHelpers.test.js.map +1 -0
  442. package/codeyam-cli/src/utils/__tests__/editorPreview.test.js +353 -0
  443. package/codeyam-cli/src/utils/__tests__/editorPreview.test.js.map +1 -0
  444. package/codeyam-cli/src/utils/__tests__/editorProxySession.test.js +153 -0
  445. package/codeyam-cli/src/utils/__tests__/editorProxySession.test.js.map +1 -0
  446. package/codeyam-cli/src/utils/__tests__/editorScenarioLookup.test.js +139 -0
  447. package/codeyam-cli/src/utils/__tests__/editorScenarioLookup.test.js.map +1 -0
  448. package/codeyam-cli/src/utils/__tests__/editorScenarioSwitch.test.js +221 -0
  449. package/codeyam-cli/src/utils/__tests__/editorScenarioSwitch.test.js.map +1 -0
  450. package/codeyam-cli/src/utils/__tests__/editorScenarios.test.js +1059 -0
  451. package/codeyam-cli/src/utils/__tests__/editorScenarios.test.js.map +1 -0
  452. package/codeyam-cli/src/utils/__tests__/editorSeedAdapter.test.js +213 -0
  453. package/codeyam-cli/src/utils/__tests__/editorSeedAdapter.test.js.map +1 -0
  454. package/codeyam-cli/src/utils/__tests__/entityChangeStatus.test.js +1742 -0
  455. package/codeyam-cli/src/utils/__tests__/entityChangeStatus.test.js.map +1 -0
  456. package/codeyam-cli/src/utils/__tests__/git.editor.test.js +134 -0
  457. package/codeyam-cli/src/utils/__tests__/git.editor.test.js.map +1 -0
  458. package/codeyam-cli/src/utils/__tests__/journalCaptureStabilization.test.js +107 -0
  459. package/codeyam-cli/src/utils/__tests__/journalCaptureStabilization.test.js.map +1 -0
  460. package/codeyam-cli/src/utils/__tests__/npmVersionCheck.test.js +185 -0
  461. package/codeyam-cli/src/utils/__tests__/npmVersionCheck.test.js.map +1 -0
  462. package/codeyam-cli/src/utils/__tests__/parseRegisterArg.test.js +129 -0
  463. package/codeyam-cli/src/utils/__tests__/parseRegisterArg.test.js.map +1 -0
  464. package/codeyam-cli/src/utils/__tests__/pathIgnoring.test.js +9 -0
  465. package/codeyam-cli/src/utils/__tests__/pathIgnoring.test.js.map +1 -1
  466. package/codeyam-cli/src/utils/__tests__/project.test.js +65 -0
  467. package/codeyam-cli/src/utils/__tests__/project.test.js.map +1 -0
  468. package/codeyam-cli/src/utils/__tests__/scenarioCoverage.test.js +227 -0
  469. package/codeyam-cli/src/utils/__tests__/scenarioCoverage.test.js.map +1 -0
  470. package/codeyam-cli/src/utils/__tests__/scenarioMarkers.test.js +121 -0
  471. package/codeyam-cli/src/utils/__tests__/scenarioMarkers.test.js.map +1 -0
  472. package/codeyam-cli/src/utils/__tests__/scenariosManifest.test.js +454 -0
  473. package/codeyam-cli/src/utils/__tests__/scenariosManifest.test.js.map +1 -0
  474. package/codeyam-cli/src/utils/__tests__/serverVersionStaleness.test.js +81 -0
  475. package/codeyam-cli/src/utils/__tests__/serverVersionStaleness.test.js.map +1 -0
  476. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js +174 -82
  477. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js.map +1 -1
  478. package/codeyam-cli/src/utils/__tests__/templateConsistency.test.js +51 -0
  479. package/codeyam-cli/src/utils/__tests__/templateConsistency.test.js.map +1 -0
  480. package/codeyam-cli/src/utils/__tests__/webappDetection.test.js +142 -0
  481. package/codeyam-cli/src/utils/__tests__/webappDetection.test.js.map +1 -0
  482. package/codeyam-cli/src/utils/analysisRunner.js +29 -15
  483. package/codeyam-cli/src/utils/analysisRunner.js.map +1 -1
  484. package/codeyam-cli/src/utils/analyzer.js +16 -0
  485. package/codeyam-cli/src/utils/analyzer.js.map +1 -1
  486. package/codeyam-cli/src/utils/analyzerFinalization.js +100 -0
  487. package/codeyam-cli/src/utils/analyzerFinalization.js.map +1 -0
  488. package/codeyam-cli/src/utils/backgroundServer.js +202 -29
  489. package/codeyam-cli/src/utils/backgroundServer.js.map +1 -1
  490. package/codeyam-cli/src/utils/buildFlags.js +4 -0
  491. package/codeyam-cli/src/utils/buildFlags.js.map +1 -0
  492. package/codeyam-cli/src/utils/database.js +37 -2
  493. package/codeyam-cli/src/utils/database.js.map +1 -1
  494. package/codeyam-cli/src/utils/devModeEvents.js +40 -0
  495. package/codeyam-cli/src/utils/devModeEvents.js.map +1 -0
  496. package/codeyam-cli/src/utils/devServerState.js +71 -0
  497. package/codeyam-cli/src/utils/devServerState.js.map +1 -0
  498. package/codeyam-cli/src/utils/editorApi.js +79 -0
  499. package/codeyam-cli/src/utils/editorApi.js.map +1 -0
  500. package/codeyam-cli/src/utils/editorAudit.js +210 -0
  501. package/codeyam-cli/src/utils/editorAudit.js.map +1 -0
  502. package/codeyam-cli/src/utils/editorCapture.js +102 -0
  503. package/codeyam-cli/src/utils/editorCapture.js.map +1 -0
  504. package/codeyam-cli/src/utils/editorDevServer.js +197 -0
  505. package/codeyam-cli/src/utils/editorDevServer.js.map +1 -0
  506. package/codeyam-cli/src/utils/editorEntityChangeStatus.js +44 -0
  507. package/codeyam-cli/src/utils/editorEntityChangeStatus.js.map +1 -0
  508. package/codeyam-cli/src/utils/editorImageVerifier.js +155 -0
  509. package/codeyam-cli/src/utils/editorImageVerifier.js.map +1 -0
  510. package/codeyam-cli/src/utils/editorJournal.js +225 -0
  511. package/codeyam-cli/src/utils/editorJournal.js.map +1 -0
  512. package/codeyam-cli/src/utils/editorLoaderHelpers.js +113 -0
  513. package/codeyam-cli/src/utils/editorLoaderHelpers.js.map +1 -0
  514. package/codeyam-cli/src/utils/editorMockState.js +248 -0
  515. package/codeyam-cli/src/utils/editorMockState.js.map +1 -0
  516. package/codeyam-cli/src/utils/editorPreloadHelpers.js +135 -0
  517. package/codeyam-cli/src/utils/editorPreloadHelpers.js.map +1 -0
  518. package/codeyam-cli/src/utils/editorPreview.js +137 -0
  519. package/codeyam-cli/src/utils/editorPreview.js.map +1 -0
  520. package/codeyam-cli/src/utils/editorScenarioSwitch.js +112 -0
  521. package/codeyam-cli/src/utils/editorScenarioSwitch.js.map +1 -0
  522. package/codeyam-cli/src/utils/editorScenarios.js +387 -0
  523. package/codeyam-cli/src/utils/editorScenarios.js.map +1 -0
  524. package/codeyam-cli/src/utils/editorSeedAdapter.js +173 -0
  525. package/codeyam-cli/src/utils/editorSeedAdapter.js.map +1 -0
  526. package/codeyam-cli/src/utils/entityChangeStatus.js +349 -0
  527. package/codeyam-cli/src/utils/entityChangeStatus.js.map +1 -0
  528. package/codeyam-cli/src/utils/entityChangeStatus.server.js +158 -0
  529. package/codeyam-cli/src/utils/entityChangeStatus.server.js.map +1 -0
  530. package/codeyam-cli/src/utils/fileMetadata.js +5 -0
  531. package/codeyam-cli/src/utils/fileMetadata.js.map +1 -1
  532. package/codeyam-cli/src/utils/fileWatcher.js +25 -9
  533. package/codeyam-cli/src/utils/fileWatcher.js.map +1 -1
  534. package/codeyam-cli/src/utils/generateReport.js +4 -3
  535. package/codeyam-cli/src/utils/generateReport.js.map +1 -1
  536. package/codeyam-cli/src/utils/git.js +103 -0
  537. package/codeyam-cli/src/utils/git.js.map +1 -1
  538. package/codeyam-cli/src/utils/install-skills.js +120 -39
  539. package/codeyam-cli/src/utils/install-skills.js.map +1 -1
  540. package/codeyam-cli/src/utils/interactiveSyncWatcher.js +126 -0
  541. package/codeyam-cli/src/utils/interactiveSyncWatcher.js.map +1 -0
  542. package/codeyam-cli/src/utils/labsAutoCheck.js +19 -0
  543. package/codeyam-cli/src/utils/labsAutoCheck.js.map +1 -0
  544. package/codeyam-cli/src/utils/npmVersionCheck.js +76 -0
  545. package/codeyam-cli/src/utils/npmVersionCheck.js.map +1 -0
  546. package/codeyam-cli/src/utils/parseRegisterArg.js +31 -0
  547. package/codeyam-cli/src/utils/parseRegisterArg.js.map +1 -0
  548. package/codeyam-cli/src/utils/pathIgnoring.js +19 -7
  549. package/codeyam-cli/src/utils/pathIgnoring.js.map +1 -1
  550. package/codeyam-cli/src/utils/progress.js +8 -1
  551. package/codeyam-cli/src/utils/progress.js.map +1 -1
  552. package/codeyam-cli/src/utils/project.js +15 -5
  553. package/codeyam-cli/src/utils/project.js.map +1 -1
  554. package/codeyam-cli/src/utils/queue/__tests__/heartbeat.test.js +11 -11
  555. package/codeyam-cli/src/utils/queue/__tests__/heartbeat.test.js.map +1 -1
  556. package/codeyam-cli/src/utils/queue/__tests__/manager.test.js +22 -0
  557. package/codeyam-cli/src/utils/queue/__tests__/manager.test.js.map +1 -1
  558. package/codeyam-cli/src/utils/queue/heartbeat.js +13 -5
  559. package/codeyam-cli/src/utils/queue/heartbeat.js.map +1 -1
  560. package/codeyam-cli/src/utils/queue/job.js +75 -1
  561. package/codeyam-cli/src/utils/queue/job.js.map +1 -1
  562. package/codeyam-cli/src/utils/queue/manager.js +7 -0
  563. package/codeyam-cli/src/utils/queue/manager.js.map +1 -1
  564. package/codeyam-cli/src/utils/requireSimulations.js +10 -0
  565. package/codeyam-cli/src/utils/requireSimulations.js.map +1 -0
  566. package/codeyam-cli/src/utils/ruleReflection/__tests__/confusionDetector.test.js +82 -0
  567. package/codeyam-cli/src/utils/ruleReflection/__tests__/confusionDetector.test.js.map +1 -0
  568. package/codeyam-cli/src/utils/ruleReflection/__tests__/contextBuilder.test.js +229 -0
  569. package/codeyam-cli/src/utils/ruleReflection/__tests__/contextBuilder.test.js.map +1 -0
  570. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/assertRules.js +67 -0
  571. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/assertRules.js.map +1 -0
  572. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/captureFixture.js +105 -0
  573. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/captureFixture.js.map +1 -0
  574. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/loadCapturedFixture.js +34 -0
  575. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/loadCapturedFixture.js.map +1 -0
  576. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/runClaude.js +162 -0
  577. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/runClaude.js.map +1 -0
  578. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/setupTempProject.js +74 -0
  579. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/setupTempProject.js.map +1 -0
  580. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/ruleReflectionE2E.test.js +376 -0
  581. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/ruleReflectionE2E.test.js.map +1 -0
  582. package/codeyam-cli/src/utils/ruleReflection/__tests__/promptBuilder.test.js +113 -0
  583. package/codeyam-cli/src/utils/ruleReflection/__tests__/promptBuilder.test.js.map +1 -0
  584. package/codeyam-cli/src/utils/ruleReflection/__tests__/transcriptParser.test.js +127 -0
  585. package/codeyam-cli/src/utils/ruleReflection/__tests__/transcriptParser.test.js.map +1 -0
  586. package/codeyam-cli/src/utils/ruleReflection/confusionDetector.js +50 -0
  587. package/codeyam-cli/src/utils/ruleReflection/confusionDetector.js.map +1 -0
  588. package/codeyam-cli/src/utils/ruleReflection/contextBuilder.js +116 -0
  589. package/codeyam-cli/src/utils/ruleReflection/contextBuilder.js.map +1 -0
  590. package/codeyam-cli/src/utils/ruleReflection/index.js +5 -0
  591. package/codeyam-cli/src/utils/ruleReflection/index.js.map +1 -0
  592. package/codeyam-cli/src/utils/ruleReflection/promptBuilder.js +44 -0
  593. package/codeyam-cli/src/utils/ruleReflection/promptBuilder.js.map +1 -0
  594. package/codeyam-cli/src/utils/ruleReflection/transcriptParser.js +85 -0
  595. package/codeyam-cli/src/utils/ruleReflection/transcriptParser.js.map +1 -0
  596. package/codeyam-cli/src/utils/ruleReflection/types.js +5 -0
  597. package/codeyam-cli/src/utils/ruleReflection/types.js.map +1 -0
  598. package/codeyam-cli/src/utils/rules/__tests__/parser.test.js +83 -0
  599. package/codeyam-cli/src/utils/rules/__tests__/parser.test.js.map +1 -0
  600. package/codeyam-cli/src/utils/rules/__tests__/pathMatcher.test.js +118 -0
  601. package/codeyam-cli/src/utils/rules/__tests__/pathMatcher.test.js.map +1 -0
  602. package/codeyam-cli/src/utils/rules/__tests__/rulePlacement.test.js +72 -0
  603. package/codeyam-cli/src/utils/rules/__tests__/rulePlacement.test.js.map +1 -0
  604. package/codeyam-cli/src/utils/rules/__tests__/ruleState.test.js +293 -0
  605. package/codeyam-cli/src/utils/rules/__tests__/ruleState.test.js.map +1 -0
  606. package/codeyam-cli/src/utils/rules/__tests__/sourceFiles.test.js +76 -0
  607. package/codeyam-cli/src/utils/rules/__tests__/sourceFiles.test.js.map +1 -0
  608. package/codeyam-cli/src/utils/rules/index.js +7 -0
  609. package/codeyam-cli/src/utils/rules/index.js.map +1 -0
  610. package/codeyam-cli/src/utils/rules/parser.js +93 -0
  611. package/codeyam-cli/src/utils/rules/parser.js.map +1 -0
  612. package/codeyam-cli/src/utils/rules/pathMatcher.js +49 -0
  613. package/codeyam-cli/src/utils/rules/pathMatcher.js.map +1 -0
  614. package/codeyam-cli/src/utils/rules/rulePlacement.js +65 -0
  615. package/codeyam-cli/src/utils/rules/rulePlacement.js.map +1 -0
  616. package/codeyam-cli/src/utils/rules/ruleState.js +150 -0
  617. package/codeyam-cli/src/utils/rules/ruleState.js.map +1 -0
  618. package/codeyam-cli/src/utils/rules/sourceFiles.js +43 -0
  619. package/codeyam-cli/src/utils/rules/sourceFiles.js.map +1 -0
  620. package/codeyam-cli/src/utils/rules/staleness.js +137 -0
  621. package/codeyam-cli/src/utils/rules/staleness.js.map +1 -0
  622. package/codeyam-cli/src/utils/scenarioCoverage.js +75 -0
  623. package/codeyam-cli/src/utils/scenarioCoverage.js.map +1 -0
  624. package/codeyam-cli/src/utils/scenarioMarkers.js +134 -0
  625. package/codeyam-cli/src/utils/scenarioMarkers.js.map +1 -0
  626. package/codeyam-cli/src/utils/scenariosManifest.js +241 -0
  627. package/codeyam-cli/src/utils/scenariosManifest.js.map +1 -0
  628. package/codeyam-cli/src/utils/serverState.js +94 -12
  629. package/codeyam-cli/src/utils/serverState.js.map +1 -1
  630. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js +95 -45
  631. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js.map +1 -1
  632. package/codeyam-cli/src/utils/simulationGateMiddleware.js +166 -0
  633. package/codeyam-cli/src/utils/simulationGateMiddleware.js.map +1 -0
  634. package/codeyam-cli/src/utils/slugUtils.js +25 -0
  635. package/codeyam-cli/src/utils/slugUtils.js.map +1 -0
  636. package/codeyam-cli/src/utils/syncMocksMiddleware.js +7 -26
  637. package/codeyam-cli/src/utils/syncMocksMiddleware.js.map +1 -1
  638. package/codeyam-cli/src/utils/testRunner.js +158 -0
  639. package/codeyam-cli/src/utils/testRunner.js.map +1 -0
  640. package/codeyam-cli/src/utils/transcriptPruning.js +67 -0
  641. package/codeyam-cli/src/utils/transcriptPruning.js.map +1 -0
  642. package/codeyam-cli/src/utils/versionInfo.js +67 -15
  643. package/codeyam-cli/src/utils/versionInfo.js.map +1 -1
  644. package/codeyam-cli/src/utils/webappDetection.js +35 -2
  645. package/codeyam-cli/src/utils/webappDetection.js.map +1 -1
  646. package/codeyam-cli/src/webserver/__tests__/clientErrors.test.js +40 -0
  647. package/codeyam-cli/src/webserver/__tests__/clientErrors.test.js.map +1 -0
  648. package/codeyam-cli/src/webserver/__tests__/dependency-smoke.test.js +66 -0
  649. package/codeyam-cli/src/webserver/__tests__/dependency-smoke.test.js.map +1 -0
  650. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js +567 -0
  651. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js.map +1 -0
  652. package/codeyam-cli/src/webserver/__tests__/idleDetector.test.js +146 -0
  653. package/codeyam-cli/src/webserver/__tests__/idleDetector.test.js.map +1 -0
  654. package/codeyam-cli/src/webserver/app/lib/clientErrors.js +65 -0
  655. package/codeyam-cli/src/webserver/app/lib/clientErrors.js.map +1 -0
  656. package/codeyam-cli/src/webserver/app/lib/database.js +63 -33
  657. package/codeyam-cli/src/webserver/app/lib/database.js.map +1 -1
  658. package/codeyam-cli/src/webserver/app/lib/dbNotifier.js.map +1 -1
  659. package/codeyam-cli/src/webserver/app/lib/git.js +397 -0
  660. package/codeyam-cli/src/webserver/app/lib/git.js.map +1 -0
  661. package/codeyam-cli/src/webserver/backgroundServer.js +166 -16
  662. package/codeyam-cli/src/webserver/backgroundServer.js.map +1 -1
  663. package/codeyam-cli/src/webserver/bootstrap.js +51 -0
  664. package/codeyam-cli/src/webserver/bootstrap.js.map +1 -1
  665. package/codeyam-cli/src/webserver/build/client/assets/CopyButton-BPXZwM4t.js +1 -0
  666. package/codeyam-cli/src/webserver/build/client/assets/EntityItem-BcgbViKV.js +11 -0
  667. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeBadge-DLqD3qNt.js → EntityTypeBadge-g3saevPb.js} +1 -1
  668. package/codeyam-cli/src/webserver/build/client/assets/EntityTypeIcon-CQIG2qda.js +41 -0
  669. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-Bu6c6aDe.js +1 -0
  670. package/codeyam-cli/src/webserver/build/client/assets/InteractivePreview-DYFW3lDD.js +25 -0
  671. package/codeyam-cli/src/webserver/build/client/assets/{LibraryFunctionPreview-CVtiBnY5.js → LibraryFunctionPreview-DLeucoVX.js} +1 -1
  672. package/codeyam-cli/src/webserver/build/client/assets/{LoadingDots-B0GLXMsr.js → LoadingDots-BU_OAEMP.js} +1 -1
  673. package/codeyam-cli/src/webserver/build/client/assets/{LogViewer-xgeCVgSM.js → LogViewer-ceAyBX-H.js} +1 -1
  674. package/codeyam-cli/src/webserver/build/client/assets/ReportIssueModal-BzHcG7SE.js +11 -0
  675. package/codeyam-cli/src/webserver/build/client/assets/{SafeScreenshot-DuDvi0jm.js → SafeScreenshot-BED4B6sP.js} +1 -1
  676. package/codeyam-cli/src/webserver/build/client/assets/ScenarioViewer-TSD3C211.js +10 -0
  677. package/codeyam-cli/src/webserver/build/client/assets/Spinner-Bb5uFQ5V.js +34 -0
  678. package/codeyam-cli/src/webserver/build/client/assets/{TruncatedFilePath-DyFZkK0l.js → TruncatedFilePath-C8OKAR5x.js} +1 -1
  679. package/codeyam-cli/src/webserver/build/client/assets/ViewportInspectBar-oAf2Kqsf.js +1 -0
  680. package/codeyam-cli/src/webserver/build/client/assets/_index-DLxKhri3.js +11 -0
  681. package/codeyam-cli/src/webserver/build/client/assets/activity.(_tab)-BcY3q6nt.js +27 -0
  682. package/codeyam-cli/src/webserver/build/client/assets/addon-canvas-DpzMmAy5.js +1 -0
  683. package/codeyam-cli/src/webserver/build/client/assets/addon-fit-YJmn1quW.js +12 -0
  684. package/codeyam-cli/src/webserver/build/client/assets/addon-web-links-Duc5hnl7.js +1 -0
  685. package/codeyam-cli/src/webserver/build/client/assets/addon-webgl-DI8QOUvO.js +58 -0
  686. package/codeyam-cli/src/webserver/build/client/assets/agent-transcripts-Bni3iiUj.js +22 -0
  687. package/codeyam-cli/src/webserver/build/client/assets/api.editor-capture-scenario-l0sNRNKZ.js +1 -0
  688. package/codeyam-cli/src/webserver/build/client/assets/api.editor-client-errors-l0sNRNKZ.js +1 -0
  689. package/codeyam-cli/src/webserver/build/client/assets/api.editor-commit-l0sNRNKZ.js +1 -0
  690. package/codeyam-cli/src/webserver/build/client/assets/api.editor-dev-server-l0sNRNKZ.js +1 -0
  691. package/codeyam-cli/src/webserver/build/client/assets/api.editor-entity-status-l0sNRNKZ.js +1 -0
  692. package/codeyam-cli/src/webserver/build/client/assets/api.editor-file-diff-l0sNRNKZ.js +1 -0
  693. package/codeyam-cli/src/webserver/build/client/assets/api.editor-file-l0sNRNKZ.js +1 -0
  694. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-entry-l0sNRNKZ.js +1 -0
  695. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-image._-l0sNRNKZ.js +1 -0
  696. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-l0sNRNKZ.js +1 -0
  697. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-screenshot-l0sNRNKZ.js +1 -0
  698. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-update-l0sNRNKZ.js +1 -0
  699. package/codeyam-cli/src/webserver/build/client/assets/api.editor-load-commit-l0sNRNKZ.js +1 -0
  700. package/codeyam-cli/src/webserver/build/client/assets/api.editor-project-info-l0sNRNKZ.js +1 -0
  701. package/codeyam-cli/src/webserver/build/client/assets/api.editor-refresh-l0sNRNKZ.js +1 -0
  702. package/codeyam-cli/src/webserver/build/client/assets/api.editor-register-scenario-l0sNRNKZ.js +1 -0
  703. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-coverage-l0sNRNKZ.js +1 -0
  704. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-data-l0sNRNKZ.js +1 -0
  705. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-image._-l0sNRNKZ.js +1 -0
  706. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenarios-l0sNRNKZ.js +1 -0
  707. package/codeyam-cli/src/webserver/build/client/assets/api.editor-switch-scenario-l0sNRNKZ.js +1 -0
  708. package/codeyam-cli/src/webserver/build/client/assets/api.editor-test-results-l0sNRNKZ.js +1 -0
  709. package/codeyam-cli/src/webserver/build/client/assets/api.health-l0sNRNKZ.js +1 -0
  710. package/codeyam-cli/src/webserver/build/client/assets/api.labs-unlock-l0sNRNKZ.js +1 -0
  711. package/codeyam-cli/src/webserver/build/client/assets/api.memory-profile-l0sNRNKZ.js +1 -0
  712. package/codeyam-cli/src/webserver/build/client/assets/api.restart-server-l0sNRNKZ.js +1 -0
  713. package/codeyam-cli/src/webserver/build/client/assets/api.rule-path-l0sNRNKZ.js +1 -0
  714. package/codeyam-cli/src/webserver/build/client/assets/api.save-fixture-l0sNRNKZ.js +1 -0
  715. package/codeyam-cli/src/webserver/build/client/assets/book-open-BYOypzCa.js +6 -0
  716. package/codeyam-cli/src/webserver/build/client/assets/{chevron-down-Cx24_aWc.js → chevron-down-C_Pmso5S.js} +2 -2
  717. package/codeyam-cli/src/webserver/build/client/assets/{chunk-EPOLDU6W-CXRTFQ3F.js → chunk-JZWAC4HX-C4pqxYJB.js} +12 -12
  718. package/codeyam-cli/src/webserver/build/client/assets/{circle-check-BOARzkeR.js → circle-check-BVMi9VA5.js} +2 -2
  719. package/codeyam-cli/src/webserver/build/client/assets/copy-n2FB0_Sw.js +11 -0
  720. package/codeyam-cli/src/webserver/build/client/assets/createLucideIcon-CC6AbExI.js +41 -0
  721. package/codeyam-cli/src/webserver/build/client/assets/{cy-logo-cli-C1gnJVOL.svg → cy-logo-cli-CCKUIm0S.svg} +2 -2
  722. package/codeyam-cli/src/webserver/build/client/assets/cy-logo-cli-DcX-ZS3p.js +1 -0
  723. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-Ii3inc0_.js +1 -0
  724. package/codeyam-cli/src/webserver/build/client/assets/editor-COWCNVyV.js +10 -0
  725. package/codeyam-cli/src/webserver/build/client/assets/editorPreview-CNB06EIa.js +41 -0
  726. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha._-D0-YwkBh.js → entity._sha._-DwCV5__E.js} +13 -13
  727. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.dev-CXSi2aeZ.js +6 -0
  728. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-CHMiAog3.js +6 -0
  729. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-p9hhkjJM.js +6 -0
  730. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha_.edit._scenarioId-C1H_a_Y3.js → entity._sha_.edit._scenarioId-BMvVHNXU.js} +2 -2
  731. package/codeyam-cli/src/webserver/build/client/assets/{entry.client-CS2cb_eZ.js → entry.client-DTvKq3TY.js} +1 -1
  732. package/codeyam-cli/src/webserver/build/client/assets/executionFlowCoverage-BWhdfn70.js +1 -0
  733. package/codeyam-cli/src/webserver/build/client/assets/{fileTableUtils-DMJ7zii9.js → fileTableUtils-cPo8LiG3.js} +1 -1
  734. package/codeyam-cli/src/webserver/build/client/assets/files-BZrlFE1F.js +1 -0
  735. package/codeyam-cli/src/webserver/build/client/assets/git-DdZcvjGh.js +1 -0
  736. package/codeyam-cli/src/webserver/build/client/assets/globals-phvmGvat.css +1 -0
  737. package/codeyam-cli/src/webserver/build/client/assets/{index-B1h680n5.js → index-10oVnAAH.js} +1 -1
  738. package/codeyam-cli/src/webserver/build/client/assets/{index-lzqtyFU8.js → index-BcvgDzbZ.js} +1 -1
  739. package/codeyam-cli/src/webserver/build/client/assets/index-yHOVb4rc.js +15 -0
  740. package/codeyam-cli/src/webserver/build/client/assets/labs-Zk7ryIM1.js +1 -0
  741. package/codeyam-cli/src/webserver/build/client/assets/{loader-circle-B7B9V-bu.js → loader-circle-DaAZ_H2w.js} +2 -2
  742. package/codeyam-cli/src/webserver/build/client/assets/manifest-6134dc40.js +1 -0
  743. package/codeyam-cli/src/webserver/build/client/assets/memory-9gnxSZlb.js +101 -0
  744. package/codeyam-cli/src/webserver/build/client/assets/pause-f5-1lKBt.js +11 -0
  745. package/codeyam-cli/src/webserver/build/client/assets/root-BWAyuj0r.js +67 -0
  746. package/codeyam-cli/src/webserver/build/client/assets/{search-CxXUmBSd.js → search-Di64LWVb.js} +2 -2
  747. package/codeyam-cli/src/webserver/build/client/assets/settings-0OrEMU6J.js +1 -0
  748. package/codeyam-cli/src/webserver/build/client/assets/simulations-DWT-CvLy.js +1 -0
  749. package/codeyam-cli/src/webserver/build/client/assets/terminal-Br7MOqts.js +11 -0
  750. package/codeyam-cli/src/webserver/build/client/assets/{triangle-alert-B6LgvRJg.js → triangle-alert-BLdiCuG-.js} +2 -2
  751. package/codeyam-cli/src/webserver/build/client/assets/useCustomSizes-C-_hOl_g.js +1 -0
  752. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-C14nCb1q.js +2 -0
  753. package/codeyam-cli/src/webserver/build/client/assets/useReportContext-O-jkvSPx.js +1 -0
  754. package/codeyam-cli/src/webserver/build/client/assets/{useToast-mBRpZPiu.js → useToast-9FIWuYfK.js} +1 -1
  755. package/codeyam-cli/src/webserver/build/client/assets/xterm-BqvuqXEL.js +27 -0
  756. package/codeyam-cli/src/webserver/build/client/sound-test.html +98 -0
  757. package/codeyam-cli/src/webserver/build/server/assets/index-ChX0hPcu.js +1 -0
  758. package/codeyam-cli/src/webserver/build/server/assets/init-kSNsMjj8.js +10 -0
  759. package/codeyam-cli/src/webserver/build/server/assets/server-build-Bm2xIhmh.js +439 -0
  760. package/codeyam-cli/src/webserver/build/server/index.js +1 -1
  761. package/codeyam-cli/src/webserver/build-info.json +5 -5
  762. package/codeyam-cli/src/webserver/devServer.js +39 -5
  763. package/codeyam-cli/src/webserver/devServer.js.map +1 -1
  764. package/codeyam-cli/src/webserver/editorProxy.js +877 -0
  765. package/codeyam-cli/src/webserver/editorProxy.js.map +1 -0
  766. package/codeyam-cli/src/webserver/idleDetector.js +73 -0
  767. package/codeyam-cli/src/webserver/idleDetector.js.map +1 -0
  768. package/codeyam-cli/src/webserver/public/sound-test.html +98 -0
  769. package/codeyam-cli/src/webserver/scripts/codeyam-preload.mjs +414 -0
  770. package/codeyam-cli/src/webserver/scripts/journalCapture.ts +230 -0
  771. package/codeyam-cli/src/webserver/server.js +335 -26
  772. package/codeyam-cli/src/webserver/server.js.map +1 -1
  773. package/codeyam-cli/src/webserver/terminalServer.js +735 -0
  774. package/codeyam-cli/src/webserver/terminalServer.js.map +1 -0
  775. package/codeyam-cli/templates/chrome-extension-react/EXTENSION_SETUP.md +75 -0
  776. package/codeyam-cli/templates/chrome-extension-react/README.md +46 -0
  777. package/codeyam-cli/templates/chrome-extension-react/gitignore +15 -0
  778. package/codeyam-cli/templates/chrome-extension-react/index.html +12 -0
  779. package/codeyam-cli/templates/chrome-extension-react/package.json +27 -0
  780. package/codeyam-cli/templates/chrome-extension-react/popup.html +12 -0
  781. package/codeyam-cli/templates/chrome-extension-react/public/manifest.json +15 -0
  782. package/codeyam-cli/templates/chrome-extension-react/src/background/service-worker.ts +7 -0
  783. package/codeyam-cli/templates/chrome-extension-react/src/globals.css +6 -0
  784. package/codeyam-cli/templates/chrome-extension-react/src/lib/storage.ts +37 -0
  785. package/codeyam-cli/templates/chrome-extension-react/src/popup/App.tsx +12 -0
  786. package/codeyam-cli/templates/chrome-extension-react/src/popup/main.tsx +10 -0
  787. package/codeyam-cli/templates/chrome-extension-react/tsconfig.json +24 -0
  788. package/codeyam-cli/templates/chrome-extension-react/vite.config.ts +41 -0
  789. package/codeyam-cli/templates/codeyam-editor-claude.md +147 -0
  790. package/codeyam-cli/templates/codeyam-memory-hook.sh +199 -0
  791. package/codeyam-cli/templates/commands/codeyam-diagnose.md +481 -0
  792. package/codeyam-cli/templates/editor-step-hook.py +237 -0
  793. package/codeyam-cli/templates/expo-react-native/MOBILE_SETUP.md +89 -0
  794. package/codeyam-cli/templates/expo-react-native/README.md +41 -0
  795. package/codeyam-cli/templates/expo-react-native/app/(tabs)/_layout.tsx +33 -0
  796. package/codeyam-cli/templates/expo-react-native/app/(tabs)/index.tsx +12 -0
  797. package/codeyam-cli/templates/expo-react-native/app/(tabs)/settings.tsx +12 -0
  798. package/codeyam-cli/templates/expo-react-native/app/_layout.tsx +12 -0
  799. package/codeyam-cli/templates/expo-react-native/app.json +18 -0
  800. package/codeyam-cli/templates/expo-react-native/babel.config.js +9 -0
  801. package/codeyam-cli/templates/expo-react-native/gitignore +12 -0
  802. package/codeyam-cli/templates/expo-react-native/global.css +3 -0
  803. package/codeyam-cli/templates/expo-react-native/lib/storage.ts +32 -0
  804. package/codeyam-cli/templates/expo-react-native/metro.config.js +6 -0
  805. package/codeyam-cli/templates/expo-react-native/nativewind-env.d.ts +1 -0
  806. package/codeyam-cli/templates/expo-react-native/package.json +38 -0
  807. package/codeyam-cli/templates/expo-react-native/tailwind.config.js +10 -0
  808. package/codeyam-cli/templates/expo-react-native/tsconfig.json +10 -0
  809. package/codeyam-cli/templates/hooks/staleness-check.sh +43 -0
  810. package/codeyam-cli/templates/isolation-route/next-app.tsx.template +80 -0
  811. package/codeyam-cli/templates/isolation-route/next-pages.tsx.template +79 -0
  812. package/codeyam-cli/templates/isolation-route/vite-react.tsx.template +78 -0
  813. package/codeyam-cli/templates/msw/browser-setup.ts.template +47 -0
  814. package/codeyam-cli/templates/msw/handler-router.ts.template +47 -0
  815. package/codeyam-cli/templates/msw/server-setup.ts.template +52 -0
  816. package/codeyam-cli/templates/nextjs-prisma-sqlite/AUTH_PATTERNS.md +308 -0
  817. package/codeyam-cli/templates/nextjs-prisma-sqlite/AUTH_UPGRADE.md +304 -0
  818. package/codeyam-cli/templates/nextjs-prisma-sqlite/DATABASE.md +126 -0
  819. package/codeyam-cli/templates/nextjs-prisma-sqlite/FEATURE_PATTERNS.md +37 -0
  820. package/codeyam-cli/templates/nextjs-prisma-sqlite/README.md +53 -0
  821. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/api/todos/route.ts +17 -0
  822. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/codeyam-isolate/layout.tsx +12 -0
  823. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/globals.css +26 -0
  824. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/layout.tsx +34 -0
  825. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/lib/prisma.ts +24 -0
  826. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/page.tsx +10 -0
  827. package/codeyam-cli/templates/nextjs-prisma-sqlite/env +4 -0
  828. package/codeyam-cli/templates/nextjs-prisma-sqlite/eslint.config.mjs +11 -0
  829. package/codeyam-cli/templates/nextjs-prisma-sqlite/gitignore +64 -0
  830. package/codeyam-cli/templates/nextjs-prisma-sqlite/next.config.ts +14 -0
  831. package/codeyam-cli/templates/nextjs-prisma-sqlite/package.json +39 -0
  832. package/codeyam-cli/templates/nextjs-prisma-sqlite/postcss.config.mjs +7 -0
  833. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma/schema.prisma +27 -0
  834. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma/seed.ts +40 -0
  835. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma.config.ts +12 -0
  836. package/codeyam-cli/templates/nextjs-prisma-sqlite/seed-adapter.ts +92 -0
  837. package/codeyam-cli/templates/nextjs-prisma-sqlite/tsconfig.json +34 -0
  838. package/codeyam-cli/templates/nextjs-prisma-sqlite/vitest.config.ts +13 -0
  839. package/codeyam-cli/templates/nextjs-prisma-supabase/README.md +52 -0
  840. package/codeyam-cli/templates/nextjs-prisma-supabase/SUPABASE_SETUP.md +104 -0
  841. package/codeyam-cli/templates/nextjs-prisma-supabase/app/api/todos/route.ts +17 -0
  842. package/codeyam-cli/templates/nextjs-prisma-supabase/app/globals.css +26 -0
  843. package/codeyam-cli/templates/nextjs-prisma-supabase/app/layout.tsx +34 -0
  844. package/codeyam-cli/templates/nextjs-prisma-supabase/app/lib/prisma.ts +20 -0
  845. package/codeyam-cli/templates/nextjs-prisma-supabase/app/lib/supabase.ts +12 -0
  846. package/codeyam-cli/templates/nextjs-prisma-supabase/app/page.tsx +10 -0
  847. package/codeyam-cli/templates/nextjs-prisma-supabase/env +9 -0
  848. package/codeyam-cli/templates/nextjs-prisma-supabase/eslint.config.mjs +11 -0
  849. package/codeyam-cli/templates/nextjs-prisma-supabase/gitignore +40 -0
  850. package/codeyam-cli/templates/nextjs-prisma-supabase/next.config.ts +11 -0
  851. package/codeyam-cli/templates/nextjs-prisma-supabase/package.json +37 -0
  852. package/codeyam-cli/templates/nextjs-prisma-supabase/postcss.config.mjs +7 -0
  853. package/codeyam-cli/templates/nextjs-prisma-supabase/prisma/schema.prisma +27 -0
  854. package/codeyam-cli/templates/nextjs-prisma-supabase/prisma/seed.ts +39 -0
  855. package/codeyam-cli/templates/nextjs-prisma-supabase/prisma.config.ts +12 -0
  856. package/codeyam-cli/templates/nextjs-prisma-supabase/tsconfig.json +34 -0
  857. package/codeyam-cli/templates/prompts/conversation-guidance.txt +44 -0
  858. package/codeyam-cli/templates/prompts/conversation-prompt.txt +28 -0
  859. package/codeyam-cli/templates/prompts/interruption-prompt.txt +31 -0
  860. package/codeyam-cli/templates/prompts/stale-rules-prompt.txt +24 -0
  861. package/codeyam-cli/templates/rule-notification-hook.py +83 -0
  862. package/codeyam-cli/templates/rule-reflection-hook.py +647 -0
  863. package/codeyam-cli/templates/rules-instructions.md +78 -0
  864. package/codeyam-cli/templates/{codeyam-debug-skill.md → skills/codeyam-debug/SKILL.md} +48 -4
  865. package/codeyam-cli/templates/skills/codeyam-dev-mode/SKILL.md +237 -0
  866. package/codeyam-cli/templates/skills/codeyam-editor/SKILL.md +149 -0
  867. package/codeyam-cli/templates/skills/codeyam-memory/SKILL.md +611 -0
  868. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/deprecated-prompt.md +100 -0
  869. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/detect-deprecated-patterns.mjs +139 -0
  870. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/find-exports.mjs +52 -0
  871. package/codeyam-cli/templates/skills/codeyam-memory/scripts/holistic-analysis/misleading-api-prompt.md +117 -0
  872. package/codeyam-cli/templates/skills/codeyam-memory/scripts/lib/read-json-field.mjs +61 -0
  873. package/codeyam-cli/templates/skills/codeyam-memory/scripts/lib/ripgrep-fallback.mjs +155 -0
  874. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/analyze-prompt.md +46 -0
  875. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/cleanup.mjs +13 -0
  876. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/filter-session.mjs +95 -0
  877. package/codeyam-cli/templates/skills/codeyam-memory/scripts/session-mining/preprocess.mjs +160 -0
  878. package/codeyam-cli/templates/skills/codeyam-new-rule/SKILL.md +11 -0
  879. package/codeyam-cli/templates/{codeyam-setup-skill.md → skills/codeyam-setup/SKILL.md} +13 -1
  880. package/codeyam-cli/templates/{codeyam-sim-skill.md → skills/codeyam-sim/SKILL.md} +1 -1
  881. package/codeyam-cli/templates/{codeyam-test-skill.md → skills/codeyam-test/SKILL.md} +1 -1
  882. package/codeyam-cli/templates/{codeyam-verify-skill.md → skills/codeyam-verify/SKILL.md} +1 -1
  883. package/package.json +32 -22
  884. package/packages/ai/index.js +8 -6
  885. package/packages/ai/index.js.map +1 -1
  886. package/packages/ai/src/lib/analyzeScope.js +179 -13
  887. package/packages/ai/src/lib/analyzeScope.js.map +1 -1
  888. package/packages/ai/src/lib/astScopes/arrayDerivationDetector.js +150 -0
  889. package/packages/ai/src/lib/astScopes/arrayDerivationDetector.js.map +1 -0
  890. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js +160 -13
  891. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js.map +1 -1
  892. package/packages/ai/src/lib/astScopes/conditionalEffectsExtractor.js +435 -0
  893. package/packages/ai/src/lib/astScopes/conditionalEffectsExtractor.js.map +1 -0
  894. package/packages/ai/src/lib/astScopes/methodSemantics.js +138 -23
  895. package/packages/ai/src/lib/astScopes/methodSemantics.js.map +1 -1
  896. package/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.js +10 -14
  897. package/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.js.map +1 -1
  898. package/packages/ai/src/lib/astScopes/patterns/ifStatementHandler.js +8 -0
  899. package/packages/ai/src/lib/astScopes/patterns/ifStatementHandler.js.map +1 -1
  900. package/packages/ai/src/lib/astScopes/patterns/switchStatementHandler.js +7 -0
  901. package/packages/ai/src/lib/astScopes/patterns/switchStatementHandler.js.map +1 -1
  902. package/packages/ai/src/lib/astScopes/patterns/variableDeclarationHandler.js +138 -1
  903. package/packages/ai/src/lib/astScopes/patterns/variableDeclarationHandler.js.map +1 -1
  904. package/packages/ai/src/lib/astScopes/processExpression.js +931 -29
  905. package/packages/ai/src/lib/astScopes/processExpression.js.map +1 -1
  906. package/packages/ai/src/lib/astScopes/sharedPatterns.js +25 -0
  907. package/packages/ai/src/lib/astScopes/sharedPatterns.js.map +1 -1
  908. package/packages/ai/src/lib/completionCall.js +188 -38
  909. package/packages/ai/src/lib/completionCall.js.map +1 -1
  910. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js +1600 -189
  911. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js.map +1 -1
  912. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.js +5 -1
  913. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.js.map +1 -1
  914. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/MuiManager.js +179 -0
  915. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/MuiManager.js.map +1 -1
  916. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/ReactFrameworkManager.js +7 -1
  917. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/ReactFrameworkManager.js.map +1 -1
  918. package/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.js +13 -3
  919. package/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.js.map +1 -1
  920. package/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.js +6 -4
  921. package/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.js.map +1 -1
  922. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js +52 -3
  923. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js.map +1 -1
  924. package/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.js +111 -14
  925. package/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.js.map +1 -1
  926. package/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.js +63 -0
  927. package/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.js.map +1 -0
  928. package/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.js +54 -0
  929. package/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.js.map +1 -0
  930. package/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.js +122 -12
  931. package/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.js.map +1 -1
  932. package/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.js +173 -0
  933. package/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.js.map +1 -0
  934. package/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.js +37 -20
  935. package/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.js.map +1 -1
  936. package/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.js +333 -81
  937. package/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.js.map +1 -1
  938. package/packages/ai/src/lib/dataStructure/helpers/fixNullIdsBySchema.js +107 -0
  939. package/packages/ai/src/lib/dataStructure/helpers/fixNullIdsBySchema.js.map +1 -0
  940. package/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.js +34 -0
  941. package/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.js.map +1 -0
  942. package/packages/ai/src/lib/dataStructureChunking.js +130 -0
  943. package/packages/ai/src/lib/dataStructureChunking.js.map +1 -0
  944. package/packages/ai/src/lib/e2eDataTracking.js +241 -0
  945. package/packages/ai/src/lib/e2eDataTracking.js.map +1 -0
  946. package/packages/ai/src/lib/extractCriticalDataKeys.js +96 -0
  947. package/packages/ai/src/lib/extractCriticalDataKeys.js.map +1 -0
  948. package/packages/ai/src/lib/generateChangesEntityScenarioData.js +4 -3
  949. package/packages/ai/src/lib/generateChangesEntityScenarioData.js.map +1 -1
  950. package/packages/ai/src/lib/generateChangesEntityScenarios.js +78 -120
  951. package/packages/ai/src/lib/generateChangesEntityScenarios.js.map +1 -1
  952. package/packages/ai/src/lib/generateEntityDataStructure.js +47 -2
  953. package/packages/ai/src/lib/generateEntityDataStructure.js.map +1 -1
  954. package/packages/ai/src/lib/generateEntityScenarioData.js +1153 -60
  955. package/packages/ai/src/lib/generateEntityScenarioData.js.map +1 -1
  956. package/packages/ai/src/lib/generateEntityScenarios.js +177 -163
  957. package/packages/ai/src/lib/generateEntityScenarios.js.map +1 -1
  958. package/packages/ai/src/lib/generateExecutionFlows.js +484 -0
  959. package/packages/ai/src/lib/generateExecutionFlows.js.map +1 -0
  960. package/packages/ai/src/lib/generateExecutionFlowsFromConditionalEffects.js +380 -0
  961. package/packages/ai/src/lib/generateExecutionFlowsFromConditionalEffects.js.map +1 -0
  962. package/packages/ai/src/lib/generateExecutionFlowsFromConditionals.js +1807 -0
  963. package/packages/ai/src/lib/generateExecutionFlowsFromConditionals.js.map +1 -0
  964. package/packages/ai/src/lib/generateExecutionFlowsFromJsxUsages.js +194 -0
  965. package/packages/ai/src/lib/generateExecutionFlowsFromJsxUsages.js.map +1 -0
  966. package/packages/ai/src/lib/guessScenarioDataFromDescription.js +2 -2
  967. package/packages/ai/src/lib/guessScenarioDataFromDescription.js.map +1 -1
  968. package/packages/ai/src/lib/isolateScopes.js +270 -7
  969. package/packages/ai/src/lib/isolateScopes.js.map +1 -1
  970. package/packages/ai/src/lib/mergeJsonTypeDefinitions.js +5 -0
  971. package/packages/ai/src/lib/mergeJsonTypeDefinitions.js.map +1 -1
  972. package/packages/ai/src/lib/mergeStatements.js +88 -46
  973. package/packages/ai/src/lib/mergeStatements.js.map +1 -1
  974. package/packages/ai/src/lib/promptGenerators/collapseNullableObjects.js +97 -0
  975. package/packages/ai/src/lib/promptGenerators/collapseNullableObjects.js.map +1 -0
  976. package/packages/ai/src/lib/promptGenerators/gatherAttributesMap.js +10 -4
  977. package/packages/ai/src/lib/promptGenerators/gatherAttributesMap.js.map +1 -1
  978. package/packages/ai/src/lib/promptGenerators/generateChangesEntityScenarioDataGenerator.js +1 -1
  979. package/packages/ai/src/lib/promptGenerators/generateChangesEntityScenarioDataGenerator.js.map +1 -1
  980. package/packages/ai/src/lib/promptGenerators/generateChangesEntityScenariosGenerator.js +21 -119
  981. package/packages/ai/src/lib/promptGenerators/generateChangesEntityScenariosGenerator.js.map +1 -1
  982. package/packages/ai/src/lib/promptGenerators/generateChunkPrompt.js +54 -0
  983. package/packages/ai/src/lib/promptGenerators/generateChunkPrompt.js.map +1 -0
  984. package/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.js +83 -6
  985. package/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.js.map +1 -1
  986. package/packages/ai/src/lib/promptGenerators/generateEntityScenariosGenerator.js +10 -70
  987. package/packages/ai/src/lib/promptGenerators/generateEntityScenariosGenerator.js.map +1 -1
  988. package/packages/ai/src/lib/promptGenerators/generateMissingKeysPrompt.js +45 -0
  989. package/packages/ai/src/lib/promptGenerators/generateMissingKeysPrompt.js.map +1 -0
  990. package/packages/ai/src/lib/promptGenerators/guessNewScenarioDataFromDescriptionGenerator.js +9 -9
  991. package/packages/ai/src/lib/promptGenerators/guessNewScenarioDataFromDescriptionGenerator.js.map +1 -1
  992. package/packages/ai/src/lib/promptGenerators/simplifyKeysForLLM.js +335 -0
  993. package/packages/ai/src/lib/promptGenerators/simplifyKeysForLLM.js.map +1 -0
  994. package/packages/ai/src/lib/resolvePathToControllable.js +677 -0
  995. package/packages/ai/src/lib/resolvePathToControllable.js.map +1 -0
  996. package/packages/ai/src/lib/splitOutsideParentheses.js +3 -1
  997. package/packages/ai/src/lib/splitOutsideParentheses.js.map +1 -1
  998. package/packages/ai/src/lib/worker/SerializableDataStructure.js +29 -0
  999. package/packages/ai/src/lib/worker/SerializableDataStructure.js.map +1 -1
  1000. package/packages/ai/src/lib/worker/analyzeScopeWorker.js +94 -1
  1001. package/packages/ai/src/lib/worker/analyzeScopeWorker.js.map +1 -1
  1002. package/packages/analyze/index.js +1 -0
  1003. package/packages/analyze/index.js.map +1 -1
  1004. package/packages/analyze/src/lib/FileAnalyzer.js +60 -36
  1005. package/packages/analyze/src/lib/FileAnalyzer.js.map +1 -1
  1006. package/packages/analyze/src/lib/ProjectAnalyzer.js +109 -30
  1007. package/packages/analyze/src/lib/ProjectAnalyzer.js.map +1 -1
  1008. package/packages/analyze/src/lib/analysisContext.js +30 -5
  1009. package/packages/analyze/src/lib/analysisContext.js.map +1 -1
  1010. package/packages/analyze/src/lib/asts/index.js +4 -2
  1011. package/packages/analyze/src/lib/asts/index.js.map +1 -1
  1012. package/packages/analyze/src/lib/asts/nodes/getNodeType.js +1 -0
  1013. package/packages/analyze/src/lib/asts/nodes/getNodeType.js.map +1 -1
  1014. package/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.js +14 -0
  1015. package/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.js.map +1 -1
  1016. package/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.js +14 -0
  1017. package/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.js.map +1 -1
  1018. package/packages/analyze/src/lib/asts/sourceFiles/getAllExports.js +6 -0
  1019. package/packages/analyze/src/lib/asts/sourceFiles/getAllExports.js.map +1 -1
  1020. package/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.js +6 -0
  1021. package/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.js.map +1 -1
  1022. package/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.js +39 -1
  1023. package/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.js.map +1 -1
  1024. package/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.js +2 -1
  1025. package/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.js.map +1 -1
  1026. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js +189 -41
  1027. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js.map +1 -1
  1028. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js +28 -4
  1029. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js.map +1 -1
  1030. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js +9 -0
  1031. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js.map +1 -1
  1032. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js +2 -1
  1033. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js.map +1 -1
  1034. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js +31 -10
  1035. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js.map +1 -1
  1036. package/packages/analyze/src/lib/files/analyzeChange.js +10 -10
  1037. package/packages/analyze/src/lib/files/analyzeChange.js.map +1 -1
  1038. package/packages/analyze/src/lib/files/analyzeEntity.js +4 -4
  1039. package/packages/analyze/src/lib/files/analyzeEntity.js.map +1 -1
  1040. package/packages/analyze/src/lib/files/analyzeInitial.js +9 -10
  1041. package/packages/analyze/src/lib/files/analyzeInitial.js.map +1 -1
  1042. package/packages/analyze/src/lib/files/analyzeRemixRoute.js +3 -2
  1043. package/packages/analyze/src/lib/files/analyzeRemixRoute.js.map +1 -1
  1044. package/packages/analyze/src/lib/files/enums/steps.js +1 -1
  1045. package/packages/analyze/src/lib/files/enums/steps.js.map +1 -1
  1046. package/packages/analyze/src/lib/files/getImportedExports.js +11 -7
  1047. package/packages/analyze/src/lib/files/getImportedExports.js.map +1 -1
  1048. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js +907 -0
  1049. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js.map +1 -0
  1050. package/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.js +164 -68
  1051. package/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.js.map +1 -1
  1052. package/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.js +178 -31
  1053. package/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.js.map +1 -1
  1054. package/packages/analyze/src/lib/files/scenarios/generateChangesScenarios.js +29 -129
  1055. package/packages/analyze/src/lib/files/scenarios/generateChangesScenarios.js.map +1 -1
  1056. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js +252 -21
  1057. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js.map +1 -1
  1058. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js +104 -0
  1059. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js.map +1 -0
  1060. package/packages/analyze/src/lib/files/scenarios/generateScenarioData.js +1 -0
  1061. package/packages/analyze/src/lib/files/scenarios/generateScenarioData.js.map +1 -1
  1062. package/packages/analyze/src/lib/files/scenarios/generateScenarios.js +2 -3
  1063. package/packages/analyze/src/lib/files/scenarios/generateScenarios.js.map +1 -1
  1064. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js +686 -55
  1065. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js.map +1 -1
  1066. package/packages/analyze/src/lib/files/scenarios/mergeValidatedDataStructures.js +46 -9
  1067. package/packages/analyze/src/lib/files/scenarios/mergeValidatedDataStructures.js.map +1 -1
  1068. package/packages/analyze/src/lib/files/setImportedExports.js +2 -1
  1069. package/packages/analyze/src/lib/files/setImportedExports.js.map +1 -1
  1070. package/packages/analyze/src/lib/index.js +1 -0
  1071. package/packages/analyze/src/lib/index.js.map +1 -1
  1072. package/packages/analyze/src/lib/utils/getFileByPath.js +12 -0
  1073. package/packages/analyze/src/lib/utils/getFileByPath.js.map +1 -0
  1074. package/packages/database/index.js +1 -0
  1075. package/packages/database/index.js.map +1 -1
  1076. package/packages/database/src/lib/analysisBranchToDb.js +1 -1
  1077. package/packages/database/src/lib/analysisBranchToDb.js.map +1 -1
  1078. package/packages/database/src/lib/analysisToDb.js +1 -1
  1079. package/packages/database/src/lib/analysisToDb.js.map +1 -1
  1080. package/packages/database/src/lib/branchToDb.js +1 -1
  1081. package/packages/database/src/lib/branchToDb.js.map +1 -1
  1082. package/packages/database/src/lib/commitBranchToDb.js +1 -1
  1083. package/packages/database/src/lib/commitBranchToDb.js.map +1 -1
  1084. package/packages/database/src/lib/commitToDb.js +1 -1
  1085. package/packages/database/src/lib/commitToDb.js.map +1 -1
  1086. package/packages/database/src/lib/fileToDb.js +1 -1
  1087. package/packages/database/src/lib/fileToDb.js.map +1 -1
  1088. package/packages/database/src/lib/kysely/db.js +16 -1
  1089. package/packages/database/src/lib/kysely/db.js.map +1 -1
  1090. package/packages/database/src/lib/kysely/tables/commitsTable.js +3 -0
  1091. package/packages/database/src/lib/kysely/tables/commitsTable.js.map +1 -1
  1092. package/packages/database/src/lib/kysely/tables/debugReportsTable.js.map +1 -1
  1093. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js +121 -0
  1094. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -0
  1095. package/packages/database/src/lib/kysely/tables/labsRequestsTable.js +35 -0
  1096. package/packages/database/src/lib/kysely/tables/labsRequestsTable.js.map +1 -0
  1097. package/packages/database/src/lib/loadAnalyses.js +45 -2
  1098. package/packages/database/src/lib/loadAnalyses.js.map +1 -1
  1099. package/packages/database/src/lib/loadAnalysis.js +8 -0
  1100. package/packages/database/src/lib/loadAnalysis.js.map +1 -1
  1101. package/packages/database/src/lib/loadBranch.js +11 -1
  1102. package/packages/database/src/lib/loadBranch.js.map +1 -1
  1103. package/packages/database/src/lib/loadCommit.js +7 -0
  1104. package/packages/database/src/lib/loadCommit.js.map +1 -1
  1105. package/packages/database/src/lib/loadCommits.js +45 -14
  1106. package/packages/database/src/lib/loadCommits.js.map +1 -1
  1107. package/packages/database/src/lib/loadEntities.js +23 -10
  1108. package/packages/database/src/lib/loadEntities.js.map +1 -1
  1109. package/packages/database/src/lib/loadEntityBranches.js +9 -0
  1110. package/packages/database/src/lib/loadEntityBranches.js.map +1 -1
  1111. package/packages/database/src/lib/loadReadyToBeCapturedAnalyses.js +5 -5
  1112. package/packages/database/src/lib/loadReadyToBeCapturedAnalyses.js.map +1 -1
  1113. package/packages/database/src/lib/projectToDb.js +1 -1
  1114. package/packages/database/src/lib/projectToDb.js.map +1 -1
  1115. package/packages/database/src/lib/saveFiles.js +1 -1
  1116. package/packages/database/src/lib/saveFiles.js.map +1 -1
  1117. package/packages/database/src/lib/scenarioToDb.js +1 -1
  1118. package/packages/database/src/lib/scenarioToDb.js.map +1 -1
  1119. package/packages/database/src/lib/updateCommitMetadata.js +76 -89
  1120. package/packages/database/src/lib/updateCommitMetadata.js.map +1 -1
  1121. package/packages/database/src/lib/updateFreshAnalysisStatus.js +41 -30
  1122. package/packages/database/src/lib/updateFreshAnalysisStatus.js.map +1 -1
  1123. package/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.js +68 -57
  1124. package/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.js.map +1 -1
  1125. package/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js +29 -1
  1126. package/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js.map +1 -1
  1127. package/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js +33 -5
  1128. package/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js.map +1 -1
  1129. package/packages/types/index.js +0 -1
  1130. package/packages/types/index.js.map +1 -1
  1131. package/packages/types/src/enums/ProjectFramework.js +2 -0
  1132. package/packages/types/src/enums/ProjectFramework.js.map +1 -1
  1133. package/packages/types/src/types/Scenario.js +1 -21
  1134. package/packages/types/src/types/Scenario.js.map +1 -1
  1135. package/packages/utils/src/lib/fs/rsyncCopy.js +98 -3
  1136. package/packages/utils/src/lib/fs/rsyncCopy.js.map +1 -1
  1137. package/packages/utils/src/lib/safeFileName.js +29 -3
  1138. package/packages/utils/src/lib/safeFileName.js.map +1 -1
  1139. package/scripts/npm-post-install.cjs +34 -0
  1140. package/analyzer-template/packages/ai/src/lib/findMatchingAttribute.ts +0 -109
  1141. package/analyzer-template/packages/ai/src/lib/gatherRelevantDependentKeyAttributes.ts +0 -584
  1142. package/analyzer-template/packages/ai/src/lib/generateChangesEntityKeyAttributes.ts +0 -341
  1143. package/analyzer-template/packages/ai/src/lib/generateEntityKeyAttributes.ts +0 -495
  1144. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateEntityKeyAttributesGenerator.ts +0 -67
  1145. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateKeyAttributes.ts +0 -120
  1146. package/codeyam-cli/src/commands/detect-universal-mocks.js +0 -118
  1147. package/codeyam-cli/src/commands/detect-universal-mocks.js.map +0 -1
  1148. package/codeyam-cli/src/commands/list.js +0 -31
  1149. package/codeyam-cli/src/commands/list.js.map +0 -1
  1150. package/codeyam-cli/src/commands/webapp-info.js +0 -146
  1151. package/codeyam-cli/src/commands/webapp-info.js.map +0 -1
  1152. package/codeyam-cli/src/utils/universal-mocks.js +0 -152
  1153. package/codeyam-cli/src/utils/universal-mocks.js.map +0 -1
  1154. package/codeyam-cli/src/webserver/build/client/assets/EntityItem-Cmysw5OP.js +0 -1
  1155. package/codeyam-cli/src/webserver/build/client/assets/EntityTypeIcon-CAneekK2.js +0 -41
  1156. package/codeyam-cli/src/webserver/build/client/assets/InteractivePreview-Cu16OUmx.js +0 -25
  1157. package/codeyam-cli/src/webserver/build/client/assets/ReportIssueModal-DcAUIpD_.js +0 -11
  1158. package/codeyam-cli/src/webserver/build/client/assets/ScenarioViewer-BMKg0SAF.js +0 -15
  1159. package/codeyam-cli/src/webserver/build/client/assets/_index-DSmTpjmK.js +0 -11
  1160. package/codeyam-cli/src/webserver/build/client/assets/activity.(_tab)-BF_aK4y6.js +0 -32
  1161. package/codeyam-cli/src/webserver/build/client/assets/createLucideIcon-BdhJEx6B.js +0 -21
  1162. package/codeyam-cli/src/webserver/build/client/assets/cy-logo-cli-CKnwPCDr.js +0 -1
  1163. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-RJCf3Tvw.js +0 -1
  1164. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-EylcgScH.js +0 -1
  1165. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-DMe7kvgo.js +0 -1
  1166. package/codeyam-cli/src/webserver/build/client/assets/files-BW7Cyeyi.js +0 -1
  1167. package/codeyam-cli/src/webserver/build/client/assets/git-CZu4fif0.js +0 -15
  1168. package/codeyam-cli/src/webserver/build/client/assets/globals-wHVy_II5.css +0 -1
  1169. package/codeyam-cli/src/webserver/build/client/assets/keyAttributeCoverage-CTlFMihX.js +0 -1
  1170. package/codeyam-cli/src/webserver/build/client/assets/manifest-2d191949.js +0 -1
  1171. package/codeyam-cli/src/webserver/build/client/assets/root-FHgpM6gc.js +0 -56
  1172. package/codeyam-cli/src/webserver/build/client/assets/settings-6D8k8Jp5.js +0 -1
  1173. package/codeyam-cli/src/webserver/build/client/assets/simulations-CDJZnWhN.js +0 -1
  1174. package/codeyam-cli/src/webserver/build/client/assets/useCustomSizes-Dv18q8LD.js +0 -1
  1175. package/codeyam-cli/src/webserver/build/client/assets/useInteractiveMode-0ToGk4K3.js +0 -1
  1176. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-aSv48UbS.js +0 -2
  1177. package/codeyam-cli/src/webserver/build/client/assets/useReportContext-1BX144Eg.js +0 -1
  1178. package/codeyam-cli/src/webserver/build/server/assets/index-pU0o5t1o.js +0 -1
  1179. package/codeyam-cli/src/webserver/build/server/assets/server-build-YzfkRwdn.js +0 -178
  1180. package/codeyam-cli/templates/codeyam-stop-hook.sh +0 -284
  1181. package/codeyam-cli/templates/debug-codeyam.md +0 -625
  1182. package/packages/ai/src/lib/findMatchingAttribute.js +0 -81
  1183. package/packages/ai/src/lib/findMatchingAttribute.js.map +0 -1
  1184. package/packages/ai/src/lib/gatherRelevantDependentKeyAttributes.js +0 -425
  1185. package/packages/ai/src/lib/gatherRelevantDependentKeyAttributes.js.map +0 -1
  1186. package/packages/ai/src/lib/generateChangesEntityKeyAttributes.js +0 -267
  1187. package/packages/ai/src/lib/generateChangesEntityKeyAttributes.js.map +0 -1
  1188. package/packages/ai/src/lib/generateEntityKeyAttributes.js +0 -408
  1189. package/packages/ai/src/lib/generateEntityKeyAttributes.js.map +0 -1
  1190. package/packages/ai/src/lib/isFrontend.js +0 -5
  1191. package/packages/ai/src/lib/isFrontend.js.map +0 -1
  1192. package/packages/ai/src/lib/promptGenerators/generateEntityKeyAttributesGenerator.js +0 -40
  1193. package/packages/ai/src/lib/promptGenerators/generateEntityKeyAttributesGenerator.js.map +0 -1
  1194. package/packages/analyze/src/lib/files/scenarios/generateKeyAttributes.js +0 -77
  1195. package/packages/analyze/src/lib/files/scenarios/generateKeyAttributes.js.map +0 -1
  1196. package/scripts/finalize-analyzer.cjs +0 -81
  1197. /package/codeyam-cli/src/webserver/build/client/assets/{api.link-scenario-value-l0sNRNKZ.js → api.agent-transcripts-l0sNRNKZ.js} +0 -0
  1198. /package/codeyam-cli/src/webserver/build/client/assets/{api.update-key-attributes-l0sNRNKZ.js → api.dev-mode-events-l0sNRNKZ.js} +0 -0
  1199. /package/codeyam-cli/src/webserver/build/client/assets/{api.update-valid-values-l0sNRNKZ.js → api.editor-audit-l0sNRNKZ.js} +0 -0
@@ -0,0 +1,1742 @@
1
+ import { buildReverseDependencyGraph, classifyDirectChanges, computeEntityChangeStatus, buildChangedFilesMap, pageNameFromUrl, scenarioEntityName, buildEntityInfosFromScenarios, buildEntityInfosFromGlossary, filterGroupsByChangeStatus, filterGlossaryByChangeStatus, filterScenarioScreenshotsByChangeStatus, } from "../entityChangeStatus.js";
2
+ import { scanPageFilePaths, detectFirstFeature, readFeatureStartedAt, } from "../entityChangeStatus.server.js";
3
+ import * as fsModule from 'fs';
4
+ import * as pathModule from 'path';
5
+ import * as os from 'os';
6
+ // ── Helpers ──────────────────────────────────────────────────────────────
7
+ /** Shorthand to build an EntityInfo with importedBy metadata */
8
+ function entity(name, filePath, importedBy) {
9
+ if (!importedBy)
10
+ return { name, filePath };
11
+ // Convert simplified { filePath: [importerName, ...] } to DB format
12
+ const dbFormat = {};
13
+ for (const [fp, importers] of Object.entries(importedBy)) {
14
+ dbFormat[fp] = {};
15
+ for (const imp of importers) {
16
+ dbFormat[fp][imp] = { shas: ['test-sha'] };
17
+ }
18
+ }
19
+ return { name, filePath, importedBy: dbFormat };
20
+ }
21
+ /** Helper: get sorted impactedBy names from a result */
22
+ function impactNames(result, entityName) {
23
+ return (result[entityName]?.impactedBy || []).map((d) => d.name).sort();
24
+ }
25
+ // ── Tests ────────────────────────────────────────────────────────────────
26
+ describe('entityChangeStatus', () => {
27
+ // ── buildChangedFilesMap ───────────────────────────────────────────────
28
+ describe('buildChangedFilesMap', () => {
29
+ it('should map "added" files to "new"', () => {
30
+ const files = [
31
+ { path: 'components/New.tsx', status: 'added' },
32
+ ];
33
+ const result = buildChangedFilesMap(files, false);
34
+ expect(result.get('components/New.tsx')).toBe('new');
35
+ });
36
+ it('should map "untracked" files to "new"', () => {
37
+ const files = [
38
+ { path: 'components/Untracked.tsx', status: 'untracked' },
39
+ ];
40
+ const result = buildChangedFilesMap(files, false);
41
+ expect(result.get('components/Untracked.tsx')).toBe('new');
42
+ });
43
+ it('should map "modified" files to "edited"', () => {
44
+ const files = [
45
+ { path: 'components/Modified.tsx', status: 'modified' },
46
+ ];
47
+ const result = buildChangedFilesMap(files, false);
48
+ expect(result.get('components/Modified.tsx')).toBe('edited');
49
+ });
50
+ it('should ignore "deleted" files', () => {
51
+ const files = [
52
+ { path: 'components/Deleted.tsx', status: 'deleted' },
53
+ ];
54
+ const result = buildChangedFilesMap(files, false);
55
+ expect(result.size).toBe(0);
56
+ });
57
+ it('should ignore "renamed" files', () => {
58
+ const files = [
59
+ { path: 'components/Renamed.tsx', status: 'renamed' },
60
+ ];
61
+ const result = buildChangedFilesMap(files, false);
62
+ expect(result.size).toBe(0);
63
+ });
64
+ it('should treat ALL non-deleted files as "new" when isFirstFeature is true', () => {
65
+ const files = [
66
+ { path: 'components/Modified.tsx', status: 'modified' },
67
+ { path: 'components/Added.tsx', status: 'added' },
68
+ { path: 'components/Untracked.tsx', status: 'untracked' },
69
+ { path: 'components/Renamed.tsx', status: 'renamed' },
70
+ { path: 'components/Deleted.tsx', status: 'deleted' },
71
+ ];
72
+ const result = buildChangedFilesMap(files, true);
73
+ // Everything except deleted should be 'new'
74
+ expect(result.get('components/Modified.tsx')).toBe('new');
75
+ expect(result.get('components/Added.tsx')).toBe('new');
76
+ expect(result.get('components/Untracked.tsx')).toBe('new');
77
+ expect(result.get('components/Renamed.tsx')).toBe('new');
78
+ expect(result.has('components/Deleted.tsx')).toBe(false);
79
+ });
80
+ it('should return empty map for empty input', () => {
81
+ expect(buildChangedFilesMap([], false).size).toBe(0);
82
+ expect(buildChangedFilesMap([], true).size).toBe(0);
83
+ });
84
+ it('should handle mix of statuses correctly', () => {
85
+ const files = [
86
+ { path: 'a.tsx', status: 'added' },
87
+ { path: 'b.tsx', status: 'modified' },
88
+ { path: 'c.tsx', status: 'untracked' },
89
+ { path: 'd.tsx', status: 'deleted' },
90
+ ];
91
+ const result = buildChangedFilesMap(files, false);
92
+ expect(result.get('a.tsx')).toBe('new');
93
+ expect(result.get('b.tsx')).toBe('edited');
94
+ expect(result.get('c.tsx')).toBe('new');
95
+ expect(result.has('d.tsx')).toBe(false);
96
+ expect(result.size).toBe(3);
97
+ });
98
+ });
99
+ // ── pageNameFromUrl ───────────────────────────────────────────────────
100
+ describe('pageNameFromUrl', () => {
101
+ it('should return "Home" for null', () => {
102
+ expect(pageNameFromUrl(null)).toBe('Home');
103
+ });
104
+ it('should return "Home" for undefined', () => {
105
+ expect(pageNameFromUrl(undefined)).toBe('Home');
106
+ });
107
+ it('should return "Home" for root path "/"', () => {
108
+ expect(pageNameFromUrl('/')).toBe('Home');
109
+ });
110
+ it('should capitalize first segment for simple paths', () => {
111
+ expect(pageNameFromUrl('/about')).toBe('About');
112
+ expect(pageNameFromUrl('/drinks')).toBe('Drinks');
113
+ expect(pageNameFromUrl('/settings')).toBe('Settings');
114
+ });
115
+ it('should use only the first segment for nested paths', () => {
116
+ expect(pageNameFromUrl('/drinks/123')).toBe('Drinks');
117
+ expect(pageNameFromUrl('/blog/posts/latest')).toBe('Blog');
118
+ });
119
+ it('should strip query parameters', () => {
120
+ expect(pageNameFromUrl('/settings?tab=profile')).toBe('Settings');
121
+ expect(pageNameFromUrl('/drinks?sort=name&filter=ipa')).toBe('Drinks');
122
+ });
123
+ it('should handle paths without leading slash', () => {
124
+ expect(pageNameFromUrl('about')).toBe('About');
125
+ });
126
+ it('should handle empty string as Home', () => {
127
+ // Empty string: after stripping '/', split gives [''], charAt(0) is ''
128
+ // This is an edge case — empty string is falsy so returns Home
129
+ expect(pageNameFromUrl('')).toBe('Home');
130
+ });
131
+ it('should return Home for root path with query string', () => {
132
+ expect(pageNameFromUrl('/?demo=empty')).toBe('Home');
133
+ expect(pageNameFromUrl('/?state=seeded')).toBe('Home');
134
+ expect(pageNameFromUrl('/?tab=overview&sort=date')).toBe('Home');
135
+ });
136
+ it('should handle dynamic route segments', () => {
137
+ expect(pageNameFromUrl('/drinks/[id]')).toBe('Drinks');
138
+ expect(pageNameFromUrl('/users/[userId]/posts')).toBe('Users');
139
+ });
140
+ });
141
+ // ── buildEntityInfosFromScenarios ─────────────────────────────────────
142
+ describe('buildEntityInfosFromScenarios', () => {
143
+ it('should extract component entity from scenario with componentName/componentPath', () => {
144
+ const scenarios = [
145
+ {
146
+ componentName: 'DrinkCard',
147
+ componentPath: 'components/DrinkCard.tsx',
148
+ },
149
+ ];
150
+ const result = buildEntityInfosFromScenarios(scenarios, {}, []);
151
+ expect(result).toEqual([
152
+ { name: 'DrinkCard', filePath: 'components/DrinkCard.tsx' },
153
+ ]);
154
+ });
155
+ it('should extract page entity from scenario with URL', () => {
156
+ const scenarios = [
157
+ { componentName: null, componentPath: null, url: '/' },
158
+ ];
159
+ const pageFilePaths = { Home: 'app/page.tsx' };
160
+ const result = buildEntityInfosFromScenarios(scenarios, pageFilePaths, []);
161
+ expect(result).toEqual([{ name: 'Home', filePath: 'app/page.tsx' }]);
162
+ });
163
+ it('should derive page name from URL using pageNameFromUrl', () => {
164
+ const scenarios = [
165
+ { componentName: null, componentPath: null, url: '/drinks/123' },
166
+ ];
167
+ const pageFilePaths = { Drinks: 'app/drinks/page.tsx' };
168
+ const result = buildEntityInfosFromScenarios(scenarios, pageFilePaths, []);
169
+ expect(result).toEqual([
170
+ { name: 'Drinks', filePath: 'app/drinks/page.tsx' },
171
+ ]);
172
+ });
173
+ it('should skip page scenarios when page file path is not found', () => {
174
+ const scenarios = [
175
+ { componentName: null, componentPath: null, url: '/nonexistent' },
176
+ ];
177
+ const result = buildEntityInfosFromScenarios(scenarios, {}, []);
178
+ expect(result).toEqual([]);
179
+ });
180
+ it('should deduplicate by entity name (first occurrence wins)', () => {
181
+ const scenarios = [
182
+ {
183
+ componentName: 'DrinkCard',
184
+ componentPath: 'components/DrinkCard.tsx',
185
+ },
186
+ {
187
+ componentName: 'DrinkCard',
188
+ componentPath: 'components/DrinkCard.tsx',
189
+ },
190
+ {
191
+ componentName: 'DrinkCard',
192
+ componentPath: 'components/DrinkCard2.tsx',
193
+ },
194
+ ];
195
+ const result = buildEntityInfosFromScenarios(scenarios, {}, []);
196
+ expect(result).toHaveLength(1);
197
+ expect(result[0].filePath).toBe('components/DrinkCard.tsx');
198
+ });
199
+ it('should enrich entities with importedBy metadata from entity DB', () => {
200
+ const scenarios = [
201
+ {
202
+ componentName: 'DrinkCard',
203
+ componentPath: 'components/DrinkCard.tsx',
204
+ },
205
+ ];
206
+ const entitiesWithMetadata = [
207
+ {
208
+ name: 'DrinkCard',
209
+ metadata: {
210
+ importedBy: {
211
+ 'components/DrinkCard.tsx': {
212
+ DrinkList: { shas: ['abc'] },
213
+ },
214
+ },
215
+ },
216
+ },
217
+ ];
218
+ const result = buildEntityInfosFromScenarios(scenarios, {}, entitiesWithMetadata);
219
+ expect(result[0].importedBy).toEqual({
220
+ 'components/DrinkCard.tsx': {
221
+ DrinkList: { shas: ['abc'] },
222
+ },
223
+ });
224
+ });
225
+ it('should handle entities without matching metadata (no importedBy)', () => {
226
+ const scenarios = [
227
+ {
228
+ componentName: 'DrinkCard',
229
+ componentPath: 'components/DrinkCard.tsx',
230
+ },
231
+ ];
232
+ const entitiesWithMetadata = [
233
+ { name: 'UnrelatedEntity', metadata: null },
234
+ ];
235
+ const result = buildEntityInfosFromScenarios(scenarios, {}, entitiesWithMetadata);
236
+ expect(result[0].importedBy).toBeUndefined();
237
+ });
238
+ it('should handle mixed component and page scenarios', () => {
239
+ const scenarios = [
240
+ {
241
+ componentName: 'DrinkCard',
242
+ componentPath: 'components/DrinkCard.tsx',
243
+ },
244
+ { componentName: null, componentPath: null, url: '/' },
245
+ { componentName: 'Header', componentPath: 'components/Header.tsx' },
246
+ { componentName: null, componentPath: null, url: '/drinks' },
247
+ ];
248
+ const pageFilePaths = {
249
+ Home: 'app/page.tsx',
250
+ Drinks: 'app/drinks/page.tsx',
251
+ };
252
+ const result = buildEntityInfosFromScenarios(scenarios, pageFilePaths, []);
253
+ expect(result.map((e) => e.name)).toEqual([
254
+ 'DrinkCard',
255
+ 'Home',
256
+ 'Header',
257
+ 'Drinks',
258
+ ]);
259
+ });
260
+ it('should skip scenarios with componentName but no componentPath', () => {
261
+ const scenarios = [
262
+ { componentName: 'DrinkCard', componentPath: null },
263
+ ];
264
+ const result = buildEntityInfosFromScenarios(scenarios, {}, []);
265
+ expect(result).toEqual([]);
266
+ });
267
+ it('should handle empty inputs', () => {
268
+ expect(buildEntityInfosFromScenarios([], {}, [])).toEqual([]);
269
+ });
270
+ it('should handle scenarios where url is undefined (not a page scenario)', () => {
271
+ // A scenario with no componentName and no url — should be skipped
272
+ const scenarios = [
273
+ { componentName: null, componentPath: null },
274
+ ];
275
+ const result = buildEntityInfosFromScenarios(scenarios, {}, []);
276
+ expect(result).toEqual([]);
277
+ });
278
+ it('should handle entity metadata with null metadata field', () => {
279
+ const scenarios = [
280
+ { componentName: 'Card', componentPath: 'components/Card.tsx' },
281
+ ];
282
+ const metadata = [
283
+ { name: 'Card', metadata: null },
284
+ ];
285
+ const result = buildEntityInfosFromScenarios(scenarios, {}, metadata);
286
+ expect(result[0].importedBy).toBeUndefined();
287
+ });
288
+ });
289
+ // ── buildReverseDependencyGraph ──────────────────────────────────────
290
+ describe('buildReverseDependencyGraph', () => {
291
+ it('should return empty map for empty entities', () => {
292
+ expect(buildReverseDependencyGraph([]).size).toBe(0);
293
+ });
294
+ it('should build reverse graph from a single importer', () => {
295
+ const result = buildReverseDependencyGraph([
296
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
297
+ 'components/DrinkCard.tsx': ['DrinkList'],
298
+ }),
299
+ ]);
300
+ expect(result.get('DrinkCard')).toEqual(new Set(['DrinkList']));
301
+ });
302
+ it('should collect multiple importers across file paths', () => {
303
+ const result = buildReverseDependencyGraph([
304
+ entity('Button', 'components/Button.tsx', {
305
+ 'components/Button.tsx': ['LoginForm'],
306
+ 'lib/ui/Button.tsx': ['SignupForm'],
307
+ }),
308
+ ]);
309
+ expect(result.get('Button')).toEqual(new Set(['LoginForm', 'SignupForm']));
310
+ });
311
+ it('should deduplicate importers appearing under multiple file paths', () => {
312
+ const result = buildReverseDependencyGraph([
313
+ entity('Icon', 'components/Icon.tsx', {
314
+ 'components/Icon.tsx': ['Header'],
315
+ 'lib/Icon.tsx': ['Header'],
316
+ }),
317
+ ]);
318
+ expect(result.get('Icon')).toEqual(new Set(['Header']));
319
+ });
320
+ it('should skip entities with no importedBy metadata', () => {
321
+ const result = buildReverseDependencyGraph([
322
+ entity('Orphan', 'components/Orphan.tsx'),
323
+ entity('Used', 'components/Used.tsx', {
324
+ 'components/Used.tsx': ['Parent'],
325
+ }),
326
+ ]);
327
+ expect(result.has('Orphan')).toBe(false);
328
+ expect(result.get('Used')).toEqual(new Set(['Parent']));
329
+ });
330
+ it('should handle empty importedBy object gracefully', () => {
331
+ const entities = [
332
+ { name: 'Empty', filePath: 'components/Empty.tsx', importedBy: {} },
333
+ ];
334
+ const result = buildReverseDependencyGraph(entities);
335
+ expect(result.has('Empty')).toBe(false);
336
+ });
337
+ it('should handle multiple importers under a single file path', () => {
338
+ const result = buildReverseDependencyGraph([
339
+ entity('utils', 'lib/utils.ts', {
340
+ 'lib/utils.ts': ['DrinkCard', 'DrinkList', 'Header'],
341
+ }),
342
+ ]);
343
+ expect(result.get('utils')).toEqual(new Set(['DrinkCard', 'DrinkList', 'Header']));
344
+ });
345
+ });
346
+ // ── classifyDirectChanges ────────────────────────────────────────────
347
+ describe('classifyDirectChanges', () => {
348
+ it('should classify entities whose file paths match changed files', () => {
349
+ const changedFiles = new Map([
350
+ ['components/DrinkCard.tsx', 'edited'],
351
+ ['app/page.tsx', 'new'],
352
+ ]);
353
+ const entities = [
354
+ entity('DrinkCard', 'components/DrinkCard.tsx'),
355
+ entity('Home', 'app/page.tsx'),
356
+ entity('Header', 'components/Header.tsx'),
357
+ ];
358
+ const result = classifyDirectChanges(changedFiles, entities);
359
+ expect(result.get('DrinkCard')).toBe('edited');
360
+ expect(result.get('Home')).toBe('new');
361
+ expect(result.has('Header')).toBe(false);
362
+ });
363
+ it('should return empty map when no files match', () => {
364
+ const changedFiles = new Map([
365
+ ['unrelated/file.ts', 'edited'],
366
+ ]);
367
+ const result = classifyDirectChanges(changedFiles, [
368
+ entity('DrinkCard', 'components/DrinkCard.tsx'),
369
+ ]);
370
+ expect(result.size).toBe(0);
371
+ });
372
+ it('should return empty map for empty inputs', () => {
373
+ expect(classifyDirectChanges(new Map(), [])).toEqual(new Map());
374
+ });
375
+ it('should match file paths exactly (no normalization)', () => {
376
+ const changedFiles = new Map([
377
+ ['components/DrinkCard.tsx', 'edited'],
378
+ ]);
379
+ // Slightly different path should NOT match
380
+ const result = classifyDirectChanges(changedFiles, [
381
+ entity('DrinkCard', './components/DrinkCard.tsx'),
382
+ ]);
383
+ expect(result.size).toBe(0);
384
+ });
385
+ });
386
+ // ── computeEntityChangeStatus ────────────────────────────────────────
387
+ describe('computeEntityChangeStatus', () => {
388
+ // ── Basic classification ───────────────────────────────────────────
389
+ it('should mark entity as "new" when its file is added', () => {
390
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'new']]), [entity('DrinkCard', 'components/DrinkCard.tsx')]);
391
+ expect(result['DrinkCard']).toEqual({ status: 'new' });
392
+ });
393
+ it('should mark entity as "edited" when its file is modified', () => {
394
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'edited']]), [entity('DrinkCard', 'components/DrinkCard.tsx')]);
395
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
396
+ });
397
+ it('should omit unchanged entities with no impacted deps', () => {
398
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'edited']]), [
399
+ entity('DrinkCard', 'components/DrinkCard.tsx'),
400
+ entity('Header', 'components/Header.tsx'),
401
+ ]);
402
+ expect(result['DrinkCard']).toBeDefined();
403
+ expect(result['Header']).toBeUndefined();
404
+ });
405
+ it('should return empty map for empty inputs', () => {
406
+ expect(computeEntityChangeStatus(new Map(), [])).toEqual({});
407
+ });
408
+ it('should return empty map when changedFiles has entries but no entity matches', () => {
409
+ const result = computeEntityChangeStatus(new Map([['unrelated/file.ts', 'edited']]), [entity('DrinkCard', 'components/DrinkCard.tsx')]);
410
+ expect(result).toEqual({});
411
+ });
412
+ // ── Single-level impact ────────────────────────────────────────────
413
+ it('should mark single-level impact: A imports B, B edited → A impacted by B', () => {
414
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'edited']]), [
415
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
416
+ 'components/DrinkCard.tsx': ['DrinkList'],
417
+ }),
418
+ entity('DrinkList', 'components/DrinkList.tsx'),
419
+ ]);
420
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
421
+ expect(result['DrinkList']).toEqual({
422
+ status: 'impacted',
423
+ impactedBy: [
424
+ {
425
+ name: 'DrinkCard',
426
+ filePath: 'components/DrinkCard.tsx',
427
+ changeType: 'edited',
428
+ },
429
+ ],
430
+ });
431
+ });
432
+ it('should preserve changeType correctly: "new" root cause shows as "new" in impactedBy', () => {
433
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'new']]), [
434
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
435
+ 'components/DrinkCard.tsx': ['DrinkList'],
436
+ }),
437
+ entity('DrinkList', 'components/DrinkList.tsx'),
438
+ ]);
439
+ expect(result['DrinkList']?.impactedBy?.[0]?.changeType).toBe('new');
440
+ });
441
+ // ── Multi-level transitive ─────────────────────────────────────────
442
+ it('should trace multi-level transitive: Icon→DrinkCard→DrinkList, Icon edited → all impacted by Icon', () => {
443
+ const result = computeEntityChangeStatus(new Map([['components/Icon.tsx', 'edited']]), [
444
+ entity('Icon', 'components/Icon.tsx', {
445
+ 'components/Icon.tsx': ['DrinkCard'],
446
+ }),
447
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
448
+ 'components/DrinkCard.tsx': ['DrinkList'],
449
+ }),
450
+ entity('DrinkList', 'components/DrinkList.tsx'),
451
+ ]);
452
+ expect(result['Icon']).toEqual({ status: 'edited' });
453
+ // Both should trace back to Icon as root cause (not intermediate DrinkCard)
454
+ expect(result['DrinkCard']?.impactedBy).toEqual([
455
+ { name: 'Icon', filePath: 'components/Icon.tsx', changeType: 'edited' },
456
+ ]);
457
+ expect(result['DrinkList']?.impactedBy).toEqual([
458
+ { name: 'Icon', filePath: 'components/Icon.tsx', changeType: 'edited' },
459
+ ]);
460
+ });
461
+ // ── Multiple root causes ───────────────────────────────────────────
462
+ it('should track multiple root causes: DrinkCard imports Icon and utils (both changed)', () => {
463
+ const result = computeEntityChangeStatus(new Map([
464
+ ['components/Icon.tsx', 'edited'],
465
+ ['lib/utils.ts', 'new'],
466
+ ]), [
467
+ entity('Icon', 'components/Icon.tsx', {
468
+ 'components/Icon.tsx': ['DrinkCard'],
469
+ }),
470
+ entity('utils', 'lib/utils.ts', {
471
+ 'lib/utils.ts': ['DrinkCard'],
472
+ }),
473
+ entity('DrinkCard', 'components/DrinkCard.tsx'),
474
+ ]);
475
+ expect(result['DrinkCard']?.status).toBe('impacted');
476
+ expect(impactNames(result, 'DrinkCard')).toEqual(['Icon', 'utils']);
477
+ // Verify changeTypes are preserved
478
+ const iconDep = result['DrinkCard']?.impactedBy?.find((d) => d.name === 'Icon');
479
+ const utilsDep = result['DrinkCard']?.impactedBy?.find((d) => d.name === 'utils');
480
+ expect(iconDep?.changeType).toBe('edited');
481
+ expect(utilsDep?.changeType).toBe('new');
482
+ });
483
+ // ── Diamond dependency (BFS root cause propagation) ────────────────
484
+ it('should propagate ALL root causes through diamond: D1→A→C→E, D2→B→C→E', () => {
485
+ // This tests that when two paths converge at C and C has downstream
486
+ // importers (E), ALL root causes propagate through — not just the
487
+ // ones from whichever path was processed first.
488
+ //
489
+ // D1(edited) ──importedBy──→ A ──importedBy──→ C ──importedBy──→ E
490
+ // D2(new) ──importedBy──→ B ──importedBy──→ C
491
+ //
492
+ const result = computeEntityChangeStatus(new Map([
493
+ ['components/D1.tsx', 'edited'],
494
+ ['components/D2.tsx', 'new'],
495
+ ]), [
496
+ entity('D1', 'components/D1.tsx', {
497
+ 'components/D1.tsx': ['A'],
498
+ }),
499
+ entity('D2', 'components/D2.tsx', {
500
+ 'components/D2.tsx': ['B'],
501
+ }),
502
+ entity('A', 'components/A.tsx', {
503
+ 'components/A.tsx': ['C'],
504
+ }),
505
+ entity('B', 'components/B.tsx', {
506
+ 'components/B.tsx': ['C'],
507
+ }),
508
+ entity('C', 'components/C.tsx', {
509
+ 'components/C.tsx': ['E'],
510
+ }),
511
+ entity('E', 'components/E.tsx'),
512
+ ]);
513
+ // C should have both root causes
514
+ expect(result['C']?.status).toBe('impacted');
515
+ expect(impactNames(result, 'C')).toEqual(['D1', 'D2']);
516
+ // E (downstream of C) must also have both root causes
517
+ expect(result['E']?.status).toBe('impacted');
518
+ expect(impactNames(result, 'E')).toEqual(['D1', 'D2']);
519
+ });
520
+ it('should propagate root causes through diamond with unequal path lengths', () => {
521
+ // D1 reaches C directly (depth 1), D2 reaches C via B (depth 2).
522
+ // C has downstream E. E must get both root causes.
523
+ //
524
+ // D1(edited) ──importedBy──→ C ──importedBy──→ E
525
+ // D2(new) ──importedBy──→ B ──importedBy──→ C
526
+ //
527
+ const result = computeEntityChangeStatus(new Map([
528
+ ['components/D1.tsx', 'edited'],
529
+ ['components/D2.tsx', 'new'],
530
+ ]), [
531
+ entity('D1', 'components/D1.tsx', {
532
+ 'components/D1.tsx': ['C'],
533
+ }),
534
+ entity('D2', 'components/D2.tsx', {
535
+ 'components/D2.tsx': ['B'],
536
+ }),
537
+ entity('B', 'components/B.tsx', {
538
+ 'components/B.tsx': ['C'],
539
+ }),
540
+ entity('C', 'components/C.tsx', {
541
+ 'components/C.tsx': ['E'],
542
+ }),
543
+ entity('E', 'components/E.tsx'),
544
+ ]);
545
+ expect(impactNames(result, 'C')).toEqual(['D1', 'D2']);
546
+ expect(impactNames(result, 'E')).toEqual(['D1', 'D2']);
547
+ });
548
+ // ── Cycles ─────────────────────────────────────────────────────────
549
+ it('should handle simple cycle without infinite loop: A↔B, A edited', () => {
550
+ const result = computeEntityChangeStatus(new Map([['components/A.tsx', 'edited']]), [
551
+ entity('A', 'components/A.tsx', {
552
+ 'components/A.tsx': ['B'],
553
+ }),
554
+ entity('B', 'components/B.tsx', {
555
+ 'components/B.tsx': ['A'],
556
+ }),
557
+ ]);
558
+ expect(result['A']).toEqual({ status: 'edited' });
559
+ expect(result['B']).toEqual({
560
+ status: 'impacted',
561
+ impactedBy: [
562
+ { name: 'A', filePath: 'components/A.tsx', changeType: 'edited' },
563
+ ],
564
+ });
565
+ });
566
+ it('should handle three-node cycle: A→B→C→A, A edited', () => {
567
+ const result = computeEntityChangeStatus(new Map([['components/A.tsx', 'edited']]), [
568
+ entity('A', 'components/A.tsx', {
569
+ 'components/A.tsx': ['B'],
570
+ }),
571
+ entity('B', 'components/B.tsx', {
572
+ 'components/B.tsx': ['C'],
573
+ }),
574
+ entity('C', 'components/C.tsx', {
575
+ 'components/C.tsx': ['A'],
576
+ }),
577
+ ]);
578
+ expect(result['A']).toEqual({ status: 'edited' });
579
+ expect(result['B']?.status).toBe('impacted');
580
+ expect(result['C']?.status).toBe('impacted');
581
+ expect(impactNames(result, 'B')).toEqual(['A']);
582
+ expect(impactNames(result, 'C')).toEqual(['A']);
583
+ });
584
+ it('should handle cycle with multiple seeds: A→B→C→A, A and C both edited', () => {
585
+ const result = computeEntityChangeStatus(new Map([
586
+ ['components/A.tsx', 'edited'],
587
+ ['components/C.tsx', 'new'],
588
+ ]), [
589
+ entity('A', 'components/A.tsx', {
590
+ 'components/A.tsx': ['B'],
591
+ }),
592
+ entity('B', 'components/B.tsx', {
593
+ 'components/B.tsx': ['C'],
594
+ }),
595
+ entity('C', 'components/C.tsx', {
596
+ 'components/C.tsx': ['A'],
597
+ }),
598
+ ]);
599
+ // A and C directly changed
600
+ expect(result['A']).toEqual({ status: 'edited' });
601
+ expect(result['C']).toEqual({ status: 'new' });
602
+ // B is impacted by both (A importedBy B via A→B, C importedBy B via C→...→B)
603
+ expect(result['B']?.status).toBe('impacted');
604
+ expect(impactNames(result, 'B')).toEqual(['A', 'C']);
605
+ });
606
+ it('should handle self-import cycle gracefully', () => {
607
+ // Entity lists itself as its own importer
608
+ const result = computeEntityChangeStatus(new Map([['components/A.tsx', 'edited']]), [
609
+ entity('A', 'components/A.tsx', {
610
+ 'components/A.tsx': ['A'],
611
+ }),
612
+ ]);
613
+ // A is directly changed — self-loop should not create an impacted entry
614
+ expect(result['A']).toEqual({ status: 'edited' });
615
+ });
616
+ // ── Direct change priority ─────────────────────────────────────────
617
+ it('should NOT mark directly-changed entities as impacted even if they import other changed entities', () => {
618
+ const result = computeEntityChangeStatus(new Map([
619
+ ['components/A.tsx', 'edited'],
620
+ ['components/B.tsx', 'new'],
621
+ ]), [
622
+ entity('A', 'components/A.tsx', {
623
+ 'components/A.tsx': ['B'],
624
+ }),
625
+ entity('B', 'components/B.tsx'),
626
+ ]);
627
+ expect(result['A']).toEqual({ status: 'edited' });
628
+ expect(result['B']).toEqual({ status: 'new' });
629
+ });
630
+ it('should not produce impacted entries when ALL entities are directly changed', () => {
631
+ const result = computeEntityChangeStatus(new Map([
632
+ ['components/A.tsx', 'edited'],
633
+ ['components/B.tsx', 'new'],
634
+ ['components/C.tsx', 'edited'],
635
+ ]), [
636
+ entity('A', 'components/A.tsx', {
637
+ 'components/A.tsx': ['B'],
638
+ }),
639
+ entity('B', 'components/B.tsx', {
640
+ 'components/B.tsx': ['C'],
641
+ }),
642
+ entity('C', 'components/C.tsx'),
643
+ ]);
644
+ expect(Object.values(result).every((s) => s.status !== 'impacted')).toBe(true);
645
+ expect(result['A']?.status).toBe('edited');
646
+ expect(result['B']?.status).toBe('new');
647
+ expect(result['C']?.status).toBe('edited');
648
+ });
649
+ // ── Missing metadata / graceful degradation ────────────────────────
650
+ it('should handle missing importedBy metadata gracefully (no impact propagation)', () => {
651
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'edited']]), [
652
+ entity('DrinkCard', 'components/DrinkCard.tsx'),
653
+ entity('DrinkList', 'components/DrinkList.tsx'),
654
+ ]);
655
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
656
+ expect(result['DrinkList']).toBeUndefined();
657
+ });
658
+ it('should skip phantom importers (importedBy references entity not in entities list)', () => {
659
+ // DrinkCard's importedBy references "PhantomPage" which is not in entities
660
+ const result = computeEntityChangeStatus(new Map([['components/DrinkCard.tsx', 'edited']]), [
661
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
662
+ 'components/DrinkCard.tsx': ['PhantomPage', 'DrinkList'],
663
+ }),
664
+ entity('DrinkList', 'components/DrinkList.tsx'),
665
+ ]);
666
+ // PhantomPage should not appear in results (not in entities list)
667
+ expect(result['PhantomPage']).toBeUndefined();
668
+ // DrinkList should still be impacted
669
+ expect(result['DrinkList']?.status).toBe('impacted');
670
+ });
671
+ // ── maxDepth ───────────────────────────────────────────────────────
672
+ it('should enforce maxDepth and stop propagation beyond limit', () => {
673
+ // Chain: D → C → B → A
674
+ const result = computeEntityChangeStatus(new Map([['components/D.tsx', 'edited']]), [
675
+ entity('D', 'components/D.tsx', { 'components/D.tsx': ['C'] }),
676
+ entity('C', 'components/C.tsx', { 'components/C.tsx': ['B'] }),
677
+ entity('B', 'components/B.tsx', { 'components/B.tsx': ['A'] }),
678
+ entity('A', 'components/A.tsx'),
679
+ ], 2);
680
+ expect(result['D']).toEqual({ status: 'edited' });
681
+ expect(result['C']?.status).toBe('impacted');
682
+ expect(result['B']?.status).toBe('impacted');
683
+ // A at depth 3 is cut off
684
+ expect(result['A']).toBeUndefined();
685
+ });
686
+ it('should use default maxDepth of 20 when not specified', () => {
687
+ // Build a chain of depth 15 — should all be reached with default maxDepth
688
+ const entities = [];
689
+ for (let i = 0; i < 16; i++) {
690
+ const name = `E${i}`;
691
+ const filePath = `components/${name}.tsx`;
692
+ if (i < 15) {
693
+ entities.push(entity(name, filePath, { [filePath]: [`E${i + 1}`] }));
694
+ }
695
+ else {
696
+ entities.push(entity(name, filePath));
697
+ }
698
+ }
699
+ const result = computeEntityChangeStatus(new Map([['components/E0.tsx', 'edited']]), entities);
700
+ // E15 at depth 15 should still be reached (within default maxDepth of 20)
701
+ expect(result['E15']?.status).toBe('impacted');
702
+ });
703
+ // ── Fan-out ────────────────────────────────────────────────────────
704
+ it('should handle large fan-out: one changed entity imported by many', () => {
705
+ const importers = Array.from({ length: 10 }, (_, i) => `Comp${i}`);
706
+ const entities = [
707
+ entity('SharedUtil', 'lib/shared.ts', {
708
+ 'lib/shared.ts': importers,
709
+ }),
710
+ ...importers.map((name) => entity(name, `components/${name}.tsx`)),
711
+ ];
712
+ const result = computeEntityChangeStatus(new Map([['lib/shared.ts', 'edited']]), entities);
713
+ expect(result['SharedUtil']).toEqual({ status: 'edited' });
714
+ for (const name of importers) {
715
+ expect(result[name]?.status).toBe('impacted');
716
+ expect(result[name]?.impactedBy).toEqual([
717
+ {
718
+ name: 'SharedUtil',
719
+ filePath: 'lib/shared.ts',
720
+ changeType: 'edited',
721
+ },
722
+ ]);
723
+ }
724
+ });
725
+ // ── Deterministic output ───────────────────────────────────────────
726
+ it('should sort impactedBy array by name for deterministic output', () => {
727
+ const result = computeEntityChangeStatus(new Map([
728
+ ['components/Zebra.tsx', 'edited'],
729
+ ['components/Apple.tsx', 'new'],
730
+ ['components/Mango.tsx', 'edited'],
731
+ ]), [
732
+ entity('Zebra', 'components/Zebra.tsx', {
733
+ 'components/Zebra.tsx': ['Target'],
734
+ }),
735
+ entity('Apple', 'components/Apple.tsx', {
736
+ 'components/Apple.tsx': ['Target'],
737
+ }),
738
+ entity('Mango', 'components/Mango.tsx', {
739
+ 'components/Mango.tsx': ['Target'],
740
+ }),
741
+ entity('Target', 'components/Target.tsx'),
742
+ ]);
743
+ expect(impactNames(result, 'Target')).toEqual([
744
+ 'Apple',
745
+ 'Mango',
746
+ 'Zebra',
747
+ ]);
748
+ });
749
+ // ── Realistic integration-style scenarios ──────────────────────────
750
+ it('should compute correct status for a realistic Next.js app scenario', () => {
751
+ // Simulates a real editor session:
752
+ // - User edited DrinkCard component and added a new formatPrice utility
753
+ // - DrinkCard is imported by DrinkList component and Home page
754
+ // - formatPrice is imported by DrinkCard
755
+ // - Header is unchanged and imports nothing changed
756
+ //
757
+ // Expected:
758
+ // - DrinkCard: edited (directly changed)
759
+ // - formatPrice: new (directly changed)
760
+ // - DrinkList: impacted by [DrinkCard, formatPrice] (DrinkCard is edited,
761
+ // and formatPrice flows through DrinkCard)
762
+ // - Home: impacted by [DrinkCard, formatPrice] (same reasoning)
763
+ // - Header: not in results (unchanged, no changed deps)
764
+ const result = computeEntityChangeStatus(new Map([
765
+ ['components/DrinkCard.tsx', 'edited'],
766
+ ['lib/formatPrice.ts', 'new'],
767
+ ]), [
768
+ entity('formatPrice', 'lib/formatPrice.ts', {
769
+ 'lib/formatPrice.ts': ['DrinkCard'],
770
+ }),
771
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
772
+ 'components/DrinkCard.tsx': ['DrinkList', 'Home'],
773
+ }),
774
+ entity('DrinkList', 'components/DrinkList.tsx'),
775
+ entity('Home', 'app/page.tsx'),
776
+ entity('Header', 'components/Header.tsx'),
777
+ ]);
778
+ expect(result['formatPrice']).toEqual({ status: 'new' });
779
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
780
+ expect(result['Header']).toBeUndefined();
781
+ // DrinkList and Home are impacted by DrinkCard (directly, its an importer)
782
+ // AND by formatPrice (transitively via DrinkCard)
783
+ expect(result['DrinkList']?.status).toBe('impacted');
784
+ expect(impactNames(result, 'DrinkList')).toEqual([
785
+ 'DrinkCard',
786
+ 'formatPrice',
787
+ ]);
788
+ expect(result['Home']?.status).toBe('impacted');
789
+ expect(impactNames(result, 'Home')).toEqual(['DrinkCard', 'formatPrice']);
790
+ });
791
+ it('should handle a realistic scenario where component is both directly changed AND an intermediate', () => {
792
+ // DrinkCard is edited AND it imports formatPrice (also changed).
793
+ // DrinkCard should be marked 'edited' (direct), not 'impacted'.
794
+ // DrinkList imports DrinkCard → impacted by both DrinkCard and formatPrice.
795
+ const result = computeEntityChangeStatus(new Map([
796
+ ['components/DrinkCard.tsx', 'edited'],
797
+ ['lib/formatPrice.ts', 'new'],
798
+ ]), [
799
+ entity('formatPrice', 'lib/formatPrice.ts', {
800
+ 'lib/formatPrice.ts': ['DrinkCard'],
801
+ }),
802
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
803
+ 'components/DrinkCard.tsx': ['DrinkList'],
804
+ }),
805
+ entity('DrinkList', 'components/DrinkList.tsx'),
806
+ ]);
807
+ // DrinkCard is directly changed — status should be 'edited', no impactedBy
808
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
809
+ // DrinkList is impacted by BOTH root causes
810
+ expect(result['DrinkList']?.status).toBe('impacted');
811
+ expect(impactNames(result, 'DrinkList')).toEqual([
812
+ 'DrinkCard',
813
+ 'formatPrice',
814
+ ]);
815
+ });
816
+ // ── End-to-end pipeline test ───────────────────────────────────────
817
+ it('should produce correct results through the full pipeline: git files → scenarios → entity infos → status', () => {
818
+ // This tests the complete data flow that both route handlers execute:
819
+ // 1. buildChangedFilesMap from git status
820
+ // 2. buildEntityInfosFromScenarios from scenario data + metadata
821
+ // 3. computeEntityChangeStatus from the above
822
+ // Step 1: Git status
823
+ const gitFiles = [
824
+ { path: 'components/DrinkCard.tsx', status: 'modified' },
825
+ { path: 'lib/formatPrice.ts', status: 'added' },
826
+ { path: 'components/Header.tsx', status: 'deleted' },
827
+ ];
828
+ const changedFiles = buildChangedFilesMap(gitFiles, false);
829
+ expect(changedFiles.get('components/DrinkCard.tsx')).toBe('edited');
830
+ expect(changedFiles.get('lib/formatPrice.ts')).toBe('new');
831
+ expect(changedFiles.has('components/Header.tsx')).toBe(false);
832
+ // Step 2: Build entity infos from scenarios
833
+ const scenarios = [
834
+ {
835
+ componentName: 'DrinkCard',
836
+ componentPath: 'components/DrinkCard.tsx',
837
+ },
838
+ { componentName: null, componentPath: null, url: '/' },
839
+ {
840
+ componentName: 'DrinkList',
841
+ componentPath: 'components/DrinkList.tsx',
842
+ },
843
+ ];
844
+ const pageFilePaths = { Home: 'app/page.tsx' };
845
+ const metadata = [
846
+ {
847
+ name: 'DrinkCard',
848
+ metadata: {
849
+ importedBy: {
850
+ 'components/DrinkCard.tsx': {
851
+ DrinkList: { shas: ['sha1'] },
852
+ Home: { shas: ['sha2'] },
853
+ },
854
+ },
855
+ },
856
+ },
857
+ { name: 'DrinkList', metadata: null },
858
+ ];
859
+ const entityInfos = buildEntityInfosFromScenarios(scenarios, pageFilePaths, metadata);
860
+ expect(entityInfos).toHaveLength(3);
861
+ expect(entityInfos.find((e) => e.name === 'DrinkCard')?.importedBy).toBeDefined();
862
+ // Step 3: Compute status
863
+ const result = computeEntityChangeStatus(changedFiles, entityInfos);
864
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
865
+ expect(result['DrinkList']?.status).toBe('impacted');
866
+ expect(result['Home']?.status).toBe('impacted');
867
+ expect(impactNames(result, 'DrinkList')).toEqual(['DrinkCard']);
868
+ expect(impactNames(result, 'Home')).toEqual(['DrinkCard']);
869
+ });
870
+ it('should produce correct results for first-feature pipeline (all files new)', () => {
871
+ const gitFiles = [
872
+ { path: 'components/DrinkCard.tsx', status: 'modified' },
873
+ { path: 'app/page.tsx', status: 'modified' },
874
+ { path: 'components/Header.tsx', status: 'untracked' },
875
+ ];
876
+ // isFirstFeature = true → all become 'new'
877
+ const changedFiles = buildChangedFilesMap(gitFiles, true);
878
+ expect(changedFiles.get('components/DrinkCard.tsx')).toBe('new');
879
+ expect(changedFiles.get('app/page.tsx')).toBe('new');
880
+ expect(changedFiles.get('components/Header.tsx')).toBe('new');
881
+ const scenarios = [
882
+ {
883
+ componentName: 'DrinkCard',
884
+ componentPath: 'components/DrinkCard.tsx',
885
+ },
886
+ { componentName: null, componentPath: null, url: '/' },
887
+ { componentName: 'Header', componentPath: 'components/Header.tsx' },
888
+ ];
889
+ const entityInfos = buildEntityInfosFromScenarios(scenarios, { Home: 'app/page.tsx' }, []);
890
+ const result = computeEntityChangeStatus(changedFiles, entityInfos);
891
+ // All should be 'new' — none impacted
892
+ expect(result['DrinkCard']).toEqual({ status: 'new' });
893
+ expect(result['Home']).toEqual({ status: 'new' });
894
+ expect(result['Header']).toEqual({ status: 'new' });
895
+ expect(Object.values(result).some((s) => s.status === 'impacted')).toBe(false);
896
+ });
897
+ // ── Pass-through propagation ─────────────────────────────────────
898
+ it('should propagate root causes THROUGH a directly-changed intermediate entity', () => {
899
+ // D is edited, A is also edited, A importedBy B.
900
+ // D importedBy A (so D flows through A).
901
+ // B should be impacted by BOTH D (transitive through A) and A (direct importer).
902
+ //
903
+ // D(edited) ──importedBy──→ A(edited) ──importedBy──→ B
904
+ //
905
+ const result = computeEntityChangeStatus(new Map([
906
+ ['components/D.tsx', 'edited'],
907
+ ['components/A.tsx', 'edited'],
908
+ ]), [
909
+ entity('D', 'components/D.tsx', {
910
+ 'components/D.tsx': ['A'],
911
+ }),
912
+ entity('A', 'components/A.tsx', {
913
+ 'components/A.tsx': ['B'],
914
+ }),
915
+ entity('B', 'components/B.tsx'),
916
+ ]);
917
+ expect(result['D']).toEqual({ status: 'edited' });
918
+ expect(result['A']).toEqual({ status: 'edited' });
919
+ // B must see BOTH root causes — D propagated through A
920
+ expect(result['B']?.status).toBe('impacted');
921
+ expect(impactNames(result, 'B')).toEqual(['A', 'D']);
922
+ });
923
+ it('should propagate through a chain of directly-changed entities: D→C→B→A all changed except A', () => {
924
+ // D, C, B all directly changed. A imports B.
925
+ // A should see D, C, B as root causes.
926
+ const result = computeEntityChangeStatus(new Map([
927
+ ['components/D.tsx', 'edited'],
928
+ ['components/C.tsx', 'new'],
929
+ ['components/B.tsx', 'edited'],
930
+ ]), [
931
+ entity('D', 'components/D.tsx', { 'components/D.tsx': ['C'] }),
932
+ entity('C', 'components/C.tsx', { 'components/C.tsx': ['B'] }),
933
+ entity('B', 'components/B.tsx', { 'components/B.tsx': ['A'] }),
934
+ entity('A', 'components/A.tsx'),
935
+ ]);
936
+ expect(result['A']?.status).toBe('impacted');
937
+ expect(impactNames(result, 'A')).toEqual(['B', 'C', 'D']);
938
+ });
939
+ // ── Wide diamond ─────────────────────────────────────────────────
940
+ it('should handle wide diamond: 3+ paths converging then fanning out', () => {
941
+ // D1, D2, D3 all edited, all importedBy Merge, Merge importedBy [Out1, Out2]
942
+ //
943
+ // D1 ─┐
944
+ // D2 ─┼──importedBy──→ Merge ──importedBy──→ Out1
945
+ // D3 ─┘ └──importedBy──→ Out2
946
+ //
947
+ const result = computeEntityChangeStatus(new Map([
948
+ ['components/D1.tsx', 'edited'],
949
+ ['components/D2.tsx', 'new'],
950
+ ['components/D3.tsx', 'edited'],
951
+ ]), [
952
+ entity('D1', 'components/D1.tsx', { 'components/D1.tsx': ['Merge'] }),
953
+ entity('D2', 'components/D2.tsx', { 'components/D2.tsx': ['Merge'] }),
954
+ entity('D3', 'components/D3.tsx', { 'components/D3.tsx': ['Merge'] }),
955
+ entity('Merge', 'components/Merge.tsx', {
956
+ 'components/Merge.tsx': ['Out1', 'Out2'],
957
+ }),
958
+ entity('Out1', 'components/Out1.tsx'),
959
+ entity('Out2', 'components/Out2.tsx'),
960
+ ]);
961
+ // Merge gets all 3 root causes
962
+ expect(impactNames(result, 'Merge')).toEqual(['D1', 'D2', 'D3']);
963
+ // Out1 and Out2 also get all 3 root causes (propagated through Merge)
964
+ expect(impactNames(result, 'Out1')).toEqual(['D1', 'D2', 'D3']);
965
+ expect(impactNames(result, 'Out2')).toEqual(['D1', 'D2', 'D3']);
966
+ });
967
+ // ── changedFiles with non-entity entries ─────────────────────────
968
+ it('should ignore changedFiles entries that do not match any entity file path', () => {
969
+ // Git shows many changed files, but only some are entities
970
+ const result = computeEntityChangeStatus(new Map([
971
+ ['components/DrinkCard.tsx', 'edited'],
972
+ ['package.json', 'edited'],
973
+ ['README.md', 'new'],
974
+ ['.env', 'edited'],
975
+ ['tsconfig.json', 'edited'],
976
+ ]), [
977
+ entity('DrinkCard', 'components/DrinkCard.tsx', {
978
+ 'components/DrinkCard.tsx': ['DrinkList'],
979
+ }),
980
+ entity('DrinkList', 'components/DrinkList.tsx'),
981
+ ]);
982
+ expect(result['DrinkCard']).toEqual({ status: 'edited' });
983
+ expect(result['DrinkList']?.status).toBe('impacted');
984
+ // Only entities should appear in results
985
+ expect(Object.keys(result)).toEqual(['DrinkCard', 'DrinkList']);
986
+ });
987
+ });
988
+ // ── buildEntityInfosFromScenarios — additional edge cases ────────
989
+ describe('buildEntityInfosFromScenarios (edge cases)', () => {
990
+ it('should deduplicate page scenarios with different URLs resolving to same name', () => {
991
+ // /drinks and /drinks/123 both resolve to "Drinks"
992
+ const scenarios = [
993
+ { componentName: null, componentPath: null, url: '/drinks' },
994
+ { componentName: null, componentPath: null, url: '/drinks/123' },
995
+ {
996
+ componentName: null,
997
+ componentPath: null,
998
+ url: '/drinks/456?tab=reviews',
999
+ },
1000
+ ];
1001
+ const pageFilePaths = { Drinks: 'app/drinks/page.tsx' };
1002
+ const result = buildEntityInfosFromScenarios(scenarios, pageFilePaths, []);
1003
+ // Should deduplicate to a single "Drinks" entity
1004
+ expect(result).toHaveLength(1);
1005
+ expect(result[0]).toEqual({
1006
+ name: 'Drinks',
1007
+ filePath: 'app/drinks/page.tsx',
1008
+ });
1009
+ });
1010
+ it('should handle component and page scenario with same conceptual name (component wins)', () => {
1011
+ // A component named "Home" AND a page URL "/" both produce name "Home"
1012
+ // Component scenario comes first — it should win
1013
+ const scenarios = [
1014
+ { componentName: 'Home', componentPath: 'components/Home.tsx' },
1015
+ { componentName: null, componentPath: null, url: '/' },
1016
+ ];
1017
+ const pageFilePaths = { Home: 'app/page.tsx' };
1018
+ const result = buildEntityInfosFromScenarios(scenarios, pageFilePaths, []);
1019
+ expect(result).toHaveLength(1);
1020
+ // Component path should win (first occurrence)
1021
+ expect(result[0].filePath).toBe('components/Home.tsx');
1022
+ });
1023
+ it('should handle multiple scenarios for the same component (only first is included)', () => {
1024
+ const scenarios = [
1025
+ {
1026
+ componentName: 'DrinkCard',
1027
+ componentPath: 'components/DrinkCard.tsx',
1028
+ },
1029
+ {
1030
+ componentName: 'DrinkCard',
1031
+ componentPath: 'components/DrinkCard.tsx',
1032
+ },
1033
+ {
1034
+ componentName: 'DrinkCard',
1035
+ componentPath: 'components/DrinkCard.tsx',
1036
+ },
1037
+ ];
1038
+ const metadata = [
1039
+ {
1040
+ name: 'DrinkCard',
1041
+ metadata: {
1042
+ importedBy: {
1043
+ 'components/DrinkCard.tsx': { DrinkList: { shas: ['abc'] } },
1044
+ },
1045
+ },
1046
+ },
1047
+ ];
1048
+ const result = buildEntityInfosFromScenarios(scenarios, {}, metadata);
1049
+ expect(result).toHaveLength(1);
1050
+ expect(result[0].importedBy).toBeDefined();
1051
+ });
1052
+ });
1053
+ // ── Full pipeline regression tests ───────────────────────────────
1054
+ describe('full pipeline regression tests', () => {
1055
+ it('regression: entity with changed file AND imported by other unchanged entities should propagate', () => {
1056
+ // This catches the old regex-based bug: previously, "impacted" was marked
1057
+ // for every entity not directly changed, even if no dependency was actually changed.
1058
+ // With the new system, only entities that transitively import a changed entity get impacted.
1059
+ const gitFiles = [
1060
+ { path: 'lib/formatPrice.ts', status: 'added' },
1061
+ ];
1062
+ const changedFiles = buildChangedFilesMap(gitFiles, false);
1063
+ const scenarios = [
1064
+ {
1065
+ componentName: 'DrinkCard',
1066
+ componentPath: 'components/DrinkCard.tsx',
1067
+ },
1068
+ { componentName: 'Header', componentPath: 'components/Header.tsx' },
1069
+ { componentName: null, componentPath: null, url: '/' },
1070
+ ];
1071
+ const pageFilePaths = { Home: 'app/page.tsx' };
1072
+ const metadata = [
1073
+ {
1074
+ name: 'formatPrice',
1075
+ metadata: {
1076
+ importedBy: {
1077
+ 'lib/formatPrice.ts': { DrinkCard: { shas: ['sha1'] } },
1078
+ },
1079
+ },
1080
+ },
1081
+ {
1082
+ name: 'DrinkCard',
1083
+ metadata: {
1084
+ importedBy: {
1085
+ 'components/DrinkCard.tsx': { Home: { shas: ['sha2'] } },
1086
+ },
1087
+ },
1088
+ },
1089
+ ];
1090
+ // formatPrice is not a scenario entity — only DrinkCard, Header, Home are.
1091
+ // But formatPrice IS in the metadata as importedBy DrinkCard.
1092
+ // We need to include formatPrice in entityInfos for the graph to work.
1093
+ // However, buildEntityInfosFromScenarios only builds from scenarios,
1094
+ // so formatPrice won't be in the entities list → DrinkCard won't be impacted.
1095
+ //
1096
+ // This test documents the current behavior: only entities from scenarios
1097
+ // participate in the change status graph. Changed files that aren't entities
1098
+ // don't propagate impact (they'd need to be in the entities list).
1099
+ const entityInfos = buildEntityInfosFromScenarios(scenarios, pageFilePaths, metadata);
1100
+ const result = computeEntityChangeStatus(changedFiles, entityInfos);
1101
+ // formatPrice is not in scenarios, so not in entityInfos
1102
+ expect(entityInfos.find((e) => e.name === 'formatPrice')).toBeUndefined();
1103
+ // No entity files match changed files → empty result
1104
+ // DrinkCard and Home are NOT impacted because formatPrice is not in entities
1105
+ expect(result['DrinkCard']).toBeUndefined();
1106
+ expect(result['Header']).toBeUndefined();
1107
+ expect(result['Home']).toBeUndefined();
1108
+ });
1109
+ it('pipeline: changed utility file that IS an entity propagates impact correctly', () => {
1110
+ // When the changed utility IS registered as a scenario entity,
1111
+ // the full chain works end-to-end.
1112
+ const gitFiles = [
1113
+ { path: 'lib/formatPrice.ts', status: 'added' },
1114
+ ];
1115
+ const changedFiles = buildChangedFilesMap(gitFiles, false);
1116
+ const scenarios = [
1117
+ { componentName: 'formatPrice', componentPath: 'lib/formatPrice.ts' },
1118
+ {
1119
+ componentName: 'DrinkCard',
1120
+ componentPath: 'components/DrinkCard.tsx',
1121
+ },
1122
+ { componentName: null, componentPath: null, url: '/' },
1123
+ ];
1124
+ const pageFilePaths = { Home: 'app/page.tsx' };
1125
+ const metadata = [
1126
+ {
1127
+ name: 'formatPrice',
1128
+ metadata: {
1129
+ importedBy: {
1130
+ 'lib/formatPrice.ts': { DrinkCard: { shas: ['sha1'] } },
1131
+ },
1132
+ },
1133
+ },
1134
+ {
1135
+ name: 'DrinkCard',
1136
+ metadata: {
1137
+ importedBy: {
1138
+ 'components/DrinkCard.tsx': { Home: { shas: ['sha2'] } },
1139
+ },
1140
+ },
1141
+ },
1142
+ ];
1143
+ const entityInfos = buildEntityInfosFromScenarios(scenarios, pageFilePaths, metadata);
1144
+ const result = computeEntityChangeStatus(changedFiles, entityInfos);
1145
+ expect(result['formatPrice']).toEqual({ status: 'new' });
1146
+ expect(result['DrinkCard']?.status).toBe('impacted');
1147
+ expect(impactNames(result, 'DrinkCard')).toEqual(['formatPrice']);
1148
+ expect(result['Home']?.status).toBe('impacted');
1149
+ expect(impactNames(result, 'Home')).toEqual(['formatPrice']);
1150
+ });
1151
+ it('pipeline: renamed file status is correctly ignored', () => {
1152
+ const gitFiles = [
1153
+ { path: 'components/OldName.tsx', status: 'renamed' },
1154
+ { path: 'components/DrinkCard.tsx', status: 'modified' },
1155
+ ];
1156
+ const changedFiles = buildChangedFilesMap(gitFiles, false);
1157
+ // Renamed file should not appear
1158
+ expect(changedFiles.has('components/OldName.tsx')).toBe(false);
1159
+ expect(changedFiles.get('components/DrinkCard.tsx')).toBe('edited');
1160
+ });
1161
+ it('pipeline: first feature mode with import graph still marks everything new', () => {
1162
+ // Even with importedBy metadata, first-feature mode should mark
1163
+ // all entities as 'new' with no 'impacted' entries.
1164
+ const gitFiles = [
1165
+ { path: 'components/DrinkCard.tsx', status: 'modified' },
1166
+ { path: 'app/page.tsx', status: 'modified' },
1167
+ ];
1168
+ const changedFiles = buildChangedFilesMap(gitFiles, true); // first feature!
1169
+ const scenarios = [
1170
+ {
1171
+ componentName: 'DrinkCard',
1172
+ componentPath: 'components/DrinkCard.tsx',
1173
+ },
1174
+ { componentName: null, componentPath: null, url: '/' },
1175
+ ];
1176
+ const metadata = [
1177
+ {
1178
+ name: 'DrinkCard',
1179
+ metadata: {
1180
+ importedBy: {
1181
+ 'components/DrinkCard.tsx': { Home: { shas: ['sha1'] } },
1182
+ },
1183
+ },
1184
+ },
1185
+ ];
1186
+ const entityInfos = buildEntityInfosFromScenarios(scenarios, { Home: 'app/page.tsx' }, metadata);
1187
+ const result = computeEntityChangeStatus(changedFiles, entityInfos);
1188
+ // Both are directly changed (new) → no impact propagation
1189
+ expect(result['DrinkCard']).toEqual({ status: 'new' });
1190
+ expect(result['Home']).toEqual({ status: 'new' });
1191
+ });
1192
+ });
1193
+ // ── filterGroupsByChangeStatus ──────────────────────────────────────────
1194
+ describe('filterGroupsByChangeStatus', () => {
1195
+ const groups = [
1196
+ ['Home', ['s1', 's2']],
1197
+ ['About', ['s3']],
1198
+ ['Header', ['s4']],
1199
+ ];
1200
+ it('should return all groups when entityChangeStatus is undefined', () => {
1201
+ expect(filterGroupsByChangeStatus(groups, undefined)).toEqual(groups);
1202
+ });
1203
+ it('should return all groups when entityChangeStatus is empty', () => {
1204
+ expect(filterGroupsByChangeStatus(groups, {})).toEqual(groups);
1205
+ });
1206
+ it('should filter to only groups with a change status', () => {
1207
+ const status = {
1208
+ Home: { status: 'new' },
1209
+ Header: { status: 'edited' },
1210
+ };
1211
+ const result = filterGroupsByChangeStatus(groups, status);
1212
+ expect(result).toEqual([
1213
+ ['Home', ['s1', 's2']],
1214
+ ['Header', ['s4']],
1215
+ ]);
1216
+ });
1217
+ it('should return empty array when no groups match', () => {
1218
+ const status = {
1219
+ Footer: { status: 'new' },
1220
+ };
1221
+ const result = filterGroupsByChangeStatus(groups, status);
1222
+ expect(result).toEqual([]);
1223
+ });
1224
+ });
1225
+ // ── buildEntityInfosFromGlossary ────────────────────────────────────
1226
+ describe('buildEntityInfosFromGlossary', () => {
1227
+ it('should convert glossary entries to EntityInfo with name and filePath', () => {
1228
+ const glossary = [
1229
+ { name: 'pad', filePath: 'app/lib/calendar.ts' },
1230
+ { name: 'formatDate', filePath: 'app/lib/calendar.ts' },
1231
+ ];
1232
+ const result = buildEntityInfosFromGlossary(glossary);
1233
+ expect(result).toEqual([
1234
+ { name: 'pad', filePath: 'app/lib/calendar.ts' },
1235
+ { name: 'formatDate', filePath: 'app/lib/calendar.ts' },
1236
+ ]);
1237
+ });
1238
+ it('should deduplicate by name (first occurrence wins)', () => {
1239
+ const glossary = [
1240
+ { name: 'pad', filePath: 'app/lib/calendar.ts' },
1241
+ { name: 'pad', filePath: 'app/lib/other.ts' },
1242
+ ];
1243
+ const result = buildEntityInfosFromGlossary(glossary);
1244
+ expect(result).toHaveLength(1);
1245
+ expect(result[0].filePath).toBe('app/lib/calendar.ts');
1246
+ });
1247
+ it('should skip entries already present in existingNames set', () => {
1248
+ const glossary = [
1249
+ { name: 'Home', filePath: 'app/lib/calendar.ts' },
1250
+ { name: 'pad', filePath: 'app/lib/calendar.ts' },
1251
+ ];
1252
+ const existing = new Set(['Home']);
1253
+ const result = buildEntityInfosFromGlossary(glossary, existing);
1254
+ expect(result).toHaveLength(1);
1255
+ expect(result[0].name).toBe('pad');
1256
+ });
1257
+ it('should return empty array for empty input', () => {
1258
+ expect(buildEntityInfosFromGlossary([])).toEqual([]);
1259
+ });
1260
+ });
1261
+ // ── filterGlossaryByChangeStatus ──────────────────────────────────────
1262
+ describe('filterGlossaryByChangeStatus', () => {
1263
+ const functions = [
1264
+ {
1265
+ name: 'pad',
1266
+ filePath: 'lib/calendar.ts',
1267
+ description: '',
1268
+ testFile: 'test.ts',
1269
+ },
1270
+ {
1271
+ name: 'formatDate',
1272
+ filePath: 'lib/calendar.ts',
1273
+ description: '',
1274
+ testFile: 'test.ts',
1275
+ },
1276
+ {
1277
+ name: 'getDays',
1278
+ filePath: 'lib/calendar.ts',
1279
+ description: '',
1280
+ testFile: 'test.ts',
1281
+ },
1282
+ ];
1283
+ it('should return all functions when entityChangeStatus is undefined', () => {
1284
+ expect(filterGlossaryByChangeStatus(functions, undefined)).toEqual(functions);
1285
+ });
1286
+ it('should return all functions when entityChangeStatus is empty', () => {
1287
+ expect(filterGlossaryByChangeStatus(functions, {})).toEqual(functions);
1288
+ });
1289
+ it('should filter to only functions with a change status', () => {
1290
+ const status = {
1291
+ pad: { status: 'new' },
1292
+ getDays: { status: 'edited' },
1293
+ };
1294
+ const result = filterGlossaryByChangeStatus(functions, status);
1295
+ expect(result).toHaveLength(2);
1296
+ expect(result.map((f) => f.name)).toEqual(['pad', 'getDays']);
1297
+ });
1298
+ it('should return empty array when no functions match', () => {
1299
+ const status = {
1300
+ SomeOtherThing: { status: 'new' },
1301
+ };
1302
+ const result = filterGlossaryByChangeStatus(functions, status);
1303
+ expect(result).toEqual([]);
1304
+ });
1305
+ });
1306
+ // ── scenarioEntityName ──────────────────────────────────────────────
1307
+ describe('scenarioEntityName', () => {
1308
+ it('should return componentName when set', () => {
1309
+ expect(scenarioEntityName({
1310
+ componentName: 'ReviewCard',
1311
+ url: '/isolated-components/ReviewCard?s=Default',
1312
+ })).toBe('ReviewCard');
1313
+ });
1314
+ it('should return page name from URL when componentName is null', () => {
1315
+ expect(scenarioEntityName({ componentName: null, url: '/drinks/1' })).toBe('Drinks');
1316
+ });
1317
+ it('should return Home for root URL when componentName is null', () => {
1318
+ expect(scenarioEntityName({ componentName: null, url: '/' })).toBe('Home');
1319
+ });
1320
+ it('should return Home when both componentName and url are null', () => {
1321
+ expect(scenarioEntityName({ componentName: null, url: null })).toBe('Home');
1322
+ });
1323
+ it('should return page name when componentName is undefined', () => {
1324
+ expect(scenarioEntityName({ url: '/settings' })).toBe('Settings');
1325
+ });
1326
+ it('should return Home when no fields provided', () => {
1327
+ expect(scenarioEntityName({})).toBe('Home');
1328
+ });
1329
+ });
1330
+ // ── filterScenarioScreenshotsByChangeStatus ───────────────────────────
1331
+ describe('filterScenarioScreenshotsByChangeStatus', () => {
1332
+ it('should return all screenshots when entityChangeStatus is undefined', () => {
1333
+ const screenshots = [
1334
+ { name: 'Busy Month', path: 'busy.png', componentName: null, url: '/' },
1335
+ {
1336
+ name: 'EventPill - Default',
1337
+ path: 'pill.png',
1338
+ componentName: 'EventPill',
1339
+ url: '/isolated-components/EventPill',
1340
+ },
1341
+ ];
1342
+ expect(filterScenarioScreenshotsByChangeStatus(screenshots, undefined)).toEqual(screenshots);
1343
+ });
1344
+ it('should return all screenshots when entityChangeStatus is empty', () => {
1345
+ const screenshots = [
1346
+ { name: 'Busy Month', path: 'busy.png', componentName: null, url: '/' },
1347
+ ];
1348
+ expect(filterScenarioScreenshotsByChangeStatus(screenshots, {})).toEqual(screenshots);
1349
+ });
1350
+ it('should include component scenarios whose entity has a change status', () => {
1351
+ const screenshots = [
1352
+ {
1353
+ name: 'EventPill - Default',
1354
+ path: 'pill.png',
1355
+ componentName: 'EventPill',
1356
+ url: '/isolated-components/EventPill',
1357
+ },
1358
+ {
1359
+ name: 'ErrorState - Server Error',
1360
+ path: 'error.png',
1361
+ componentName: 'ErrorState',
1362
+ url: '/isolated-components/ErrorState',
1363
+ },
1364
+ {
1365
+ name: 'CalendarGrid - Default',
1366
+ path: 'grid.png',
1367
+ componentName: 'CalendarGrid',
1368
+ url: '/isolated-components/CalendarGrid',
1369
+ },
1370
+ ];
1371
+ const status = {
1372
+ ErrorState: { status: 'new' },
1373
+ EventPill: { status: 'edited' },
1374
+ };
1375
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1376
+ expect(result.map((s) => s.name)).toEqual([
1377
+ 'EventPill - Default',
1378
+ 'ErrorState - Server Error',
1379
+ ]);
1380
+ });
1381
+ it('should include page scenarios whose page entity has a change status', () => {
1382
+ const screenshots = [
1383
+ { name: 'Busy Month', path: 'busy.png', componentName: null, url: '/' },
1384
+ {
1385
+ name: 'Empty Calendar',
1386
+ path: 'empty.png',
1387
+ componentName: null,
1388
+ url: '/',
1389
+ },
1390
+ ];
1391
+ const status = {
1392
+ Home: { status: 'edited' },
1393
+ };
1394
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1395
+ expect(result).toEqual(screenshots);
1396
+ });
1397
+ it('should exclude page scenarios when their page entity has no status', () => {
1398
+ const screenshots = [
1399
+ { name: 'Busy Month', path: 'busy.png', componentName: null, url: '/' },
1400
+ {
1401
+ name: 'EventPill - Default',
1402
+ path: 'pill.png',
1403
+ componentName: 'EventPill',
1404
+ url: '/isolated-components/EventPill',
1405
+ },
1406
+ ];
1407
+ const status = {
1408
+ SomeComponent: { status: 'new' },
1409
+ };
1410
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1411
+ // Home page has no status, so "Busy Month" is excluded
1412
+ // SomeComponent doesn't match EventPill either
1413
+ expect(result).toEqual([]);
1414
+ });
1415
+ it('should include both page and component scenarios when both entities have status', () => {
1416
+ const screenshots = [
1417
+ { name: 'Busy Month', path: 'busy.png', componentName: null, url: '/' },
1418
+ {
1419
+ name: 'Empty Calendar',
1420
+ path: 'empty.png',
1421
+ componentName: null,
1422
+ url: '/',
1423
+ },
1424
+ {
1425
+ name: 'EventPill - Default',
1426
+ path: 'pill.png',
1427
+ componentName: 'EventPill',
1428
+ url: '/isolated-components/EventPill',
1429
+ },
1430
+ ];
1431
+ const status = {
1432
+ Home: { status: 'edited' },
1433
+ EventPill: { status: 'new' },
1434
+ };
1435
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1436
+ expect(result.map((s) => s.name)).toEqual([
1437
+ 'Busy Month',
1438
+ 'Empty Calendar',
1439
+ 'EventPill - Default',
1440
+ ]);
1441
+ });
1442
+ it('should handle page scenarios with dashes in names using URL, not name parsing', () => {
1443
+ // The original bug: "Drink Detail - Earl Grey" was parsed as component "Drink Detail"
1444
+ const screenshots = [
1445
+ {
1446
+ name: 'Drink Detail - Earl Grey',
1447
+ path: 'earl.png',
1448
+ componentName: null,
1449
+ url: '/drinks/1',
1450
+ },
1451
+ {
1452
+ name: 'Drink Detail - No Reviews',
1453
+ path: 'no-reviews.png',
1454
+ componentName: null,
1455
+ url: '/drinks/1',
1456
+ },
1457
+ {
1458
+ name: 'ReviewCard - Default',
1459
+ path: 'review.png',
1460
+ componentName: 'ReviewCard',
1461
+ url: '/isolated-components/ReviewCard',
1462
+ },
1463
+ {
1464
+ name: 'Full Catalog',
1465
+ path: 'catalog.png',
1466
+ componentName: null,
1467
+ url: '/',
1468
+ },
1469
+ ];
1470
+ const status = {
1471
+ Drinks: { status: 'new' },
1472
+ ReviewCard: { status: 'new' },
1473
+ Home: { status: 'edited' },
1474
+ };
1475
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1476
+ expect(result.map((s) => s.name)).toEqual([
1477
+ 'Drink Detail - Earl Grey',
1478
+ 'Drink Detail - No Reviews',
1479
+ 'ReviewCard - Default',
1480
+ 'Full Catalog',
1481
+ ]);
1482
+ });
1483
+ it('should exclude page scenarios for unchanged pages even with dashes in names', () => {
1484
+ const screenshots = [
1485
+ {
1486
+ name: 'Drink Detail - Earl Grey',
1487
+ path: 'earl.png',
1488
+ componentName: null,
1489
+ url: '/drinks/1',
1490
+ },
1491
+ {
1492
+ name: 'ReviewCard - Default',
1493
+ path: 'review.png',
1494
+ componentName: 'ReviewCard',
1495
+ url: '/isolated-components/ReviewCard',
1496
+ },
1497
+ ];
1498
+ const status = {
1499
+ ReviewCard: { status: 'edited' },
1500
+ };
1501
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1502
+ // Drinks page has no status, so "Drink Detail" page scenarios are excluded
1503
+ expect(result.map((s) => s.name)).toEqual(['ReviewCard - Default']);
1504
+ });
1505
+ it('should use componentName for matching, not the name prefix', () => {
1506
+ const screenshots = [
1507
+ {
1508
+ name: 'Card Loading State - Spinner',
1509
+ path: 'spinner.png',
1510
+ componentName: 'LoadingCard',
1511
+ url: '/isolated-components/LoadingCard',
1512
+ },
1513
+ ];
1514
+ const status = {
1515
+ LoadingCard: { status: 'new' },
1516
+ };
1517
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1518
+ expect(result.map((s) => s.name)).toEqual([
1519
+ 'Card Loading State - Spinner',
1520
+ ]);
1521
+ });
1522
+ it('should exclude component scenario when its entity has no status', () => {
1523
+ const screenshots = [
1524
+ {
1525
+ name: 'Card Loading State - Spinner',
1526
+ path: 'spinner.png',
1527
+ componentName: 'LoadingCard',
1528
+ url: '/isolated-components/LoadingCard',
1529
+ },
1530
+ ];
1531
+ const status = {
1532
+ SomeOtherComponent: { status: 'new' },
1533
+ };
1534
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1535
+ expect(result).toEqual([]);
1536
+ });
1537
+ it('should filter per-page independently (not all-or-nothing for pages)', () => {
1538
+ // Only Drinks page changed, not Home — scenarios for each page should be filtered independently
1539
+ const screenshots = [
1540
+ {
1541
+ name: 'Full Catalog',
1542
+ path: 'catalog.png',
1543
+ componentName: null,
1544
+ url: '/',
1545
+ },
1546
+ {
1547
+ name: 'Drink Detail - Earl Grey',
1548
+ path: 'earl.png',
1549
+ componentName: null,
1550
+ url: '/drinks/1',
1551
+ },
1552
+ ];
1553
+ const status = {
1554
+ Drinks: { status: 'new' },
1555
+ };
1556
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1557
+ // Home page has no status, so "Full Catalog" is excluded
1558
+ // Drinks page has status, so "Drink Detail" is included
1559
+ expect(result.map((s) => s.name)).toEqual(['Drink Detail - Earl Grey']);
1560
+ });
1561
+ it('should fall back to name-based parsing when componentName is not present (backward compat)', () => {
1562
+ const screenshots = [
1563
+ { name: 'EventPill - Default', path: 'pill.png' },
1564
+ { name: 'Full Calendar', path: 'cal.png' },
1565
+ ];
1566
+ const status = {
1567
+ EventPill: { status: 'new' },
1568
+ Home: { status: 'edited' },
1569
+ };
1570
+ const result = filterScenarioScreenshotsByChangeStatus(screenshots, status);
1571
+ expect(result.map((s) => s.name)).toEqual([
1572
+ 'EventPill - Default',
1573
+ 'Full Calendar',
1574
+ ]);
1575
+ });
1576
+ });
1577
+ // ── scanPageFilePaths ───────────────────────────────────────────────────
1578
+ describe('scanPageFilePaths', () => {
1579
+ let tempDir;
1580
+ beforeEach(() => {
1581
+ tempDir = fsModule.mkdtempSync(pathModule.join(os.tmpdir(), 'scanpages-test-'));
1582
+ });
1583
+ afterEach(() => {
1584
+ fsModule.rmSync(tempDir, { recursive: true, force: true });
1585
+ });
1586
+ it('should find page.tsx files in nested app/ structure', () => {
1587
+ const appDir = pathModule.join(tempDir, 'app');
1588
+ fsModule.mkdirSync(appDir, { recursive: true });
1589
+ fsModule.writeFileSync(pathModule.join(appDir, 'page.tsx'), '');
1590
+ fsModule.mkdirSync(pathModule.join(appDir, 'drinks'), {
1591
+ recursive: true,
1592
+ });
1593
+ fsModule.writeFileSync(pathModule.join(appDir, 'drinks', 'page.tsx'), '');
1594
+ fsModule.mkdirSync(pathModule.join(appDir, 'settings', 'profile'), {
1595
+ recursive: true,
1596
+ });
1597
+ fsModule.writeFileSync(pathModule.join(appDir, 'settings', 'profile', 'page.tsx'), '');
1598
+ const result = scanPageFilePaths(tempDir);
1599
+ expect(result).toEqual({
1600
+ Home: 'app/page.tsx',
1601
+ Drinks: 'app/drinks/page.tsx',
1602
+ Settings: 'app/settings/profile/page.tsx',
1603
+ });
1604
+ });
1605
+ it('should find page.js files', () => {
1606
+ const appDir = pathModule.join(tempDir, 'app');
1607
+ fsModule.mkdirSync(appDir, { recursive: true });
1608
+ fsModule.writeFileSync(pathModule.join(appDir, 'page.js'), '');
1609
+ const result = scanPageFilePaths(tempDir);
1610
+ expect(result).toEqual({ Home: 'app/page.js' });
1611
+ });
1612
+ it('should skip isolated-components directories', () => {
1613
+ const appDir = pathModule.join(tempDir, 'app');
1614
+ fsModule.mkdirSync(pathModule.join(appDir, 'isolated-components', 'test'), {
1615
+ recursive: true,
1616
+ });
1617
+ fsModule.writeFileSync(pathModule.join(appDir, 'isolated-components', 'test', 'page.tsx'), '');
1618
+ const result = scanPageFilePaths(tempDir);
1619
+ expect(result).toEqual({});
1620
+ });
1621
+ it('should return empty map when app/ does not exist', () => {
1622
+ const result = scanPageFilePaths(tempDir);
1623
+ expect(result).toEqual({});
1624
+ });
1625
+ // ── Expo Router support ──────────────────────────────────────────────
1626
+ it('should find index.tsx as Home page (Expo Router)', () => {
1627
+ const appDir = pathModule.join(tempDir, 'app');
1628
+ fsModule.mkdirSync(appDir, { recursive: true });
1629
+ fsModule.writeFileSync(pathModule.join(appDir, 'index.tsx'), '');
1630
+ const result = scanPageFilePaths(tempDir);
1631
+ expect(result).toEqual({ Home: 'app/index.tsx' });
1632
+ });
1633
+ it('should find named route files as pages (Expo Router)', () => {
1634
+ const appDir = pathModule.join(tempDir, 'app');
1635
+ fsModule.mkdirSync(appDir, { recursive: true });
1636
+ fsModule.writeFileSync(pathModule.join(appDir, 'index.tsx'), '');
1637
+ fsModule.writeFileSync(pathModule.join(appDir, 'add-tea.tsx'), '');
1638
+ fsModule.mkdirSync(pathModule.join(appDir, 'tea'), { recursive: true });
1639
+ fsModule.writeFileSync(pathModule.join(appDir, 'tea', '[id].tsx'), '');
1640
+ const result = scanPageFilePaths(tempDir);
1641
+ expect(result).toEqual({
1642
+ Home: 'app/index.tsx',
1643
+ 'Add-tea': 'app/add-tea.tsx',
1644
+ Tea: 'app/tea/[id].tsx',
1645
+ });
1646
+ });
1647
+ it('should skip _layout.tsx files (Expo Router)', () => {
1648
+ const appDir = pathModule.join(tempDir, 'app');
1649
+ fsModule.mkdirSync(appDir, { recursive: true });
1650
+ fsModule.writeFileSync(pathModule.join(appDir, '_layout.tsx'), '');
1651
+ fsModule.writeFileSync(pathModule.join(appDir, 'index.tsx'), '');
1652
+ const result = scanPageFilePaths(tempDir);
1653
+ expect(result).toEqual({ Home: 'app/index.tsx' });
1654
+ });
1655
+ it('should handle route groups like (tabs) transparently (Expo Router)', () => {
1656
+ const appDir = pathModule.join(tempDir, 'app');
1657
+ const tabsDir = pathModule.join(appDir, '(tabs)');
1658
+ fsModule.mkdirSync(tabsDir, { recursive: true });
1659
+ fsModule.writeFileSync(pathModule.join(tabsDir, '_layout.tsx'), '');
1660
+ fsModule.writeFileSync(pathModule.join(tabsDir, 'index.tsx'), '');
1661
+ fsModule.writeFileSync(pathModule.join(tabsDir, 'settings.tsx'), '');
1662
+ const result = scanPageFilePaths(tempDir);
1663
+ expect(result).toEqual({
1664
+ Home: 'app/(tabs)/index.tsx',
1665
+ Settings: 'app/(tabs)/settings.tsx',
1666
+ });
1667
+ });
1668
+ it('should prefer page.tsx over index.tsx when both exist (Next.js priority)', () => {
1669
+ const appDir = pathModule.join(tempDir, 'app');
1670
+ fsModule.mkdirSync(appDir, { recursive: true });
1671
+ fsModule.writeFileSync(pathModule.join(appDir, 'page.tsx'), '');
1672
+ fsModule.writeFileSync(pathModule.join(appDir, 'index.tsx'), '');
1673
+ const result = scanPageFilePaths(tempDir);
1674
+ expect(result).toEqual({ Home: 'app/page.tsx' });
1675
+ });
1676
+ });
1677
+ // ── detectFirstFeature ─────────────────────────────────────────────────
1678
+ describe('detectFirstFeature', () => {
1679
+ it('should return true when git has 0 or 1 commits', () => {
1680
+ const tempDir = fsModule.mkdtempSync(pathModule.join(os.tmpdir(), 'firstfeat-test-'));
1681
+ try {
1682
+ // Init a git repo with no commits
1683
+ require('child_process').execSync('git init', {
1684
+ cwd: tempDir,
1685
+ stdio: 'ignore',
1686
+ });
1687
+ expect(detectFirstFeature(tempDir)).toBe(true);
1688
+ }
1689
+ finally {
1690
+ fsModule.rmSync(tempDir, { recursive: true, force: true });
1691
+ }
1692
+ });
1693
+ it('should return false when git has multiple commits', () => {
1694
+ const tempDir = fsModule.mkdtempSync(pathModule.join(os.tmpdir(), 'firstfeat-test-'));
1695
+ try {
1696
+ require('child_process').execSync('git init && git config user.email "test@test.com" && git config user.name "test" && git commit --allow-empty -m "first" && git commit --allow-empty -m "second"', { cwd: tempDir, stdio: 'ignore' });
1697
+ expect(detectFirstFeature(tempDir)).toBe(false);
1698
+ }
1699
+ finally {
1700
+ fsModule.rmSync(tempDir, { recursive: true, force: true });
1701
+ }
1702
+ });
1703
+ it('should return true when not a git repo', () => {
1704
+ const tempDir = fsModule.mkdtempSync(pathModule.join(os.tmpdir(), 'firstfeat-test-'));
1705
+ try {
1706
+ expect(detectFirstFeature(tempDir)).toBe(true);
1707
+ }
1708
+ finally {
1709
+ fsModule.rmSync(tempDir, { recursive: true, force: true });
1710
+ }
1711
+ });
1712
+ });
1713
+ // ── readFeatureStartedAt ───────────────────────────────────────────────
1714
+ describe('readFeatureStartedAt', () => {
1715
+ let tempDir;
1716
+ beforeEach(() => {
1717
+ tempDir = fsModule.mkdtempSync(pathModule.join(os.tmpdir(), 'featurets-test-'));
1718
+ fsModule.mkdirSync(pathModule.join(tempDir, '.codeyam'), {
1719
+ recursive: true,
1720
+ });
1721
+ });
1722
+ afterEach(() => {
1723
+ fsModule.rmSync(tempDir, { recursive: true, force: true });
1724
+ });
1725
+ it('should return featureStartedAt from editor-step.json', () => {
1726
+ fsModule.writeFileSync(pathModule.join(tempDir, '.codeyam', 'editor-step.json'), JSON.stringify({ featureStartedAt: '2026-03-01T10:00:00.000Z' }));
1727
+ expect(readFeatureStartedAt(tempDir)).toBe('2026-03-01T10:00:00.000Z');
1728
+ });
1729
+ it('should return null when file does not exist', () => {
1730
+ expect(readFeatureStartedAt(tempDir)).toBeNull();
1731
+ });
1732
+ it('should return null when featureStartedAt is not set', () => {
1733
+ fsModule.writeFileSync(pathModule.join(tempDir, '.codeyam', 'editor-step.json'), JSON.stringify({ step: 5 }));
1734
+ expect(readFeatureStartedAt(tempDir)).toBeNull();
1735
+ });
1736
+ it('should return null for invalid JSON', () => {
1737
+ fsModule.writeFileSync(pathModule.join(tempDir, '.codeyam', 'editor-step.json'), 'not json');
1738
+ expect(readFeatureStartedAt(tempDir)).toBeNull();
1739
+ });
1740
+ });
1741
+ });
1742
+ //# sourceMappingURL=entityChangeStatus.test.js.map