@codeyam/codeyam-cli 0.1.0-staging.dbc742d → 0.1.0-staging.df25827

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 (479) 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 +6 -6
  4. package/analyzer-template/packages/ai/package.json +1 -1
  5. package/analyzer-template/packages/ai/src/lib/astScopes/methodSemantics.ts +135 -0
  6. package/analyzer-template/packages/ai/src/lib/astScopes/nodeToSource.ts +19 -0
  7. package/analyzer-template/packages/ai/src/lib/astScopes/paths.ts +11 -4
  8. package/analyzer-template/packages/ai/src/lib/dataStructure/ScopeDataStructure.ts +36 -9
  9. package/analyzer-template/packages/ai/src/lib/dataStructure/equivalencyManagers/ParentScopeManager.ts +10 -3
  10. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.ts +16 -6
  11. package/analyzer-template/packages/analyze/index.ts +4 -1
  12. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.ts +28 -2
  13. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities.ts +5 -36
  14. package/analyzer-template/packages/analyze/src/lib/files/analyze/findOrCreateEntity.ts +10 -6
  15. package/analyzer-template/packages/analyze/src/lib/files/analyze/gatherEntityMap.ts +9 -12
  16. package/analyzer-template/packages/analyze/src/lib/files/analyze/trackEntityCircularDependencies.ts +21 -0
  17. package/analyzer-template/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.ts +82 -10
  18. package/analyzer-template/packages/analyze/src/lib/files/analyzeChange.ts +4 -0
  19. package/analyzer-template/packages/analyze/src/lib/files/analyzeInitial.ts +4 -0
  20. package/analyzer-template/packages/analyze/src/lib/files/analyzeNextRoute.ts +8 -3
  21. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateDataStructure.ts +239 -58
  22. package/analyzer-template/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.ts +1684 -1462
  23. package/analyzer-template/packages/aws/package.json +7 -7
  24. package/analyzer-template/packages/database/package.json +3 -3
  25. package/analyzer-template/packages/database/src/lib/kysely/tables/editorScenariosTable.ts +77 -6
  26. package/analyzer-template/packages/database/src/lib/loadAnalysis.ts +25 -15
  27. package/analyzer-template/packages/database/src/lib/loadEntity.ts +19 -8
  28. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts +5 -1
  29. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.d.ts.map +1 -1
  30. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js +79 -6
  31. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -1
  32. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalysis.d.ts.map +1 -1
  33. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalysis.js +7 -1
  34. package/analyzer-template/packages/github/dist/database/src/lib/loadAnalysis.js.map +1 -1
  35. package/analyzer-template/packages/github/dist/database/src/lib/loadEntity.d.ts +4 -1
  36. package/analyzer-template/packages/github/dist/database/src/lib/loadEntity.d.ts.map +1 -1
  37. package/analyzer-template/packages/github/dist/database/src/lib/loadEntity.js +5 -5
  38. package/analyzer-template/packages/github/dist/database/src/lib/loadEntity.js.map +1 -1
  39. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.d.ts +3 -1
  40. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.d.ts.map +1 -1
  41. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.js +22 -1
  42. package/analyzer-template/packages/utils/dist/utils/src/lib/fs/rsyncCopy.js.map +1 -1
  43. package/analyzer-template/packages/utils/src/lib/fs/rsyncCopy.ts +27 -0
  44. package/analyzer-template/project/analyzeFileEntities.ts +26 -0
  45. package/analyzer-template/project/runMultiScenarioServer.ts +26 -3
  46. package/background/src/lib/virtualized/project/analyzeFileEntities.js +22 -0
  47. package/background/src/lib/virtualized/project/analyzeFileEntities.js.map +1 -1
  48. package/background/src/lib/virtualized/project/runMultiScenarioServer.js +23 -3
  49. package/background/src/lib/virtualized/project/runMultiScenarioServer.js.map +1 -1
  50. package/codeyam-cli/src/cli.js +24 -0
  51. package/codeyam-cli/src/cli.js.map +1 -1
  52. package/codeyam-cli/src/commands/__tests__/editor.analyzeImportsArgs.test.js +47 -0
  53. package/codeyam-cli/src/commands/__tests__/editor.analyzeImportsArgs.test.js.map +1 -0
  54. package/codeyam-cli/src/commands/__tests__/editor.auditNoAutoAnalysis.test.js +71 -0
  55. package/codeyam-cli/src/commands/__tests__/editor.auditNoAutoAnalysis.test.js.map +1 -0
  56. package/codeyam-cli/src/commands/__tests__/editor.designSystem.test.js +30 -0
  57. package/codeyam-cli/src/commands/__tests__/editor.designSystem.test.js.map +1 -0
  58. package/codeyam-cli/src/commands/__tests__/editor.isolateArgs.test.js +51 -0
  59. package/codeyam-cli/src/commands/__tests__/editor.isolateArgs.test.js.map +1 -0
  60. package/codeyam-cli/src/commands/__tests__/editor.statePersistence.test.js +55 -0
  61. package/codeyam-cli/src/commands/__tests__/editor.statePersistence.test.js.map +1 -0
  62. package/codeyam-cli/src/commands/__tests__/editor.stepDispatch.test.js +56 -0
  63. package/codeyam-cli/src/commands/__tests__/editor.stepDispatch.test.js.map +1 -0
  64. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js +39 -3
  65. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js.map +1 -1
  66. package/codeyam-cli/src/commands/editor.js +3819 -624
  67. package/codeyam-cli/src/commands/editor.js.map +1 -1
  68. package/codeyam-cli/src/commands/editorAnalyzeImportsArgs.js +23 -0
  69. package/codeyam-cli/src/commands/editorAnalyzeImportsArgs.js.map +1 -0
  70. package/codeyam-cli/src/commands/editorIsolateArgs.js +25 -0
  71. package/codeyam-cli/src/commands/editorIsolateArgs.js.map +1 -0
  72. package/codeyam-cli/src/commands/init.js +22 -0
  73. package/codeyam-cli/src/commands/init.js.map +1 -1
  74. package/codeyam-cli/src/commands/telemetry.js +37 -0
  75. package/codeyam-cli/src/commands/telemetry.js.map +1 -0
  76. package/codeyam-cli/src/data/designSystems.js +27 -0
  77. package/codeyam-cli/src/data/designSystems.js.map +1 -0
  78. package/codeyam-cli/src/data/techStacks.js +1 -1
  79. package/codeyam-cli/src/utils/__tests__/analyzerFinalization.test.js +173 -0
  80. package/codeyam-cli/src/utils/__tests__/analyzerFinalization.test.js.map +1 -0
  81. package/codeyam-cli/src/utils/__tests__/devServerState.test.js +93 -1
  82. package/codeyam-cli/src/utils/__tests__/devServerState.test.js.map +1 -1
  83. package/codeyam-cli/src/utils/__tests__/editorApi.test.js +62 -8
  84. package/codeyam-cli/src/utils/__tests__/editorApi.test.js.map +1 -1
  85. package/codeyam-cli/src/utils/__tests__/editorAudit.test.js +3526 -1
  86. package/codeyam-cli/src/utils/__tests__/editorAudit.test.js.map +1 -1
  87. package/codeyam-cli/src/utils/__tests__/editorBroadcastViewport.test.js +76 -0
  88. package/codeyam-cli/src/utils/__tests__/editorBroadcastViewport.test.js.map +1 -0
  89. package/codeyam-cli/src/utils/__tests__/editorCaptureScenarioSeeding.test.js +137 -0
  90. package/codeyam-cli/src/utils/__tests__/editorCaptureScenarioSeeding.test.js.map +1 -0
  91. package/codeyam-cli/src/utils/__tests__/editorDeleteScenario.test.js +100 -0
  92. package/codeyam-cli/src/utils/__tests__/editorDeleteScenario.test.js.map +1 -0
  93. package/codeyam-cli/src/utils/__tests__/editorEntityChangeStatus.test.js +76 -3
  94. package/codeyam-cli/src/utils/__tests__/editorEntityChangeStatus.test.js.map +1 -1
  95. package/codeyam-cli/src/utils/__tests__/editorEntityHelpers.test.js +381 -0
  96. package/codeyam-cli/src/utils/__tests__/editorEntityHelpers.test.js.map +1 -0
  97. package/codeyam-cli/src/utils/__tests__/editorGuardMiddleware.test.js +67 -0
  98. package/codeyam-cli/src/utils/__tests__/editorGuardMiddleware.test.js.map +1 -0
  99. package/codeyam-cli/src/utils/__tests__/editorLoaderHelpers.test.js +75 -1
  100. package/codeyam-cli/src/utils/__tests__/editorLoaderHelpers.test.js.map +1 -1
  101. package/codeyam-cli/src/utils/__tests__/editorMigration.test.js +435 -0
  102. package/codeyam-cli/src/utils/__tests__/editorMigration.test.js.map +1 -0
  103. package/codeyam-cli/src/utils/__tests__/editorPreview.test.js +73 -1
  104. package/codeyam-cli/src/utils/__tests__/editorPreview.test.js.map +1 -1
  105. package/codeyam-cli/src/utils/__tests__/editorProxySession.test.js +98 -1
  106. package/codeyam-cli/src/utils/__tests__/editorProxySession.test.js.map +1 -1
  107. package/codeyam-cli/src/utils/__tests__/editorRoadmap.test.js +1108 -0
  108. package/codeyam-cli/src/utils/__tests__/editorRoadmap.test.js.map +1 -0
  109. package/codeyam-cli/src/utils/__tests__/editorScenarioSwitch.test.js +190 -0
  110. package/codeyam-cli/src/utils/__tests__/editorScenarioSwitch.test.js.map +1 -1
  111. package/codeyam-cli/src/utils/__tests__/editorScenarios.test.js +936 -9
  112. package/codeyam-cli/src/utils/__tests__/editorScenarios.test.js.map +1 -1
  113. package/codeyam-cli/src/utils/__tests__/editorSeedAdapter.test.js +201 -1
  114. package/codeyam-cli/src/utils/__tests__/editorSeedAdapter.test.js.map +1 -1
  115. package/codeyam-cli/src/utils/__tests__/editorSeedAdapterPrismaValidation.test.js +143 -0
  116. package/codeyam-cli/src/utils/__tests__/editorSeedAdapterPrismaValidation.test.js.map +1 -0
  117. package/codeyam-cli/src/utils/__tests__/editorSessionFilter.test.js +66 -0
  118. package/codeyam-cli/src/utils/__tests__/editorSessionFilter.test.js.map +1 -0
  119. package/codeyam-cli/src/utils/__tests__/editorShouldRevalidate.test.js +53 -0
  120. package/codeyam-cli/src/utils/__tests__/editorShouldRevalidate.test.js.map +1 -0
  121. package/codeyam-cli/src/utils/__tests__/entityChangeStatus.test.js +390 -11
  122. package/codeyam-cli/src/utils/__tests__/entityChangeStatus.test.js.map +1 -1
  123. package/codeyam-cli/src/utils/__tests__/envFile.test.js +125 -0
  124. package/codeyam-cli/src/utils/__tests__/envFile.test.js.map +1 -0
  125. package/codeyam-cli/src/utils/__tests__/glossaryAdd.test.js +177 -0
  126. package/codeyam-cli/src/utils/__tests__/glossaryAdd.test.js.map +1 -0
  127. package/codeyam-cli/src/utils/__tests__/handoffContext.test.js +500 -0
  128. package/codeyam-cli/src/utils/__tests__/handoffContext.test.js.map +1 -0
  129. package/codeyam-cli/src/utils/__tests__/journalCaptureStabilization.test.js +16 -1
  130. package/codeyam-cli/src/utils/__tests__/journalCaptureStabilization.test.js.map +1 -1
  131. package/codeyam-cli/src/utils/__tests__/manualEntityAnalysis.test.js +302 -0
  132. package/codeyam-cli/src/utils/__tests__/manualEntityAnalysis.test.js.map +1 -0
  133. package/codeyam-cli/src/utils/__tests__/parseRegisterArg.test.js +30 -2
  134. package/codeyam-cli/src/utils/__tests__/parseRegisterArg.test.js.map +1 -1
  135. package/codeyam-cli/src/utils/__tests__/registerScenarioResult.test.js +127 -0
  136. package/codeyam-cli/src/utils/__tests__/registerScenarioResult.test.js.map +1 -0
  137. package/codeyam-cli/src/utils/__tests__/routePatternMatching.test.js +118 -0
  138. package/codeyam-cli/src/utils/__tests__/routePatternMatching.test.js.map +1 -0
  139. package/codeyam-cli/src/utils/__tests__/scenarioCoverage.test.js +57 -0
  140. package/codeyam-cli/src/utils/__tests__/scenarioCoverage.test.js.map +1 -1
  141. package/codeyam-cli/src/utils/__tests__/scenariosManifest.test.js +373 -1
  142. package/codeyam-cli/src/utils/__tests__/scenariosManifest.test.js.map +1 -1
  143. package/codeyam-cli/src/utils/__tests__/screenshotHash.test.js +84 -0
  144. package/codeyam-cli/src/utils/__tests__/screenshotHash.test.js.map +1 -0
  145. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js +1 -0
  146. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js.map +1 -1
  147. package/codeyam-cli/src/utils/__tests__/telemetry.test.js +159 -0
  148. package/codeyam-cli/src/utils/__tests__/telemetry.test.js.map +1 -0
  149. package/codeyam-cli/src/utils/__tests__/testRunner.test.js +216 -0
  150. package/codeyam-cli/src/utils/__tests__/testRunner.test.js.map +1 -0
  151. package/codeyam-cli/src/utils/__tests__/webappDetection.test.js +6 -0
  152. package/codeyam-cli/src/utils/__tests__/webappDetection.test.js.map +1 -1
  153. package/codeyam-cli/src/utils/analysisRunner.js +39 -8
  154. package/codeyam-cli/src/utils/analysisRunner.js.map +1 -1
  155. package/codeyam-cli/src/utils/analyzer.js +11 -1
  156. package/codeyam-cli/src/utils/analyzer.js.map +1 -1
  157. package/codeyam-cli/src/utils/analyzerFinalization.js +100 -0
  158. package/codeyam-cli/src/utils/analyzerFinalization.js.map +1 -0
  159. package/codeyam-cli/src/utils/backgroundServer.js +1 -1
  160. package/codeyam-cli/src/utils/backgroundServer.js.map +1 -1
  161. package/codeyam-cli/src/utils/designSystemShowcase.js +810 -0
  162. package/codeyam-cli/src/utils/designSystemShowcase.js.map +1 -0
  163. package/codeyam-cli/src/utils/devServerState.js +32 -0
  164. package/codeyam-cli/src/utils/devServerState.js.map +1 -1
  165. package/codeyam-cli/src/utils/editorApi.js +27 -5
  166. package/codeyam-cli/src/utils/editorApi.js.map +1 -1
  167. package/codeyam-cli/src/utils/editorAudit.js +700 -10
  168. package/codeyam-cli/src/utils/editorAudit.js.map +1 -1
  169. package/codeyam-cli/src/utils/editorBroadcastViewport.js +26 -0
  170. package/codeyam-cli/src/utils/editorBroadcastViewport.js.map +1 -0
  171. package/codeyam-cli/src/utils/editorDeleteScenario.js +67 -0
  172. package/codeyam-cli/src/utils/editorDeleteScenario.js.map +1 -0
  173. package/codeyam-cli/src/utils/editorEntityChangeStatus.js +13 -7
  174. package/codeyam-cli/src/utils/editorEntityChangeStatus.js.map +1 -1
  175. package/codeyam-cli/src/utils/editorEntityHelpers.js +144 -0
  176. package/codeyam-cli/src/utils/editorEntityHelpers.js.map +1 -0
  177. package/codeyam-cli/src/utils/editorGuard.js +36 -0
  178. package/codeyam-cli/src/utils/editorGuard.js.map +1 -0
  179. package/codeyam-cli/src/utils/editorLoaderHelpers.js +40 -1
  180. package/codeyam-cli/src/utils/editorLoaderHelpers.js.map +1 -1
  181. package/codeyam-cli/src/utils/editorMigration.js +224 -0
  182. package/codeyam-cli/src/utils/editorMigration.js.map +1 -0
  183. package/codeyam-cli/src/utils/editorPreview.js +33 -0
  184. package/codeyam-cli/src/utils/editorPreview.js.map +1 -1
  185. package/codeyam-cli/src/utils/editorRecapture.js +109 -0
  186. package/codeyam-cli/src/utils/editorRecapture.js.map +1 -0
  187. package/codeyam-cli/src/utils/editorRoadmap.js +574 -0
  188. package/codeyam-cli/src/utils/editorRoadmap.js.map +1 -0
  189. package/codeyam-cli/src/utils/editorScenarioSwitch.js +39 -2
  190. package/codeyam-cli/src/utils/editorScenarioSwitch.js.map +1 -1
  191. package/codeyam-cli/src/utils/editorScenarios.js +375 -16
  192. package/codeyam-cli/src/utils/editorScenarios.js.map +1 -1
  193. package/codeyam-cli/src/utils/editorSeedAdapter.js +308 -6
  194. package/codeyam-cli/src/utils/editorSeedAdapter.js.map +1 -1
  195. package/codeyam-cli/src/utils/editorShouldRevalidate.js +21 -0
  196. package/codeyam-cli/src/utils/editorShouldRevalidate.js.map +1 -0
  197. package/codeyam-cli/src/utils/entityChangeStatus.js +50 -5
  198. package/codeyam-cli/src/utils/entityChangeStatus.js.map +1 -1
  199. package/codeyam-cli/src/utils/entityChangeStatus.server.js +72 -3
  200. package/codeyam-cli/src/utils/entityChangeStatus.server.js.map +1 -1
  201. package/codeyam-cli/src/utils/envFile.js +90 -0
  202. package/codeyam-cli/src/utils/envFile.js.map +1 -0
  203. package/codeyam-cli/src/utils/fileWatcher.js +38 -0
  204. package/codeyam-cli/src/utils/fileWatcher.js.map +1 -1
  205. package/codeyam-cli/src/utils/glossaryAdd.js +74 -0
  206. package/codeyam-cli/src/utils/glossaryAdd.js.map +1 -0
  207. package/codeyam-cli/src/utils/handoffContext.js +257 -0
  208. package/codeyam-cli/src/utils/handoffContext.js.map +1 -0
  209. package/codeyam-cli/src/utils/install-skills.js +50 -6
  210. package/codeyam-cli/src/utils/install-skills.js.map +1 -1
  211. package/codeyam-cli/src/utils/manualEntityAnalysis.js +196 -0
  212. package/codeyam-cli/src/utils/manualEntityAnalysis.js.map +1 -0
  213. package/codeyam-cli/src/utils/parseRegisterArg.js.map +1 -1
  214. package/codeyam-cli/src/utils/progress.js +2 -2
  215. package/codeyam-cli/src/utils/progress.js.map +1 -1
  216. package/codeyam-cli/src/utils/queue/__tests__/job.interactiveStart.test.js +159 -0
  217. package/codeyam-cli/src/utils/queue/__tests__/job.interactiveStart.test.js.map +1 -0
  218. package/codeyam-cli/src/utils/queue/job.js +35 -6
  219. package/codeyam-cli/src/utils/queue/job.js.map +1 -1
  220. package/codeyam-cli/src/utils/registerScenarioResult.js +52 -0
  221. package/codeyam-cli/src/utils/registerScenarioResult.js.map +1 -0
  222. package/codeyam-cli/src/utils/routePatternMatching.js +129 -0
  223. package/codeyam-cli/src/utils/routePatternMatching.js.map +1 -0
  224. package/codeyam-cli/src/utils/scenarioCoverage.js +12 -10
  225. package/codeyam-cli/src/utils/scenarioCoverage.js.map +1 -1
  226. package/codeyam-cli/src/utils/scenariosManifest.js +154 -0
  227. package/codeyam-cli/src/utils/scenariosManifest.js.map +1 -1
  228. package/codeyam-cli/src/utils/screenshotHash.js +26 -0
  229. package/codeyam-cli/src/utils/screenshotHash.js.map +1 -0
  230. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js +1 -0
  231. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js.map +1 -1
  232. package/codeyam-cli/src/utils/simulationGateMiddleware.js +9 -0
  233. package/codeyam-cli/src/utils/simulationGateMiddleware.js.map +1 -1
  234. package/codeyam-cli/src/utils/techStackConfig.js +38 -0
  235. package/codeyam-cli/src/utils/techStackConfig.js.map +1 -0
  236. package/codeyam-cli/src/utils/techStackConfig.test.js +85 -0
  237. package/codeyam-cli/src/utils/techStackConfig.test.js.map +1 -0
  238. package/codeyam-cli/src/utils/telemetry.js +106 -0
  239. package/codeyam-cli/src/utils/telemetry.js.map +1 -0
  240. package/codeyam-cli/src/utils/telemetryMiddleware.js +22 -0
  241. package/codeyam-cli/src/utils/telemetryMiddleware.js.map +1 -0
  242. package/codeyam-cli/src/utils/testResultCache.js +53 -0
  243. package/codeyam-cli/src/utils/testResultCache.js.map +1 -0
  244. package/codeyam-cli/src/utils/testResultCache.server.js +81 -0
  245. package/codeyam-cli/src/utils/testResultCache.server.js.map +1 -0
  246. package/codeyam-cli/src/utils/testResultCache.server.test.js +187 -0
  247. package/codeyam-cli/src/utils/testResultCache.server.test.js.map +1 -0
  248. package/codeyam-cli/src/utils/testResultCache.test.js +230 -0
  249. package/codeyam-cli/src/utils/testResultCache.test.js.map +1 -0
  250. package/codeyam-cli/src/utils/testRunner.js +193 -1
  251. package/codeyam-cli/src/utils/testRunner.js.map +1 -1
  252. package/codeyam-cli/src/utils/webappDetection.js +4 -2
  253. package/codeyam-cli/src/utils/webappDetection.js.map +1 -1
  254. package/codeyam-cli/src/webserver/__tests__/api.interactive-switch-scenario.test.js +99 -0
  255. package/codeyam-cli/src/webserver/__tests__/api.interactive-switch-scenario.test.js.map +1 -0
  256. package/codeyam-cli/src/webserver/__tests__/buildPtyEnv.test.js +153 -0
  257. package/codeyam-cli/src/webserver/__tests__/buildPtyEnv.test.js.map +1 -0
  258. package/codeyam-cli/src/webserver/__tests__/clientErrors.test.js +68 -1
  259. package/codeyam-cli/src/webserver/__tests__/clientErrors.test.js.map +1 -1
  260. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js +454 -4
  261. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js.map +1 -1
  262. package/codeyam-cli/src/webserver/__tests__/idleDetector.test.js +315 -0
  263. package/codeyam-cli/src/webserver/__tests__/idleDetector.test.js.map +1 -0
  264. package/codeyam-cli/src/webserver/__tests__/stripClaudeCommand.test.js +135 -0
  265. package/codeyam-cli/src/webserver/__tests__/stripClaudeCommand.test.js.map +1 -0
  266. package/codeyam-cli/src/webserver/app/lib/clientErrors.js +22 -1
  267. package/codeyam-cli/src/webserver/app/lib/clientErrors.js.map +1 -1
  268. package/codeyam-cli/src/webserver/app/lib/database.js.map +1 -1
  269. package/codeyam-cli/src/webserver/app/lib/git.js +3 -2
  270. package/codeyam-cli/src/webserver/app/lib/git.js.map +1 -1
  271. package/codeyam-cli/src/webserver/app/routes/api.interactive-switch-scenario.js +34 -0
  272. package/codeyam-cli/src/webserver/app/routes/api.interactive-switch-scenario.js.map +1 -0
  273. package/codeyam-cli/src/webserver/app/types/editor.js +8 -0
  274. package/codeyam-cli/src/webserver/app/types/editor.js.map +1 -0
  275. package/codeyam-cli/src/webserver/backgroundServer.js +60 -61
  276. package/codeyam-cli/src/webserver/backgroundServer.js.map +1 -1
  277. package/codeyam-cli/src/webserver/build/client/assets/CopyButton-DTBZZfSk.js +1 -0
  278. package/codeyam-cli/src/webserver/build/client/assets/{EntityItem-BcgbViKV.js → EntityItem-BxclONWq.js} +3 -3
  279. package/codeyam-cli/src/webserver/build/client/assets/EntityTypeBadge-CQgyEGV-.js +1 -0
  280. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeIcon-CQIG2qda.js → EntityTypeIcon-BsnEOJZ_.js} +1 -1
  281. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-ByaELMbv.js +1 -0
  282. package/codeyam-cli/src/webserver/build/client/assets/InteractivePreview-6WjVfhxX.js +25 -0
  283. package/codeyam-cli/src/webserver/build/client/assets/LibraryFunctionPreview-ChX-Hp7W.js +3 -0
  284. package/codeyam-cli/src/webserver/build/client/assets/{LoadingDots-BU_OAEMP.js → LoadingDots-By5zI316.js} +1 -1
  285. package/codeyam-cli/src/webserver/build/client/assets/{LogViewer-ceAyBX-H.js → LogViewer-C-9zQdXg.js} +3 -3
  286. package/codeyam-cli/src/webserver/build/client/assets/MiniClaudeChat-Bs2_Oua4.js +36 -0
  287. package/codeyam-cli/src/webserver/build/client/assets/{ReportIssueModal-BzHcG7SE.js → ReportIssueModal-DQsceHVv.js} +2 -2
  288. package/codeyam-cli/src/webserver/build/client/assets/SafeScreenshot-DThcm_9M.js +1 -0
  289. package/codeyam-cli/src/webserver/build/client/assets/{ScenarioViewer-0DY_NKil.js → ScenarioViewer-Cl4oOA3A.js} +3 -3
  290. package/codeyam-cli/src/webserver/build/client/assets/Spinner-CIil5-gb.js +34 -0
  291. package/codeyam-cli/src/webserver/build/client/assets/TruncatedFilePath-CK7-NaPZ.js +1 -0
  292. package/codeyam-cli/src/webserver/build/client/assets/ViewportInspectBar-BqkA9zyZ.js +1 -0
  293. package/codeyam-cli/src/webserver/build/client/assets/{_index-DLxKhri3.js → _index-DnOgyseQ.js} +2 -2
  294. package/codeyam-cli/src/webserver/build/client/assets/{activity.(_tab)-BcY3q6nt.js → activity.(_tab)-DqM9hbNE.js} +3 -3
  295. package/codeyam-cli/src/webserver/build/client/assets/{addon-web-links-Duc5hnl7.js → addon-web-links-C58dYPwR.js} +1 -1
  296. package/codeyam-cli/src/webserver/build/client/assets/{agent-transcripts-Bni3iiUj.js → agent-transcripts-B8NCeOrm.js} +3 -3
  297. package/codeyam-cli/src/webserver/build/client/assets/api.editor-database-verify-l0sNRNKZ.js +1 -0
  298. package/codeyam-cli/src/webserver/build/client/assets/api.editor-github-verify-l0sNRNKZ.js +1 -0
  299. package/codeyam-cli/src/webserver/build/client/assets/api.editor-handoff-l0sNRNKZ.js +1 -0
  300. package/codeyam-cli/src/webserver/build/client/assets/api.editor-hosting-verify-l0sNRNKZ.js +1 -0
  301. package/codeyam-cli/src/webserver/build/client/assets/api.editor-recapture-stale-l0sNRNKZ.js +1 -0
  302. package/codeyam-cli/src/webserver/build/client/assets/api.editor-rename-scenario-l0sNRNKZ.js +1 -0
  303. package/codeyam-cli/src/webserver/build/client/assets/api.editor-roadmap-l0sNRNKZ.js +1 -0
  304. package/codeyam-cli/src/webserver/build/client/assets/api.editor-save-scenario-data-l0sNRNKZ.js +1 -0
  305. package/codeyam-cli/src/webserver/build/client/assets/api.editor-save-seed-state-l0sNRNKZ.js +1 -0
  306. package/codeyam-cli/src/webserver/build/client/assets/api.editor-scenario-prompt-l0sNRNKZ.js +1 -0
  307. package/codeyam-cli/src/webserver/build/client/assets/api.editor-schema-l0sNRNKZ.js +1 -0
  308. package/codeyam-cli/src/webserver/build/client/assets/api.editor-session-l0sNRNKZ.js +1 -0
  309. package/codeyam-cli/src/webserver/build/client/assets/api.editor-verify-routes-l0sNRNKZ.js +1 -0
  310. package/codeyam-cli/src/webserver/build/client/assets/api.interactive-switch-scenario-l0sNRNKZ.js +1 -0
  311. package/codeyam-cli/src/webserver/build/client/assets/{book-open-BYOypzCa.js → book-open-BFSIqZgO.js} +1 -1
  312. package/codeyam-cli/src/webserver/build/client/assets/{chevron-down-C_Pmso5S.js → chevron-down-B9fDzFVh.js} +1 -1
  313. package/codeyam-cli/src/webserver/build/client/assets/chunk-UVKPFVEO-Bmq2apuh.js +43 -0
  314. package/codeyam-cli/src/webserver/build/client/assets/{circle-check-BVMi9VA5.js → circle-check-DLPObLUx.js} +1 -1
  315. package/codeyam-cli/src/webserver/build/client/assets/{copy-n2FB0_Sw.js → copy-DXEmO0TD.js} +1 -1
  316. package/codeyam-cli/src/webserver/build/client/assets/{createLucideIcon-CC6AbExI.js → createLucideIcon-BwyFiRot.js} +1 -1
  317. package/codeyam-cli/src/webserver/build/client/assets/cy-logo-cli-Coe5NhbS.js +1 -0
  318. package/codeyam-cli/src/webserver/build/client/assets/{cy-logo-cli-CCKUIm0S.svg → cy-logo-cli-DoA97ML3.svg} +2 -2
  319. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-iRhRIFlp.js +1 -0
  320. package/codeyam-cli/src/webserver/build/client/assets/editor._tab-BZPBzV73.js +1 -0
  321. package/codeyam-cli/src/webserver/build/client/assets/editor.entity.(_sha)-DhtVC4aI.js +161 -0
  322. package/codeyam-cli/src/webserver/build/client/assets/editorPreview-C6fEYHrh.js +41 -0
  323. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha._-BF4oLwaE.js → entity._sha._-pc-vc6wO.js} +13 -12
  324. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.dev-C8AyYgYT.js +6 -0
  325. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-DziaVQX1.js +6 -0
  326. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-BTcpgIpC.js +6 -0
  327. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha_.edit._scenarioId-BMvVHNXU.js → entity._sha_.edit._scenarioId-D_O_ajfZ.js} +2 -2
  328. package/codeyam-cli/src/webserver/build/client/assets/{entry.client-DTvKq3TY.js → entry.client-j1Vi0bco.js} +6 -6
  329. package/codeyam-cli/src/webserver/build/client/assets/fileTableUtils-Daa96Fr1.js +1 -0
  330. package/codeyam-cli/src/webserver/build/client/assets/files-kuny2Q_s.js +1 -0
  331. package/codeyam-cli/src/webserver/build/client/assets/git-DgCZPMie.js +1 -0
  332. package/codeyam-cli/src/webserver/build/client/assets/globals-L-aUIeux.css +1 -0
  333. package/codeyam-cli/src/webserver/build/client/assets/{index-BcvgDzbZ.js → index-BliGSSpl.js} +1 -1
  334. package/codeyam-cli/src/webserver/build/client/assets/{index-yHOVb4rc.js → index-SqjQKTdH.js} +1 -1
  335. package/codeyam-cli/src/webserver/build/client/assets/{index-10oVnAAH.js → index-vyrZD2g4.js} +1 -1
  336. package/codeyam-cli/src/webserver/build/client/assets/jsx-runtime-D_zvdyIk.js +9 -0
  337. package/codeyam-cli/src/webserver/build/client/assets/labs-c3yLxSEp.js +1 -0
  338. package/codeyam-cli/src/webserver/build/client/assets/{loader-circle-DaAZ_H2w.js → loader-circle-D-q28GLF.js} +1 -1
  339. package/codeyam-cli/src/webserver/build/client/assets/manifest-b0d69c06.js +1 -0
  340. package/codeyam-cli/src/webserver/build/client/assets/{memory-9gnxSZlb.js → memory-CEWIUC4t.js} +2 -2
  341. package/codeyam-cli/src/webserver/build/client/assets/{pause-f5-1lKBt.js → pause-BP6fitdh.js} +1 -1
  342. package/codeyam-cli/src/webserver/build/client/assets/root-CLedrjXQ.js +80 -0
  343. package/codeyam-cli/src/webserver/build/client/assets/{search-Di64LWVb.js → search-BooqacKS.js} +1 -1
  344. package/codeyam-cli/src/webserver/build/client/assets/settings-BM0nbryO.js +1 -0
  345. package/codeyam-cli/src/webserver/build/client/assets/simulations-ovy6FjRY.js +1 -0
  346. package/codeyam-cli/src/webserver/build/client/assets/{terminal-Br7MOqts.js → terminal-DHemCJIs.js} +1 -1
  347. package/codeyam-cli/src/webserver/build/client/assets/{triangle-alert-BLdiCuG-.js → triangle-alert-D87ekDl8.js} +1 -1
  348. package/codeyam-cli/src/webserver/build/client/assets/useCustomSizes-Dk0Tciqg.js +1 -0
  349. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-C8QvIe05.js +2 -0
  350. package/codeyam-cli/src/webserver/build/client/assets/useReportContext-jkCytuYz.js +1 -0
  351. package/codeyam-cli/src/webserver/build/client/assets/useToast-BgqkixU9.js +1 -0
  352. package/codeyam-cli/src/webserver/build/client/sound-test.html +98 -0
  353. package/codeyam-cli/src/webserver/build/server/assets/analysisRunner-DPUEhrWo.js +16 -0
  354. package/codeyam-cli/src/webserver/build/server/assets/{index-BWoRb5RY.js → index-oF2amaGI.js} +1 -1
  355. package/codeyam-cli/src/webserver/build/server/assets/init-C42BvUGp.js +14 -0
  356. package/codeyam-cli/src/webserver/build/server/assets/progress-CHTtrxFG.js +1 -0
  357. package/codeyam-cli/src/webserver/build/server/assets/server-build-DiCdDL5d.js +853 -0
  358. package/codeyam-cli/src/webserver/build/server/index.js +1 -1
  359. package/codeyam-cli/src/webserver/build-info.json +5 -5
  360. package/codeyam-cli/src/webserver/editorProxy.js +443 -35
  361. package/codeyam-cli/src/webserver/editorProxy.js.map +1 -1
  362. package/codeyam-cli/src/webserver/idleDetector.js +130 -0
  363. package/codeyam-cli/src/webserver/idleDetector.js.map +1 -0
  364. package/codeyam-cli/src/webserver/mockStateEvents.js +28 -0
  365. package/codeyam-cli/src/webserver/mockStateEvents.js.map +1 -0
  366. package/codeyam-cli/src/webserver/public/sound-test.html +98 -0
  367. package/codeyam-cli/src/webserver/scripts/journalCapture.ts +53 -0
  368. package/codeyam-cli/src/webserver/server.js +192 -4
  369. package/codeyam-cli/src/webserver/server.js.map +1 -1
  370. package/codeyam-cli/src/webserver/terminalServer.js +369 -52
  371. package/codeyam-cli/src/webserver/terminalServer.js.map +1 -1
  372. package/codeyam-cli/templates/__tests__/editor-step-hook.prompt-capture.test.ts +118 -0
  373. package/codeyam-cli/templates/codeyam-editor-claude.md +3 -1
  374. package/codeyam-cli/templates/codeyam-editor-codex.md +61 -0
  375. package/codeyam-cli/templates/codeyam-editor-gemini.md +59 -0
  376. package/codeyam-cli/templates/codeyam-editor-reference.md +216 -0
  377. package/codeyam-cli/templates/design-systems/clean-dashboard-design-system.md +255 -0
  378. package/codeyam-cli/templates/design-systems/editorial-design-system.md +267 -0
  379. package/codeyam-cli/templates/design-systems/mono-brutalist-design-system.md +256 -0
  380. package/codeyam-cli/templates/design-systems/neo-brutalist-design-system.md +294 -0
  381. package/codeyam-cli/templates/editor-step-hook.py +193 -56
  382. package/codeyam-cli/templates/expo-react-native/MOBILE_SETUP.md +204 -5
  383. package/codeyam-cli/templates/expo-react-native/__tests__/.gitkeep +0 -0
  384. package/codeyam-cli/templates/expo-react-native/app/_layout.tsx +6 -3
  385. package/codeyam-cli/templates/expo-react-native/app/index.tsx +36 -0
  386. package/codeyam-cli/templates/expo-react-native/app.json +11 -0
  387. package/codeyam-cli/templates/expo-react-native/babel.config.js +1 -0
  388. package/codeyam-cli/templates/expo-react-native/gitignore +2 -0
  389. package/codeyam-cli/templates/expo-react-native/global.css +7 -0
  390. package/codeyam-cli/templates/expo-react-native/lib/theme.ts +73 -0
  391. package/codeyam-cli/templates/expo-react-native/package.json +32 -16
  392. package/codeyam-cli/templates/expo-react-native/patches/expo-modules-autolinking+3.0.24.patch +29 -0
  393. package/codeyam-cli/templates/isolation-route/expo-router.tsx.template +54 -0
  394. package/codeyam-cli/templates/nextjs-prisma-sqlite/gitignore +1 -0
  395. package/codeyam-cli/templates/nextjs-prisma-sqlite/package.json +1 -1
  396. package/codeyam-cli/templates/nextjs-prisma-sqlite/seed-adapter.ts +88 -40
  397. package/codeyam-cli/templates/nextjs-prisma-supabase/package.json +1 -1
  398. package/codeyam-cli/templates/seed-adapters/supabase.ts +475 -0
  399. package/codeyam-cli/templates/skills/codeyam-editor/SKILL.md +106 -10
  400. package/package.json +2 -1
  401. package/packages/ai/src/lib/astScopes/methodSemantics.js +99 -0
  402. package/packages/ai/src/lib/astScopes/methodSemantics.js.map +1 -1
  403. package/packages/ai/src/lib/astScopes/nodeToSource.js +16 -0
  404. package/packages/ai/src/lib/astScopes/nodeToSource.js.map +1 -1
  405. package/packages/ai/src/lib/astScopes/paths.js +12 -3
  406. package/packages/ai/src/lib/astScopes/paths.js.map +1 -1
  407. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js +27 -10
  408. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js.map +1 -1
  409. package/packages/ai/src/lib/dataStructure/equivalencyManagers/ParentScopeManager.js +9 -2
  410. package/packages/ai/src/lib/dataStructure/equivalencyManagers/ParentScopeManager.js.map +1 -1
  411. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js +14 -4
  412. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js.map +1 -1
  413. package/packages/analyze/index.js +1 -1
  414. package/packages/analyze/index.js.map +1 -1
  415. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js +16 -2
  416. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js.map +1 -1
  417. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js +6 -26
  418. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js.map +1 -1
  419. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js +3 -2
  420. package/packages/analyze/src/lib/files/analyze/findOrCreateEntity.js.map +1 -1
  421. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js +9 -7
  422. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js.map +1 -1
  423. package/packages/analyze/src/lib/files/analyze/trackEntityCircularDependencies.js +14 -0
  424. package/packages/analyze/src/lib/files/analyze/trackEntityCircularDependencies.js.map +1 -1
  425. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js +44 -11
  426. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js.map +1 -1
  427. package/packages/analyze/src/lib/files/analyzeChange.js +1 -0
  428. package/packages/analyze/src/lib/files/analyzeChange.js.map +1 -1
  429. package/packages/analyze/src/lib/files/analyzeInitial.js +1 -0
  430. package/packages/analyze/src/lib/files/analyzeInitial.js.map +1 -1
  431. package/packages/analyze/src/lib/files/analyzeNextRoute.js +5 -1
  432. package/packages/analyze/src/lib/files/analyzeNextRoute.js.map +1 -1
  433. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js +120 -28
  434. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js.map +1 -1
  435. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js +1368 -1193
  436. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js.map +1 -1
  437. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js +79 -6
  438. package/packages/database/src/lib/kysely/tables/editorScenariosTable.js.map +1 -1
  439. package/packages/database/src/lib/loadAnalysis.js +7 -1
  440. package/packages/database/src/lib/loadAnalysis.js.map +1 -1
  441. package/packages/database/src/lib/loadEntity.js +5 -5
  442. package/packages/database/src/lib/loadEntity.js.map +1 -1
  443. package/packages/utils/src/lib/fs/rsyncCopy.js +22 -1
  444. package/packages/utils/src/lib/fs/rsyncCopy.js.map +1 -1
  445. package/codeyam-cli/src/webserver/build/client/assets/CopyButton-BPXZwM4t.js +0 -1
  446. package/codeyam-cli/src/webserver/build/client/assets/EntityTypeBadge-g3saevPb.js +0 -1
  447. package/codeyam-cli/src/webserver/build/client/assets/InlineSpinner-Bu6c6aDe.js +0 -1
  448. package/codeyam-cli/src/webserver/build/client/assets/InteractivePreview-DYFW3lDD.js +0 -25
  449. package/codeyam-cli/src/webserver/build/client/assets/LibraryFunctionPreview-DLeucoVX.js +0 -3
  450. package/codeyam-cli/src/webserver/build/client/assets/SafeScreenshot-BED4B6sP.js +0 -1
  451. package/codeyam-cli/src/webserver/build/client/assets/Spinner-Bb5uFQ5V.js +0 -34
  452. package/codeyam-cli/src/webserver/build/client/assets/TruncatedFilePath-C8OKAR5x.js +0 -1
  453. package/codeyam-cli/src/webserver/build/client/assets/ViewportInspectBar-oAf2Kqsf.js +0 -1
  454. package/codeyam-cli/src/webserver/build/client/assets/chunk-JZWAC4HX-C4pqxYJB.js +0 -51
  455. package/codeyam-cli/src/webserver/build/client/assets/cy-logo-cli-DcX-ZS3p.js +0 -1
  456. package/codeyam-cli/src/webserver/build/client/assets/dev.empty-Csi0_PMl.js +0 -1
  457. package/codeyam-cli/src/webserver/build/client/assets/editor-BBAGP_mE.js +0 -10
  458. package/codeyam-cli/src/webserver/build/client/assets/editorPreview-BLQMSKZa.js +0 -41
  459. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.dev-C7YX6r3H.js +0 -6
  460. package/codeyam-cli/src/webserver/build/client/assets/entity._sha.scenarios._scenarioId.fullscreen-CF164ouH.js +0 -6
  461. package/codeyam-cli/src/webserver/build/client/assets/entity._sha_.create-scenario-p9hhkjJM.js +0 -6
  462. package/codeyam-cli/src/webserver/build/client/assets/fileTableUtils-cPo8LiG3.js +0 -1
  463. package/codeyam-cli/src/webserver/build/client/assets/files-BZrlFE1F.js +0 -1
  464. package/codeyam-cli/src/webserver/build/client/assets/git-DdZcvjGh.js +0 -1
  465. package/codeyam-cli/src/webserver/build/client/assets/globals-COUSHTyZ.css +0 -1
  466. package/codeyam-cli/src/webserver/build/client/assets/labs-Zk7ryIM1.js +0 -1
  467. package/codeyam-cli/src/webserver/build/client/assets/manifest-9c70d1f3.js +0 -1
  468. package/codeyam-cli/src/webserver/build/client/assets/root-CHOdrM6Y.js +0 -67
  469. package/codeyam-cli/src/webserver/build/client/assets/settings-0OrEMU6J.js +0 -1
  470. package/codeyam-cli/src/webserver/build/client/assets/simulations-DWT-CvLy.js +0 -1
  471. package/codeyam-cli/src/webserver/build/client/assets/useCustomSizes-CrAK28Bc.js +0 -1
  472. package/codeyam-cli/src/webserver/build/client/assets/useLastLogLine-C14nCb1q.js +0 -2
  473. package/codeyam-cli/src/webserver/build/client/assets/useReportContext-O-jkvSPx.js +0 -1
  474. package/codeyam-cli/src/webserver/build/client/assets/useToast-9FIWuYfK.js +0 -1
  475. package/codeyam-cli/src/webserver/build/server/assets/init-DbChSUQP.js +0 -10
  476. package/codeyam-cli/src/webserver/build/server/assets/server-build-BtbLQkKd.js +0 -433
  477. package/codeyam-cli/templates/expo-react-native/app/(tabs)/_layout.tsx +0 -33
  478. package/codeyam-cli/templates/expo-react-native/app/(tabs)/index.tsx +0 -12
  479. package/codeyam-cli/templates/expo-react-native/app/(tabs)/settings.tsx +0 -12
