@codeyam/codeyam-cli 0.1.0-staging.323686 → 0.1.0-staging.4813bf3

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 (444) 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 +5 -5
  4. package/analyzer-template/packages/ai/index.ts +7 -1
  5. package/analyzer-template/packages/ai/package.json +2 -2
  6. package/analyzer-template/packages/ai/src/lib/analyzeScope.ts +62 -18
  7. package/analyzer-template/packages/ai/src/lib/astScopes/astScopeAnalyzer.ts +67 -9
  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/dataStructure/ScopeDataStructure.ts +992 -249
  13. package/analyzer-template/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.ts +5 -1
  14. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.ts +16 -3
  15. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.ts +6 -4
  16. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.ts +31 -3
  17. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.ts +37 -15
  18. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.ts +70 -0
  19. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.ts +126 -11
  20. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.ts +179 -0
  21. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.ts +40 -30
  22. package/analyzer-template/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.ts +367 -96
  23. package/analyzer-template/packages/ai/src/lib/dataStructureChunking.ts +33 -15
  24. package/analyzer-template/packages/ai/src/lib/generateEntityDataStructure.ts +58 -3
  25. package/analyzer-template/packages/ai/src/lib/generateEntityScenarioData.ts +315 -6
  26. package/analyzer-template/packages/ai/src/lib/generateEntityScenarios.ts +9 -5
  27. package/analyzer-template/packages/ai/src/lib/generateExecutionFlows.ts +49 -5
  28. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromConditionalEffects.ts +1 -1
  29. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromConditionals.ts +649 -142
  30. package/analyzer-template/packages/ai/src/lib/generateExecutionFlowsFromJsxUsages.ts +1 -1
  31. package/analyzer-template/packages/ai/src/lib/isolateScopes.ts +51 -3
  32. package/analyzer-template/packages/ai/src/lib/mergeJsonTypeDefinitions.ts +5 -0
  33. package/analyzer-template/packages/ai/src/lib/mergeStatements.ts +90 -96
  34. package/analyzer-template/packages/ai/src/lib/promptGenerators/collapseNullableObjects.ts +118 -0
  35. package/analyzer-template/packages/ai/src/lib/promptGenerators/gatherAttributesMap.ts +10 -7
  36. package/analyzer-template/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.ts +24 -4
  37. package/analyzer-template/packages/ai/src/lib/resolvePathToControllable.ts +25 -13
  38. package/analyzer-template/packages/ai/src/lib/worker/SerializableDataStructure.ts +4 -3
  39. package/analyzer-template/packages/analyze/index.ts +2 -0
  40. package/analyzer-template/packages/analyze/src/lib/FileAnalyzer.ts +65 -59
  41. package/analyzer-template/packages/analyze/src/lib/ProjectAnalyzer.ts +113 -26
  42. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.ts +19 -0
  43. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.ts +19 -0
  44. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getAllExports.ts +11 -0
  45. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.ts +8 -0
  46. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.ts +49 -1
  47. package/analyzer-template/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.ts +2 -1
  48. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.ts +89 -9
  49. package/analyzer-template/packages/analyze/src/lib/files/analyze/analyzeEntities.ts +19 -4
  50. package/analyzer-template/packages/analyze/src/lib/files/analyze/gatherEntityMap.ts +4 -2
  51. package/analyzer-template/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.ts +0 -3
  52. package/analyzer-template/packages/analyze/src/lib/files/analyzeRemixRoute.ts +4 -5
  53. package/analyzer-template/packages/analyze/src/lib/files/getImportedExports.ts +14 -12
  54. package/analyzer-template/packages/analyze/src/lib/files/scenarios/TransformationTracer.ts +1315 -0
  55. package/analyzer-template/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.ts +61 -13
  56. package/analyzer-template/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.ts +37 -0
  57. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateDataStructure.ts +229 -19
  58. package/analyzer-template/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.ts +117 -9
  59. package/analyzer-template/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.ts +459 -39
  60. package/analyzer-template/packages/analyze/src/lib/files/scenarios/propagateArrayItemSchemas.ts +474 -0
  61. package/analyzer-template/packages/analyze/src/lib/files/setImportedExports.ts +2 -1
  62. package/analyzer-template/packages/analyze/src/lib/index.ts +1 -0
  63. package/analyzer-template/packages/analyze/src/lib/utils/getFileByPath.ts +19 -0
  64. package/analyzer-template/packages/aws/package.json +1 -1
  65. package/analyzer-template/packages/database/package.json +1 -1
  66. package/analyzer-template/packages/database/src/lib/analysisBranchToDb.ts +1 -1
  67. package/analyzer-template/packages/database/src/lib/analysisToDb.ts +1 -1
  68. package/analyzer-template/packages/database/src/lib/branchToDb.ts +1 -1
  69. package/analyzer-template/packages/database/src/lib/commitBranchToDb.ts +1 -1
  70. package/analyzer-template/packages/database/src/lib/commitToDb.ts +1 -1
  71. package/analyzer-template/packages/database/src/lib/fileToDb.ts +1 -1
  72. package/analyzer-template/packages/database/src/lib/kysely/db.ts +6 -0
  73. package/analyzer-template/packages/database/src/lib/kysely/tables/debugReportsTable.ts +1 -1
  74. package/analyzer-template/packages/database/src/lib/kysely/tables/labsRequestsTable.ts +52 -0
  75. package/analyzer-template/packages/database/src/lib/projectToDb.ts +1 -1
  76. package/analyzer-template/packages/database/src/lib/saveFiles.ts +1 -1
  77. package/analyzer-template/packages/database/src/lib/scenarioToDb.ts +1 -1
  78. package/analyzer-template/packages/database/src/lib/userScenarioToDb.ts +1 -1
  79. package/analyzer-template/packages/github/dist/database/src/lib/analysisBranchToDb.js +1 -1
  80. package/analyzer-template/packages/github/dist/database/src/lib/analysisBranchToDb.js.map +1 -1
  81. package/analyzer-template/packages/github/dist/database/src/lib/analysisToDb.js +1 -1
  82. package/analyzer-template/packages/github/dist/database/src/lib/analysisToDb.js.map +1 -1
  83. package/analyzer-template/packages/github/dist/database/src/lib/branchToDb.js +1 -1
  84. package/analyzer-template/packages/github/dist/database/src/lib/branchToDb.js.map +1 -1
  85. package/analyzer-template/packages/github/dist/database/src/lib/commitBranchToDb.js +1 -1
  86. package/analyzer-template/packages/github/dist/database/src/lib/commitBranchToDb.js.map +1 -1
  87. package/analyzer-template/packages/github/dist/database/src/lib/commitToDb.js +1 -1
  88. package/analyzer-template/packages/github/dist/database/src/lib/commitToDb.js.map +1 -1
  89. package/analyzer-template/packages/github/dist/database/src/lib/fileToDb.js +1 -1
  90. package/analyzer-template/packages/github/dist/database/src/lib/fileToDb.js.map +1 -1
  91. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts +2 -0
  92. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.d.ts.map +1 -1
  93. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js +3 -0
  94. package/analyzer-template/packages/github/dist/database/src/lib/kysely/db.js.map +1 -1
  95. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/debugReportsTable.d.ts +1 -1
  96. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.d.ts +23 -0
  97. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.d.ts.map +1 -0
  98. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.js +35 -0
  99. package/analyzer-template/packages/github/dist/database/src/lib/kysely/tables/labsRequestsTable.js.map +1 -0
  100. package/analyzer-template/packages/github/dist/database/src/lib/projectToDb.js +1 -1
  101. package/analyzer-template/packages/github/dist/database/src/lib/projectToDb.js.map +1 -1
  102. package/analyzer-template/packages/github/dist/database/src/lib/saveFiles.js +1 -1
  103. package/analyzer-template/packages/github/dist/database/src/lib/saveFiles.js.map +1 -1
  104. package/analyzer-template/packages/github/dist/database/src/lib/scenarioToDb.js +1 -1
  105. package/analyzer-template/packages/github/dist/database/src/lib/scenarioToDb.js.map +1 -1
  106. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts +7 -0
  107. package/analyzer-template/packages/github/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  108. package/analyzer-template/packages/github/dist/types/src/types/ScenariosDataStructure.d.ts +5 -5
  109. package/analyzer-template/packages/github/dist/types/src/types/ScenariosDataStructure.d.ts.map +1 -1
  110. package/analyzer-template/packages/github/dist/types/src/types/ScopeAnalysis.d.ts +6 -1
  111. package/analyzer-template/packages/github/dist/types/src/types/ScopeAnalysis.d.ts.map +1 -1
  112. package/analyzer-template/packages/github/package.json +1 -1
  113. package/analyzer-template/packages/types/src/types/ProjectMetadata.ts +7 -0
  114. package/analyzer-template/packages/types/src/types/ScenariosDataStructure.ts +6 -5
  115. package/analyzer-template/packages/types/src/types/ScopeAnalysis.ts +6 -1
  116. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts +7 -0
  117. package/analyzer-template/packages/utils/dist/types/src/types/ProjectMetadata.d.ts.map +1 -1
  118. package/analyzer-template/packages/utils/dist/types/src/types/ScenariosDataStructure.d.ts +5 -5
  119. package/analyzer-template/packages/utils/dist/types/src/types/ScenariosDataStructure.d.ts.map +1 -1
  120. package/analyzer-template/packages/utils/dist/types/src/types/ScopeAnalysis.d.ts +6 -1
  121. package/analyzer-template/packages/utils/dist/types/src/types/ScopeAnalysis.d.ts.map +1 -1
  122. package/analyzer-template/project/constructMockCode.ts +90 -10
  123. package/analyzer-template/project/writeMockDataTsx.ts +181 -8
  124. package/analyzer-template/project/writeScenarioComponents.ts +60 -12
  125. package/analyzer-template/project/writeSimpleRoot.ts +21 -11
  126. package/background/src/lib/local/createLocalAnalyzer.js +1 -1
  127. package/background/src/lib/local/createLocalAnalyzer.js.map +1 -1
  128. package/background/src/lib/virtualized/project/constructMockCode.js +75 -4
  129. package/background/src/lib/virtualized/project/constructMockCode.js.map +1 -1
  130. package/background/src/lib/virtualized/project/writeMockDataTsx.js +162 -4
  131. package/background/src/lib/virtualized/project/writeMockDataTsx.js.map +1 -1
  132. package/background/src/lib/virtualized/project/writeScenarioComponents.js +60 -15
  133. package/background/src/lib/virtualized/project/writeScenarioComponents.js.map +1 -1
  134. package/background/src/lib/virtualized/project/writeSimpleRoot.js +21 -11
  135. package/background/src/lib/virtualized/project/writeSimpleRoot.js.map +1 -1
  136. package/codeyam-cli/scripts/apply-setup.js +180 -0
  137. package/codeyam-cli/scripts/apply-setup.js.map +1 -1
  138. package/codeyam-cli/src/cli.js +2 -0
  139. package/codeyam-cli/src/cli.js.map +1 -1
  140. package/codeyam-cli/src/codeyam-cli.js +18 -2
  141. package/codeyam-cli/src/codeyam-cli.js.map +1 -1
  142. package/codeyam-cli/src/commands/analyze.js +4 -2
  143. package/codeyam-cli/src/commands/analyze.js.map +1 -1
  144. package/codeyam-cli/src/commands/baseline.js +2 -0
  145. package/codeyam-cli/src/commands/baseline.js.map +1 -1
  146. package/codeyam-cli/src/commands/debug.js +9 -5
  147. package/codeyam-cli/src/commands/debug.js.map +1 -1
  148. package/codeyam-cli/src/commands/default.js +31 -20
  149. package/codeyam-cli/src/commands/default.js.map +1 -1
  150. package/codeyam-cli/src/commands/detect-universal-mocks.js +2 -0
  151. package/codeyam-cli/src/commands/detect-universal-mocks.js.map +1 -1
  152. package/codeyam-cli/src/commands/init.js +49 -257
  153. package/codeyam-cli/src/commands/init.js.map +1 -1
  154. package/codeyam-cli/src/commands/memory.js +17 -26
  155. package/codeyam-cli/src/commands/memory.js.map +1 -1
  156. package/codeyam-cli/src/commands/recapture.js +2 -0
  157. package/codeyam-cli/src/commands/recapture.js.map +1 -1
  158. package/codeyam-cli/src/commands/setup-sandbox.js +2 -0
  159. package/codeyam-cli/src/commands/setup-sandbox.js.map +1 -1
  160. package/codeyam-cli/src/commands/setup-simulations.js +284 -0
  161. package/codeyam-cli/src/commands/setup-simulations.js.map +1 -0
  162. package/codeyam-cli/src/commands/test-startup.js +2 -0
  163. package/codeyam-cli/src/commands/test-startup.js.map +1 -1
  164. package/codeyam-cli/src/commands/verify.js +14 -2
  165. package/codeyam-cli/src/commands/verify.js.map +1 -1
  166. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js +128 -86
  167. package/codeyam-cli/src/utils/__tests__/setupClaudeCodeSettings.test.js.map +1 -1
  168. package/codeyam-cli/src/utils/analyzer.js +7 -0
  169. package/codeyam-cli/src/utils/analyzer.js.map +1 -1
  170. package/codeyam-cli/src/utils/backgroundServer.js +5 -0
  171. package/codeyam-cli/src/utils/backgroundServer.js.map +1 -1
  172. package/codeyam-cli/src/utils/generateReport.js +2 -2
  173. package/codeyam-cli/src/utils/install-skills.js +70 -45
  174. package/codeyam-cli/src/utils/install-skills.js.map +1 -1
  175. package/codeyam-cli/src/utils/labsAutoCheck.js +19 -0
  176. package/codeyam-cli/src/utils/labsAutoCheck.js.map +1 -0
  177. package/codeyam-cli/src/utils/progress.js +7 -0
  178. package/codeyam-cli/src/utils/progress.js.map +1 -1
  179. package/codeyam-cli/src/utils/queue/job.js +4 -0
  180. package/codeyam-cli/src/utils/queue/job.js.map +1 -1
  181. package/codeyam-cli/src/utils/requireSimulations.js +10 -0
  182. package/codeyam-cli/src/utils/requireSimulations.js.map +1 -0
  183. package/codeyam-cli/src/utils/ruleReflection/__tests__/confusionDetector.test.js +82 -0
  184. package/codeyam-cli/src/utils/ruleReflection/__tests__/confusionDetector.test.js.map +1 -0
  185. package/codeyam-cli/src/utils/ruleReflection/__tests__/contextBuilder.test.js +230 -0
  186. package/codeyam-cli/src/utils/ruleReflection/__tests__/contextBuilder.test.js.map +1 -0
  187. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/assertRules.js +67 -0
  188. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/assertRules.js.map +1 -0
  189. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/captureFixture.js +105 -0
  190. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/captureFixture.js.map +1 -0
  191. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/loadCapturedFixture.js +34 -0
  192. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/loadCapturedFixture.js.map +1 -0
  193. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/runClaude.js +162 -0
  194. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/runClaude.js.map +1 -0
  195. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/setupTempProject.js +75 -0
  196. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/helpers/setupTempProject.js.map +1 -0
  197. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/ruleReflectionE2E.test.js +378 -0
  198. package/codeyam-cli/src/utils/ruleReflection/__tests__/integration/ruleReflectionE2E.test.js.map +1 -0
  199. package/codeyam-cli/src/utils/ruleReflection/__tests__/promptBuilder.test.js +115 -0
  200. package/codeyam-cli/src/utils/ruleReflection/__tests__/promptBuilder.test.js.map +1 -0
  201. package/codeyam-cli/src/utils/ruleReflection/__tests__/transcriptParser.test.js +127 -0
  202. package/codeyam-cli/src/utils/ruleReflection/__tests__/transcriptParser.test.js.map +1 -0
  203. package/codeyam-cli/src/utils/ruleReflection/confusionDetector.js +50 -0
  204. package/codeyam-cli/src/utils/ruleReflection/confusionDetector.js.map +1 -0
  205. package/codeyam-cli/src/utils/ruleReflection/contextBuilder.js +116 -0
  206. package/codeyam-cli/src/utils/ruleReflection/contextBuilder.js.map +1 -0
  207. package/codeyam-cli/src/utils/ruleReflection/index.js +5 -0
  208. package/codeyam-cli/src/utils/ruleReflection/index.js.map +1 -0
  209. package/codeyam-cli/src/utils/ruleReflection/promptBuilder.js +44 -0
  210. package/codeyam-cli/src/utils/ruleReflection/promptBuilder.js.map +1 -0
  211. package/codeyam-cli/src/utils/ruleReflection/transcriptParser.js +85 -0
  212. package/codeyam-cli/src/utils/ruleReflection/transcriptParser.js.map +1 -0
  213. package/codeyam-cli/src/utils/ruleReflection/types.js +5 -0
  214. package/codeyam-cli/src/utils/ruleReflection/types.js.map +1 -0
  215. package/codeyam-cli/src/utils/rules/__tests__/ruleState.test.js +293 -0
  216. package/codeyam-cli/src/utils/rules/__tests__/ruleState.test.js.map +1 -0
  217. package/codeyam-cli/src/utils/rules/index.js +1 -0
  218. package/codeyam-cli/src/utils/rules/index.js.map +1 -1
  219. package/codeyam-cli/src/utils/rules/parser.js +2 -25
  220. package/codeyam-cli/src/utils/rules/parser.js.map +1 -1
  221. package/codeyam-cli/src/utils/rules/ruleState.js +150 -0
  222. package/codeyam-cli/src/utils/rules/ruleState.js.map +1 -0
  223. package/codeyam-cli/src/utils/rules/staleness.js +16 -11
  224. package/codeyam-cli/src/utils/rules/staleness.js.map +1 -1
  225. package/codeyam-cli/src/utils/serverState.js +37 -10
  226. package/codeyam-cli/src/utils/serverState.js.map +1 -1
  227. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js +21 -44
  228. package/codeyam-cli/src/utils/setupClaudeCodeSettings.js.map +1 -1
  229. package/codeyam-cli/src/webserver/app/lib/database.js +15 -3
  230. package/codeyam-cli/src/webserver/app/lib/database.js.map +1 -1
  231. package/codeyam-cli/src/webserver/backgroundServer.js +24 -0
  232. package/codeyam-cli/src/webserver/backgroundServer.js.map +1 -1
  233. package/codeyam-cli/src/webserver/build/client/assets/CopyButton-CA3JxPb7.js +1 -0
  234. package/codeyam-cli/src/webserver/build/client/assets/{EntityItem-DsN1wKrm.js → EntityItem-B86KKU7e.js} +1 -1
  235. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeBadge-DLqD3qNt.js → EntityTypeBadge-B5ctlSYt.js} +1 -1
  236. package/codeyam-cli/src/webserver/build/client/assets/{EntityTypeIcon-Ba2JVPzP.js → EntityTypeIcon-BqY8gDAW.js} +1 -1
  237. package/codeyam-cli/src/webserver/build/client/assets/{InlineSpinner-C8lyxW9k.js → InlineSpinner-ClaLpuOo.js} +1 -1
  238. package/codeyam-cli/src/webserver/build/client/assets/{InteractivePreview-aht4aafF.js → InteractivePreview-BDhPilK7.js} +2 -2
  239. package/codeyam-cli/src/webserver/build/client/assets/{LibraryFunctionPreview-CVtiBnY5.js → LibraryFunctionPreview-VeqEBv9v.js} +1 -1
  240. package/codeyam-cli/src/webserver/build/client/assets/{LoadingDots-B0GLXMsr.js → LoadingDots-Bs7Nn1Jr.js} +1 -1
  241. package/codeyam-cli/src/webserver/build/client/assets/{LogViewer-xgeCVgSM.js → LogViewer-Bm3PmcCz.js} +1 -1
  242. package/codeyam-cli/src/webserver/build/client/assets/{ReportIssueModal-OApQuNyq.js → ReportIssueModal-CgMEzchJ.js} +3 -8
  243. package/codeyam-cli/src/webserver/build/client/assets/{SafeScreenshot-DuDvi0jm.js → SafeScreenshot-Gq3Ocjo6.js} +1 -1
  244. package/codeyam-cli/src/webserver/build/client/assets/{ScenarioViewer-DzccYyI8.js → ScenarioViewer-CBui0id_.js} +2 -2
  245. package/codeyam-cli/src/webserver/build/client/assets/{TruncatedFilePath-DyFZkK0l.js → TruncatedFilePath-CiwXDxLh.js} +1 -1
  246. package/codeyam-cli/src/webserver/build/client/assets/{_index-BwqWJOgH.js → _index-B3TDXxnk.js} +1 -1
  247. package/codeyam-cli/src/webserver/build/client/assets/{activity.(_tab)-BwavGCpm.js → activity.(_tab)-BtBFH820.js} +6 -11
  248. package/codeyam-cli/src/webserver/build/client/assets/agent-transcripts-CN61MOMa.js +11 -0
  249. package/codeyam-cli/src/webserver/build/client/assets/api.agent-transcripts-l0sNRNKZ.js +1 -0
  250. package/codeyam-cli/src/webserver/build/client/assets/api.labs-unlock-l0sNRNKZ.js +1 -0
  251. package/codeyam-cli/src/webserver/build/client/assets/api.save-fixture-l0sNRNKZ.js +1 -0
  252. package/codeyam-cli/src/webserver/build/client/assets/book-open-PttOB2SF.js +6 -0
  253. package/codeyam-cli/src/webserver/build/client/assets/{chevron-down-Cx24_aWc.js → chevron-down-TJp6ofnp.js} +1 -1
  254. package/codeyam-cli/src/webserver/build/client/assets/{chunk-EPOLDU6W-CXRTFQ3F.js → chunk-JZWAC4HX-JE9ZIoBl.js} +12 -12
  255. package/codeyam-cli/src/webserver/build/client/assets/{circle-check-BOARzkeR.js → circle-check-CXhHQYrI.js} +1 -1
  256. package/codeyam-cli/src/webserver/build/client/assets/copy-6y9ALfGT.js +11 -0
  257. package/codeyam-cli/src/webserver/build/client/assets/{createLucideIcon-BdhJEx6B.js → createLucideIcon-Ca9fAY46.js} +1 -1
  258. package/codeyam-cli/src/webserver/build/client/assets/{dev.empty-BBnGWYga.js → dev.empty-C0epRiVn.js} +1 -1
  259. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha._-BJUiQqZF.js → entity._sha._-BVnB8a9L.js} +10 -10
  260. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha.scenarios._scenarioId.fullscreen-DavjRmOY.js → entity._sha.scenarios._scenarioId.fullscreen-CBoafmVs.js} +1 -1
  261. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha_.create-scenario-D1T4TGjf.js → entity._sha_.create-scenario-DGgZjdFg.js} +1 -1
  262. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha_.edit._scenarioId-CTBG2mmz.js → entity._sha_.edit._scenarioId-38yPijoD.js} +1 -1
  263. package/codeyam-cli/src/webserver/build/client/assets/{entry.client-CS2cb_eZ.js → entry.client-BSHEfydn.js} +1 -1
  264. package/codeyam-cli/src/webserver/build/client/assets/{fileTableUtils-DMJ7zii9.js → fileTableUtils-DCPhhSMo.js} +1 -1
  265. package/codeyam-cli/src/webserver/build/client/assets/{files-CJ6lTdTA.js → files-0N0YJQv7.js} +1 -1
  266. package/codeyam-cli/src/webserver/build/client/assets/{git-CPTZZ-JZ.js → git-DXnyr8uP.js} +1 -1
  267. package/codeyam-cli/src/webserver/build/client/assets/globals-CKT08Djd.css +1 -0
  268. package/codeyam-cli/src/webserver/build/client/assets/{index-lzqtyFU8.js → index-CcsFv748.js} +1 -1
  269. package/codeyam-cli/src/webserver/build/client/assets/{index-B1h680n5.js → index-ChN9-fAY.js} +1 -1
  270. package/codeyam-cli/src/webserver/build/client/assets/labs-BLJ7HxOC.js +1 -0
  271. package/codeyam-cli/src/webserver/build/client/assets/{loader-circle-B7B9V-bu.js → loader-circle-CTqLEAGU.js} +1 -1
  272. package/codeyam-cli/src/webserver/build/client/assets/manifest-b171b9d3.js +1 -0
  273. package/codeyam-cli/src/webserver/build/client/assets/memory-CCQd4aZA.js +78 -0
  274. package/codeyam-cli/src/webserver/build/client/assets/pause-D6vreykR.js +11 -0
  275. package/codeyam-cli/src/webserver/build/client/assets/root-CHhiHoo_.js +62 -0
  276. package/codeyam-cli/src/webserver/build/client/assets/{search-CxXUmBSd.js → search-B8VUL8nl.js} +1 -1
  277. package/codeyam-cli/src/webserver/build/client/assets/settings-BejnUJ6R.js +1 -0
  278. package/codeyam-cli/src/webserver/build/client/assets/{simulations-DwFIBT09.js → simulations-CPoAg7Zo.js} +1 -1
  279. package/codeyam-cli/src/webserver/build/client/assets/terminal-BrCP7uQo.js +11 -0
  280. package/codeyam-cli/src/webserver/build/client/assets/{triangle-alert-B6LgvRJg.js → triangle-alert-BZz2NjYa.js} +1 -1
  281. package/codeyam-cli/src/webserver/build/client/assets/{useCustomSizes-C1v1PQzo.js → useCustomSizes-DNwUduNu.js} +1 -1
  282. package/codeyam-cli/src/webserver/build/client/assets/{useLastLogLine-aSv48UbS.js → useLastLogLine-COky1GVF.js} +1 -1
  283. package/codeyam-cli/src/webserver/build/client/assets/{useReportContext-DYxHZQuP.js → useReportContext-CpZgwliL.js} +1 -1
  284. package/codeyam-cli/src/webserver/build/client/assets/{useToast-mBRpZPiu.js → useToast-Bv9JFvUO.js} +1 -1
  285. package/codeyam-cli/src/webserver/build/server/assets/{index-DVzYx8PN.js → index-8Fv-lH1-.js} +1 -1
  286. package/codeyam-cli/src/webserver/build/server/assets/server-build-Akn3iYFP.js +257 -0
  287. package/codeyam-cli/src/webserver/build/server/index.js +1 -1
  288. package/codeyam-cli/src/webserver/build-info.json +5 -5
  289. package/codeyam-cli/templates/{codeyam:debug.md → codeyam-debug.md} +1 -1
  290. package/codeyam-cli/templates/codeyam-diagnose.md +481 -0
  291. package/codeyam-cli/templates/codeyam-memory-hook.sh +19 -20
  292. package/codeyam-cli/templates/codeyam-memory.md +392 -0
  293. package/codeyam-cli/templates/codeyam-new-rule.md +13 -0
  294. package/codeyam-cli/templates/{codeyam:setup.md → codeyam-setup.md} +13 -1
  295. package/codeyam-cli/templates/{codeyam:sim.md → codeyam-sim.md} +1 -1
  296. package/codeyam-cli/templates/{codeyam:test.md → codeyam-test.md} +1 -1
  297. package/codeyam-cli/templates/{codeyam:verify.md → codeyam-verify.md} +1 -1
  298. package/codeyam-cli/templates/rule-notification-hook.py +56 -0
  299. package/codeyam-cli/templates/rule-reflection-hook.py +627 -0
  300. package/codeyam-cli/templates/rules-instructions.md +132 -0
  301. package/package.json +2 -2
  302. package/packages/ai/index.js +3 -2
  303. package/packages/ai/index.js.map +1 -1
  304. package/packages/ai/src/lib/analyzeScope.js +50 -13
  305. package/packages/ai/src/lib/analyzeScope.js.map +1 -1
  306. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js +54 -8
  307. package/packages/ai/src/lib/astScopes/astScopeAnalyzer.js.map +1 -1
  308. package/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.js +10 -14
  309. package/packages/ai/src/lib/astScopes/patterns/forInStatementHandler.js.map +1 -1
  310. package/packages/ai/src/lib/astScopes/processExpression.js +317 -44
  311. package/packages/ai/src/lib/astScopes/processExpression.js.map +1 -1
  312. package/packages/ai/src/lib/astScopes/sharedPatterns.js +25 -0
  313. package/packages/ai/src/lib/astScopes/sharedPatterns.js.map +1 -1
  314. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js +763 -171
  315. package/packages/ai/src/lib/dataStructure/ScopeDataStructure.js.map +1 -1
  316. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.js +5 -1
  317. package/packages/ai/src/lib/dataStructure/equivalencyManagers/frameworks/JavascriptFrameworkManager.js.map +1 -1
  318. package/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.js +13 -3
  319. package/packages/ai/src/lib/dataStructure/helpers/BatchSchemaProcessor.js.map +1 -1
  320. package/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.js +6 -4
  321. package/packages/ai/src/lib/dataStructure/helpers/ScopeTreeManager.js.map +1 -1
  322. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js +33 -3
  323. package/packages/ai/src/lib/dataStructure/helpers/cleanKnownObjectFunctions.js.map +1 -1
  324. package/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.js +36 -11
  325. package/packages/ai/src/lib/dataStructure/helpers/cleanNonObjectFunctions.js.map +1 -1
  326. package/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.js +63 -0
  327. package/packages/ai/src/lib/dataStructure/helpers/coerceObjectsToPrimitivesBySchema.js.map +1 -0
  328. package/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.js +113 -11
  329. package/packages/ai/src/lib/dataStructure/helpers/convertDotNotation.js.map +1 -1
  330. package/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.js +173 -0
  331. package/packages/ai/src/lib/dataStructure/helpers/convertTypeAnnotationsToValues.js.map +1 -0
  332. package/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.js +37 -20
  333. package/packages/ai/src/lib/dataStructure/helpers/deduplicateFunctionSchemas.js.map +1 -1
  334. package/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.js +309 -84
  335. package/packages/ai/src/lib/dataStructure/helpers/fillInSchemaGapsAndUnknowns.js.map +1 -1
  336. package/packages/ai/src/lib/dataStructureChunking.js +26 -11
  337. package/packages/ai/src/lib/dataStructureChunking.js.map +1 -1
  338. package/packages/ai/src/lib/generateEntityDataStructure.js +46 -2
  339. package/packages/ai/src/lib/generateEntityDataStructure.js.map +1 -1
  340. package/packages/ai/src/lib/generateEntityScenarioData.js +227 -4
  341. package/packages/ai/src/lib/generateEntityScenarioData.js.map +1 -1
  342. package/packages/ai/src/lib/generateEntityScenarios.js +7 -1
  343. package/packages/ai/src/lib/generateEntityScenarios.js.map +1 -1
  344. package/packages/ai/src/lib/generateExecutionFlows.js +26 -4
  345. package/packages/ai/src/lib/generateExecutionFlows.js.map +1 -1
  346. package/packages/ai/src/lib/generateExecutionFlowsFromConditionals.js +447 -80
  347. package/packages/ai/src/lib/generateExecutionFlowsFromConditionals.js.map +1 -1
  348. package/packages/ai/src/lib/isolateScopes.js +39 -3
  349. package/packages/ai/src/lib/isolateScopes.js.map +1 -1
  350. package/packages/ai/src/lib/mergeJsonTypeDefinitions.js +5 -0
  351. package/packages/ai/src/lib/mergeJsonTypeDefinitions.js.map +1 -1
  352. package/packages/ai/src/lib/mergeStatements.js +70 -51
  353. package/packages/ai/src/lib/mergeStatements.js.map +1 -1
  354. package/packages/ai/src/lib/promptGenerators/collapseNullableObjects.js +97 -0
  355. package/packages/ai/src/lib/promptGenerators/collapseNullableObjects.js.map +1 -0
  356. package/packages/ai/src/lib/promptGenerators/gatherAttributesMap.js +10 -4
  357. package/packages/ai/src/lib/promptGenerators/gatherAttributesMap.js.map +1 -1
  358. package/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.js +17 -2
  359. package/packages/ai/src/lib/promptGenerators/generateEntityScenarioDataGenerator.js.map +1 -1
  360. package/packages/ai/src/lib/resolvePathToControllable.js +24 -14
  361. package/packages/ai/src/lib/resolvePathToControllable.js.map +1 -1
  362. package/packages/ai/src/lib/worker/SerializableDataStructure.js.map +1 -1
  363. package/packages/analyze/index.js +1 -0
  364. package/packages/analyze/index.js.map +1 -1
  365. package/packages/analyze/src/lib/FileAnalyzer.js +60 -36
  366. package/packages/analyze/src/lib/FileAnalyzer.js.map +1 -1
  367. package/packages/analyze/src/lib/ProjectAnalyzer.js +96 -26
  368. package/packages/analyze/src/lib/ProjectAnalyzer.js.map +1 -1
  369. package/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.js +14 -0
  370. package/packages/analyze/src/lib/asts/sourceFiles/getAllDeclaredEntityNodes.js.map +1 -1
  371. package/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.js +14 -0
  372. package/packages/analyze/src/lib/asts/sourceFiles/getAllEntityNodes.js.map +1 -1
  373. package/packages/analyze/src/lib/asts/sourceFiles/getAllExports.js +6 -0
  374. package/packages/analyze/src/lib/asts/sourceFiles/getAllExports.js.map +1 -1
  375. package/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.js +6 -0
  376. package/packages/analyze/src/lib/asts/sourceFiles/getImportsAnalysis.js.map +1 -1
  377. package/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.js +39 -1
  378. package/packages/analyze/src/lib/asts/sourceFiles/getResolvedModule.js.map +1 -1
  379. package/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.js +2 -1
  380. package/packages/analyze/src/lib/asts/sourceFiles/getSourceFilesForAllImports.js.map +1 -1
  381. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js +65 -7
  382. package/packages/analyze/src/lib/files/analyze/analyzeEntities/prepareDataStructures.js.map +1 -1
  383. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js +17 -4
  384. package/packages/analyze/src/lib/files/analyze/analyzeEntities.js.map +1 -1
  385. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js +2 -1
  386. package/packages/analyze/src/lib/files/analyze/gatherEntityMap.js.map +1 -1
  387. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js +0 -3
  388. package/packages/analyze/src/lib/files/analyze/validateDependencyAnalyses.js.map +1 -1
  389. package/packages/analyze/src/lib/files/analyzeRemixRoute.js +3 -2
  390. package/packages/analyze/src/lib/files/analyzeRemixRoute.js.map +1 -1
  391. package/packages/analyze/src/lib/files/getImportedExports.js +11 -7
  392. package/packages/analyze/src/lib/files/getImportedExports.js.map +1 -1
  393. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js +880 -0
  394. package/packages/analyze/src/lib/files/scenarios/TransformationTracer.js.map +1 -0
  395. package/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.js +56 -10
  396. package/packages/analyze/src/lib/files/scenarios/enrichArrayTypesFromChildSignatures.js.map +1 -1
  397. package/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.js +33 -8
  398. package/packages/analyze/src/lib/files/scenarios/gatherDataForMocks.js.map +1 -1
  399. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js +150 -17
  400. package/packages/analyze/src/lib/files/scenarios/generateDataStructure.js.map +1 -1
  401. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js +56 -8
  402. package/packages/analyze/src/lib/files/scenarios/generateExecutionFlows.js.map +1 -1
  403. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js +399 -31
  404. package/packages/analyze/src/lib/files/scenarios/mergeInDependentDataStructure.js.map +1 -1
  405. package/packages/analyze/src/lib/files/setImportedExports.js +2 -1
  406. package/packages/analyze/src/lib/files/setImportedExports.js.map +1 -1
  407. package/packages/analyze/src/lib/index.js +1 -0
  408. package/packages/analyze/src/lib/index.js.map +1 -1
  409. package/packages/analyze/src/lib/utils/getFileByPath.js +12 -0
  410. package/packages/analyze/src/lib/utils/getFileByPath.js.map +1 -0
  411. package/packages/database/src/lib/analysisBranchToDb.js +1 -1
  412. package/packages/database/src/lib/analysisBranchToDb.js.map +1 -1
  413. package/packages/database/src/lib/analysisToDb.js +1 -1
  414. package/packages/database/src/lib/analysisToDb.js.map +1 -1
  415. package/packages/database/src/lib/branchToDb.js +1 -1
  416. package/packages/database/src/lib/branchToDb.js.map +1 -1
  417. package/packages/database/src/lib/commitBranchToDb.js +1 -1
  418. package/packages/database/src/lib/commitBranchToDb.js.map +1 -1
  419. package/packages/database/src/lib/commitToDb.js +1 -1
  420. package/packages/database/src/lib/commitToDb.js.map +1 -1
  421. package/packages/database/src/lib/fileToDb.js +1 -1
  422. package/packages/database/src/lib/fileToDb.js.map +1 -1
  423. package/packages/database/src/lib/kysely/db.js +3 -0
  424. package/packages/database/src/lib/kysely/db.js.map +1 -1
  425. package/packages/database/src/lib/kysely/tables/labsRequestsTable.js +35 -0
  426. package/packages/database/src/lib/kysely/tables/labsRequestsTable.js.map +1 -0
  427. package/packages/database/src/lib/projectToDb.js +1 -1
  428. package/packages/database/src/lib/projectToDb.js.map +1 -1
  429. package/packages/database/src/lib/saveFiles.js +1 -1
  430. package/packages/database/src/lib/saveFiles.js.map +1 -1
  431. package/packages/database/src/lib/scenarioToDb.js +1 -1
  432. package/packages/database/src/lib/scenarioToDb.js.map +1 -1
  433. package/scripts/finalize-analyzer.cjs +8 -76
  434. package/codeyam-cli/src/webserver/build/client/assets/copy-Bb-80kDT.js +0 -6
  435. package/codeyam-cli/src/webserver/build/client/assets/file-code-Dhef1kWN.js +0 -6
  436. package/codeyam-cli/src/webserver/build/client/assets/globals-D3yhhV8x.css +0 -1
  437. package/codeyam-cli/src/webserver/build/client/assets/manifest-7522edd4.js +0 -1
  438. package/codeyam-cli/src/webserver/build/client/assets/memory-yxFcrxBX.js +0 -92
  439. package/codeyam-cli/src/webserver/build/client/assets/root-eVAaavTS.js +0 -62
  440. package/codeyam-cli/src/webserver/build/client/assets/settings-CS5f3WzT.js +0 -1
  441. package/codeyam-cli/src/webserver/build/server/assets/server-build-4Cr0uToj.js +0 -257
  442. package/codeyam-cli/templates/codeyam:diagnose.md +0 -803
  443. package/codeyam-cli/templates/codeyam:memory.md +0 -462
  444. package/codeyam-cli/templates/codeyam:new-rule.md +0 -13
