@codeyam/codeyam-cli 0.1.0-staging.323686 → 0.1.0-staging.415103

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 (683) hide show
  1. package/analyzer-template/.build-info.json +7 -7
  2. package/analyzer-template/log.txt +3 -3
  3. package/analyzer-template/package.json +18 -18
  4. package/analyzer-template/packages/ai/index.ts +7 -1
  5. package/analyzer-template/packages/ai/package.json +3 -3
  6. package/analyzer-template/packages/ai/src/lib/analyzeScope.ts +62 -18
  7. package/analyzer-template/packages/ai/src/lib/astScopes/astScopeAnalyzer.ts +101 -12
  8. package/analyzer-template/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.ts +10 -17
  9. package/analyzer-template/packages/ai/src/lib/astScopes/processExpression.ts +409 -50
  10. package/analyzer-template/packages/ai/src/lib/astScopes/sharedPatterns.ts +28 -0
  11. package/analyzer-template/packages/ai/src/lib/astScopes/types.ts +21 -6
  12. package/analyzer-template/packages/ai/src/lib/completionCall.ts +114 -113
  13. package/analyzer-template/packages/ai/src/lib/dataStructure/ScopeDataStructure.ts +1250 -253
  14. package/analyzer-template/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.ts +5 -1
  15. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.ts +16 -3
  16. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.ts +6 -4
  17. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.ts +31 -3
  18. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.ts +37 -15
  19. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.ts +70 -0
  20. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.ts +62 -0
  21. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.ts +140 -14
  22. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.ts +179 -0
  23. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.ts +40 -30
  24. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.ts +367 -96
  25. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.ts +35 -0
  26. package/analyzer-template/packages/ai/src/lib/dataStructureChunking.ts +40 -13
  27. package/analyzer-template/packages/ai/src/lib/generateEntityDataStructure.ts +58 -3
  28. package/analyzer-template/packages/ai/src/lib/generateEntityScenarioData.ts +393 -8
  29. package/analyzer-template/packages/ai/src/lib/generateEntityScenarios.ts +9 -5
  30. package/analyzer-template/packages/ai/src/lib/generateExecutionFlows.ts +145 -5
  31. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromConditionalEffects.ts +1 -1
  32. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromConditionals.ts +649 -142
  33. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromJsxUsages.ts +1 -1
  34. package/analyzer-template/packages/ai/src/lib/isolateScopes.ts +51 -3
  35. package/analyzer-template/packages/ai/src/lib/mergeJsonTypeDefinitions.ts +5 -0
  36. package/analyzer-template/packages/ai/src/lib/mergeStatements.ts +90 -96
  37. package/analyzer-template/packages/ai/src/lib/promptGenerators/collapseNullableObjects.ts +118 -0
  38. package/analyzer-template/packages/ai/src/lib/promptGenerators/gatherAttributesMap.ts +10 -7
  39. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.ts +24 -4
  40. package/analyzer-template/packages/ai/src/lib/resolvePathToControllable.ts +25 -13
  41. package/analyzer-template/packages/ai/src/lib/worker/SerializableDataStructure.ts +4 -3
  42. package/analyzer-template/packages/analyze/index.ts +2 -0
  43. package/analyzer-template/packages/analyze/src/lib/FileAnalyzer.ts +65 -59
  44. package/analyzer-template/packages/analyze/src/lib/ProjectAnalyzer.ts +119 -26
  45. package/analyzer-template/packages/analyze/src/lib/asts/nodes/getNodeType.ts +1 -0
  46. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.ts +19 -0
  47. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.ts +19 -0
  48. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllExports.ts +11 -0
  49. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.ts +8 -0
  50. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.ts +49 -1
  51. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.ts +2 -1
  52. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.ts +89 -9
  53. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities.ts +27 -4
  54. package/analyzer-template/packages/analyze/src/lib/files/analyze/dependencyResolver.ts +0 -6
  55. package/analyzer-template/packages/analyze/src/lib/files/analyze/findOrCreateEntity.ts +12 -0
  56. package/analyzer-template/packages/analyze/src/lib/files/analyze/gatherEntityMap.ts +4 -2
  57. package/analyzer-template/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.ts +0 -3
  58. package/analyzer-template/packages/analyze/src/lib/files/analyzeRemixRoute.ts +4 -5
  59. package/analyzer-template/packages/analyze/src/lib/files/getImportedExports.ts +14 -12
  60. package/analyzer-template/packages/analyze/src/lib/files/scenarios/TransformationTracer.ts +1352 -0
  61. package/analyzer-template/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.ts +61 -13
  62. package/analyzer-template/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.ts +87 -25
  63. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateDataStructure.ts +312 -19
  64. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.ts +117 -9
  65. package/analyzer-template/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.ts +591 -75
  66. package/analyzer-template/packages/analyze/src/lib/files/scenarios/propagateArrayItemSchemas.ts +474 -0
  67. package/analyzer-template/packages/analyze/src/lib/files/setImportedExports.ts +2 -1
  68. package/analyzer-template/packages/analyze/src/lib/index.ts +1 -0
  69. package/analyzer-template/packages/analyze/src/lib/utils/getFileByPath.ts +19 -0
  70. package/analyzer-template/packages/aws/package.json +10 -10
  71. package/analyzer-template/packages/database/index.ts +1 -0
  72. package/analyzer-template/packages/database/package.json +1 -1
  73. package/analyzer-template/packages/database/src/lib/analysisBranchToDb.ts +1 -1
  74. package/analyzer-template/packages/database/src/lib/analysisToDb.ts +1 -1
  75. package/analyzer-template/packages/database/src/lib/branchToDb.ts +1 -1
  76. package/analyzer-template/packages/database/src/lib/commitBranchToDb.ts +1 -1
  77. package/analyzer-template/packages/database/src/lib/commitToDb.ts +1 -1
  78. package/analyzer-template/packages/database/src/lib/fileToDb.ts +1 -1
  79. package/analyzer-template/packages/database/src/lib/kysely/db.ts +14 -0
  80. package/analyzer-template/packages/database/src/lib/kysely/tables/debugReportsTable.ts +1 -1
  81. package/analyzer-template/packages/database/src/lib/kysely/tables/editorScenariosTable.ts +62 -0
  82. package/analyzer-template/packages/database/src/lib/kysely/tables/labsRequestsTable.ts +52 -0
  83. package/analyzer-template/packages/database/src/lib/loadCommits.ts +31 -20
  84. package/analyzer-template/packages/database/src/lib/loadReadyToBeCapturedAnalyses.ts +0 -5
  85. package/analyzer-template/packages/database/src/lib/projectToDb.ts +1 -1
  86. package/analyzer-template/packages/database/src/lib/saveFiles.ts +1 -1
  87. package/analyzer-template/packages/database/src/lib/scenarioToDb.ts +1 -1
  88. package/analyzer-template/packages/database/src/lib/updateCommitMetadata.ts +151 -135
  89. package/analyzer-template/packages/database/src/lib/updateFreshAnalysisStatus.ts +58 -42
  90. package/analyzer-template/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.ts +81 -65
  91. package/analyzer-template/packages/database/src/lib/userScenarioToDb.ts +1 -1
  92. package/analyzer-template/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.ts +29 -1
  93. package/analyzer-template/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.ts +33 -5
  94. package/analyzer-template/packages/github/dist/database/index.d.ts +1 -0
  95. package/analyzer-template/packages/github/dist/database/index.d.ts.map +1 -1
  96. package/analyzer-template/packages/github/dist/database/index.js +1 -0
  97. package/analyzer-template/packages/github/dist/database/index.js.map +1 -1
  98. package/analyzer-template/packages/github/dist/database/src/lib/analysisBranchToDb.js +1 -1
  99. package/analyzer-template/packages/github/dist/database/src/lib/analysisBranchToDb.js.map +1 -1
  100. package/analyzer-template/packages/github/dist/database/src/lib/analysisToDb.js +1 -1
  101. package/analyzer-template/packages/github/dist/database/src/lib/analysisToDb.js.map +1 -1
  102. package/analyzer-template/packages/github/dist/database/src/lib/branchToDb.js +1 -1
  103. package/analyzer-template/packages/github/dist/database/src/lib/branchToDb.js.map +1 -1
  104. package/analyzer-template/packages/github/dist/database/src/lib/commitBranchToDb.js +1 -1
  105. package/analyzer-template/packages/github/dist/database/src/lib/commitBranchToDb.js.map +1 -1
  106. package/analyzer-template/packages/github/dist/database/src/lib/commitToDb.js +1 -1
  107. package/analyzer-template/packages/github/dist/database/src/lib/commitToDb.js.map +1 -1
  108. package/analyzer-template/packages/github/dist/database/src/lib/fileToDb.js +1 -1
  109. package/analyzer-template/packages/github/dist/database/src/lib/fileToDb.js.map +1 -1
  110. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts +4 -0
  111. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts.map +1 -1
  112. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js +8 -0
  113. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js.map +1 -1
  114. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/debugReportsTable.d.ts +1 -1
  115. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts +20 -0
  116. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts.map +1 -0
  117. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js +45 -0
  118. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -0
  119. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.d.ts +23 -0
  120. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.d.ts.map +1 -0
  121. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.js +35 -0
  122. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.js.map +1 -0
  123. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/scenariosTable.d.ts +5 -0
  124. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/scenariosTable.d.ts.map +1 -1
  125. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.d.ts.map +1 -1
  126. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.js +23 -13
  127. package/analyzer-template/packages/github/dist/database/src/lib/loadCommits.js.map +1 -1
  128. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.d.ts.map +1 -1
  129. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.js +1 -4
  130. package/analyzer-template/packages/github/dist/database/src/lib/loadReadyToBeCapturedAnalyses.js.map +1 -1
  131. package/analyzer-template/packages/github/dist/database/src/lib/projectToDb.js +1 -1
  132. package/analyzer-template/packages/github/dist/database/src/lib/projectToDb.js.map +1 -1
  133. package/analyzer-template/packages/github/dist/database/src/lib/saveFiles.js +1 -1
  134. package/analyzer-template/packages/github/dist/database/src/lib/saveFiles.js.map +1 -1
  135. package/analyzer-template/packages/github/dist/database/src/lib/scenarioToDb.js +1 -1
  136. package/analyzer-template/packages/github/dist/database/src/lib/scenarioToDb.js.map +1 -1
  137. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.d.ts.map +1 -1
  138. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.js +100 -89
  139. package/analyzer-template/packages/github/dist/database/src/lib/updateCommitMetadata.js.map +1 -1
  140. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.d.ts.map +1 -1
  141. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.js +41 -30
  142. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatus.js.map +1 -1
  143. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.d.ts.map +1 -1
  144. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.js +68 -57
  145. package/analyzer-template/packages/github/dist/database/src/lib/updateFreshAnalysisStatusWithScenarios.js.map +1 -1
  146. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.d.ts.map +1 -1
  147. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js +29 -1
  148. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js.map +1 -1
  149. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.d.ts.map +1 -1
  150. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js +33 -5
  151. package/analyzer-template/packages/github/dist/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js.map +1 -1
  152. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts +8 -0
  153. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  154. package/analyzer-template/packages/github/dist/types/src/types/Scenario.d.ts +10 -0
  155. package/analyzer-template/packages/github/dist/types/src/types/Scenario.d.ts.map +1 -1
  156. package/analyzer-template/packages/github/dist/types/src/types/ScenariosDataStructure.d.ts +5 -5
  157. package/analyzer-template/packages/github/dist/types/src/types/ScenariosDataStructure.d.ts.map +1 -1
  158. package/analyzer-template/packages/github/dist/types/src/types/ScopeAnalysis.d.ts +6 -1
  159. package/analyzer-template/packages/github/dist/types/src/types/ScopeAnalysis.d.ts.map +1 -1
  160. package/analyzer-template/packages/github/package.json +1 -1
  161. package/analyzer-template/packages/types/src/types/ProjectMetadata.ts +8 -0
  162. package/analyzer-template/packages/types/src/types/Scenario.ts +10 -0
  163. package/analyzer-template/packages/types/src/types/ScenariosDataStructure.ts +6 -5
  164. package/analyzer-template/packages/types/src/types/ScopeAnalysis.ts +6 -1
  165. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts +8 -0
  166. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  167. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.d.ts +10 -0
  168. package/analyzer-template/packages/utils/dist/types/src/types/Scenario.d.ts.map +1 -1
  169. package/analyzer-template/packages/utils/dist/types/src/types/ScenariosDataStructure.d.ts +5 -5
  170. package/analyzer-template/packages/utils/dist/types/src/types/ScenariosDataStructure.d.ts.map +1 -1
  171. package/analyzer-template/packages/utils/dist/types/src/types/ScopeAnalysis.d.ts +6 -1
  172. package/analyzer-template/packages/utils/dist/types/src/types/ScopeAnalysis.d.ts.map +1 -1
  173. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.d.ts.map +1 -1
  174. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.js +98 -3
  175. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.js.map +1 -1
  176. package/analyzer-template/packages/utils/src/lib/fs/rsyncCopy.ts +121 -3
  177. package/analyzer-template/playwright/captureFromUrl.ts +89 -82
  178. package/analyzer-template/project/constructMockCode.ts +260 -60
  179. package/analyzer-template/project/orchestrateCapture.ts +4 -1
  180. package/analyzer-template/project/reconcileMockDataKeys.ts +19 -14
  181. package/analyzer-template/project/start.ts +3 -0
  182. package/analyzer-template/project/startScenarioCapture.ts +9 -0
  183. package/analyzer-template/project/writeClientLogRoute.ts +125 -0
  184. package/analyzer-template/project/writeMockDataTsx.ts +198 -8
  185. package/analyzer-template/project/writeScenarioComponents.ts +170 -29
  186. package/analyzer-template/project/writeSimpleRoot.ts +21 -11
  187. package/analyzer-template/tsconfig.json +13 -1
  188. package/background/src/lib/local/createLocalAnalyzer.js +1 -1
  189. package/background/src/lib/local/createLocalAnalyzer.js.map +1 -1
  190. package/background/src/lib/virtualized/project/constructMockCode.js +220 -45
  191. package/background/src/lib/virtualized/project/constructMockCode.js.map +1 -1
  192. package/background/src/lib/virtualized/project/orchestrateCapture.js +4 -1
  193. package/background/src/lib/virtualized/project/orchestrateCapture.js.map +1 -1
  194. package/background/src/lib/virtualized/project/reconcileMockDataKeys.js +17 -11
  195. package/background/src/lib/virtualized/project/reconcileMockDataKeys.js.map +1 -1
  196. package/background/src/lib/virtualized/project/start.js +2 -0
  197. package/background/src/lib/virtualized/project/start.js.map +1 -1
  198. package/background/src/lib/virtualized/project/startScenarioCapture.js +5 -0
  199. package/background/src/lib/virtualized/project/startScenarioCapture.js.map +1 -1
  200. package/background/src/lib/virtualized/project/writeClientLogRoute.js +110 -0
  201. package/background/src/lib/virtualized/project/writeClientLogRoute.js.map +1 -0
  202. package/background/src/lib/virtualized/project/writeMockDataTsx.js +174 -4
  203. package/background/src/lib/virtualized/project/writeMockDataTsx.js.map +1 -1
  204. package/background/src/lib/virtualized/project/writeScenarioComponents.js +143 -27
  205. package/background/src/lib/virtualized/project/writeScenarioComponents.js.map +1 -1
  206. package/background/src/lib/virtualized/project/writeSimpleRoot.js +21 -11
  207. package/background/src/lib/virtualized/project/writeSimpleRoot.js.map +1 -1
  208. package/codeyam-cli/scripts/apply-setup.js +386 -9
  209. package/codeyam-cli/scripts/apply-setup.js.map +1 -1
  210. package/codeyam-cli/src/cli.js +33 -24
  211. package/codeyam-cli/src/cli.js.map +1 -1
  212. package/codeyam-cli/src/codeyam-cli.js +18 -2
  213. package/codeyam-cli/src/codeyam-cli.js.map +1 -1
  214. package/codeyam-cli/src/commands/analyze.js +21 -9
  215. package/codeyam-cli/src/commands/analyze.js.map +1 -1
  216. package/codeyam-cli/src/commands/baseline.js +2 -0
  217. package/codeyam-cli/src/commands/baseline.js.map +1 -1
  218. package/codeyam-cli/src/commands/debug.js +9 -5
  219. package/codeyam-cli/src/commands/debug.js.map +1 -1
  220. package/codeyam-cli/src/commands/default.js +87 -21
  221. package/codeyam-cli/src/commands/default.js.map +1 -1
  222. package/codeyam-cli/src/commands/editor.js +696 -0
  223. package/codeyam-cli/src/commands/editor.js.map +1 -0
  224. package/codeyam-cli/src/commands/init.js +75 -259
  225. package/codeyam-cli/src/commands/init.js.map +1 -1
  226. package/codeyam-cli/src/commands/memory.js +97 -92
  227. package/codeyam-cli/src/commands/memory.js.map +1 -1
  228. package/codeyam-cli/src/commands/recapture.js +2 -0
  229. package/codeyam-cli/src/commands/recapture.js.map +1 -1
  230. package/codeyam-cli/src/commands/setup-sandbox.js +2 -0
  231. package/codeyam-cli/src/commands/setup-sandbox.js.map +1 -1
  232. package/codeyam-cli/src/commands/setup-simulations.js +284 -0
  233. package/codeyam-cli/src/commands/setup-simulations.js.map +1 -0
  234. package/codeyam-cli/src/commands/test-startup.js +2 -0
  235. package/codeyam-cli/src/commands/test-startup.js.map +1 -1
  236. package/codeyam-cli/src/commands/verify.js +14 -2
  237. package/codeyam-cli/src/commands/verify.js.map +1 -1
  238. package/codeyam-cli/src/utils/__tests__/npmVersionCheck.test.js +185 -0
  239. package/codeyam-cli/src/utils/__tests__/npmVersionCheck.test.js.map +1 -0
  240. package/codeyam-cli/src/utils/__tests__/pathIgnoring.test.js +9 -0
  241. package/codeyam-cli/src/utils/__tests__/pathIgnoring.test.js.map +1 -1
  242. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js +154 -86
  243. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js.map +1 -1
  244. package/codeyam-cli/src/utils/analyzer.js +7 -0
  245. package/codeyam-cli/src/utils/analyzer.js.map +1 -1
  246. package/codeyam-cli/src/utils/backgroundServer.js +109 -22
  247. package/codeyam-cli/src/utils/backgroundServer.js.map +1 -1
  248. package/codeyam-cli/src/utils/devModeEvents.js +40 -0
  249. package/codeyam-cli/src/utils/devModeEvents.js.map +1 -0
  250. package/codeyam-cli/src/utils/fileMetadata.js +5 -0
  251. package/codeyam-cli/src/utils/fileMetadata.js.map +1 -1
  252. package/codeyam-cli/src/utils/fileWatcher.js +25 -9
  253. package/codeyam-cli/src/utils/fileWatcher.js.map +1 -1
  254. package/codeyam-cli/src/utils/generateReport.js +2 -2
  255. package/codeyam-cli/src/utils/git.js +52 -0
  256. package/codeyam-cli/src/utils/git.js.map +1 -1
  257. package/codeyam-cli/src/utils/install-skills.js +101 -45
  258. package/codeyam-cli/src/utils/install-skills.js.map +1 -1
  259. package/codeyam-cli/src/utils/interactiveSyncWatcher.js +126 -0
  260. package/codeyam-cli/src/utils/interactiveSyncWatcher.js.map +1 -0
  261. package/codeyam-cli/src/utils/labsAutoCheck.js +19 -0
  262. package/codeyam-cli/src/utils/labsAutoCheck.js.map +1 -0
  263. package/codeyam-cli/src/utils/npmVersionCheck.js +76 -0
  264. package/codeyam-cli/src/utils/npmVersionCheck.js.map +1 -0
  265. package/codeyam-cli/src/utils/pathIgnoring.js +19 -7
  266. package/codeyam-cli/src/utils/pathIgnoring.js.map +1 -1
  267. package/codeyam-cli/src/utils/progress.js +7 -0
  268. package/codeyam-cli/src/utils/progress.js.map +1 -1
  269. package/codeyam-cli/src/utils/queue/__tests__/heartbeat.test.js +11 -11
  270. package/codeyam-cli/src/utils/queue/__tests__/heartbeat.test.js.map +1 -1
  271. package/codeyam-cli/src/utils/queue/__tests__/manager.test.js +22 -0
  272. package/codeyam-cli/src/utils/queue/__tests__/manager.test.js.map +1 -1
  273. package/codeyam-cli/src/utils/queue/heartbeat.js +13 -5
  274. package/codeyam-cli/src/utils/queue/heartbeat.js.map +1 -1
  275. package/codeyam-cli/src/utils/queue/job.js +74 -1
  276. package/codeyam-cli/src/utils/queue/job.js.map +1 -1
  277. package/codeyam-cli/src/utils/queue/manager.js +7 -6
  278. package/codeyam-cli/src/utils/queue/manager.js.map +1 -1
  279. package/codeyam-cli/src/utils/requireSimulations.js +10 -0
  280. package/codeyam-cli/src/utils/requireSimulations.js.map +1 -0
  281. package/codeyam-cli/src/utils/ruleReflection/__tests__/confusionDetector.test.js +82 -0
  282. package/codeyam-cli/src/utils/ruleReflection/__tests__/confusionDetector.test.js.map +1 -0
  283. package/codeyam-cli/src/utils/ruleReflection/__tests__/contextBuilder.test.js +229 -0
  284. package/codeyam-cli/src/utils/ruleReflection/__tests__/contextBuilder.test.js.map +1 -0
  285. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/assertRules.js +67 -0
  286. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/assertRules.js.map +1 -0
  287. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/captureFixture.js +105 -0
  288. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/captureFixture.js.map +1 -0
  289. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/loadCapturedFixture.js +34 -0
  290. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/loadCapturedFixture.js.map +1 -0
  291. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/runClaude.js +162 -0
  292. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/runClaude.js.map +1 -0
  293. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/setupTempProject.js +74 -0
  294. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/setupTempProject.js.map +1 -0
  295. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/ruleReflectionE2E.test.js +376 -0
  296. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/ruleReflectionE2E.test.js.map +1 -0
  297. package/codeyam-cli/src/utils/ruleReflection/__tests__/promptBuilder.test.js +113 -0
  298. package/codeyam-cli/src/utils/ruleReflection/__tests__/promptBuilder.test.js.map +1 -0
  299. package/codeyam-cli/src/utils/ruleReflection/__tests__/transcriptParser.test.js +127 -0
  300. package/codeyam-cli/src/utils/ruleReflection/__tests__/transcriptParser.test.js.map +1 -0
  301. package/codeyam-cli/src/utils/ruleReflection/confusionDetector.js +50 -0
  302. package/codeyam-cli/src/utils/ruleReflection/confusionDetector.js.map +1 -0
  303. package/codeyam-cli/src/utils/ruleReflection/contextBuilder.js +116 -0
  304. package/codeyam-cli/src/utils/ruleReflection/contextBuilder.js.map +1 -0
  305. package/codeyam-cli/src/utils/ruleReflection/index.js +5 -0
  306. package/codeyam-cli/src/utils/ruleReflection/index.js.map +1 -0
  307. package/codeyam-cli/src/utils/ruleReflection/promptBuilder.js +44 -0
  308. package/codeyam-cli/src/utils/ruleReflection/promptBuilder.js.map +1 -0
  309. package/codeyam-cli/src/utils/ruleReflection/transcriptParser.js +85 -0
  310. package/codeyam-cli/src/utils/ruleReflection/transcriptParser.js.map +1 -0
  311. package/codeyam-cli/src/utils/ruleReflection/types.js +5 -0
  312. package/codeyam-cli/src/utils/ruleReflection/types.js.map +1 -0
  313. package/codeyam-cli/src/utils/rules/__tests__/parser.test.js +83 -0
  314. package/codeyam-cli/src/utils/rules/__tests__/parser.test.js.map +1 -0
  315. package/codeyam-cli/src/utils/rules/__tests__/pathMatcher.test.js +118 -0
  316. package/codeyam-cli/src/utils/rules/__tests__/pathMatcher.test.js.map +1 -0
  317. package/codeyam-cli/src/utils/rules/__tests__/rulePlacement.test.js +72 -0
  318. package/codeyam-cli/src/utils/rules/__tests__/rulePlacement.test.js.map +1 -0
  319. package/codeyam-cli/src/utils/rules/__tests__/ruleState.test.js +293 -0
  320. package/codeyam-cli/src/utils/rules/__tests__/ruleState.test.js.map +1 -0
  321. package/codeyam-cli/src/utils/rules/__tests__/sourceFiles.test.js +76 -0
  322. package/codeyam-cli/src/utils/rules/__tests__/sourceFiles.test.js.map +1 -0
  323. package/codeyam-cli/src/utils/rules/index.js +2 -0
  324. package/codeyam-cli/src/utils/rules/index.js.map +1 -1
  325. package/codeyam-cli/src/utils/rules/parser.js +16 -29
  326. package/codeyam-cli/src/utils/rules/parser.js.map +1 -1
  327. package/codeyam-cli/src/utils/rules/pathMatcher.js +34 -3
  328. package/codeyam-cli/src/utils/rules/pathMatcher.js.map +1 -1
  329. package/codeyam-cli/src/utils/rules/rulePlacement.js +65 -0
  330. package/codeyam-cli/src/utils/rules/rulePlacement.js.map +1 -0
  331. package/codeyam-cli/src/utils/rules/ruleState.js +150 -0
  332. package/codeyam-cli/src/utils/rules/ruleState.js.map +1 -0
  333. package/codeyam-cli/src/utils/rules/sourceFiles.js +43 -0
  334. package/codeyam-cli/src/utils/rules/sourceFiles.js.map +1 -0
  335. package/codeyam-cli/src/utils/rules/staleness.js +16 -11
  336. package/codeyam-cli/src/utils/rules/staleness.js.map +1 -1
  337. package/codeyam-cli/src/utils/serverState.js +64 -12
  338. package/codeyam-cli/src/utils/serverState.js.map +1 -1
  339. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js +61 -43
  340. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js.map +1 -1
  341. package/codeyam-cli/src/utils/simulationGateMiddleware.js +159 -0
  342. package/codeyam-cli/src/utils/simulationGateMiddleware.js.map +1 -0
  343. package/codeyam-cli/src/utils/syncMocksMiddleware.js +5 -24
  344. package/codeyam-cli/src/utils/syncMocksMiddleware.js.map +1 -1
  345. package/codeyam-cli/src/utils/testRunner.js +158 -0
  346. package/codeyam-cli/src/utils/testRunner.js.map +1 -0
  347. package/codeyam-cli/src/utils/transcriptPruning.js +67 -0
  348. package/codeyam-cli/src/utils/transcriptPruning.js.map +1 -0
  349. package/codeyam-cli/src/utils/versionInfo.js +46 -0
  350. package/codeyam-cli/src/utils/versionInfo.js.map +1 -1
  351. package/codeyam-cli/src/utils/webappDetection.js +14 -2
  352. package/codeyam-cli/src/utils/webappDetection.js.map +1 -1
  353. package/codeyam-cli/src/webserver/__tests__/dependency-smoke.test.js +66 -0
  354. package/codeyam-cli/src/webserver/__tests__/dependency-smoke.test.js.map +1 -0
  355. package/codeyam-cli/src/webserver/app/lib/database.js +15 -3
  356. package/codeyam-cli/src/webserver/app/lib/database.js.map +1 -1
  357. package/codeyam-cli/src/webserver/app/lib/dbNotifier.js.map +1 -1
  358. package/codeyam-cli/src/webserver/backgroundServer.js +166 -16
  359. package/codeyam-cli/src/webserver/backgroundServer.js.map +1 -1
  360. package/codeyam-cli/src/webserver/bootstrap.js +11 -0
  361. package/codeyam-cli/src/webserver/bootstrap.js.map +1 -1
  362. package/codeyam-cli/src/webserver/build/client/assets/CopyButton-DmJveP3T.js +1 -0
  363. package/codeyam-cli/src/webserver/build/client/assets/{EntityItem-DsN1wKrm.js → EntityItem-C76mRRiF.js} +1 -1
  364. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeBadge-DLqD3qNt.js → EntityTypeBadge-g3saevPb.js} +1 -1
  365. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeIcon-Ba2JVPzP.js → EntityTypeIcon-CobE682z.js} +1 -1
  366. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-Bu6c6aDe.js +1 -0
  367. package/codeyam-cli/src/webserver/build/client/assets/{InteractivePreview-aht4aafF.js → InteractivePreview-DYFW3lDD.js} +3 -3
  368. package/codeyam-cli/src/webserver/build/client/assets/{LibraryFunctionPreview-CVtiBnY5.js → LibraryFunctionPreview-DLeucoVX.js} +1 -1
  369. package/codeyam-cli/src/webserver/build/client/assets/{LoadingDots-B0GLXMsr.js → LoadingDots-BU_OAEMP.js} +1 -1
  370. package/codeyam-cli/src/webserver/build/client/assets/{LogViewer-xgeCVgSM.js → LogViewer-ceAyBX-H.js} +1 -1
  371. package/codeyam-cli/src/webserver/build/client/assets/{ReportIssueModal-OApQuNyq.js → ReportIssueModal-djPLI-WV.js} +3 -8
  372. package/codeyam-cli/src/webserver/build/client/assets/{SafeScreenshot-DuDvi0jm.js → SafeScreenshot-BED4B6sP.js} +1 -1
  373. package/codeyam-cli/src/webserver/build/client/assets/{ScenarioViewer-DzccYyI8.js → ScenarioViewer-B76aig_2.js} +2 -2
  374. package/codeyam-cli/src/webserver/build/client/assets/Spinner-Bb5uFQ5V.js +34 -0
  375. package/codeyam-cli/src/webserver/build/client/assets/Terminal-BaIiqg_w.js +41 -0
  376. package/codeyam-cli/src/webserver/build/client/assets/{TruncatedFilePath-DyFZkK0l.js → TruncatedFilePath-C8OKAR5x.js} +1 -1
  377. package/codeyam-cli/src/webserver/build/client/assets/{_index-BwqWJOgH.js → _index-C96V0n15.js} +1 -1
  378. package/codeyam-cli/src/webserver/build/client/assets/{activity.(_tab)-BwavGCpm.js → activity.(_tab)-BpKzcsJz.js} +6 -11
  379. package/codeyam-cli/src/webserver/build/client/assets/addon-fit-CUXOrorO.js +1 -0
  380. package/codeyam-cli/src/webserver/build/client/assets/addon-web-links-Duc5hnl7.js +1 -0
  381. package/codeyam-cli/src/webserver/build/client/assets/agent-transcripts-D9hemwl6.js +22 -0
  382. package/codeyam-cli/src/webserver/build/client/assets/api.agent-transcripts-l0sNRNKZ.js +1 -0
  383. package/codeyam-cli/src/webserver/build/client/assets/api.dev-mode-events-l0sNRNKZ.js +1 -0
  384. package/codeyam-cli/src/webserver/build/client/assets/api.editor-capture-scenario-l0sNRNKZ.js +1 -0
  385. package/codeyam-cli/src/webserver/build/client/assets/api.editor-client-errors-l0sNRNKZ.js +1 -0
  386. package/codeyam-cli/src/webserver/build/client/assets/api.editor-commit-l0sNRNKZ.js +1 -0
  387. package/codeyam-cli/src/webserver/build/client/assets/api.editor-dev-server-l0sNRNKZ.js +1 -0
  388. package/codeyam-cli/src/webserver/build/client/assets/api.editor-entity-status-l0sNRNKZ.js +1 -0
  389. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-entry-l0sNRNKZ.js +1 -0
  390. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-image._-l0sNRNKZ.js +1 -0
  391. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-l0sNRNKZ.js +1 -0
  392. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-screenshot-l0sNRNKZ.js +1 -0
  393. package/codeyam-cli/src/webserver/build/client/assets/api.editor-journal-update-l0sNRNKZ.js +1 -0
  394. package/codeyam-cli/src/webserver/build/client/assets/api.editor-refresh-l0sNRNKZ.js +1 -0
  395. package/codeyam-cli/src/webserver/build/client/assets/api.editor-register-scenario-l0sNRNKZ.js +1 -0
  396. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-data-l0sNRNKZ.js +1 -0
  397. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-image._-l0sNRNKZ.js +1 -0
  398. package/codeyam-cli/src/webserver/build/client/assets/api.editor-switch-scenario-l0sNRNKZ.js +1 -0
  399. package/codeyam-cli/src/webserver/build/client/assets/api.editor-test-results-l0sNRNKZ.js +1 -0
  400. package/codeyam-cli/src/webserver/build/client/assets/api.labs-unlock-l0sNRNKZ.js +1 -0
  401. package/codeyam-cli/src/webserver/build/client/assets/api.rule-path-l0sNRNKZ.js +1 -0
  402. package/codeyam-cli/src/webserver/build/client/assets/api.save-fixture-l0sNRNKZ.js +1 -0
  403. package/codeyam-cli/src/webserver/build/client/assets/book-open-D_nMCFmP.js +6 -0
  404. package/codeyam-cli/src/webserver/build/client/assets/{chevron-down-Cx24_aWc.js → chevron-down-BH2h1Ea2.js} +1 -1
  405. package/codeyam-cli/src/webserver/build/client/assets/{chunk-EPOLDU6W-CXRTFQ3F.js → chunk-JZWAC4HX-C4pqxYJB.js} +12 -12
  406. package/codeyam-cli/src/webserver/build/client/assets/{circle-check-BOARzkeR.js → circle-check-DyIKORY6.js} +1 -1
  407. package/codeyam-cli/src/webserver/build/client/assets/copy-NDbZjXao.js +11 -0
  408. package/codeyam-cli/src/webserver/build/client/assets/{createLucideIcon-BdhJEx6B.js → createLucideIcon-CMT1jU2q.js} +1 -1
  409. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-BiM6z3Do.js +1 -0
  410. package/codeyam-cli/src/webserver/build/client/assets/editor-BaC8lHDG.js +7 -0
  411. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha._-BJUiQqZF.js → entity._sha._-CrjR3zZW.js} +11 -11
  412. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.dev-DloHYjtt.js +6 -0
  413. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-C28BiQzt.js +6 -0
  414. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-p9hhkjJM.js +6 -0
  415. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha_.edit._scenarioId-CTBG2mmz.js → entity._sha_.edit._scenarioId-BMvVHNXU.js} +2 -2
  416. package/codeyam-cli/src/webserver/build/client/assets/{entry.client-CS2cb_eZ.js → entry.client-DTvKq3TY.js} +1 -1
  417. package/codeyam-cli/src/webserver/build/client/assets/{fileTableUtils-DMJ7zii9.js → fileTableUtils-cPo8LiG3.js} +1 -1
  418. package/codeyam-cli/src/webserver/build/client/assets/{files-CJ6lTdTA.js → files-DO4CZ16O.js} +1 -1
  419. package/codeyam-cli/src/webserver/build/client/assets/{git-CPTZZ-JZ.js → git-CFCTYk9I.js} +1 -1
  420. package/codeyam-cli/src/webserver/build/client/assets/globals-BH6uYxPM.css +1 -0
  421. package/codeyam-cli/src/webserver/build/client/assets/{index-B1h680n5.js → index-10oVnAAH.js} +1 -1
  422. package/codeyam-cli/src/webserver/build/client/assets/{index-lzqtyFU8.js → index-BcvgDzbZ.js} +1 -1
  423. package/codeyam-cli/src/webserver/build/client/assets/labs-Zk7ryIM1.js +1 -0
  424. package/codeyam-cli/src/webserver/build/client/assets/{loader-circle-B7B9V-bu.js → loader-circle-BAXYRVEO.js} +1 -1
  425. package/codeyam-cli/src/webserver/build/client/assets/manifest-fb3128c3.js +1 -0
  426. package/codeyam-cli/src/webserver/build/client/assets/memory-FweZHj5U.js +93 -0
  427. package/codeyam-cli/src/webserver/build/client/assets/pause-DTAcYxBt.js +11 -0
  428. package/codeyam-cli/src/webserver/build/client/assets/root-Dzn8nIkU.js +67 -0
  429. package/codeyam-cli/src/webserver/build/client/assets/{search-CxXUmBSd.js → search-fKo7v0Zo.js} +1 -1
  430. package/codeyam-cli/src/webserver/build/client/assets/settings-DfuTtcJP.js +1 -0
  431. package/codeyam-cli/src/webserver/build/client/assets/{simulations-DwFIBT09.js → simulations-B3aOzpCZ.js} +1 -1
  432. package/codeyam-cli/src/webserver/build/client/assets/terminal-BG4heKCG.js +11 -0
  433. package/codeyam-cli/src/webserver/build/client/assets/{triangle-alert-B6LgvRJg.js → triangle-alert-DtSmdtM4.js} +1 -1
  434. package/codeyam-cli/src/webserver/build/client/assets/{useCustomSizes-C1v1PQzo.js → useCustomSizes-ByhSyh0W.js} +1 -1
  435. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-C14nCb1q.js +2 -0
  436. package/codeyam-cli/src/webserver/build/client/assets/{useReportContext-DYxHZQuP.js → useReportContext-O-jkvSPx.js} +1 -1
  437. package/codeyam-cli/src/webserver/build/client/assets/{useToast-mBRpZPiu.js → useToast-9FIWuYfK.js} +1 -1
  438. package/codeyam-cli/src/webserver/build/client/assets/xterm-DMSzMhqy.js +9 -0
  439. package/codeyam-cli/src/webserver/build/server/assets/index-DeSuKbSF.js +1 -0
  440. package/codeyam-cli/src/webserver/build/server/assets/server-build-BbQ8YBP-.js +362 -0
  441. package/codeyam-cli/src/webserver/build/server/index.js +1 -1
  442. package/codeyam-cli/src/webserver/build-info.json +5 -5
  443. package/codeyam-cli/src/webserver/devServer.js +39 -5
  444. package/codeyam-cli/src/webserver/devServer.js.map +1 -1
  445. package/codeyam-cli/src/webserver/editorProxy.js +272 -0
  446. package/codeyam-cli/src/webserver/editorProxy.js.map +1 -0
  447. package/codeyam-cli/src/webserver/scripts/journalCapture.ts +121 -0
  448. package/codeyam-cli/src/webserver/server.js +177 -1
  449. package/codeyam-cli/src/webserver/server.js.map +1 -1
  450. package/codeyam-cli/src/webserver/terminalServer.js +606 -0
  451. package/codeyam-cli/src/webserver/terminalServer.js.map +1 -0
  452. package/codeyam-cli/templates/{codeyam:debug.md → codeyam-debug.md} +1 -1
  453. package/codeyam-cli/templates/codeyam-dev-mode.md +237 -0
  454. package/codeyam-cli/templates/codeyam-diagnose.md +481 -0
  455. package/codeyam-cli/templates/codeyam-editor-claude.md +68 -0
  456. package/codeyam-cli/templates/codeyam-editor.md +67 -0
  457. package/codeyam-cli/templates/codeyam-memory-hook.sh +19 -20
  458. package/codeyam-cli/templates/codeyam-memory.md +396 -0
  459. package/codeyam-cli/templates/codeyam-new-rule.md +11 -0
  460. package/codeyam-cli/templates/{codeyam:setup.md → codeyam-setup.md} +13 -1
  461. package/codeyam-cli/templates/{codeyam:sim.md → codeyam-sim.md} +1 -1
  462. package/codeyam-cli/templates/{codeyam:test.md → codeyam-test.md} +1 -1
  463. package/codeyam-cli/templates/{codeyam:verify.md → codeyam-verify.md} +1 -1
  464. package/codeyam-cli/templates/editor-step-hook.py +143 -0
  465. package/codeyam-cli/templates/hooks/staleness-check.sh +43 -0
  466. package/codeyam-cli/templates/isolation-route/next-app.tsx.template +80 -0
  467. package/codeyam-cli/templates/isolation-route/next-pages.tsx.template +79 -0
  468. package/codeyam-cli/templates/isolation-route/vite-react.tsx.template +78 -0
  469. package/codeyam-cli/templates/msw/browser-setup.ts.template +47 -0
  470. package/codeyam-cli/templates/msw/handler-router.ts.template +47 -0
  471. package/codeyam-cli/templates/msw/server-setup.ts.template +52 -0
  472. package/codeyam-cli/templates/nextjs-prisma-sqlite/PRISMA_SETUP.md +84 -0
  473. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/api/todos/route.ts +17 -0
  474. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/globals.css +26 -0
  475. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/layout.tsx +34 -0
  476. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/lib/prisma.ts +19 -0
  477. package/codeyam-cli/templates/nextjs-prisma-sqlite/app/page.tsx +10 -0
  478. package/codeyam-cli/templates/nextjs-prisma-sqlite/eslint.config.mjs +11 -0
  479. package/codeyam-cli/templates/nextjs-prisma-sqlite/next.config.ts +14 -0
  480. package/codeyam-cli/templates/nextjs-prisma-sqlite/package.json +35 -0
  481. package/codeyam-cli/templates/nextjs-prisma-sqlite/postcss.config.mjs +7 -0
  482. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma/schema.prisma +27 -0
  483. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma/seed.ts +37 -0
  484. package/codeyam-cli/templates/nextjs-prisma-sqlite/prisma.config.ts +12 -0
  485. package/codeyam-cli/templates/nextjs-prisma-sqlite/tsconfig.json +34 -0
  486. package/codeyam-cli/templates/prompts/conversation-guidance.txt +44 -0
  487. package/codeyam-cli/templates/prompts/conversation-prompt.txt +28 -0
  488. package/codeyam-cli/templates/prompts/interruption-prompt.txt +31 -0
  489. package/codeyam-cli/templates/prompts/stale-rules-prompt.txt +24 -0
  490. package/codeyam-cli/templates/rule-notification-hook.py +83 -0
  491. package/codeyam-cli/templates/rule-reflection-hook.py +647 -0
  492. package/codeyam-cli/templates/rules-instructions.md +78 -0
  493. package/package.json +16 -14
  494. package/packages/ai/index.js +3 -2
  495. package/packages/ai/index.js.map +1 -1
  496. package/packages/ai/src/lib/analyzeScope.js +50 -13
  497. package/packages/ai/src/lib/analyzeScope.js.map +1 -1
  498. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js +76 -12
  499. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js.map +1 -1
  500. package/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.js +10 -14
  501. package/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.js.map +1 -1
  502. package/packages/ai/src/lib/astScopes/processExpression.js +317 -44
  503. package/packages/ai/src/lib/astScopes/processExpression.js.map +1 -1
  504. package/packages/ai/src/lib/astScopes/sharedPatterns.js +25 -0
  505. package/packages/ai/src/lib/astScopes/sharedPatterns.js.map +1 -1
  506. package/packages/ai/src/lib/completionCall.js +10 -7
  507. package/packages/ai/src/lib/completionCall.js.map +1 -1
  508. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js +996 -173
  509. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js.map +1 -1
  510. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.js +5 -1
  511. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.js.map +1 -1
  512. package/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.js +13 -3
  513. package/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.js.map +1 -1
  514. package/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.js +6 -4
  515. package/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.js.map +1 -1
  516. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js +33 -3
  517. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js.map +1 -1
  518. package/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.js +36 -11
  519. package/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.js.map +1 -1
  520. package/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.js +63 -0
  521. package/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.js.map +1 -0
  522. package/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.js +54 -0
  523. package/packages/ai/src/lib/dataStructure/helpers/coercePrimitivesToArraysBySchema.js.map +1 -0
  524. package/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.js +122 -12
  525. package/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.js.map +1 -1
  526. package/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.js +173 -0
  527. package/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.js.map +1 -0
  528. package/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.js +37 -20
  529. package/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.js.map +1 -1
  530. package/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.js +309 -84
  531. package/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.js.map +1 -1
  532. package/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.js +34 -0
  533. package/packages/ai/src/lib/dataStructure/helpers/stripNullableMarkers.js.map +1 -0
  534. package/packages/ai/src/lib/dataStructureChunking.js +30 -11
  535. package/packages/ai/src/lib/dataStructureChunking.js.map +1 -1
  536. package/packages/ai/src/lib/generateEntityDataStructure.js +46 -2
  537. package/packages/ai/src/lib/generateEntityDataStructure.js.map +1 -1
  538. package/packages/ai/src/lib/generateEntityScenarioData.js +284 -6
  539. package/packages/ai/src/lib/generateEntityScenarioData.js.map +1 -1
  540. package/packages/ai/src/lib/generateEntityScenarios.js +7 -1
  541. package/packages/ai/src/lib/generateEntityScenarios.js.map +1 -1
  542. package/packages/ai/src/lib/generateExecutionFlows.js +107 -4
  543. package/packages/ai/src/lib/generateExecutionFlows.js.map +1 -1
  544. package/packages/ai/src/lib/generateExecutionFlowsFromConditionals.js +447 -80
  545. package/packages/ai/src/lib/generateExecutionFlowsFromConditionals.js.map +1 -1
  546. package/packages/ai/src/lib/isolateScopes.js +39 -3
  547. package/packages/ai/src/lib/isolateScopes.js.map +1 -1
  548. package/packages/ai/src/lib/mergeJsonTypeDefinitions.js +5 -0
  549. package/packages/ai/src/lib/mergeJsonTypeDefinitions.js.map +1 -1
  550. package/packages/ai/src/lib/mergeStatements.js +70 -51
  551. package/packages/ai/src/lib/mergeStatements.js.map +1 -1
  552. package/packages/ai/src/lib/promptGenerators/collapseNullableObjects.js +97 -0
  553. package/packages/ai/src/lib/promptGenerators/collapseNullableObjects.js.map +1 -0
  554. package/packages/ai/src/lib/promptGenerators/gatherAttributesMap.js +10 -4
  555. package/packages/ai/src/lib/promptGenerators/gatherAttributesMap.js.map +1 -1
  556. package/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.js +17 -2
  557. package/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.js.map +1 -1
  558. package/packages/ai/src/lib/resolvePathToControllable.js +24 -14
  559. package/packages/ai/src/lib/resolvePathToControllable.js.map +1 -1
  560. package/packages/ai/src/lib/worker/SerializableDataStructure.js.map +1 -1
  561. package/packages/analyze/index.js +1 -0
  562. package/packages/analyze/index.js.map +1 -1
  563. package/packages/analyze/src/lib/FileAnalyzer.js +60 -36
  564. package/packages/analyze/src/lib/FileAnalyzer.js.map +1 -1
  565. package/packages/analyze/src/lib/ProjectAnalyzer.js +99 -26
  566. package/packages/analyze/src/lib/ProjectAnalyzer.js.map +1 -1
  567. package/packages/analyze/src/lib/asts/nodes/getNodeType.js +1 -0
  568. package/packages/analyze/src/lib/asts/nodes/getNodeType.js.map +1 -1
  569. package/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.js +14 -0
  570. package/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.js.map +1 -1
  571. package/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.js +14 -0
  572. package/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.js.map +1 -1
  573. package/packages/analyze/src/lib/asts/sourceFiles/getAllExports.js +6 -0
  574. package/packages/analyze/src/lib/asts/sourceFiles/getAllExports.js.map +1 -1
  575. package/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.js +6 -0
  576. package/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.js.map +1 -1
  577. package/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.js +39 -1
  578. package/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.js.map +1 -1
  579. package/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.js +2 -1
  580. package/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.js.map +1 -1
  581. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js +65 -7
  582. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js.map +1 -1
  583. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js +24 -4
  584. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js.map +1 -1
  585. package/packages/analyze/src/lib/files/analyze/dependencyResolver.js +0 -5
  586. package/packages/analyze/src/lib/files/analyze/dependencyResolver.js.map +1 -1
  587. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js +9 -0
  588. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js.map +1 -1
  589. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js +2 -1
  590. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js.map +1 -1
  591. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js +0 -3
  592. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js.map +1 -1
  593. package/packages/analyze/src/lib/files/analyzeRemixRoute.js +3 -2
  594. package/packages/analyze/src/lib/files/analyzeRemixRoute.js.map +1 -1
  595. package/packages/analyze/src/lib/files/getImportedExports.js +11 -7
  596. package/packages/analyze/src/lib/files/getImportedExports.js.map +1 -1
  597. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js +907 -0
  598. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js.map +1 -0
  599. package/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.js +56 -10
  600. package/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.js.map +1 -1
  601. package/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.js +75 -21
  602. package/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.js.map +1 -1
  603. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js +215 -17
  604. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js.map +1 -1
  605. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js +56 -8
  606. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js.map +1 -1
  607. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js +494 -56
  608. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js.map +1 -1
  609. package/packages/analyze/src/lib/files/setImportedExports.js +2 -1
  610. package/packages/analyze/src/lib/files/setImportedExports.js.map +1 -1
  611. package/packages/analyze/src/lib/index.js +1 -0
  612. package/packages/analyze/src/lib/index.js.map +1 -1
  613. package/packages/analyze/src/lib/utils/getFileByPath.js +12 -0
  614. package/packages/analyze/src/lib/utils/getFileByPath.js.map +1 -0
  615. package/packages/database/index.js +1 -0
  616. package/packages/database/index.js.map +1 -1
  617. package/packages/database/src/lib/analysisBranchToDb.js +1 -1
  618. package/packages/database/src/lib/analysisBranchToDb.js.map +1 -1
  619. package/packages/database/src/lib/analysisToDb.js +1 -1
  620. package/packages/database/src/lib/analysisToDb.js.map +1 -1
  621. package/packages/database/src/lib/branchToDb.js +1 -1
  622. package/packages/database/src/lib/branchToDb.js.map +1 -1
  623. package/packages/database/src/lib/commitBranchToDb.js +1 -1
  624. package/packages/database/src/lib/commitBranchToDb.js.map +1 -1
  625. package/packages/database/src/lib/commitToDb.js +1 -1
  626. package/packages/database/src/lib/commitToDb.js.map +1 -1
  627. package/packages/database/src/lib/fileToDb.js +1 -1
  628. package/packages/database/src/lib/fileToDb.js.map +1 -1
  629. package/packages/database/src/lib/kysely/db.js +8 -0
  630. package/packages/database/src/lib/kysely/db.js.map +1 -1
  631. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js +45 -0
  632. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -0
  633. package/packages/database/src/lib/kysely/tables/labsRequestsTable.js +35 -0
  634. package/packages/database/src/lib/kysely/tables/labsRequestsTable.js.map +1 -0
  635. package/packages/database/src/lib/loadCommits.js +23 -13
  636. package/packages/database/src/lib/loadCommits.js.map +1 -1
  637. package/packages/database/src/lib/loadReadyToBeCapturedAnalyses.js +1 -4
  638. package/packages/database/src/lib/loadReadyToBeCapturedAnalyses.js.map +1 -1
  639. package/packages/database/src/lib/projectToDb.js +1 -1
  640. package/packages/database/src/lib/projectToDb.js.map +1 -1
  641. package/packages/database/src/lib/saveFiles.js +1 -1
  642. package/packages/database/src/lib/saveFiles.js.map +1 -1
  643. package/packages/database/src/lib/scenarioToDb.js +1 -1
  644. package/packages/database/src/lib/scenarioToDb.js.map +1 -1
  645. package/packages/database/src/lib/updateCommitMetadata.js +100 -89
  646. package/packages/database/src/lib/updateCommitMetadata.js.map +1 -1
  647. package/packages/database/src/lib/updateFreshAnalysisStatus.js +41 -30
  648. package/packages/database/src/lib/updateFreshAnalysisStatus.js.map +1 -1
  649. package/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.js +68 -57
  650. package/packages/database/src/lib/updateFreshAnalysisStatusWithScenarios.js.map +1 -1
  651. package/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js +29 -1
  652. package/packages/generate/src/lib/componentScenarioPage/generateScenarioClientWrapper.js.map +1 -1
  653. package/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js +33 -5
  654. package/packages/generate/src/lib/componentScenarioPage/getIFrameMessageListenerCode.js.map +1 -1
  655. package/packages/utils/src/lib/fs/rsyncCopy.js +98 -3
  656. package/packages/utils/src/lib/fs/rsyncCopy.js.map +1 -1
  657. package/scripts/finalize-analyzer.cjs +8 -76
  658. package/codeyam-cli/src/commands/detect-universal-mocks.js +0 -118
  659. package/codeyam-cli/src/commands/detect-universal-mocks.js.map +0 -1
  660. package/codeyam-cli/src/commands/list.js +0 -31
  661. package/codeyam-cli/src/commands/list.js.map +0 -1
  662. package/codeyam-cli/src/commands/webapp-info.js +0 -146
  663. package/codeyam-cli/src/commands/webapp-info.js.map +0 -1
  664. package/codeyam-cli/src/utils/universal-mocks.js +0 -152
  665. package/codeyam-cli/src/utils/universal-mocks.js.map +0 -1
  666. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-C8lyxW9k.js +0 -34
  667. package/codeyam-cli/src/webserver/build/client/assets/copy-Bb-80kDT.js +0 -6
  668. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-BBnGWYga.js +0 -1
  669. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-DavjRmOY.js +0 -6
  670. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-D1T4TGjf.js +0 -6
  671. package/codeyam-cli/src/webserver/build/client/assets/file-code-Dhef1kWN.js +0 -6
  672. package/codeyam-cli/src/webserver/build/client/assets/globals-D3yhhV8x.css +0 -1
  673. package/codeyam-cli/src/webserver/build/client/assets/manifest-7522edd4.js +0 -1
  674. package/codeyam-cli/src/webserver/build/client/assets/memory-yxFcrxBX.js +0 -92
  675. package/codeyam-cli/src/webserver/build/client/assets/root-eVAaavTS.js +0 -62
  676. package/codeyam-cli/src/webserver/build/client/assets/settings-CS5f3WzT.js +0 -1
  677. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-aSv48UbS.js +0 -2
  678. package/codeyam-cli/src/webserver/build/server/assets/index-DVzYx8PN.js +0 -1
  679. package/codeyam-cli/src/webserver/build/server/assets/server-build-4Cr0uToj.js +0 -257
  680. package/codeyam-cli/templates/codeyam-stop-hook.sh +0 -284
  681. package/codeyam-cli/templates/codeyam:diagnose.md +0 -803
  682. package/codeyam-cli/templates/codeyam:memory.md +0 -462
  683. package/codeyam-cli/templates/codeyam:new-rule.md +0 -13