@@ -2,9 +2,104 @@ var _a, _b;
2
2
  import { WebSocketServer, WebSocket } from 'ws';
3
3
  import crypto from 'crypto';
4
4
  import fs from 'fs';
5
+ import os from 'os';
5
6
  import path from 'path';
6
7
  import * as pty from 'node-pty';
7
8
  import { createMarkerTransformer } from "../utils/scenarioMarkers.js";
9
+ import { IdleDetector } from "./idleDetector.js";
10
+ import { mockStateEventEmitter } from "./mockStateEvents.js";
11
+ /**
12
+ * Strip the raw `claude ...` shell command from suppressed PTY output
13
+ * so it doesn't leak to the user in the fallback timeout path.
14
+ */
15
+ export function stripClaudeCommand(output) {
16
+ // Find the line containing the claude command and strip everything before
17
+ // and including it (shell prompt, ANSI codes, command echo). Preserve any
18
+ // real output that follows.
19
+ const idx = output.indexOf('claude ');
20
+ if (idx === -1)
21
+ return output;
22
+ // Find the end of the command line (\r or \n after "claude")
23
+ let end = idx;
24
+ while (end < output.length && output[end] !== '\r' && output[end] !== '\n') {
25
+ end++;
26
+ }
27
+ // Skip past trailing \r\n
28
+ while (end < output.length &&
29
+ (output[end] === '\r' || output[end] === '\n')) {
30
+ end++;
31
+ }
32
+ // Strip leading ANSI escape sequences from the remainder
33
+ // (e.g. bracketed paste mode toggles like \x1b[?2004l)
34
+ let rest = output.slice(end);
35
+ rest = rest.replace(/^(\x1b\[[\x20-\x3f]*[\x40-\x7e]|\x1b[()][A-Z0-9])*/, '');
36
+ return rest;
37
+ }
38
+ /**
39
+ * Detect whether a PTY output chunk contains Claude Code's welcome banner.
40
+ * Supports both the old box-drawing format (`╭─── Claude Code ───╮`)
41
+ * and the new block-art format (`▐▛███▜▌ Claude Code`).
42
+ */
43
+ export function isBannerChunk(data) {
44
+ // Strip ANSI escape sequences before checking — Claude Code's Ink renderer
45
+ // inserts SGR color codes between the banner characters (e.g. ▐\x1b[48;5;16m▛).
46
+ const plain = data.replace(/\x1b\[[0-9;?]*[A-Za-z]|\x1b\][^\x07]*\x07|\x1b[()][A-Z0-9]/g, '');
47
+ return plain.includes('╭') || plain.includes('▐▛');
48
+ }
49
+ /**
50
+ * Determine whether the startup output filter should be active.
51
+ * The filter strips the raw `claude --session-id ...` command from the
52
+ * terminal so the user never sees it.
53
+ */
54
+ export function shouldFilterStartup(opts) {
55
+ return opts.hasContext || opts.editorMode;
56
+ }
57
+ /**
58
+ * Manages PTY output suppression during Claude startup.
59
+ *
60
+ * Suppresses all output until the welcome banner arrives, then clears
61
+ * the screen. The claude command is stripped from any chunk that is forwarded
62
+ * (both the banner chunk on the normal path and the buffered output on the
63
+ * fallback timeout path).
64
+ */
65
+ export class StartupOutputFilter {
66
+ constructor() {
67
+ this.hasClearedScreen = false;
68
+ this.suppressedOutput = '';
69
+ }
70
+ /**
71
+ * Process a PTY data chunk. Returns an object describing what action to take:
72
+ * - `type: 'suppress'` — buffer the chunk, send nothing
73
+ * - `type: 'clear-and-send'` — clear screen, then send `data`
74
+ * - `type: 'send'` — send `data` as-is (post-startup)
75
+ */
76
+ processChunk(data) {
77
+ if (!this.hasClearedScreen && isBannerChunk(data)) {
78
+ this.hasClearedScreen = true;
79
+ const cleaned = stripClaudeCommand(data);
80
+ return { type: 'clear-and-send', data: cleaned };
81
+ }
82
+ if (!this.hasClearedScreen) {
83
+ this.suppressedOutput += data;
84
+ return { type: 'suppress' };
85
+ }
86
+ return { type: 'send', data };
87
+ }
88
+ /**
89
+ * Called when the fallback timeout fires (╭ never arrived).
90
+ * Returns the buffered output with the command stripped, or null if
91
+ * the screen was already cleared.
92
+ */
93
+ getTimeoutOutput() {
94
+ if (this.hasClearedScreen)
95
+ return null;
96
+ this.hasClearedScreen = true;
97
+ if (this.suppressedOutput.length === 0)
98
+ return null;
99
+ const cleaned = stripClaudeCommand(this.suppressedOutput);
100
+ return cleaned.length > 0 ? cleaned : null;
101
+ }
102
+ }
8
103
  // Use globalThis so the sessions Set is shared across module instances.
9
104
  // In Vite dev mode, this file is loaded twice: once by the Vite plugin (Node.js native)
10
105
  // and once by the SSR context (ssrLoadModule). Without globalThis, each instance
@@ -52,9 +147,25 @@ const ENV_PASSTHROUGH_KEYS = [
52
147
  'HOMEBREW_CELLAR',
53
148
  'HOMEBREW_REPOSITORY',
54
149
  ];
55
- function buildPtyEnv() {
150
+ // Additional env vars needed on Windows
151
+ const WINDOWS_ENV_PASSTHROUGH_KEYS = [
152
+ 'USERPROFILE',
153
+ 'APPDATA',
154
+ 'LOCALAPPDATA',
155
+ 'COMSPEC',
156
+ 'SystemRoot',
157
+ 'TEMP',
158
+ 'TMP',
159
+ // Nodist (Windows Node.js version manager)
160
+ 'NODIST_PREFIX',
161
+ 'NODIST_X64',
162
+ ];
163
+ export function buildPtyEnv(options) {
56
164
  const env = {};
57
- for (const key of ENV_PASSTHROUGH_KEYS) {
165
+ const keys = process.platform === 'win32'
166
+ ? [...ENV_PASSTHROUGH_KEYS, ...WINDOWS_ENV_PASSTHROUGH_KEYS]
167
+ : ENV_PASSTHROUGH_KEYS;
168
+ for (const key of keys) {
58
169
  const val = process.env[key];
59
170
  if (val != null)
60
171
  env[key] = val;
@@ -62,8 +173,56 @@ function buildPtyEnv() {
62
173
  // Always set TERM for proper terminal behavior
63
174
  if (!env.TERM)
64
175
  env.TERM = 'xterm-256color';
176
+ // Signal to hooks that this is an editor Build session.
177
+ // Without this, editor hooks fire in ALL Claude sessions in the project.
178
+ if (options?.editorMode) {
179
+ env.CODEYAM_EDITOR_ACTIVE = '1';
180
+ }
65
181
  return env;
66
182
  }
183
+ /**
184
+ * Returns the shell executable and arguments for PTY spawning,
185
+ * handling platform differences (Windows uses cmd.exe, Unix uses $SHELL).
186
+ * cmd.exe is preferred over PowerShell on Windows because:
187
+ * - .cmd shims run in-process (no window flash from subprocess spawning)
188
+ * - No execution policy restrictions blocking .ps1 scripts
189
+ * - Faster startup (no profile loading)
190
+ */
191
+ export function getShellConfig() {
192
+ if (process.platform === 'win32') {
193
+ return { shell: 'cmd.exe', args: [] };
194
+ }
195
+ const defaultShell = process.platform === 'darwin' ? '/bin/zsh' : '/bin/sh';
196
+ return {
197
+ shell: process.env.SHELL || defaultShell,
198
+ args: ['-l'],
199
+ };
200
+ }
201
+ /**
202
+ * Quote a shell argument for the current platform.
203
+ * Unix shells use single quotes; cmd.exe uses double quotes.
204
+ */
205
+ export function shellQuote(arg) {
206
+ if (process.platform === 'win32') {
207
+ return `"${arg.replace(/"/g, '""')}"`;
208
+ }
209
+ return `'${arg}'`;
210
+ }
211
+ /**
212
+ * Returns a safe working directory for the PTY, falling back through
213
+ * CODEYAM_ROOT_PATH → cwd → HOME/USERPROFILE → os.tmpdir().
214
+ */
215
+ export function getSafeCwd() {
216
+ const cwd = process.env.CODEYAM_ROOT_PATH || process.cwd();
217
+ if (fs.existsSync(cwd))
218
+ return cwd;
219
+ const home = process.platform === 'win32'
220
+ ? process.env.USERPROFILE || process.env.HOME
221
+ : process.env.HOME;
222
+ if (home && fs.existsSync(home))
223
+ return home;
224
+ return os.tmpdir();
225
+ }
67
226
  /**
68
227
  * Convert a name to a safe file name matching CodeYam's convention.
69
228
  * e.g. "Default Scenario" -> "Default_Scenario"
@@ -190,7 +349,7 @@ function writeDevModeContext(ctx) {
190
349
  // Server
191
350
  const serverPort = process.env.CODEYAM_PORT || '3111';
192
351
  lines.push('## Server');
193
- lines.push(`- **Refresh preview:** \`curl -X POST http://localhost:${serverPort}/api/dev-mode-preview\``);
352
+ lines.push(`- **Refresh preview:** \`curl -X POST http://localhost:${serverPort}/api/dev-mode-preview -H 'Content-Type: application/json' -d '{"dimension":"Desktop"}'\` (dimension is REQUIRED)`);
194
353
  lines.push('');
195
354
  fs.writeFileSync(contextPath, lines.join('\n'), 'utf8');
196
355
  return true;
@@ -227,21 +386,21 @@ function writeEditorModeContext(ctx) {
227
386
  `- **Proxy:** http://localhost:${proxyPort} (used by the browser iframe only — do NOT use for health checks)`,
228
387
  '',
229
388
  '## Workflow',
230
- '- **Guided steps:** Run `codeyam editor steps` for the next step (plan → prototype → confirm → deconstruct → extract → glossary → analyze → app scenarios → user scenarios → verify → journal → review → present)',
231
- '- Each feature follows 13 steps: `codeyam editor 1` through `codeyam editor 13`',
232
- '- Steps 1, 3, and 13 require user confirmation before proceeding',
389
+ '- **Guided steps:** Run `codeyam editor steps` for the next step (plan → prepare → prototype → verify prototype → confirm → deconstruct → extract → glossary → analyze → app scenarios → user scenarios → verify → journal → review → present → commit → finalize → push)',
390
+ '- Each feature follows 18 steps — run `codeyam editor steps` to see what to do next. Never expose `codeyam editor` commands to the user.',
391
+ '- **Run ONE step at a time.** Read the FULL output, complete every checklist item, then advance. NEVER batch-run steps in a loop or delegate them to a subagent.',
392
+ '- Steps 1, 5, 15, and 18 require user confirmation before proceeding',
233
393
  '',
234
394
  '## Preview Updates',
235
395
  '- **Refresh the preview frequently** — the user watches the preview as you work. Refresh after every meaningful change (new page, UI section, data seeding, styling), not just at the end.',
236
- `- **Refresh command:** \`curl -s -X POST http://localhost:${serverPort}/api/dev-mode-preview\``,
396
+ `- **Refresh command:** \`curl -s -X POST http://localhost:${serverPort}/api/dev-mode-preview -H 'Content-Type: application/json' -d '{"dimension":"Desktop"}'\` (dimension is REQUIRED)`,
237
397
  '- Aim for 4-8+ preview updates during a typical building session.',
238
398
  '',
239
399
  '## Verifying the Dev Server',
240
400
  '- Get the dev server URL: `curl -s http://localhost:' +
241
401
  serverPort +
242
402
  '/api/editor-dev-server` → look at the `"url"` field (e.g., `http://localhost:3000`)',
243
- '- Check the page loads: `curl -s -o /dev/null -w "%{http_code}" http://localhost:3000` (use the actual dev server URL, NOT the proxy)',
244
- '- Check API routes work: `curl -s http://localhost:3000/api/your-route` (should return JSON, not "Internal Server Error")',
403
+ '- Verify pages and API routes: `codeyam editor verify-routes \'{"paths":["/"],"apiRoutes":["/api/your-route"]}\'` (checks HTTP status and JSON validity)',
245
404
  '- **NEVER check localhost:' +
246
405
  proxyPort +
247
406
  ' for health** — the proxy returns 200 even when the app is broken',
@@ -249,11 +408,12 @@ function writeEditorModeContext(ctx) {
249
408
  '## API Endpoints',
250
409
  `- **Register scenario (auto-captures screenshot):** \`codeyam editor register '{"name":"...","url":"/page-to-screenshot","type":"application","seed":{...}}'\` — ALWAYS include "url" for the page to screenshot. For large payloads, write JSON to a file and use @ prefix: \`codeyam editor register @/tmp/scenario.json\``,
251
410
  `- **Get active scenario data:** \`curl http://localhost:${serverPort}/api/editor-scenario-data\``,
252
- `- **Refresh/navigate/switch preview:** \`curl -X POST http://localhost:${serverPort}/api/dev-mode-preview\` (no body = refresh; with \`{"path":"/route"}\` = navigate; with \`{"scenarioId":"..."}\` = switch scenario). Clears logs, waits for HMR, checks SSR health. This is the ONE command for all iframe preview control.`,
411
+ `- **Refresh/navigate/switch preview:** \`curl -X POST http://localhost:${serverPort}/api/dev-mode-preview -H 'Content-Type: application/json' -d '{"dimension":"Desktop"}'\` (dimension is REQUIRED in every call; add \`"path":"/route"\` to navigate; add \`"scenarioId":"..."\` to switch scenario). Clears logs, waits for HMR, checks SSR health.`,
253
412
  `- **Refresh config:** \`curl -X POST http://localhost:${serverPort}/api/editor-refresh\` (also starts dev server if a webapp is detected)`,
254
413
  `- **Dev server status:** \`curl http://localhost:${serverPort}/api/editor-dev-server\``,
255
414
  `- **Start dev server:** \`curl -X POST http://localhost:${serverPort}/api/editor-dev-server -H 'Content-Type: application/json' -d '{"action":"start"}'\``,
256
415
  `- **Restart dev server:** \`curl -X POST http://localhost:${serverPort}/api/editor-dev-server -H 'Content-Type: application/json' -d '{"action":"restart"}'\``,
416
+ `- **Feature complete (after push):** \`curl -s -X POST http://localhost:${serverPort}/api/editor-feature-complete\` — signals the UI that the feature is done. Call this after step 18 push/skip, then STOP.`,
257
417
  `- **Re-capture scenario screenshot:** \`curl -X POST http://localhost:${serverPort}/api/editor-capture-scenario -H 'Content-Type: application/json' -d '{"scenarioId":"...","url":"..."}'\` (for manual re-capture; register auto-captures)`,
258
418
  `- **Journal screenshot:** \`curl -X POST http://localhost:${serverPort}/api/editor-journal-screenshot -H 'Content-Type: application/json' -d '{"url":"...","filename":"..."}'\``,
259
419
  `- **Journal entry:** \`curl -X POST http://localhost:${serverPort}/api/editor-journal-entry -H 'Content-Type: application/json' -d '{"title":"...","type":"feature","description":"..."}'\``,
@@ -271,8 +431,10 @@ function writeEditorModeContext(ctx) {
271
431
  '- Create ONE isolation route page per component:',
272
432
  ' - **Remix:** `app/routes/isolated-components.ComponentName.tsx` → `/isolated-components/ComponentName`',
273
433
  ' - **Next.js:** `app/isolated-components/ComponentName/page.tsx` → `/isolated-components/ComponentName`',
434
+ ' - **Expo:** `app/isolated-components/ComponentName.tsx` → `/isolated-components/ComponentName` (use `useLocalSearchParams` from expo-router)',
274
435
  '- The route defines a `scenarios` object mapping scenario names to props, reads `?s=ScenarioName` from the URL, and renders the component',
275
- '- Wrap the component in a centered container: `<div style="display:flex;justify-content:center;align-items:center;min-height:100vh;padding:20px"><div style="width:100%;max-width:...">` — set max-width to match the component\'s real container (e.g. card in 3-col grid → 24rem)',
436
+ '- For web apps: Wrap the component in a centered container: `<div style="display:flex;justify-content:center;align-items:center;min-height:100vh"><div style="width:100%;max-width:...">` — set max-width to match the component\'s real container (e.g. card in 3-col grid → 24rem)',
437
+ '- For Expo: Wrap with `<View nativeID="codeyam-capture">` (nativeID maps to id on web for screenshot capture)',
276
438
  '- **Create multiple scenarios per component** (like tests): default/happy path, edge cases (empty data, long text, max items), different visual states (loading, error, disabled)',
277
439
  '- Register each scenario: `codeyam editor register \'{"name":"ComponentName - Scenario","componentName":"ComponentName","componentPath":"path/to/file.tsx","url":"/isolated-components/ComponentName?s=Scenario","mockData":{"routes":{"/api/...":{"body":[...]}}}}\'`',
278
440
  '- The url is a PATH, not a full URL — the proxy appends it and intercepts API calls the component makes',
@@ -292,9 +454,21 @@ function writeEditorModeContext(ctx) {
292
454
  const hasPackageJson = fs.existsSync(path.join(codeyamRoot, 'package.json'));
293
455
  if (!hasPackageJson) {
294
456
  lines.push('## Status');
295
- lines.push('- **Empty project** — no package.json found. Ask the user what they want to build.');
457
+ lines.push('- **Empty project** — no package.json found. Run `codeyam editor steps` to begin.');
296
458
  lines.push('');
297
459
  }
460
+ // Include handoff context if a provider switch generated one
461
+ try {
462
+ const handoffContextPath = path.join(codeyamRoot, '.codeyam', 'handoff-context.md');
463
+ if (fs.existsSync(handoffContextPath)) {
464
+ lines.push('');
465
+ lines.push(fs.readFileSync(handoffContextPath, 'utf8'));
466
+ lines.push('');
467
+ }
468
+ }
469
+ catch {
470
+ // Non-fatal
471
+ }
298
472
  fs.writeFileSync(contextPath, lines.join('\n'), 'utf8');
299
473
  return true;
300
474
  }
@@ -332,6 +506,12 @@ export function attachTerminalServer(httpServer) {
332
506
  }
333
507
  }, PING_INTERVAL_MS);
334
508
  globalThis.__codeyamPingInterval = pingInterval;
509
+ // Subscribe to mock state events and broadcast to all terminal clients
510
+ mockStateEventEmitter.on('event', (event) => {
511
+ if (event.type === 'data-mutation-forwarded') {
512
+ broadcastDataMutationForwarded(event.method, event.pathname);
513
+ }
514
+ });
335
515
  wss.on('connection', (ws, req) => {
336
516
  // Parse entity context from query params
337
517
  const url = new URL(req.url || '', `http://${req.headers.host}`);
@@ -345,6 +525,9 @@ export function attachTerminalServer(httpServer) {
345
525
  const projectSlug = url.searchParams.get('projectSlug') || '';
346
526
  const editorMode = url.searchParams.get('editorMode') === 'true';
347
527
  const reconnectId = url.searchParams.get('reconnectId') || '';
528
+ const claudeStartMode = url.searchParams.get('claudeStartMode') || '';
529
+ const claudeSessionId = url.searchParams.get('claudeSessionId') || '';
530
+ const editorStepLabel = url.searchParams.get('editorStepLabel') || '';
348
531
  // --- Reconnection: reattach to a detached PTY ---
349
532
  if (reconnectId && detachedPtys.has(reconnectId)) {
350
533
  const detached = detachedPtys.get(reconnectId);
@@ -371,13 +554,27 @@ export function attachTerminalServer(httpServer) {
371
554
  let detachedFlag = false;
372
555
  const reconnectPort = process.env.CODEYAM_PORT || '3111';
373
556
  const reconnectTransformMarkers = createMarkerTransformer(reconnectPort);
557
+ const reconnectIdleDetector = new IdleDetector({
558
+ onIdle: () => {
559
+ if (ws.readyState === WebSocket.OPEN) {
560
+ console.log('[terminalServer] Sending claude-idle (reconnected)');
561
+ ws.send(JSON.stringify({ type: 'claude-idle' }));
562
+ }
563
+ },
564
+ onActive: () => {
565
+ if (ws.readyState === WebSocket.OPEN) {
566
+ console.log('[terminalServer] Sending claude-active (reconnected)');
567
+ ws.send(JSON.stringify({ type: 'claude-active' }));
568
+ }
569
+ },
570
+ }, { activeOutputPattern: /[✶✢·◼]/ });
374
571
  detached.ptyProcess.onData((data) => {
375
572
  if (detachedFlag)
376
573
  return; // Superseded by a newer listener
377
574
  if (ws.readyState === WebSocket.OPEN) {
378
575
  if (!detached.hasClearedScreen &&
379
576
  detached.hasContext &&
380
- data.includes('╭')) {
577
+ isBannerChunk(data)) {
381
578
  detached.hasClearedScreen = true;
382
579
  ws.send(JSON.stringify({ type: 'output', data: '\x1b[2J\x1b[H' }));
383
580
  }
@@ -385,14 +582,12 @@ export function attachTerminalServer(httpServer) {
385
582
  if (transformed.length > 0) {
386
583
  ws.send(JSON.stringify({ type: 'output', data: transformed }));
387
584
  }
388
- const stripped = data.replace(/\x1b\[[0-9;]*[a-zA-Z]/g, '');
389
- if (stripped.includes('Editor Mode \u2014')) {
390
- console.log('[terminalServer] Detected claude-idle signal (reconnect)');
391
- ws.send(JSON.stringify({ type: 'claude-idle' }));
392
- }
585
+ reconnectIdleDetector.onPtyOutput(data);
393
586
  }
394
587
  });
395
588
  detached.ptyProcess.onExit(() => {
589
+ console.log('[terminalServer] PTY exited (reconnected), flushing idle state');
590
+ reconnectIdleDetector.disposeAndFlush();
396
591
  sessions.delete(session);
397
592
  if (ws.readyState === WebSocket.OPEN) {
398
593
  ws.close();
@@ -403,6 +598,7 @@ export function attachTerminalServer(httpServer) {
403
598
  try {
404
599
  const msg = JSON.parse(raw.toString());
405
600
  if (msg.type === 'input') {
601
+ reconnectIdleDetector.onUserInput();
406
602
  detached.ptyProcess.write(msg.data);
407
603
  }
408
604
  else if (msg.type === 'resize' && msg.cols && msg.rows) {
@@ -423,19 +619,17 @@ export function attachTerminalServer(httpServer) {
423
619
  return;
424
620
  }
425
621
  // --- New connection: spawn a fresh PTY ---
426
- const shell = process.env.SHELL || '/bin/zsh';
427
- const cwd = process.env.CODEYAM_ROOT_PATH || process.cwd();
428
- // Verify cwd exists, fall back to HOME
429
- const safeCwd = fs.existsSync(cwd) ? cwd : process.env.HOME || '/tmp';
622
+ const { shell, args: shellArgs } = getShellConfig();
623
+ const safeCwd = getSafeCwd();
430
624
  // Spawn PTY with a clean env (no giant JSON blobs from CODEYAM_PROJECT etc.)
431
625
  let ptyProcess;
432
626
  try {
433
- ptyProcess = pty.spawn(shell, ['-l'], {
627
+ ptyProcess = pty.spawn(shell, shellArgs, {
434
628
  name: 'xterm-256color',
435
629
  cols: 120,
436
630
  rows: 30,
437
631
  cwd: safeCwd,
438
- env: buildPtyEnv(),
632
+ env: buildPtyEnv({ editorMode }),
439
633
  });
440
634
  }
441
635
  catch (error) {
@@ -475,9 +669,41 @@ export function attachTerminalServer(httpServer) {
475
669
  projectSlug,
476
670
  });
477
671
  let hasClearedScreen = false;
478
- // Rolling buffer of trailing text to detect "Editor Mode —" across PTY chunks.
479
- let idleDetectBuffer = '';
480
- const IDLE_MARKER = 'Editor Mode \u2014';
672
+ const filterStartup = shouldFilterStartup({ hasContext, editorMode });
673
+ const startupFilter = filterStartup ? new StartupOutputFilter() : null;
674
+ // Safety timeout: stop suppressing output after 5s even if ╭ never arrives.
675
+ // Prevents "Starting Claude..." hanging forever if the banner format changes.
676
+ if (filterStartup) {
677
+ setTimeout(() => {
678
+ const output = startupFilter.getTimeoutOutput();
679
+ if (output !== null) {
680
+ hasClearedScreen = true;
681
+ console.log('[terminalServer] Suppression timeout — showing output');
682
+ if (ws.readyState === WebSocket.OPEN) {
683
+ // Clear the "Starting Claude..." message
684
+ ws.send(JSON.stringify({ type: 'output', data: '\x1b[2J\x1b[H' }));
685
+ const transformed = transformMarkers(output);
686
+ if (transformed.length > 0) {
687
+ ws.send(JSON.stringify({ type: 'output', data: transformed }));
688
+ }
689
+ }
690
+ }
691
+ }, 5000);
692
+ }
693
+ const idleDetector = new IdleDetector({
694
+ onIdle: () => {
695
+ if (ws.readyState === WebSocket.OPEN) {
696
+ console.log('[terminalServer] Sending claude-idle');
697
+ ws.send(JSON.stringify({ type: 'claude-idle' }));
698
+ }
699
+ },
700
+ onActive: () => {
701
+ if (ws.readyState === WebSocket.OPEN) {
702
+ console.log('[terminalServer] Sending claude-active');
703
+ ws.send(JSON.stringify({ type: 'claude-active' }));
704
+ }
705
+ },
706
+ }, { activeOutputPattern: /[✶✢·◼]/ });
481
707
  // PTY output -> WebSocket
482
708
  const serverPort = process.env.CODEYAM_PORT || '3111';
483
709
  // Stateful transformer that buffers partial {{scenario:...}} markers
@@ -485,35 +711,37 @@ export function attachTerminalServer(httpServer) {
485
711
  const transformMarkers = createMarkerTransformer(serverPort);
486
712
  ptyProcess.onData((data) => {
487
713
  if (ws.readyState === WebSocket.OPEN) {
488
- // When Claude Code starts (indicated by its box-drawing welcome banner),
489
- // clear the screen to hide the raw shell command.
490
- if (!hasClearedScreen && hasContext && data.includes('╭')) {
491
- hasClearedScreen = true;
492
- ws.send(JSON.stringify({ type: 'output', data: '\x1b[2J\x1b[H' }));
493
- }
494
- // Transform {{scenario:Name:ID}} markers into OSC 8 clickable hyperlinks.
495
- // Uses stateful buffering to handle markers split across PTY chunks.
496
- const transformed = transformMarkers(data);
497
- if (transformed.length > 0) {
498
- ws.send(JSON.stringify({ type: 'output', data: transformed }));
499
- }
500
- // Detect the Stop hook output — signals Claude is waiting for user input.
501
- // The hook outputs "Editor Mode —" when Claude finishes a turn.
502
- // Strip ANSI escape codes and check buffer+current to handle cross-chunk splits.
503
- const stripped = data.replace(/\x1b\[[0-9;]*[a-zA-Z]/g, '');
504
- const combined = idleDetectBuffer + stripped;
505
- if (combined.includes(IDLE_MARKER)) {
506
- console.log('[terminalServer] Detected claude-idle signal');
507
- ws.send(JSON.stringify({ type: 'claude-idle' }));
508
- idleDetectBuffer = '';
714
+ // During startup, suppress output until Claude's welcome banner (╭)
715
+ // arrives, then clear screen and strip the raw command from the chunk.
716
+ if (startupFilter && !hasClearedScreen) {
717
+ const result = startupFilter.processChunk(data);
718
+ if (result.type === 'clear-and-send') {
719
+ hasClearedScreen = true;
720
+ ws.send(JSON.stringify({ type: 'output', data: '\x1b[2J\x1b[H' }));
721
+ if (result.data.length > 0) {
722
+ const transformed = transformMarkers(result.data);
723
+ if (transformed.length > 0) {
724
+ ws.send(JSON.stringify({ type: 'output', data: transformed }));
725
+ }
726
+ }
727
+ idleDetector.onPtyOutput(data);
728
+ }
729
+ // type === 'suppress': skip sending and idle detection
509
730
  }
510
731
  else {
511
- // Keep trailing chars in case the marker spans two chunks
512
- idleDetectBuffer = stripped.slice(-(IDLE_MARKER.length - 1));
732
+ // Transform {{scenario:Name:ID}} markers into OSC 8 clickable hyperlinks.
733
+ // Uses stateful buffering to handle markers split across PTY chunks.
734
+ const transformed = transformMarkers(data);
735
+ if (transformed.length > 0) {
736
+ ws.send(JSON.stringify({ type: 'output', data: transformed }));
737
+ }
738
+ idleDetector.onPtyOutput(data);
513
739
  }
514
740
  }
515
741
  });
516
742
  ptyProcess.onExit(() => {
743
+ console.log('[terminalServer] PTY exited, flushing idle state');
744
+ idleDetector.disposeAndFlush();
517
745
  sessions.delete(session);
518
746
  // Also clean up from detachedPtys if it was detached
519
747
  if (detachedPtys.has(sessionId)) {
@@ -529,6 +757,7 @@ export function attachTerminalServer(httpServer) {
529
757
  try {
530
758
  const msg = JSON.parse(raw.toString());
531
759
  if (msg.type === 'input') {
760
+ idleDetector.onUserInput();
532
761
  ptyProcess.write(msg.data);
533
762
  }
534
763
  else if (msg.type === 'resize' && msg.cols && msg.rows) {
@@ -546,14 +775,57 @@ export function attachTerminalServer(httpServer) {
546
775
  ws.on('close', () => {
547
776
  detachSession(session, hasContext, hasClearedScreen);
548
777
  });
778
+ // Show a loading indicator while Claude Code starts up.
779
+ // PTY output is suppressed until Claude's welcome banner (╭) appears
780
+ // and triggers a screen clear, so the user never sees the raw command.
781
+ if (filterStartup) {
782
+ ws.send(JSON.stringify({
783
+ type: 'output',
784
+ data: '\x1b[2J\x1b[H\r\n\x1b[90m Starting Claude...\x1b[0m\r\n',
785
+ }));
786
+ }
549
787
  // Start Claude Code with the appropriate skill.
550
788
  // Using a skill avoids shell escaping issues and multi-line paste detection.
789
+ //
790
+ // Session recovery modes (editor only):
791
+ // resume + sessionId → claude --resume <uuid> (resumes exact conversation)
792
+ // resume, no sessionId → claude --continue (fallback for pre-existing sessions)
793
+ // fresh → generate UUID, write to .codeyam/claude-session-id.txt,
794
+ // launch claude --session-id <uuid> '/codeyam-editor'
795
+ //
796
+ // The UUID is generated HERE (not in printStep1) so it's available before
797
+ // Claude launches. printStep1 runs inside the session — too late.
551
798
  setTimeout(() => {
552
799
  if (editorMode && hasContext) {
553
- ptyProcess.write("claude '/codeyam-editor'\r");
800
+ if (claudeStartMode === 'resume' && claudeSessionId) {
801
+ const resumePrompt = editorStepLabel === 'Present'
802
+ ? 'The session was interrupted. Please rerun show-results.'
803
+ : 'The session was interrupted. Please continue where you left off.';
804
+ ptyProcess.write(`claude --resume ${shellQuote(claudeSessionId)} ${shellQuote(resumePrompt)}\r`);
805
+ }
806
+ else if (claudeStartMode === 'resume') {
807
+ // No session ID available — fall back to most recent session
808
+ const resumePrompt = editorStepLabel === 'Present'
809
+ ? 'The session was interrupted. Please rerun show-results.'
810
+ : 'The session was interrupted. Please continue where you left off.';
811
+ ptyProcess.write(`claude --continue ${shellQuote(resumePrompt)}\r`);
812
+ }
813
+ else {
814
+ // Fresh session: generate a trackable UUID
815
+ const newSessionId = crypto.randomUUID();
816
+ const sessionIdPath = path.join(safeCwd, '.codeyam', 'claude-session-id.txt');
817
+ try {
818
+ fs.mkdirSync(path.dirname(sessionIdPath), { recursive: true });
819
+ fs.writeFileSync(sessionIdPath, newSessionId, 'utf8');
820
+ }
821
+ catch (err) {
822
+ console.error('[terminalServer] Failed to write claude-session-id.txt:', err);
823
+ }
824
+ ptyProcess.write(`claude --session-id ${shellQuote(newSessionId)} ${shellQuote('/codeyam-editor')}\r`);
825
+ }
554
826
  }
555
827
  else if (hasContext) {
556
- ptyProcess.write("claude '/codeyam-dev-mode'\r");
828
+ ptyProcess.write(`claude ${shellQuote('/codeyam-dev-mode')}\r`);
557
829
  }
558
830
  else {
559
831
  ptyProcess.write('claude\r');
@@ -626,6 +898,31 @@ export function broadcastPreviewRefresh(path, scenarioId) {
626
898
  }
627
899
  return count;
628
900
  }
901
+ /**
902
+ * Notify all connected terminal WebSocket clients that a data mutation
903
+ * was forwarded to the real dev server (seed-based scenario).
904
+ * The Terminal component relays this to editor.tsx to show a save banner.
905
+ */
906
+ export function broadcastDataMutationForwarded(method, pathname) {
907
+ const msg = JSON.stringify({
908
+ type: 'data-mutation-forwarded',
909
+ method,
910
+ pathname,
911
+ });
912
+ let count = 0;
913
+ for (const session of sessions) {
914
+ try {
915
+ if (session.ws.readyState === WebSocket.OPEN) {
916
+ session.ws.send(msg);
917
+ count++;
918
+ }
919
+ }
920
+ catch {
921
+ // Ignore send errors
922
+ }
923
+ }
924
+ return count;
925
+ }
629
926
  /**
630
927
  * Show the results panel below the terminal in the Build tab.
631
928
  * Triggered by Claude at end of step 12.
@@ -665,6 +962,26 @@ export function broadcastHideResults() {
665
962
  }
666
963
  return count;
667
964
  }
965
+ /**
966
+ * Notify the editor UI that the current feature is complete (after push/skip).
967
+ * The UI shows a transition overlay letting the user start fresh or continue.
968
+ */
969
+ export function broadcastFeatureComplete() {
970
+ const msg = JSON.stringify({ type: 'feature-complete' });
971
+ let count = 0;
972
+ for (const session of sessions) {
973
+ try {
974
+ if (session.ws.readyState === WebSocket.OPEN) {
975
+ session.ws.send(msg);
976
+ count++;
977
+ }
978
+ }
979
+ catch {
980
+ // Ignore send errors
981
+ }
982
+ }
983
+ return count;
984
+ }
668
985
  /**
669
986
  * Send a set-viewport message to all connected terminal WebSocket clients.
670
987
  * The Terminal component relays this to the parent page to update the preview viewport.