@@ -72,7 +72,11 @@ export class JavascriptFrameworkManager implements EquivalencyManager {
72
72
  }
73
73
 
74
74
  finalize(scopeNode: ScopeNode, scopeDataStructure: ScopeDataStructure) {
75
- cleanNonObjectFunctions(scopeNode, scopeDataStructure, true);
75
+ // Don't call cleanNonObjectFunctions with final=true here.
76
+ // The aggressive cleanup (transformKeyMapping, clearAttributes) removes method call
77
+ // evidence like .includes() before type inference can use it.
78
+ // Instead, defer final cleanup to toSerializable() when we're done with inference.
79
+ cleanNonObjectFunctions(scopeNode, scopeDataStructure, false);
76
80
  cleanKnownObjectFunctions(scopeNode, scopeDataStructure);
77
81
  this.handleUniformArrays(scopeNode, scopeDataStructure);
78
82
  }
@@ -41,6 +41,7 @@ export interface BatchProcessorStats {
41
41
 
42
42
  export class BatchSchemaProcessor {
43
43
  private workQueue: WorkItem[] = [];
44
+ private queueIndex = 0;
44
45
  private stats: BatchProcessorStats = {
45
46
  itemsProcessed: 0,
46
47
  itemsSkipped: 0, // Kept for API compatibility, always 0
@@ -71,17 +72,28 @@ export class BatchSchemaProcessor {
71
72
  * Check if there are more items to process.
72
73
  */
73
74
  hasWork(): boolean {
74
- return this.workQueue.length > 0;
75
+ return this.queueIndex < this.workQueue.length;
75
76
  }
76
77
 
77
78
  /**
78
79
  * Get the next work item from the queue.
79
80
  */
80
81
  getNextWork(): WorkItem | undefined {
81
- const item = this.workQueue.shift();
82
+ if (this.queueIndex >= this.workQueue.length) {
83
+ return undefined;
84
+ }
85
+
86
+ const item = this.workQueue[this.queueIndex++];
82
87
  if (item) {
83
88
  this.stats.itemsProcessed++;
84
89
  }
90
+
91
+ // Once we've consumed the queue, reset to avoid unbounded growth
92
+ if (this.queueIndex >= this.workQueue.length) {
93
+ this.workQueue = [];
94
+ this.queueIndex = 0;
95
+ }
96
+
85
97
  return item;
86
98
  }
87
99
 
@@ -122,6 +134,7 @@ export class BatchSchemaProcessor {
122
134
  */
123
135
  reset(): void {
124
136
  this.workQueue = [];
137
+ this.queueIndex = 0;
125
138
  this.stats = {
126
139
  itemsProcessed: 0,
127
140
  itemsSkipped: 0,
@@ -134,6 +147,6 @@ export class BatchSchemaProcessor {
134
147
  * Get current queue size (for debugging/monitoring).
135
148
  */
136
149
  getQueueSize(): number {
137
- return this.workQueue.length;
150
+ return this.workQueue.length - this.queueIndex;
138
151
  }
139
152
  }
@@ -117,9 +117,10 @@ export class ScopeTreeManager {
117
117
  */
118
118
  findNode(name: string): ScopeTreeNode | undefined {
119
119
  const queue: ScopeTreeNode[] = [this.tree];
120
+ let index = 0;
120
121
 
121
- while (queue.length > 0) {
122
- const node = queue.shift()!;
122
+ while (index < queue.length) {
123
+ const node = queue[index++]!;
123
124
  if (node.name === name) {
124
125
  return node;
125
126
  }
@@ -135,9 +136,10 @@ export class ScopeTreeManager {
135
136
  getAllNames(): string[] {
136
137
  const names: string[] = [];
137
138
  const queue: ScopeTreeNode[] = [this.tree];
139
+ let index = 0;
138
140
 
139
- while (queue.length > 0) {
140
- const node = queue.shift()!;
141
+ while (index < queue.length) {
142
+ const node = queue[index++]!;
141
143
  names.push(node.name);
142
144
  queue.push(...node.children);
143
145
  }
@@ -35,6 +35,24 @@ const getParts = (path: string): string[] => {
35
35
  return parts;
36
36
  };
37
37
 
38
+ function buildPrefixParts(parts: string[]): string[] {
39
+ if (parts.length === 0) return [];
40
+ const prefixes = new Array(parts.length);
41
+ let current = '';
42
+ for (let i = 0; i < parts.length; i++) {
43
+ const part = parts[i];
44
+ if (i === 0) {
45
+ current = part;
46
+ } else if (part.startsWith('[') || part.startsWith('(')) {
47
+ current += part;
48
+ } else {
49
+ current += `.${part}`;
50
+ }
51
+ prefixes[i] = current;
52
+ }
53
+ return prefixes;
54
+ }
55
+
38
56
  /**
39
57
  * Clear the module-level cache.
40
58
  * Call this between entity analyses to prevent unbounded memory growth.
@@ -133,6 +151,7 @@ function scrub(
133
151
 
134
152
  for (const keyPath of sortedKeys) {
135
153
  const keyParts = getParts(keyPath);
154
+ const prefixParts = buildPrefixParts(keyParts);
136
155
 
137
156
  if (
138
157
  IGNORE_CLASSES_AND_OBJECTS.find(
@@ -145,7 +164,7 @@ function scrub(
145
164
 
146
165
  // walk back through the prefixes: foo.bar.baz.qux → foo.bar.baz → foo.bar …
147
166
  for (let cut = keyParts.length - 1; cut > 0; --cut) {
148
- const prefix = joinParenthesesAndArrays(keyParts.slice(0, cut));
167
+ const prefix = prefixParts[cut - 1];
149
168
  const rootKind = rootTable.get(prefix);
150
169
  if (!rootKind) continue;
151
170
 
@@ -153,9 +172,18 @@ function scrub(
153
172
  const rawMethod = rawToken.split('(')[0];
154
173
 
155
174
  // Check if it's a prototype method (e.g., map, filter, toString)
175
+ // Only match when the token is actually a method reference/call:
176
+ // - Has parentheses (explicit call like map(), filter(x => x))
177
+ // - OR is typed as "function" (method reference like .split)
178
+ // Bare property names that happen to match method names
179
+ // (like .entries = "array", .values = "object") are preserved.
156
180
  if (protoMethodSets[rootKind].has(rawMethod)) {
157
- onRemove(keyPath);
158
- break;
181
+ const isMethodCall = rawToken.includes('(');
182
+ const isMethodReference = mapping[keyPath] === 'function';
183
+ if (isMethodCall || isMethodReference) {
184
+ onRemove(keyPath);
185
+ break;
186
+ }
159
187
  }
160
188
 
161
189
  // Check if it's a built-in property (e.g., length on arrays/strings)
@@ -178,17 +178,36 @@ export function clearCleanNonObjectFunctionsCache(): {
178
178
  // Maps parent prefix -> array of last parts (method names without parens)
179
179
  type PrefixIndex = Map<string, string[]>;
180
180
 
181
+ function buildPrefixPartsWithRoot(parts: string[]): string[] {
182
+ const prefixes = new Array(parts.length + 1);
183
+ prefixes[0] = '';
184
+ let current = '';
185
+ for (let i = 0; i < parts.length; i++) {
186
+ const part = parts[i];
187
+ if (i === 0) {
188
+ current = part;
189
+ } else if (part.startsWith('[') || part.startsWith('(')) {
190
+ current += part;
191
+ } else {
192
+ current += `.${part}`;
193
+ }
194
+ prefixes[i + 1] = current;
195
+ }
196
+ return prefixes;
197
+ }
198
+
181
199
  function buildPrefixIndex(allPaths: string[]): PrefixIndex {
182
200
  const index: PrefixIndex = new Map();
183
201
 
184
202
  for (const path of allPaths) {
185
203
  const parts = cachedSplit(path);
186
204
  if (parts.length < 1) continue;
205
+ const prefixes = buildPrefixPartsWithRoot(parts);
187
206
 
188
207
  // For each possible prefix length, add the last part to that prefix's entry
189
208
  // This allows lookup at any depth
190
209
  for (let prefixLen = 0; prefixLen < parts.length; prefixLen++) {
191
- const prefix = joinParenthesesAndArrays(parts.slice(0, prefixLen));
210
+ const prefix = prefixes[prefixLen];
192
211
  const lastPart = parts[prefixLen]?.split('(')?.[0];
193
212
  if (lastPart) {
194
213
  let existing = index.get(prefix);
@@ -491,7 +510,7 @@ function removeNonTransformingArrayMethods(variablePath: string): string {
491
510
  const part = keyParts[i];
492
511
 
493
512
  const arrayMethod = getMethodName(part);
494
- if (isValidArrayMethod(arrayMethod)) {
513
+ if (isValidFunctionCall(part) && isValidArrayMethod(arrayMethod)) {
495
514
  const nextPart = keyParts[i + 1];
496
515
  if (
497
516
  !nonTransformingArrayMethodsSet.has(arrayMethod) ||
@@ -531,7 +550,7 @@ function removeIsolatingArrayMethods(
531
550
  const part = keyParts[i];
532
551
 
533
552
  const arrayMethod = getMethodName(part);
534
- if (isValidArrayMethod(arrayMethod)) {
553
+ if (isValidFunctionCall(part) && isValidArrayMethod(arrayMethod)) {
535
554
  const nextPart = keyParts[i + 1];
536
555
  const prevPart = i > 1 ? keyParts[i - 1] : undefined;
537
556
  if (
@@ -1085,29 +1104,32 @@ function clearAttributes(
1085
1104
  knownBasePaths.add(path);
1086
1105
  }
1087
1106
 
1107
+ // OPTIMIZATION: Instead of O(keys × basePaths), use O(keys × keyDepth) where keyDepth is small
1108
+ // For each key, find all its prefixes and check if any is a known basePath (O(1) Set lookup)
1088
1109
  for (const key in mapping) {
1089
1110
  const keyParts = cachedSplit(key);
1090
- for (const basePath of knownBasePaths) {
1091
- const basePathParts = cachedSplit(basePath);
1092
- if (keyParts.length <= basePathParts.length) continue;
1093
- if (
1094
- joinParenthesesAndArrays(keyParts.slice(0, basePathParts.length)) !==
1095
- basePath
1096
- )
1097
- continue;
1111
+ if (keyParts.length <= 1) continue; // No possible parent basePath
1112
+
1113
+ // Check each prefix of the key to see if it's a known basePath
1114
+ // Start from shortest prefix (most likely to match) for early exit
1115
+ for (let prefixLen = 1; prefixLen < keyParts.length; prefixLen++) {
1116
+ const prefix = joinParenthesesAndArrays(keyParts.slice(0, prefixLen));
1117
+ if (!knownBasePaths.has(prefix)) continue;
1118
+
1119
+ // Found a matching basePath - now check the conditions
1120
+ const basePath = prefix;
1121
+
1098
1122
  // Preserve functionCallReturnValue for non-string types, UNLESS
1099
1123
  // the method is a primitive-returning method (like some, every, includes, indexOf)
1100
1124
  // whose return values are not mockable
1101
- const methodPart = keyParts[basePathParts.length];
1125
+ const methodPart = keyParts[prefixLen];
1102
1126
  const methodName = methodPart?.split('(')[0];
1103
1127
  const isPrimitiveReturning = primitiveReturningMethodsSet.has(
1104
1128
  methodName ?? '',
1105
1129
  );
1106
1130
  if (
1107
1131
  mapping[basePath] !== 'string' &&
1108
- keyParts[basePathParts.length + 1]?.startsWith(
1109
- 'functionCallReturnValue',
1110
- ) &&
1132
+ keyParts[prefixLen + 1]?.startsWith('functionCallReturnValue') &&
1111
1133
  !isPrimitiveReturning
1112
1134
  )
1113
1135
  continue;
@@ -0,0 +1,70 @@
1
+ /**
2
+ * Coerces object/array values to primitive types when the schema expects a primitive.
3
+ *
4
+ * The LLM sometimes generates an object or array where the schema expects a primitive
5
+ * type like "string", "number", or "boolean". For example:
6
+ * Schema: { body: "string" }
7
+ * LLM output: { body: { "env": "production" } }
8
+ *
9
+ * This causes runtime errors like `TypeError: body.match is not a function` because
10
+ * the code expects a string but gets an object.
11
+ *
12
+ * Coercion rules:
13
+ * - object/array → string: JSON.stringify
14
+ * - object/array → number: 0
15
+ * - object/array → boolean: false
16
+ *
17
+ * @param data The mock data object to coerce (mutated in place)
18
+ * @param schema The schema describing expected types
19
+ */
20
+ export default function coerceObjectsToPrimitivesBySchema<T>(
21
+ data: T,
22
+ schema: unknown,
23
+ ): T {
24
+ if (data === null || data === undefined) {
25
+ return data;
26
+ }
27
+
28
+ if (typeof data !== 'object') {
29
+ return data;
30
+ }
31
+
32
+ if (Array.isArray(data)) {
33
+ const itemSchema = Array.isArray(schema) ? schema[0] : undefined;
34
+ for (let i = 0; i < data.length; i++) {
35
+ data[i] = coerceObjectsToPrimitivesBySchema(data[i], itemSchema);
36
+ }
37
+ return data;
38
+ }
39
+
40
+ const dataObj = data as Record<string, unknown>;
41
+ const schemaObj = schema as Record<string, unknown> | undefined;
42
+
43
+ for (const key of Object.keys(dataObj)) {
44
+ const value = dataObj[key];
45
+ const fieldSchema = schemaObj?.[key];
46
+
47
+ if (value === null || value === undefined) {
48
+ continue;
49
+ }
50
+
51
+ if (typeof value === 'object' && typeof fieldSchema === 'string') {
52
+ // Schema expects a primitive but the value is an object/array.
53
+ // Coerce to the expected type.
54
+ const schemaType = fieldSchema.toLowerCase();
55
+
56
+ if (schemaType.includes('string')) {
57
+ dataObj[key] = JSON.stringify(value);
58
+ } else if (schemaType.includes('number')) {
59
+ dataObj[key] = 0;
60
+ } else if (schemaType.includes('boolean')) {
61
+ dataObj[key] = false;
62
+ }
63
+ } else if (typeof value === 'object' && value !== null) {
64
+ // Both value and schema are objects — recurse
65
+ coerceObjectsToPrimitivesBySchema(value, fieldSchema);
66
+ }
67
+ }
68
+
69
+ return data;
70
+ }
@@ -17,20 +17,98 @@ const DYNAMIC_LENGTH_RE = /\[\].*\.length$/;
17
17
  // Treat these as structural placeholders (don't commit them as concrete leaves)
18
18
  function isSkippableLeafType(t: string) {
19
19
  // 'unknown' by itself is a placeholder (but 'boolean | unknown' is not)
20
+ // Also handles optional variants like 'object | undefined', 'object | null',
21
+ // 'array | undefined', etc. — these are still structural placeholders that
22
+ // should not overwrite already-populated structures.
23
+ const parts = t.split('|').map((s) => s.trim());
24
+ const base = parts.filter((s) => s !== 'undefined' && s !== 'null');
20
25
  return (
21
- t === 'object' ||
22
- t === 'array' ||
23
- t === 'function' ||
24
- t.trim() === 'unknown'
26
+ base.length === 1 &&
27
+ (base[0] === 'object' ||
28
+ base[0] === 'array' ||
29
+ base[0] === 'function' ||
30
+ base[0] === 'unknown')
25
31
  );
26
32
  }
27
33
 
34
+ // Extract the base structural type from a potentially nullable type string.
35
+ // e.g., 'object | undefined' → 'object', 'array | null' → 'array'
36
+ function getBaseSkippableType(t: string): string {
37
+ const parts = t.split('|').map((s) => s.trim());
38
+ const base = parts.filter((s) => s !== 'undefined' && s !== 'null');
39
+ return base[0];
40
+ }
41
+
42
+ // Check if a type string has nullable annotations (| undefined or | null)
43
+ function isNullableType(t: string): boolean {
44
+ const parts = t.split('|').map((s) => s.trim());
45
+ return parts.includes('undefined') || parts.includes('null');
46
+ }
47
+
48
+ // Matches paths containing [][] — e.g., "items[][]" or "items[][].text"
49
+ const DOUBLE_ARRAY_RE = /\[\]\[\]/;
50
+
51
+ // Matches a [].property segment — e.g., "items[].name" (but not "items[][]")
52
+ const ARRAY_OBJECT_PROPERTY_RE = /\[\]\.[^[\]]/;
53
+
54
+ /**
55
+ * When scope analysis produces both `prefix[].property` (array of objects) and
56
+ * `prefix[][]` (array of arrays) for the same prefix, there's a conflict about
57
+ * what the array elements are. Object-property paths are more specific and come
58
+ * from actual property access; double-array paths come from mixed iteration
59
+ * patterns in the scope analysis. Drop the double-array paths when both exist.
60
+ */
61
+ function resolveDoubleArrayConflicts(
62
+ schema: Record<string, string>,
63
+ ): Record<string, string> {
64
+ // Collect prefixes that end in [] and have object property children
65
+ const objectArrayPrefixes = new Set<string>();
66
+ for (const path of Object.keys(schema)) {
67
+ if (!ARRAY_OBJECT_PROPERTY_RE.test(path)) continue;
68
+ // Extract every "...[]" prefix that is followed by ".prop"
69
+ let idx = 0;
70
+ while ((idx = path.indexOf('[].', idx)) !== -1) {
71
+ objectArrayPrefixes.add(path.slice(0, idx + 2)); // includes the "[]"
72
+ idx += 3;
73
+ }
74
+ }
75
+
76
+ if (objectArrayPrefixes.size === 0) return schema;
77
+
78
+ // Check if any double-array paths actually conflict
79
+ let hasConflict = false;
80
+ for (const path of Object.keys(schema)) {
81
+ if (!DOUBLE_ARRAY_RE.test(path)) continue;
82
+ const idx = path.indexOf('[][]');
83
+ if (idx !== -1 && objectArrayPrefixes.has(path.slice(0, idx + 2))) {
84
+ hasConflict = true;
85
+ break;
86
+ }
87
+ }
88
+
89
+ if (!hasConflict) return schema;
90
+
91
+ // Filter out conflicting double-array paths
92
+ const resolved: Record<string, string> = {};
93
+ for (const [path, value] of Object.entries(schema)) {
94
+ if (DOUBLE_ARRAY_RE.test(path)) {
95
+ const idx = path.indexOf('[][]');
96
+ if (idx !== -1 && objectArrayPrefixes.has(path.slice(0, idx + 2))) {
97
+ continue; // drop this conflicting path
98
+ }
99
+ }
100
+ resolved[path] = value;
101
+ }
102
+ return resolved;
103
+ }
104
+
28
105
  export default function convertDotNotation(
29
106
  schema: Record<string, string>,
30
107
  ): JsonTypeDefinition {
108
+ const resolved = resolveDoubleArrayConflicts(schema);
31
109
  const result: JsonTypeDefinition = {};
32
110
 
33
- for (const [rawPath, typ] of Object.entries(schema)) {
111
+ for (const [rawPath, typ] of Object.entries(resolved)) {
34
112
  // Skip paths with .length after array notation from dynamic iteration
35
113
  // e.g., metadata[].length, metadata[][].length
36
114
  if (DYNAMIC_LENGTH_RE.test(rawPath)) continue;
@@ -218,6 +296,17 @@ export default function convertDotNotation(
218
296
  // If the next segment after '[]' is another '[]' or a standalone index,
219
297
  // we need an array at cursor[key][0], not an object (for patterns like "key[][]" or "key[][0]")
220
298
  if (afterArray === '[]' || isStandaloneIndex(afterArray)) {
299
+ // But don't overwrite an existing populated object — object properties
300
+ // (from paths like key[].property) take precedence over double-array hints
301
+ // (from paths like key[][]) which come from mixed access patterns in scope analysis
302
+ if (
303
+ cursor[key][0] &&
304
+ typeof cursor[key][0] === 'object' &&
305
+ !Array.isArray(cursor[key][0]) &&
306
+ Object.keys(cursor[key][0]).length > 0
307
+ ) {
308
+ break;
309
+ }
221
310
  if (!Array.isArray(cursor[key][0])) cursor[key][0] = [];
222
311
  } else {
223
312
  // Default case: create an object
@@ -242,15 +331,41 @@ export default function convertDotNotation(
242
331
  cursor[key] = typ;
243
332
  } else {
244
333
  // Structural/placeholder terminal
245
- if (typ === 'array') {
334
+ const nullable = isNullableType(typ);
335
+ const baseType = getBaseSkippableType(typ);
336
+
337
+ if (baseType === 'array') {
246
338
  if (!Array.isArray(cursor[key])) cursor[key] = [];
339
+ if (nullable) {
340
+ (cursor[key] as any)._nullable = true;
341
+ }
247
342
  } else if (
248
- typ === 'object' ||
249
- typ === 'function' ||
250
- typ.trim() === 'unknown'
343
+ baseType === 'object' ||
344
+ baseType === 'function' ||
345
+ baseType === 'unknown'
251
346
  ) {
252
- if (cursor[key] === undefined) {
253
- cursor[key] = typ;
347
+ if (nullable) {
348
+ // Nullable object: ensure it's an actual object (not a string
349
+ // placeholder) so _nullable can be set and child paths can
350
+ // populate properties on it.
351
+ if (
352
+ cursor[key] === undefined ||
353
+ typeof cursor[key] === 'string'
354
+ ) {
355
+ cursor[key] = {};
356
+ }
357
+ if (
358
+ typeof cursor[key] === 'object' &&
359
+ cursor[key] !== null &&
360
+ !Array.isArray(cursor[key])
361
+ ) {
362
+ (cursor[key] as any)._nullable = true;
363
+ }
364
+ } else {
365
+ // Non-nullable: preserve existing behavior (string placeholder)
366
+ if (cursor[key] === undefined) {
367
+ cursor[key] = typ;
368
+ }
254
369
  }
255
370
  }
256
371
  }
@@ -0,0 +1,179 @@
1
+ /**
2
+ * Converts type annotation strings that appear as values in mockData to actual values.
3
+ *
4
+ * The LLM sometimes echoes the schema type annotation as the actual value.
5
+ * For example, if the schema says { filePath: "string | undefined" },
6
+ * the LLM might return { filePath: "string | undefined" } instead of
7
+ * generating an actual value like { filePath: "/path/to/file.tsx" }.
8
+ *
9
+ * This causes runtime errors when the code tries to use these "values"
10
+ * as actual data (e.g., calling .length on the string "string | undefined").
11
+ *
12
+ * This function detects type annotation strings and converts them to
13
+ * appropriate default values:
14
+ * - "string" → ""
15
+ * - "number" → 0
16
+ * - "boolean" → false
17
+ * - "function" → () => {}
18
+ * - "Date" → new Date()
19
+ * - "T | undefined" → undefined
20
+ * - "T | null" → null
21
+ * - "string[]" → []
22
+ */
23
+
24
+ // Pattern to detect if a string looks like a TypeScript type annotation
25
+ // This matches common type patterns:
26
+ // - Primitive types: string, number, boolean, function, Date, object, any, unknown
27
+ // - Union types: string | undefined, number | null, T | null | undefined
28
+ // - Array types: string[], number[], T[]
29
+ // - Optional types: T?
30
+ const TYPE_ANNOTATION_PATTERN =
31
+ /^(string|number|boolean|function|Date|object|any|unknown|void|never|null|undefined|array)(\s*\|\s*(string|number|boolean|function|Date|object|any|unknown|void|never|null|undefined|array))*(\[\])?$/i;
32
+
33
+ /**
34
+ * Check if a string value looks like a TypeScript type annotation.
35
+ */
36
+ function isTypeAnnotation(value: string): boolean {
37
+ if (typeof value !== 'string') return false;
38
+
39
+ const trimmed = value.trim();
40
+
41
+ // Quick check: type annotations are typically short
42
+ if (trimmed.length > 100) return false;
43
+
44
+ // Check against the pattern
45
+ if (TYPE_ANNOTATION_PATTERN.test(trimmed)) return true;
46
+
47
+ // Also check for common patterns that the regex might miss
48
+ const commonTypePatterns = [
49
+ 'string',
50
+ 'number',
51
+ 'boolean',
52
+ 'function',
53
+ 'object',
54
+ 'array',
55
+ 'Date',
56
+ 'any',
57
+ 'unknown',
58
+ 'void',
59
+ 'never',
60
+ 'null',
61
+ 'undefined',
62
+ 'string | undefined',
63
+ 'string | null',
64
+ 'string | null | undefined',
65
+ 'number | undefined',
66
+ 'number | null',
67
+ 'number | null | undefined',
68
+ 'boolean | undefined',
69
+ 'boolean | null',
70
+ 'boolean | null | undefined',
71
+ 'Date | undefined',
72
+ 'Date | null',
73
+ 'function | undefined',
74
+ 'object | undefined',
75
+ 'object | null',
76
+ 'string[]',
77
+ 'number[]',
78
+ 'boolean[]',
79
+ 'any[]',
80
+ 'string | number',
81
+ 'string | number | undefined',
82
+ ];
83
+
84
+ if (commonTypePatterns.includes(trimmed)) return true;
85
+
86
+ // Check for arrow function type annotations like "() => void", "(event: MouseEvent) => void"
87
+ // These contain '=>' but the body is a type keyword, not a JS expression
88
+ if (trimmed.includes('=>')) {
89
+ const arrowIndex = trimmed.indexOf('=>');
90
+ const body = trimmed.slice(arrowIndex + 2).trim();
91
+ if (
92
+ /^(void|string|number|boolean|any|unknown|never|null|undefined|object|Date|Promise<\w+>)$/i.test(
93
+ body,
94
+ )
95
+ ) {
96
+ return true;
97
+ }
98
+ }
99
+
100
+ return false;
101
+ }
102
+
103
+ /**
104
+ * Convert a type annotation string to an appropriate default value.
105
+ */
106
+ function typeAnnotationToValue(typeAnnotation: string): unknown {
107
+ const trimmed = typeAnnotation.trim().toLowerCase();
108
+
109
+ // Handle undefined types first (they take precedence)
110
+ if (trimmed.includes('| undefined') || trimmed.includes('undefined |')) {
111
+ return undefined;
112
+ }
113
+
114
+ // Handle null types
115
+ if (trimmed.includes('| null') || trimmed.includes('null |')) {
116
+ return null;
117
+ }
118
+
119
+ // Handle array types
120
+ if (trimmed.endsWith('[]') || trimmed === 'array') {
121
+ return [];
122
+ }
123
+
124
+ // Handle arrow function type annotations like "() => void"
125
+ if (trimmed.includes('=>')) return () => {};
126
+
127
+ // Handle primitive types
128
+ if (trimmed === 'string') return '';
129
+ if (trimmed === 'number') return 0;
130
+ if (trimmed === 'boolean') return false;
131
+ if (trimmed === 'function') return () => {};
132
+ if (trimmed === 'date') return new Date();
133
+ if (trimmed === 'object') return {};
134
+ if (trimmed === 'null') return null;
135
+ if (trimmed === 'undefined') return undefined;
136
+ if (trimmed === 'any' || trimmed === 'unknown') return undefined;
137
+ if (trimmed === 'void' || trimmed === 'never') return undefined;
138
+
139
+ // Handle union types - prefer the first concrete type
140
+ if (trimmed.includes(' | ')) {
141
+ const parts = trimmed.split(' | ').map((p) => p.trim());
142
+ for (const part of parts) {
143
+ if (part === 'string') return '';
144
+ if (part === 'number') return 0;
145
+ if (part === 'boolean') return false;
146
+ }
147
+ }
148
+
149
+ // Fallback: return undefined for unknown type patterns
150
+ return undefined;
151
+ }
152
+
153
+ /**
154
+ * Recursively convert type annotation strings to values in a mockData object.
155
+ * Mutates the object in place.
156
+ */
157
+ export default function convertTypeAnnotationsToValues(
158
+ mockData: Record<string, unknown>,
159
+ ): void {
160
+ for (const [key, value] of Object.entries(mockData)) {
161
+ if (typeof value === 'string' && isTypeAnnotation(value)) {
162
+ mockData[key] = typeAnnotationToValue(value);
163
+ } else if (Array.isArray(value)) {
164
+ // Recursively process array elements
165
+ for (const item of value) {
166
+ if (item !== null && typeof item === 'object' && !Array.isArray(item)) {
167
+ convertTypeAnnotationsToValues(item as Record<string, unknown>);
168
+ }
169
+ }
170
+ } else if (
171
+ value !== null &&
172
+ typeof value === 'object' &&
173
+ !Array.isArray(value)
174
+ ) {
175
+ // Recursively process nested objects
176
+ convertTypeAnnotationsToValues(value as Record<string, unknown>);
177
+ }
178
+ }
179
+ }