@@ -0,0 +1,647 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Rule reflection hook for Claude Code.
4
+
5
+ Handles two hook events:
6
+ 1. Stop — Reviews completed turns for stale rules and conversation confusion signals
7
+ 2. UserPromptSubmit — Detects user interruptions (Escape/Ctrl+C) by checking if the
8
+ Stop hook's marker file is stale, then spawns a rule-reflection agent focused on
9
+ the interruption signal
10
+
11
+ Each review fires as a separate `claude -p` invocation so the LLM can focus on one task at a time.
12
+ Stays silent if there's nothing to review.
13
+
14
+ Prompt text lives in templates/prompts/*.txt (single source of truth shared with TypeScript tests).
15
+ """
16
+
17
+ import json
18
+ import os
19
+ import subprocess
20
+ import sys
21
+ from datetime import datetime
22
+ from pathlib import Path
23
+
24
+ MIN_USER_TURNS = 3 # Minimum user turns before checking for confusion
25
+
26
+ # Signals that indicate confusion, mistakes, or user corrections
27
+ CONFUSION_SIGNALS = [
28
+ 'no,', 'no ', "that's not", "thats not", "that is not",
29
+ 'wrong', 'incorrect', 'actually,', 'actually ',
30
+ "i meant", "i mean", "not what i", "stop", "wait",
31
+ "don't do", "dont do", "shouldn't", "should not",
32
+ "try again", "let me clarify", "to clarify",
33
+ "that broke", "that failed", "error", "bug",
34
+ ]
35
+
36
+ # Prompt templates directory — installed alongside this hook in .codeyam/bin/prompts/
37
+ PROMPTS_DIR = Path(__file__).parent / 'prompts'
38
+
39
+
40
+ def load_prompt_template(name, **kwargs):
41
+ """
42
+ Load a prompt template from the prompts/ directory and substitute placeholders.
43
+ Placeholders use {{KEY}} syntax (e.g., {{CONTEXT_FILE}}, {{PROJECT_DIR}}).
44
+ """
45
+ template_path = PROMPTS_DIR / name
46
+ text = template_path.read_text()
47
+ for key, value in kwargs.items():
48
+ text = text.replace(f'{{{{{key}}}}}', str(value))
49
+ return text
50
+
51
+
52
+ def has_confusion_signals(conversation_snippets):
53
+ """
54
+ Check if the conversation contains signals of confusion or user corrections.
55
+ Returns True if confusion signals are detected.
56
+ """
57
+ for snippet in conversation_snippets:
58
+ if snippet['role'] != 'user':
59
+ continue
60
+ content_lower = snippet['content'].lower()
61
+ for signal in CONFUSION_SIGNALS:
62
+ if signal in content_lower:
63
+ return True
64
+ return False
65
+
66
+
67
+ def get_file_diff(file_path, max_lines=50):
68
+ """
69
+ Get git diff for a file. Tries staged diff first, then unstaged.
70
+ Returns truncated diff string or empty string.
71
+ """
72
+ for diff_cmd in [
73
+ ['git', 'diff', 'HEAD', '--', file_path],
74
+ ['git', 'diff', '--', file_path],
75
+ ]:
76
+ try:
77
+ result = subprocess.run(
78
+ diff_cmd, capture_output=True, text=True, timeout=10
79
+ )
80
+ if result.stdout.strip():
81
+ lines = result.stdout.strip().split('\n')
82
+ # Skip the diff header (--- a/, +++ b/, @@ lines)
83
+ content_lines = [l for l in lines if not l.startswith('diff ') and
84
+ not l.startswith('index ') and not l.startswith('--- ') and
85
+ not l.startswith('+++ ')]
86
+ if len(content_lines) > max_lines:
87
+ content_lines = content_lines[:max_lines] + [f'... ({len(lines) - max_lines} more lines)']
88
+ return '\n'.join(content_lines)
89
+ except (subprocess.TimeoutExpired, FileNotFoundError):
90
+ continue
91
+ return ''
92
+
93
+
94
+ def read_rule_content(rule_name):
95
+ """
96
+ Read a rule file and return its content after YAML frontmatter.
97
+ Searches in .claude/rules/ directory.
98
+ """
99
+ # Search for the rule file
100
+ rules_dir = Path('.claude/rules')
101
+ if not rules_dir.exists():
102
+ return ''
103
+
104
+ # Find the file - could be at any depth
105
+ matches = list(rules_dir.rglob(rule_name))
106
+ if not matches:
107
+ return ''
108
+
109
+ try:
110
+ text = matches[0].read_text()
111
+ except IOError:
112
+ return ''
113
+
114
+ # Strip YAML frontmatter
115
+ if text.startswith('---'):
116
+ end = text.find('---', 3)
117
+ if end != -1:
118
+ text = text[end + 3:].strip()
119
+
120
+ return text
121
+
122
+
123
+ def get_project_slug():
124
+ """
125
+ Derive project slug from .codeyam/config.json (matching getProjectSlug() on the TS side).
126
+ Falls back to os.path.basename(project_dir) if config is absent or malformed.
127
+ """
128
+ project_dir = os.environ.get('CLAUDE_PROJECT_DIR', os.getcwd())
129
+ try:
130
+ config_path = os.path.join(project_dir, '.codeyam', 'config.json')
131
+ with open(config_path, 'r') as f:
132
+ config = json.load(f)
133
+ slug = config.get('projectSlug', '')
134
+ if slug:
135
+ return slug
136
+ except (IOError, json.JSONDecodeError, KeyError):
137
+ pass
138
+ return os.path.basename(project_dir)
139
+
140
+
141
+ def load_memory_settings():
142
+ """
143
+ Load memory settings from .codeyam/config.json.
144
+ Returns dict with safe defaults when absent or malformed.
145
+ """
146
+ defaults = {
147
+ 'conversationReflection': True,
148
+ 'ruleMaintenance': True,
149
+ 'promptModel': 'haiku',
150
+ }
151
+ try:
152
+ project_dir = os.environ.get('CLAUDE_PROJECT_DIR', os.getcwd())
153
+ config_path = os.path.join(project_dir, '.codeyam', 'config.json')
154
+ with open(config_path, 'r') as f:
155
+ config = json.load(f)
156
+ memory = config.get('memory', {})
157
+ if not isinstance(memory, dict):
158
+ return defaults
159
+ return {
160
+ 'conversationReflection': memory.get('conversationReflection', True),
161
+ 'ruleMaintenance': memory.get('ruleMaintenance', True),
162
+ 'promptModel': memory.get('promptModel', 'haiku'),
163
+ }
164
+ except (IOError, json.JSONDecodeError, KeyError):
165
+ return defaults
166
+
167
+
168
+ def get_stale_rules():
169
+ """
170
+ Run `codeyam memory status` and parse the output to find stale rules.
171
+ Returns a list of dicts with rule info, or empty list if none stale.
172
+ """
173
+ try:
174
+ # Use the local codeyam CLI via node to avoid PATH issues
175
+ project_dir = os.environ.get('CLAUDE_PROJECT_DIR', os.getcwd())
176
+ codeyam_js = os.path.join(project_dir, 'codeyam-cli', 'dist', 'codeyam-cli', 'src', 'codeyam-cli.js')
177
+ if os.path.exists(codeyam_js):
178
+ cmd = ['node', '--no-warnings', codeyam_js, 'memory', 'status']
179
+ else:
180
+ cmd = ['codeyam', 'memory', 'status']
181
+ result = subprocess.run(
182
+ cmd,
183
+ capture_output=True,
184
+ text=True,
185
+ timeout=30
186
+ )
187
+ output = result.stdout
188
+
189
+ # Check if there are stale rules
190
+ if 'Found' not in output or 'stale rule' not in output:
191
+ return []
192
+
193
+ # Parse the output to extract stale rule information
194
+ stale_rules = []
195
+ lines = output.split('\n')
196
+ i = 0
197
+ while i < len(lines):
198
+ line = lines[i].strip()
199
+ # Look for rule filenames (lines that end with .md and aren't indented much)
200
+ if line.endswith('.md') and not line.startswith('Rule') and not line.startswith('Newest'):
201
+ rule_info = {'name': line}
202
+ # Look for the next few lines for details
203
+ for j in range(i + 1, min(i + 4, len(lines))):
204
+ detail = lines[j].strip()
205
+ if detail.startswith('Last audited:'):
206
+ rule_info['last_audited'] = detail.replace('Last audited:', '').strip()
207
+ elif detail.startswith('Newest file:'):
208
+ rule_info['newest_file'] = detail.replace('Newest file:', '').strip()
209
+ elif detail.startswith('File modified:'):
210
+ rule_info['file_modified'] = detail.replace('File modified:', '').strip()
211
+ # Get diff for the changed file
212
+ if rule_info.get('newest_file'):
213
+ rule_info['diff'] = get_file_diff(rule_info['newest_file'])
214
+
215
+ # Get inline rule content
216
+ rule_info['rule_content'] = read_rule_content(rule_info['name'])
217
+
218
+ stale_rules.append(rule_info)
219
+ i += 1
220
+
221
+ return stale_rules
222
+ except (subprocess.TimeoutExpired, FileNotFoundError, Exception):
223
+ return []
224
+
225
+
226
+ def get_conversation_context(transcript_path, last_line):
227
+ """
228
+ Extract conversation snippets and modified files from transcript for rule review.
229
+ Returns (user_turn_count, conversation_snippets, modified_files, current_line_count)
230
+ """
231
+ try:
232
+ with open(transcript_path, 'r') as f:
233
+ all_lines = f.readlines()
234
+ except IOError:
235
+ return 0, [], set(), 0
236
+
237
+ current_line_count = len(all_lines)
238
+ new_lines = all_lines[last_line:]
239
+
240
+ if not new_lines:
241
+ return 0, [], set(), current_line_count
242
+
243
+ user_turn_count = 0
244
+ conversation_snippets = []
245
+ modified_files = set()
246
+
247
+ for line in new_lines:
248
+ try:
249
+ obj = json.loads(line)
250
+ msg_type = obj.get('type')
251
+
252
+ if msg_type not in ('user', 'assistant'):
253
+ continue
254
+
255
+ message = obj.get('message', {})
256
+ content = message.get('content', '')
257
+ is_external_user = msg_type == 'user' and obj.get('userType') == 'external'
258
+
259
+ # Handle string content
260
+ if isinstance(content, str):
261
+ if obj.get('isMeta') or len(content) < 20:
262
+ continue
263
+ if content.startswith('[{') or '<tool_result' in content:
264
+ continue
265
+
266
+ if is_external_user and not content.startswith('<'):
267
+ user_turn_count += 1
268
+
269
+ conversation_snippets.append({
270
+ 'role': message.get('role', msg_type),
271
+ 'content': content
272
+ })
273
+
274
+ elif isinstance(content, list):
275
+ for item in content:
276
+ if not isinstance(item, dict):
277
+ continue
278
+
279
+ # Extract text snippets
280
+ if item.get('type') == 'text':
281
+ text = item.get('text', '')
282
+ if len(text) > 20:
283
+ conversation_snippets.append({
284
+ 'role': message.get('role', msg_type),
285
+ 'content': text
286
+ })
287
+
288
+ # Track file modifications from tool_use entries
289
+ if item.get('type') == 'tool_use':
290
+ tool_name = item.get('name', '')
291
+ tool_input = item.get('input', {})
292
+ if tool_name in ('Edit', 'Write') and tool_input.get('file_path'):
293
+ modified_files.add((tool_input['file_path'], tool_name))
294
+
295
+ except (json.JSONDecodeError, KeyError):
296
+ continue
297
+
298
+ return user_turn_count, conversation_snippets, modified_files, current_line_count
299
+
300
+
301
+ def has_assistant_messages(transcript_path, start_line):
302
+ """
303
+ Check if there are any assistant messages in the transcript after start_line.
304
+ Used to confirm Claude actually started responding before treating a gap as an interruption.
305
+ """
306
+ try:
307
+ with open(transcript_path, 'r') as f:
308
+ all_lines = f.readlines()
309
+ except IOError:
310
+ return False
311
+
312
+ for line in all_lines[start_line:]:
313
+ try:
314
+ obj = json.loads(line)
315
+ if obj.get('type') == 'assistant':
316
+ return True
317
+ except (json.JSONDecodeError, KeyError):
318
+ continue
319
+
320
+ return False
321
+
322
+
323
+ def build_stale_rules_context(stale_rules):
324
+ """Build context content for stale rules review."""
325
+ parts = []
326
+ parts.append("# Reflection Step\n")
327
+ parts.append("Please review the stale rules below to determine if any need updating based on recent code changes.\n")
328
+ parts.append("Please show your thinking regarding the stale rules.\n")
329
+ parts.append("## Stale Rules to Review\n")
330
+ parts.append("The following rules have files that changed since the rule was last reviewed.")
331
+ parts.append("For each rule, review the rule content and the diff of changes, then:")
332
+ parts.append("1. Determine if the rule content needs updating based on the code changes")
333
+ parts.append("2. Update the rule if needed")
334
+ parts.append("3. Run `codeyam memory touch <rule-path1> <rule-path2> ...` with the SPECIFIC rule paths you reviewed to mark them as audited\n")
335
+
336
+ for rule in stale_rules:
337
+ parts.append(f"### {rule['name']}")
338
+ if rule.get('rule_content'):
339
+ parts.append(f" Rule content:")
340
+ for line in rule['rule_content'].split('\n'):
341
+ parts.append(f" {line}")
342
+ if rule.get('newest_file'):
343
+ parts.append(f" Changed file: {rule['newest_file']}")
344
+ if rule.get('file_modified'):
345
+ parts.append(f" File modified: {rule['file_modified']}")
346
+ if rule.get('last_audited'):
347
+ parts.append(f" Last audited: {rule['last_audited']}")
348
+ if rule.get('diff'):
349
+ parts.append(f" Changes:")
350
+ for line in rule['diff'].split('\n'):
351
+ parts.append(f" {line}")
352
+ parts.append("")
353
+
354
+ return '\n'.join(parts)
355
+
356
+
357
+ def build_conversation_context(conversation_snippets, modified_files):
358
+ """Build context content for conversation review."""
359
+ parts = []
360
+ parts.append("## Conversation Review\n")
361
+ parts.append(
362
+ "Review this session for rule-worthy learnings: architectural decisions, tribal knowledge, "
363
+ "confusion, mistakes, or corrections that future sessions would benefit from knowing.\n"
364
+ )
365
+
366
+ # Load guidance from shared template file
367
+ guidance = (PROMPTS_DIR / 'conversation-guidance.txt').read_text()
368
+ parts.append(guidance)
369
+
370
+ if modified_files:
371
+ parts.append("Files modified this session:")
372
+ for file_path, tool_name in sorted(modified_files):
373
+ parts.append(f"- {file_path} ({tool_name})")
374
+ parts.append("")
375
+
376
+ parts.append("### Session transcript\n")
377
+ summary_lines = []
378
+ for snippet in conversation_snippets:
379
+ role = snippet['role']
380
+ content = snippet['content'].replace('\n', ' ')
381
+ summary_lines.append(f"[{role}]: {content}")
382
+ parts.append('\n'.join(summary_lines))
383
+
384
+ return '\n'.join(parts)
385
+
386
+
387
+ def build_interruption_context(conversation_snippets, follow_up_prompt, modified_files):
388
+ """Build context content for an interrupted session review."""
389
+ parts = []
390
+ parts.append("## Interruption Review\n")
391
+ parts.append(
392
+ "The user interrupted Claude mid-response — a strong signal of confusion or misunderstanding. "
393
+ "Review the interrupted conversation and the user's follow-up to identify rule-worthy learnings.\n"
394
+ )
395
+
396
+ # Load guidance from shared template file
397
+ guidance = (PROMPTS_DIR / 'conversation-guidance.txt').read_text()
398
+ parts.append(guidance)
399
+
400
+ if modified_files:
401
+ parts.append("Files modified this session:")
402
+ for file_path, tool_name in sorted(modified_files):
403
+ parts.append(f"- {file_path} ({tool_name})")
404
+ parts.append("")
405
+
406
+ parts.append("### Session transcript\n")
407
+ summary_lines = []
408
+ for snippet in conversation_snippets:
409
+ role = snippet['role']
410
+ content = snippet['content'].replace('\n', ' ')
411
+ summary_lines.append(f"[{role}]: {content}")
412
+ parts.append('\n'.join(summary_lines))
413
+
414
+ parts.append("")
415
+ parts.append("### User's follow-up after interruption\n")
416
+ parts.append(follow_up_prompt)
417
+
418
+ return '\n'.join(parts)
419
+
420
+
421
+ def spawn_claude_agent(prompt, log_file, project_dir, model='haiku'):
422
+ """Spawn a detached claude -p agent as a background process."""
423
+ try:
424
+ log_fh = open(log_file, 'w')
425
+ env = os.environ.copy()
426
+ env['CODEYAM_RULE_AGENT'] = '1'
427
+ subprocess.Popen(
428
+ ['claude', '-p', prompt,
429
+ '--model', model,
430
+ '--no-session-persistence',
431
+ '--output-format', 'stream-json', '--verbose',
432
+ '--allowedTools', 'Read,Edit,Write,Bash,Glob,Grep'],
433
+ cwd=project_dir,
434
+ stdout=log_fh,
435
+ stderr=log_fh,
436
+ env=env,
437
+ start_new_session=True,
438
+ )
439
+ except FileNotFoundError:
440
+ pass # claude CLI not available, skip silently
441
+
442
+
443
+ def read_marker(marker_file):
444
+ """
445
+ Read marker file. Returns (last_line, written_by_stop).
446
+ Format: "<line_count>" or "<line_count> stop" — the suffix distinguishes
447
+ whether the Stop hook wrote this marker (normal completion) vs the
448
+ UserPromptSubmit handler (interruption detection).
449
+ """
450
+ if marker_file.exists():
451
+ try:
452
+ text = marker_file.read_text().strip()
453
+ parts = text.split()
454
+ line_count = int(parts[0])
455
+ was_stop = len(parts) > 1 and parts[1] == 'stop'
456
+ return line_count, was_stop
457
+ except (ValueError, IOError):
458
+ pass
459
+ return 0, False
460
+
461
+
462
+ def write_marker(marker_file, line_count, source='submit'):
463
+ """Write marker with source tag. source is 'stop' or 'submit'."""
464
+ marker_file.write_text(f'{line_count} {source}')
465
+
466
+
467
+ def handle_stop(hook_input):
468
+ """
469
+ Handle the Stop hook event.
470
+ Reviews completed turns for stale rules and conversation confusion signals.
471
+
472
+ Important: the fast work (transcript parsing, marker update, conversation
473
+ agent spawn) runs first. The slow `get_stale_rules()` call (~10s) runs last
474
+ so the hook timeout doesn't kill us before the critical work is done.
475
+ """
476
+ settings = load_memory_settings()
477
+
478
+ session_id = hook_input.get('session_id', '')
479
+ transcript_path = hook_input.get('transcript_path', '')
480
+
481
+ if not session_id or not transcript_path:
482
+ return
483
+
484
+ slug = get_project_slug()
485
+ marker_dir = Path('/tmp/claude-rule-markers') / slug
486
+ marker_dir.mkdir(parents=True, exist_ok=True)
487
+ marker_file = marker_dir / f'{session_id}.marker'
488
+
489
+ last_line, _ = read_marker(marker_file)
490
+
491
+ # Fast: parse transcript for conversation context
492
+ user_turn_count, conversation_snippets, modified_files, current_line_count = get_conversation_context(
493
+ transcript_path, last_line
494
+ )
495
+
496
+ # Update marker immediately so UserPromptSubmit knows Stop ran,
497
+ # even if we get killed during the slow stale rules check below
498
+ write_marker(marker_file, current_line_count, 'stop')
499
+
500
+ project_dir = os.environ.get('CLAUDE_PROJECT_DIR', os.getcwd())
501
+ invocation_ts = datetime.now().strftime('%Y%m%d-%H%M%S')
502
+ invocation_id = f'{session_id}-{invocation_ts}'
503
+
504
+ # Fast: spawn conversation review agent first (if enabled)
505
+ if settings['conversationReflection'] and len(conversation_snippets) > 0:
506
+ conv_context = build_conversation_context(conversation_snippets, modified_files)
507
+ conv_context_file = marker_dir / f'{invocation_id}-conversation.context'
508
+ conv_context_file.write_text(conv_context)
509
+
510
+ conv_log_file = marker_dir / f'{invocation_id}-conversation.log'
511
+ conv_notification_file = marker_dir / 'rule-notification-conversation.md'
512
+ conv_prompt = load_prompt_template(
513
+ 'conversation-prompt.txt',
514
+ CONTEXT_FILE=str(conv_context_file),
515
+ NOTIFICATION_FILE=str(conv_notification_file),
516
+ PROJECT_DIR=project_dir,
517
+ )
518
+ spawn_claude_agent(conv_prompt, conv_log_file, project_dir, model=settings['promptModel'])
519
+
520
+ # Slow (~10s): check for stale rules last — if the hook timeout kills us
521
+ # here, the conversation agent and marker are already handled
522
+ if settings['ruleMaintenance']:
523
+ stale_rules = get_stale_rules()
524
+
525
+ if len(stale_rules) > 0:
526
+ stale_context = build_stale_rules_context(stale_rules)
527
+ stale_context_file = marker_dir / f'{invocation_id}-stale.context'
528
+ stale_context_file.write_text(stale_context)
529
+
530
+ stale_log_file = marker_dir / f'{invocation_id}-stale.log'
531
+ stale_notification_file = marker_dir / 'rule-notification-stale.md'
532
+ stale_prompt = load_prompt_template(
533
+ 'stale-rules-prompt.txt',
534
+ CONTEXT_FILE=str(stale_context_file),
535
+ NOTIFICATION_FILE=str(stale_notification_file),
536
+ PROJECT_DIR=project_dir,
537
+ )
538
+ spawn_claude_agent(stale_prompt, stale_log_file, project_dir, model=settings['promptModel'])
539
+
540
+
541
+ def handle_user_prompt_submit(hook_input):
542
+ """
543
+ Handle the UserPromptSubmit hook event.
544
+ Detects whether the previous turn was interrupted by checking if the Stop hook's
545
+ marker file is stale (transcript has lines beyond the marker). If so, spawns a
546
+ rule-reflection agent focused on the interruption.
547
+ """
548
+ settings = load_memory_settings()
549
+
550
+ # Interruption detection is part of conversation reflection
551
+ if not settings['conversationReflection']:
552
+ return
553
+
554
+ session_id = hook_input.get('session_id', '')
555
+ transcript_path = hook_input.get('transcript_path', '')
556
+ follow_up_prompt = hook_input.get('prompt', '')
557
+
558
+ if not session_id or not transcript_path:
559
+ return
560
+
561
+ slug = get_project_slug()
562
+ marker_dir = Path('/tmp/claude-rule-markers') / slug
563
+ marker_dir.mkdir(parents=True, exist_ok=True)
564
+ marker_file = marker_dir / f'{session_id}.marker'
565
+
566
+ last_line, was_stop = read_marker(marker_file)
567
+
568
+ # If the Stop hook already ran for the previous turn, this is a normal
569
+ # completion — not an interruption. The Stop hook tags the marker with
570
+ # 'stop' when it writes it.
571
+ if was_stop:
572
+ return
573
+
574
+ # Count current transcript lines
575
+ try:
576
+ with open(transcript_path, 'r') as f:
577
+ current_line_count = sum(1 for _ in f)
578
+ except IOError:
579
+ return
580
+
581
+ # Quick exit: if nothing new since the marker, definitely no interruption
582
+ if current_line_count <= last_line:
583
+ return
584
+
585
+ # Verify Claude actually started responding in the unprocessed lines.
586
+ # Guards against false positives on first prompt (no assistant yet) or
587
+ # if the user hit Escape before Claude produced any output.
588
+ if not has_assistant_messages(transcript_path, last_line):
589
+ return
590
+
591
+ # Interruption detected! Build context and spawn agent.
592
+ _, conversation_snippets, modified_files, _ = get_conversation_context(
593
+ transcript_path, last_line
594
+ )
595
+
596
+ if not conversation_snippets:
597
+ return
598
+
599
+ # Update marker so we don't re-process these lines
600
+ write_marker(marker_file, current_line_count, 'submit')
601
+
602
+ project_dir = os.environ.get('CLAUDE_PROJECT_DIR', os.getcwd())
603
+
604
+ invocation_ts = datetime.now().strftime('%Y%m%d-%H%M%S')
605
+ invocation_id = f'{session_id}-{invocation_ts}'
606
+
607
+ interruption_context = build_interruption_context(
608
+ conversation_snippets, follow_up_prompt, modified_files
609
+ )
610
+ context_file = marker_dir / f'{invocation_id}-interruption.context'
611
+ context_file.write_text(interruption_context)
612
+
613
+ log_file = marker_dir / f'{invocation_id}-interruption.log'
614
+ notification_file = marker_dir / 'rule-notification-interruption.md'
615
+ prompt = load_prompt_template(
616
+ 'interruption-prompt.txt',
617
+ CONTEXT_FILE=str(context_file),
618
+ NOTIFICATION_FILE=str(notification_file),
619
+ PROJECT_DIR=project_dir,
620
+ )
621
+ spawn_claude_agent(prompt, log_file, project_dir, model=settings['promptModel'])
622
+
623
+
624
+ def main():
625
+ # Read hook input from stdin
626
+ try:
627
+ hook_input = json.load(sys.stdin)
628
+ except json.JSONDecodeError:
629
+ return
630
+
631
+ stop_hook_active = hook_input.get('stop_hook_active', False)
632
+
633
+ # Prevent infinite loops — stop_hook_active covers same-process recursion,
634
+ # env var covers spawned subagent sessions triggering the hook on exit
635
+ if stop_hook_active or os.environ.get('CODEYAM_RULE_AGENT'):
636
+ return
637
+
638
+ # Detect event type: UserPromptSubmit has a 'prompt' field, Stop does not
639
+ is_user_prompt = 'prompt' in hook_input
640
+ if is_user_prompt:
641
+ handle_user_prompt_submit(hook_input)
642
+ else:
643
+ handle_stop(hook_input)
644
+
645
+
646
+ if __name__ == '__main__':
647
+ main()