@soleri/core 2.1.0 → 2.5.0

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 (377) hide show
  1. package/dist/brain/brain.d.ts +10 -1
  2. package/dist/brain/brain.d.ts.map +1 -1
  3. package/dist/brain/brain.js +116 -13
  4. package/dist/brain/brain.js.map +1 -1
  5. package/dist/brain/intelligence.d.ts +36 -1
  6. package/dist/brain/intelligence.d.ts.map +1 -1
  7. package/dist/brain/intelligence.js +119 -14
  8. package/dist/brain/intelligence.js.map +1 -1
  9. package/dist/brain/types.d.ts +34 -2
  10. package/dist/brain/types.d.ts.map +1 -1
  11. package/dist/cognee/client.d.ts +3 -0
  12. package/dist/cognee/client.d.ts.map +1 -1
  13. package/dist/cognee/client.js +17 -0
  14. package/dist/cognee/client.js.map +1 -1
  15. package/dist/cognee/sync-manager.d.ts +94 -0
  16. package/dist/cognee/sync-manager.d.ts.map +1 -0
  17. package/dist/cognee/sync-manager.js +293 -0
  18. package/dist/cognee/sync-manager.js.map +1 -0
  19. package/dist/control/identity-manager.d.ts +22 -0
  20. package/dist/control/identity-manager.d.ts.map +1 -0
  21. package/dist/control/identity-manager.js +233 -0
  22. package/dist/control/identity-manager.js.map +1 -0
  23. package/dist/control/intent-router.d.ts +32 -0
  24. package/dist/control/intent-router.d.ts.map +1 -0
  25. package/dist/control/intent-router.js +242 -0
  26. package/dist/control/intent-router.js.map +1 -0
  27. package/dist/control/types.d.ts +68 -0
  28. package/dist/control/types.d.ts.map +1 -0
  29. package/dist/control/types.js +9 -0
  30. package/dist/control/types.js.map +1 -0
  31. package/dist/curator/curator.d.ts +37 -1
  32. package/dist/curator/curator.d.ts.map +1 -1
  33. package/dist/curator/curator.js +199 -1
  34. package/dist/curator/curator.js.map +1 -1
  35. package/dist/errors/classify.d.ts +13 -0
  36. package/dist/errors/classify.d.ts.map +1 -0
  37. package/dist/errors/classify.js +97 -0
  38. package/dist/errors/classify.js.map +1 -0
  39. package/dist/errors/index.d.ts +6 -0
  40. package/dist/errors/index.d.ts.map +1 -0
  41. package/dist/errors/index.js +4 -0
  42. package/dist/errors/index.js.map +1 -0
  43. package/dist/errors/retry.d.ts +40 -0
  44. package/dist/errors/retry.d.ts.map +1 -0
  45. package/dist/errors/retry.js +97 -0
  46. package/dist/errors/retry.js.map +1 -0
  47. package/dist/errors/types.d.ts +48 -0
  48. package/dist/errors/types.d.ts.map +1 -0
  49. package/dist/errors/types.js +59 -0
  50. package/dist/errors/types.js.map +1 -0
  51. package/dist/facades/types.d.ts +1 -1
  52. package/dist/governance/governance.d.ts +42 -0
  53. package/dist/governance/governance.d.ts.map +1 -0
  54. package/dist/governance/governance.js +488 -0
  55. package/dist/governance/governance.js.map +1 -0
  56. package/dist/governance/index.d.ts +3 -0
  57. package/dist/governance/index.d.ts.map +1 -0
  58. package/dist/governance/index.js +2 -0
  59. package/dist/governance/index.js.map +1 -0
  60. package/dist/governance/types.d.ts +102 -0
  61. package/dist/governance/types.d.ts.map +1 -0
  62. package/dist/governance/types.js +3 -0
  63. package/dist/governance/types.js.map +1 -0
  64. package/dist/index.d.ts +52 -3
  65. package/dist/index.d.ts.map +1 -1
  66. package/dist/index.js +47 -1
  67. package/dist/index.js.map +1 -1
  68. package/dist/intake/content-classifier.d.ts +14 -0
  69. package/dist/intake/content-classifier.d.ts.map +1 -0
  70. package/dist/intake/content-classifier.js +125 -0
  71. package/dist/intake/content-classifier.js.map +1 -0
  72. package/dist/intake/dedup-gate.d.ts +17 -0
  73. package/dist/intake/dedup-gate.d.ts.map +1 -0
  74. package/dist/intake/dedup-gate.js +66 -0
  75. package/dist/intake/dedup-gate.js.map +1 -0
  76. package/dist/intake/intake-pipeline.d.ts +63 -0
  77. package/dist/intake/intake-pipeline.d.ts.map +1 -0
  78. package/dist/intake/intake-pipeline.js +373 -0
  79. package/dist/intake/intake-pipeline.js.map +1 -0
  80. package/dist/intake/types.d.ts +65 -0
  81. package/dist/intake/types.d.ts.map +1 -0
  82. package/dist/intake/types.js +3 -0
  83. package/dist/intake/types.js.map +1 -0
  84. package/dist/intelligence/loader.js +1 -1
  85. package/dist/intelligence/loader.js.map +1 -1
  86. package/dist/intelligence/types.d.ts +3 -1
  87. package/dist/intelligence/types.d.ts.map +1 -1
  88. package/dist/logging/logger.d.ts +37 -0
  89. package/dist/logging/logger.d.ts.map +1 -0
  90. package/dist/logging/logger.js +145 -0
  91. package/dist/logging/logger.js.map +1 -0
  92. package/dist/logging/types.d.ts +19 -0
  93. package/dist/logging/types.d.ts.map +1 -0
  94. package/dist/logging/types.js +2 -0
  95. package/dist/logging/types.js.map +1 -0
  96. package/dist/loop/loop-manager.d.ts +100 -0
  97. package/dist/loop/loop-manager.d.ts.map +1 -0
  98. package/dist/loop/loop-manager.js +379 -0
  99. package/dist/loop/loop-manager.js.map +1 -0
  100. package/dist/loop/types.d.ts +103 -0
  101. package/dist/loop/types.d.ts.map +1 -0
  102. package/dist/loop/types.js +11 -0
  103. package/dist/loop/types.js.map +1 -0
  104. package/dist/persistence/index.d.ts +3 -0
  105. package/dist/persistence/index.d.ts.map +1 -0
  106. package/dist/persistence/index.js +2 -0
  107. package/dist/persistence/index.js.map +1 -0
  108. package/dist/persistence/sqlite-provider.d.ts +25 -0
  109. package/dist/persistence/sqlite-provider.d.ts.map +1 -0
  110. package/dist/persistence/sqlite-provider.js +59 -0
  111. package/dist/persistence/sqlite-provider.js.map +1 -0
  112. package/dist/persistence/types.d.ts +36 -0
  113. package/dist/persistence/types.d.ts.map +1 -0
  114. package/dist/persistence/types.js +8 -0
  115. package/dist/persistence/types.js.map +1 -0
  116. package/dist/planning/gap-analysis.d.ts +72 -0
  117. package/dist/planning/gap-analysis.d.ts.map +1 -0
  118. package/dist/planning/gap-analysis.js +442 -0
  119. package/dist/planning/gap-analysis.js.map +1 -0
  120. package/dist/planning/gap-types.d.ts +29 -0
  121. package/dist/planning/gap-types.d.ts.map +1 -0
  122. package/dist/planning/gap-types.js +28 -0
  123. package/dist/planning/gap-types.js.map +1 -0
  124. package/dist/planning/planner.d.ts +421 -4
  125. package/dist/planning/planner.d.ts.map +1 -1
  126. package/dist/planning/planner.js +949 -21
  127. package/dist/planning/planner.js.map +1 -1
  128. package/dist/playbooks/generic/brainstorming.d.ts +9 -0
  129. package/dist/playbooks/generic/brainstorming.d.ts.map +1 -0
  130. package/dist/playbooks/generic/brainstorming.js +105 -0
  131. package/dist/playbooks/generic/brainstorming.js.map +1 -0
  132. package/dist/playbooks/generic/code-review.d.ts +11 -0
  133. package/dist/playbooks/generic/code-review.d.ts.map +1 -0
  134. package/dist/playbooks/generic/code-review.js +176 -0
  135. package/dist/playbooks/generic/code-review.js.map +1 -0
  136. package/dist/playbooks/generic/subagent-execution.d.ts +9 -0
  137. package/dist/playbooks/generic/subagent-execution.d.ts.map +1 -0
  138. package/dist/playbooks/generic/subagent-execution.js +68 -0
  139. package/dist/playbooks/generic/subagent-execution.js.map +1 -0
  140. package/dist/playbooks/generic/systematic-debugging.d.ts +9 -0
  141. package/dist/playbooks/generic/systematic-debugging.d.ts.map +1 -0
  142. package/dist/playbooks/generic/systematic-debugging.js +87 -0
  143. package/dist/playbooks/generic/systematic-debugging.js.map +1 -0
  144. package/dist/playbooks/generic/tdd.d.ts +9 -0
  145. package/dist/playbooks/generic/tdd.d.ts.map +1 -0
  146. package/dist/playbooks/generic/tdd.js +70 -0
  147. package/dist/playbooks/generic/tdd.js.map +1 -0
  148. package/dist/playbooks/generic/verification.d.ts +9 -0
  149. package/dist/playbooks/generic/verification.d.ts.map +1 -0
  150. package/dist/playbooks/generic/verification.js +74 -0
  151. package/dist/playbooks/generic/verification.js.map +1 -0
  152. package/dist/playbooks/index.d.ts +4 -0
  153. package/dist/playbooks/index.d.ts.map +1 -0
  154. package/dist/playbooks/index.js +5 -0
  155. package/dist/playbooks/index.js.map +1 -0
  156. package/dist/playbooks/playbook-registry.d.ts +42 -0
  157. package/dist/playbooks/playbook-registry.d.ts.map +1 -0
  158. package/dist/playbooks/playbook-registry.js +227 -0
  159. package/dist/playbooks/playbook-registry.js.map +1 -0
  160. package/dist/playbooks/playbook-seeder.d.ts +47 -0
  161. package/dist/playbooks/playbook-seeder.d.ts.map +1 -0
  162. package/dist/playbooks/playbook-seeder.js +104 -0
  163. package/dist/playbooks/playbook-seeder.js.map +1 -0
  164. package/dist/playbooks/playbook-types.d.ts +132 -0
  165. package/dist/playbooks/playbook-types.d.ts.map +1 -0
  166. package/dist/playbooks/playbook-types.js +12 -0
  167. package/dist/playbooks/playbook-types.js.map +1 -0
  168. package/dist/project/project-registry.d.ts +79 -0
  169. package/dist/project/project-registry.d.ts.map +1 -0
  170. package/dist/project/project-registry.js +274 -0
  171. package/dist/project/project-registry.js.map +1 -0
  172. package/dist/project/types.d.ts +28 -0
  173. package/dist/project/types.d.ts.map +1 -0
  174. package/dist/project/types.js +5 -0
  175. package/dist/project/types.js.map +1 -0
  176. package/dist/prompts/index.d.ts +4 -0
  177. package/dist/prompts/index.d.ts.map +1 -0
  178. package/dist/prompts/index.js +3 -0
  179. package/dist/prompts/index.js.map +1 -0
  180. package/dist/prompts/parser.d.ts +17 -0
  181. package/dist/prompts/parser.d.ts.map +1 -0
  182. package/dist/prompts/parser.js +47 -0
  183. package/dist/prompts/parser.js.map +1 -0
  184. package/dist/prompts/template-manager.d.ts +25 -0
  185. package/dist/prompts/template-manager.d.ts.map +1 -0
  186. package/dist/prompts/template-manager.js +71 -0
  187. package/dist/prompts/template-manager.js.map +1 -0
  188. package/dist/prompts/types.d.ts +26 -0
  189. package/dist/prompts/types.d.ts.map +1 -0
  190. package/dist/prompts/types.js +5 -0
  191. package/dist/prompts/types.js.map +1 -0
  192. package/dist/runtime/admin-extra-ops.d.ts +15 -0
  193. package/dist/runtime/admin-extra-ops.d.ts.map +1 -0
  194. package/dist/runtime/admin-extra-ops.js +595 -0
  195. package/dist/runtime/admin-extra-ops.js.map +1 -0
  196. package/dist/runtime/admin-ops.d.ts +15 -0
  197. package/dist/runtime/admin-ops.d.ts.map +1 -0
  198. package/dist/runtime/admin-ops.js +329 -0
  199. package/dist/runtime/admin-ops.js.map +1 -0
  200. package/dist/runtime/capture-ops.d.ts +15 -0
  201. package/dist/runtime/capture-ops.d.ts.map +1 -0
  202. package/dist/runtime/capture-ops.js +363 -0
  203. package/dist/runtime/capture-ops.js.map +1 -0
  204. package/dist/runtime/cognee-sync-ops.d.ts +12 -0
  205. package/dist/runtime/cognee-sync-ops.d.ts.map +1 -0
  206. package/dist/runtime/cognee-sync-ops.js +55 -0
  207. package/dist/runtime/cognee-sync-ops.js.map +1 -0
  208. package/dist/runtime/core-ops.d.ts +9 -3
  209. package/dist/runtime/core-ops.d.ts.map +1 -1
  210. package/dist/runtime/core-ops.js +693 -10
  211. package/dist/runtime/core-ops.js.map +1 -1
  212. package/dist/runtime/curator-extra-ops.d.ts +9 -0
  213. package/dist/runtime/curator-extra-ops.d.ts.map +1 -0
  214. package/dist/runtime/curator-extra-ops.js +71 -0
  215. package/dist/runtime/curator-extra-ops.js.map +1 -0
  216. package/dist/runtime/domain-ops.d.ts.map +1 -1
  217. package/dist/runtime/domain-ops.js +61 -15
  218. package/dist/runtime/domain-ops.js.map +1 -1
  219. package/dist/runtime/grading-ops.d.ts +14 -0
  220. package/dist/runtime/grading-ops.d.ts.map +1 -0
  221. package/dist/runtime/grading-ops.js +105 -0
  222. package/dist/runtime/grading-ops.js.map +1 -0
  223. package/dist/runtime/intake-ops.d.ts +14 -0
  224. package/dist/runtime/intake-ops.d.ts.map +1 -0
  225. package/dist/runtime/intake-ops.js +110 -0
  226. package/dist/runtime/intake-ops.js.map +1 -0
  227. package/dist/runtime/loop-ops.d.ts +14 -0
  228. package/dist/runtime/loop-ops.d.ts.map +1 -0
  229. package/dist/runtime/loop-ops.js +251 -0
  230. package/dist/runtime/loop-ops.js.map +1 -0
  231. package/dist/runtime/memory-cross-project-ops.d.ts +12 -0
  232. package/dist/runtime/memory-cross-project-ops.d.ts.map +1 -0
  233. package/dist/runtime/memory-cross-project-ops.js +165 -0
  234. package/dist/runtime/memory-cross-project-ops.js.map +1 -0
  235. package/dist/runtime/memory-extra-ops.d.ts +13 -0
  236. package/dist/runtime/memory-extra-ops.d.ts.map +1 -0
  237. package/dist/runtime/memory-extra-ops.js +173 -0
  238. package/dist/runtime/memory-extra-ops.js.map +1 -0
  239. package/dist/runtime/orchestrate-ops.d.ts +17 -0
  240. package/dist/runtime/orchestrate-ops.d.ts.map +1 -0
  241. package/dist/runtime/orchestrate-ops.js +246 -0
  242. package/dist/runtime/orchestrate-ops.js.map +1 -0
  243. package/dist/runtime/planning-extra-ops.d.ts +25 -0
  244. package/dist/runtime/planning-extra-ops.d.ts.map +1 -0
  245. package/dist/runtime/planning-extra-ops.js +663 -0
  246. package/dist/runtime/planning-extra-ops.js.map +1 -0
  247. package/dist/runtime/playbook-ops.d.ts +14 -0
  248. package/dist/runtime/playbook-ops.d.ts.map +1 -0
  249. package/dist/runtime/playbook-ops.js +141 -0
  250. package/dist/runtime/playbook-ops.js.map +1 -0
  251. package/dist/runtime/project-ops.d.ts +15 -0
  252. package/dist/runtime/project-ops.d.ts.map +1 -0
  253. package/dist/runtime/project-ops.js +186 -0
  254. package/dist/runtime/project-ops.js.map +1 -0
  255. package/dist/runtime/runtime.d.ts.map +1 -1
  256. package/dist/runtime/runtime.js +65 -3
  257. package/dist/runtime/runtime.js.map +1 -1
  258. package/dist/runtime/types.d.ts +29 -0
  259. package/dist/runtime/types.d.ts.map +1 -1
  260. package/dist/runtime/vault-extra-ops.d.ts +10 -0
  261. package/dist/runtime/vault-extra-ops.d.ts.map +1 -0
  262. package/dist/runtime/vault-extra-ops.js +536 -0
  263. package/dist/runtime/vault-extra-ops.js.map +1 -0
  264. package/dist/telemetry/telemetry.d.ts +48 -0
  265. package/dist/telemetry/telemetry.d.ts.map +1 -0
  266. package/dist/telemetry/telemetry.js +87 -0
  267. package/dist/telemetry/telemetry.js.map +1 -0
  268. package/dist/vault/playbook.d.ts +34 -0
  269. package/dist/vault/playbook.d.ts.map +1 -0
  270. package/dist/vault/playbook.js +60 -0
  271. package/dist/vault/playbook.js.map +1 -0
  272. package/dist/vault/vault.d.ts +97 -4
  273. package/dist/vault/vault.d.ts.map +1 -1
  274. package/dist/vault/vault.js +424 -65
  275. package/dist/vault/vault.js.map +1 -1
  276. package/package.json +7 -3
  277. package/src/__tests__/admin-extra-ops.test.ts +467 -0
  278. package/src/__tests__/admin-ops.test.ts +271 -0
  279. package/src/__tests__/brain-intelligence.test.ts +205 -0
  280. package/src/__tests__/brain.test.ts +134 -3
  281. package/src/__tests__/capture-ops.test.ts +509 -0
  282. package/src/__tests__/cognee-integration.test.ts +80 -0
  283. package/src/__tests__/cognee-sync-manager.test.ts +103 -0
  284. package/src/__tests__/core-ops.test.ts +292 -2
  285. package/src/__tests__/curator-extra-ops.test.ts +381 -0
  286. package/src/__tests__/domain-ops.test.ts +66 -0
  287. package/src/__tests__/errors.test.ts +388 -0
  288. package/src/__tests__/governance.test.ts +522 -0
  289. package/src/__tests__/grading-ops.test.ts +361 -0
  290. package/src/__tests__/identity-manager.test.ts +243 -0
  291. package/src/__tests__/intake-pipeline.test.ts +162 -0
  292. package/src/__tests__/intent-router.test.ts +222 -0
  293. package/src/__tests__/logger.test.ts +200 -0
  294. package/src/__tests__/loop-ops.test.ts +469 -0
  295. package/src/__tests__/memory-cross-project-ops.test.ts +248 -0
  296. package/src/__tests__/memory-extra-ops.test.ts +352 -0
  297. package/src/__tests__/orchestrate-ops.test.ts +289 -0
  298. package/src/__tests__/persistence.test.ts +225 -0
  299. package/src/__tests__/planner.test.ts +416 -7
  300. package/src/__tests__/planning-extra-ops.test.ts +706 -0
  301. package/src/__tests__/playbook-registry.test.ts +326 -0
  302. package/src/__tests__/playbook-seeder.test.ts +163 -0
  303. package/src/__tests__/playbook.test.ts +389 -0
  304. package/src/__tests__/project-ops.test.ts +381 -0
  305. package/src/__tests__/template-manager.test.ts +222 -0
  306. package/src/__tests__/vault-extra-ops.test.ts +482 -0
  307. package/src/brain/brain.ts +185 -16
  308. package/src/brain/intelligence.ts +179 -10
  309. package/src/brain/types.ts +40 -2
  310. package/src/cognee/client.ts +18 -0
  311. package/src/cognee/sync-manager.ts +389 -0
  312. package/src/control/identity-manager.ts +354 -0
  313. package/src/control/intent-router.ts +326 -0
  314. package/src/control/types.ts +102 -0
  315. package/src/curator/curator.ts +295 -1
  316. package/src/errors/classify.ts +102 -0
  317. package/src/errors/index.ts +5 -0
  318. package/src/errors/retry.ts +132 -0
  319. package/src/errors/types.ts +81 -0
  320. package/src/governance/governance.ts +698 -0
  321. package/src/governance/index.ts +18 -0
  322. package/src/governance/types.ts +111 -0
  323. package/src/index.ts +213 -2
  324. package/src/intake/content-classifier.ts +146 -0
  325. package/src/intake/dedup-gate.ts +92 -0
  326. package/src/intake/intake-pipeline.ts +503 -0
  327. package/src/intake/types.ts +69 -0
  328. package/src/intelligence/loader.ts +1 -1
  329. package/src/intelligence/types.ts +3 -1
  330. package/src/logging/logger.ts +154 -0
  331. package/src/logging/types.ts +21 -0
  332. package/src/loop/loop-manager.ts +448 -0
  333. package/src/loop/types.ts +115 -0
  334. package/src/persistence/index.ts +7 -0
  335. package/src/persistence/sqlite-provider.ts +62 -0
  336. package/src/persistence/types.ts +44 -0
  337. package/src/planning/gap-analysis.ts +775 -0
  338. package/src/planning/gap-types.ts +61 -0
  339. package/src/planning/planner.ts +1273 -24
  340. package/src/playbooks/generic/brainstorming.ts +110 -0
  341. package/src/playbooks/generic/code-review.ts +181 -0
  342. package/src/playbooks/generic/subagent-execution.ts +74 -0
  343. package/src/playbooks/generic/systematic-debugging.ts +92 -0
  344. package/src/playbooks/generic/tdd.ts +75 -0
  345. package/src/playbooks/generic/verification.ts +79 -0
  346. package/src/playbooks/index.ts +27 -0
  347. package/src/playbooks/playbook-registry.ts +284 -0
  348. package/src/playbooks/playbook-seeder.ts +119 -0
  349. package/src/playbooks/playbook-types.ts +162 -0
  350. package/src/project/project-registry.ts +370 -0
  351. package/src/project/types.ts +31 -0
  352. package/src/prompts/index.ts +3 -0
  353. package/src/prompts/parser.ts +59 -0
  354. package/src/prompts/template-manager.ts +77 -0
  355. package/src/prompts/types.ts +28 -0
  356. package/src/runtime/admin-extra-ops.ts +652 -0
  357. package/src/runtime/admin-ops.ts +340 -0
  358. package/src/runtime/capture-ops.ts +404 -0
  359. package/src/runtime/cognee-sync-ops.ts +63 -0
  360. package/src/runtime/core-ops.ts +787 -9
  361. package/src/runtime/curator-extra-ops.ts +85 -0
  362. package/src/runtime/domain-ops.ts +67 -15
  363. package/src/runtime/grading-ops.ts +130 -0
  364. package/src/runtime/intake-ops.ts +126 -0
  365. package/src/runtime/loop-ops.ts +277 -0
  366. package/src/runtime/memory-cross-project-ops.ts +191 -0
  367. package/src/runtime/memory-extra-ops.ts +186 -0
  368. package/src/runtime/orchestrate-ops.ts +278 -0
  369. package/src/runtime/planning-extra-ops.ts +718 -0
  370. package/src/runtime/playbook-ops.ts +169 -0
  371. package/src/runtime/project-ops.ts +202 -0
  372. package/src/runtime/runtime.ts +77 -3
  373. package/src/runtime/types.ts +29 -0
  374. package/src/runtime/vault-extra-ops.ts +606 -0
  375. package/src/telemetry/telemetry.ts +118 -0
  376. package/src/vault/playbook.ts +87 -0
  377. package/src/vault/vault.ts +575 -98
@@ -0,0 +1,145 @@
1
+ /**
2
+ * Structured logger for Soleri agents.
3
+ *
4
+ * All output routes through stderr (stdout is reserved for MCP JSON-RPC).
5
+ * Supports optional file logging with daily rotation and 7-day retention.
6
+ *
7
+ * Ported from Salvador MCP's src/utils/logger.ts.
8
+ */
9
+ import { appendFileSync, mkdirSync, readdirSync, unlinkSync } from 'node:fs';
10
+ import { join } from 'node:path';
11
+ const LEVEL_ORDER = ['debug', 'info', 'warn', 'error'];
12
+ // MCP stdio servers must keep stdout clean for JSON-RPC.
13
+ // Route ALL log levels through stderr (console.error/console.warn).
14
+ const LEVEL_METHODS = {
15
+ debug: 'error',
16
+ info: 'error',
17
+ warn: 'warn',
18
+ error: 'error',
19
+ };
20
+ /** Max age for log files before auto-pruning (7 days) */
21
+ const LOG_RETENTION_DAYS = 7;
22
+ export class Logger {
23
+ prefix;
24
+ minLevel;
25
+ fileLogDir = null;
26
+ fileLogPrefix;
27
+ currentLogDate = null;
28
+ currentLogPath = null;
29
+ constructor(config) {
30
+ this.prefix = config?.prefix ?? '[Soleri]';
31
+ this.fileLogPrefix = 'agent';
32
+ const envLevel = process.env.SOLERI_LOG_LEVEL;
33
+ const level = config?.level ?? envLevel ?? 'info';
34
+ const levelIndex = LEVEL_ORDER.indexOf(level);
35
+ this.minLevel = levelIndex >= 0 ? levelIndex : LEVEL_ORDER.indexOf('info');
36
+ if (config?.fileLogDir) {
37
+ this.enableFileLog(config.fileLogDir);
38
+ }
39
+ }
40
+ /**
41
+ * Enable file-based logging. Output is teed to
42
+ * {dir}/{prefix}-YYYY-MM-DD.log in addition to stderr.
43
+ * Automatically prunes log files older than 7 days.
44
+ */
45
+ enableFileLog(dir, prefix) {
46
+ mkdirSync(dir, { recursive: true });
47
+ this.fileLogDir = dir;
48
+ if (prefix)
49
+ this.fileLogPrefix = prefix;
50
+ this.rotateFileIfNeeded();
51
+ this.pruneOldLogs();
52
+ }
53
+ debug(message, context) {
54
+ this.log('debug', message, context);
55
+ }
56
+ info(message, context) {
57
+ this.log('info', message, context);
58
+ }
59
+ warn(message, context) {
60
+ this.log('warn', message, context);
61
+ }
62
+ error(message, context) {
63
+ this.log('error', message, context);
64
+ }
65
+ log(level, message, context) {
66
+ if (LEVEL_ORDER.indexOf(level) < this.minLevel)
67
+ return;
68
+ const method = LEVEL_METHODS[level];
69
+ const tag = level.toUpperCase();
70
+ // Console output (stderr)
71
+ if (context) {
72
+ console[method](`${this.prefix}[${tag}] ${message}`, context);
73
+ }
74
+ else {
75
+ console[method](`${this.prefix}[${tag}] ${message}`);
76
+ }
77
+ // File output (if enabled)
78
+ if (this.fileLogDir) {
79
+ this.writeToFile(tag, message, context);
80
+ }
81
+ }
82
+ writeToFile(tag, message, context) {
83
+ this.rotateFileIfNeeded();
84
+ if (!this.currentLogPath)
85
+ return;
86
+ const ts = new Date().toISOString();
87
+ let line = `${ts} [${tag}] ${message}`;
88
+ if (context) {
89
+ try {
90
+ line += ' ' + JSON.stringify(context);
91
+ }
92
+ catch {
93
+ // Skip unserializable context
94
+ }
95
+ }
96
+ try {
97
+ appendFileSync(this.currentLogPath, line + '\n');
98
+ }
99
+ catch {
100
+ // Silently fail — file logging should never break the app
101
+ }
102
+ }
103
+ /** Switch to a new log file if the date has changed */
104
+ rotateFileIfNeeded() {
105
+ if (!this.fileLogDir)
106
+ return;
107
+ const today = new Date().toISOString().slice(0, 10); // YYYY-MM-DD
108
+ if (today !== this.currentLogDate) {
109
+ this.currentLogDate = today;
110
+ this.currentLogPath = join(this.fileLogDir, `${this.fileLogPrefix}-${today}.log`);
111
+ }
112
+ }
113
+ /** Remove log files older than LOG_RETENTION_DAYS */
114
+ pruneOldLogs() {
115
+ if (!this.fileLogDir)
116
+ return;
117
+ const prefix = this.fileLogPrefix;
118
+ try {
119
+ const cutoff = Date.now() - LOG_RETENTION_DAYS * 24 * 60 * 60 * 1000;
120
+ const files = readdirSync(this.fileLogDir);
121
+ for (const file of files) {
122
+ if (!file.startsWith(`${prefix}-`) || !file.endsWith('.log'))
123
+ continue;
124
+ const dateStr = file.slice(`${prefix}-`.length, -'.log'.length);
125
+ const fileDate = new Date(dateStr).getTime();
126
+ if (!isNaN(fileDate) && fileDate < cutoff) {
127
+ try {
128
+ unlinkSync(join(this.fileLogDir, file));
129
+ }
130
+ catch {
131
+ // Ignore individual file deletion errors
132
+ }
133
+ }
134
+ }
135
+ }
136
+ catch {
137
+ // Ignore pruning errors
138
+ }
139
+ }
140
+ }
141
+ /** Factory function for creating a logger. */
142
+ export function createLogger(config) {
143
+ return new Logger(config);
144
+ }
145
+ //# sourceMappingURL=logger.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/logging/logger.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ export type LogLevel = 'debug' | 'info' | 'warn' | 'error';
2
+ export interface LogContext {
3
+ [key: string]: unknown;
4
+ }
5
+ export interface LogEntry {
6
+ level: LogLevel;
7
+ message: string;
8
+ context?: LogContext;
9
+ timestamp: string;
10
+ }
11
+ export interface LoggerConfig {
12
+ /** Minimum log level. Default: 'info' (or SOLERI_LOG_LEVEL env var) */
13
+ level?: LogLevel;
14
+ /** Prefix for log messages. Default: '[Soleri]' */
15
+ prefix?: string;
16
+ /** Directory for file logging. If set, enables file output on construction. */
17
+ fileLogDir?: string;
18
+ }
19
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/logging/types.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,QAAQ,GAAG,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;AAE3D,MAAM,WAAW,UAAU;IACzB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,QAAQ;IACvB,KAAK,EAAE,QAAQ,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,UAAU,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,YAAY;IAC3B,uEAAuE;IACvE,KAAK,CAAC,EAAE,QAAQ,CAAC;IACjB,mDAAmD;IACnD,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,+EAA+E;IAC/E,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/logging/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,100 @@
1
+ /**
2
+ * Loop manager — iterative validation loop state tracking with
3
+ * output scanning and gate decision system.
4
+ *
5
+ * Ported from Salvador's loop.facade.ts with:
6
+ * - Promise tag extraction (<promise>...</promise>)
7
+ * - 5-tier heuristic completion detection
8
+ * - Gate decision pattern (allow/block) for Stop hook integration
9
+ * - Knowledge tracking for brain session recording
10
+ *
11
+ * Session-scoped (in-memory). Persistence is optional via external store.
12
+ */
13
+ import type { LoopConfig, LoopIteration, LoopMode, LoopState, LoopKnowledge, LoopHistoryEntry, LoopIterateDecision } from './types.js';
14
+ /**
15
+ * Extract text between <promise>...</promise> tags from output.
16
+ * Ported from Salvador's extractPromise.
17
+ */
18
+ export declare function extractPromise(text: string): string | null;
19
+ /**
20
+ * 5-tier heuristic completion detection.
21
+ * Returns a reason string if completion is detected, null otherwise.
22
+ * Ported from Salvador's detectImplicitCompletion.
23
+ *
24
+ * Tiers:
25
+ * 1. Validation tool score (token-migration, component-build)
26
+ * 2. Contrast mode — all PASS, no FAIL
27
+ * 3. Plan mode — grade meets target
28
+ * 4. Completion language + file modification signals (requires BOTH)
29
+ * 5. Test pass signals
30
+ */
31
+ export declare function detectImplicitCompletion(lastOutput: string, config: LoopConfig): string | null;
32
+ /**
33
+ * Detect anomalous loop iteration patterns.
34
+ * Flags fast + low-score combos that suggest the agent is spinning without making progress.
35
+ */
36
+ export declare function detectAnomaly(iteration: LoopIteration, mode: LoopMode): string | null;
37
+ export declare class LoopManager {
38
+ private activeLoop;
39
+ private completedLoops;
40
+ private historyEntries;
41
+ /**
42
+ * Start a new validation loop.
43
+ * Throws if a loop is already active.
44
+ */
45
+ startLoop(config: LoopConfig): LoopState;
46
+ /**
47
+ * Record an iteration result on the active loop (simple API).
48
+ * If the iteration passes, the loop status is NOT automatically changed —
49
+ * call completeLoop() explicitly when validation is confirmed.
50
+ * If max iterations reached and this iteration fails, status becomes 'max-iterations'.
51
+ */
52
+ iterate(result: {
53
+ validationScore?: number;
54
+ validationResult?: string;
55
+ passed: boolean;
56
+ }): LoopIteration;
57
+ /**
58
+ * Gate-based iteration — accepts LLM output and returns allow/block decision.
59
+ * Ported from Salvador's loop_iterate handler.
60
+ *
61
+ * This is the primary method for Stop hook integration:
62
+ * - Scans output for completion promise tags
63
+ * - Runs 5-tier heuristic completion detection
64
+ * - Returns 'block' with injected prompt/systemMessage for next iteration
65
+ * - Returns 'allow' when loop ends (completed, max_iterations)
66
+ *
67
+ * @param lastOutput - The LLM's last response to scan for completion signals
68
+ * @param knowledge - Optional knowledge items discovered during this iteration
69
+ */
70
+ iterateWithGate(lastOutput: string, knowledge?: LoopKnowledge, durationMs?: number): LoopIterateDecision;
71
+ /**
72
+ * Mark the active loop as completed (validation passed).
73
+ */
74
+ completeLoop(): LoopState;
75
+ /**
76
+ * Cancel the active loop.
77
+ */
78
+ cancelLoop(): LoopState;
79
+ /**
80
+ * Get current loop status. Returns null if no active loop.
81
+ */
82
+ getStatus(): LoopState | null;
83
+ /**
84
+ * Get history of all completed/cancelled/max-iterations loops.
85
+ */
86
+ getHistory(): LoopState[];
87
+ /**
88
+ * Get structured history entries (for brain session recording).
89
+ */
90
+ getHistoryEntries(): LoopHistoryEntry[];
91
+ /**
92
+ * Check if a loop is currently active.
93
+ */
94
+ isActive(): boolean;
95
+ /**
96
+ * Move active loop to history entries.
97
+ */
98
+ private moveToHistory;
99
+ }
100
+ //# sourceMappingURL=loop-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loop-manager.d.ts","sourceRoot":"","sources":["../../src/loop/loop-manager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,KAAK,EACV,UAAU,EACV,aAAa,EACb,QAAQ,EACR,SAAS,EACT,aAAa,EACb,gBAAgB,EAChB,mBAAmB,EACpB,MAAM,YAAY,CAAC;AAOpB;;;GAGG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAI1D;AAED;;;;;;;;;;;GAWG;AACH,wBAAgB,wBAAwB,CAAC,UAAU,EAAE,MAAM,EAAE,MAAM,EAAE,UAAU,GAAG,MAAM,GAAG,IAAI,CAwD9F;AAgBD;;;GAGG;AACH,wBAAgB,aAAa,CAAC,SAAS,EAAE,aAAa,EAAE,IAAI,EAAE,QAAQ,GAAG,MAAM,GAAG,IAAI,CAarF;AAID,qBAAa,WAAW;IACtB,OAAO,CAAC,UAAU,CAA0B;IAC5C,OAAO,CAAC,cAAc,CAAmB;IACzC,OAAO,CAAC,cAAc,CAA0B;IAEhD;;;OAGG;IACH,SAAS,CAAC,MAAM,EAAE,UAAU,GAAG,SAAS;IAoBxC;;;;;OAKG;IACH,OAAO,CAAC,MAAM,EAAE;QACd,eAAe,CAAC,EAAE,MAAM,CAAC;QACzB,gBAAgB,CAAC,EAAE,MAAM,CAAC;QAC1B,MAAM,EAAE,OAAO,CAAC;KACjB,GAAG,aAAa;IA8BjB;;;;;;;;;;;;OAYG;IACH,eAAe,CACb,UAAU,EAAE,MAAM,EAClB,SAAS,CAAC,EAAE,aAAa,EACzB,UAAU,CAAC,EAAE,MAAM,GAClB,mBAAmB;IA4ItB;;OAEG;IACH,YAAY,IAAI,SAAS;IAczB;;OAEG;IACH,UAAU,IAAI,SAAS;IAcvB;;OAEG;IACH,SAAS,IAAI,SAAS,GAAG,IAAI;IAI7B;;OAEG;IACH,UAAU,IAAI,SAAS,EAAE;IAIzB;;OAEG;IACH,iBAAiB,IAAI,gBAAgB,EAAE;IAIvC;;OAEG;IACH,QAAQ,IAAI,OAAO;IAInB;;OAEG;IACH,OAAO,CAAC,aAAa;CAatB"}
@@ -0,0 +1,379 @@
1
+ /**
2
+ * Loop manager — iterative validation loop state tracking with
3
+ * output scanning and gate decision system.
4
+ *
5
+ * Ported from Salvador's loop.facade.ts with:
6
+ * - Promise tag extraction (<promise>...</promise>)
7
+ * - 5-tier heuristic completion detection
8
+ * - Gate decision pattern (allow/block) for Stop hook integration
9
+ * - Knowledge tracking for brain session recording
10
+ *
11
+ * Session-scoped (in-memory). Persistence is optional via external store.
12
+ */
13
+ // ─── Grade ordering for plan-iteration mode ──────────────────────
14
+ const GRADE_ORDER = ['A+', 'A', 'B', 'C', 'D', 'F'];
15
+ // ─── Output Scanning ─────────────────────────────────────────────
16
+ /**
17
+ * Extract text between <promise>...</promise> tags from output.
18
+ * Ported from Salvador's extractPromise.
19
+ */
20
+ export function extractPromise(text) {
21
+ const match = /<promise>([\s\S]*?)<\/promise>/.exec(text);
22
+ if (!match)
23
+ return null;
24
+ return match[1].trim().replace(/\s+/g, ' ');
25
+ }
26
+ /**
27
+ * 5-tier heuristic completion detection.
28
+ * Returns a reason string if completion is detected, null otherwise.
29
+ * Ported from Salvador's detectImplicitCompletion.
30
+ *
31
+ * Tiers:
32
+ * 1. Validation tool score (token-migration, component-build)
33
+ * 2. Contrast mode — all PASS, no FAIL
34
+ * 3. Plan mode — grade meets target
35
+ * 4. Completion language + file modification signals (requires BOTH)
36
+ * 5. Test pass signals
37
+ */
38
+ export function detectImplicitCompletion(lastOutput, config) {
39
+ const text = lastOutput;
40
+ // 1. Validation tool score (token-migration, component-build)
41
+ if (config.mode === 'token-migration' || config.mode === 'component-build') {
42
+ const scoreMatch = /"score"\s*:\s*(\d+(?:\.\d+)?)/.exec(text);
43
+ if (scoreMatch) {
44
+ const score = parseFloat(scoreMatch[1]);
45
+ const target = config.targetScore ?? (config.mode === 'token-migration' ? 95 : 90);
46
+ if (score >= target) {
47
+ return `Auto-detected: validation score ${score} >= target ${target}`;
48
+ }
49
+ }
50
+ }
51
+ // 2. Contrast mode — all PASS, no FAIL
52
+ if (config.mode === 'contrast-fix') {
53
+ const hasPass = /\bPASS\b/i.test(text);
54
+ const hasFail = /\bFAIL\b/i.test(text);
55
+ if (hasPass && !hasFail) {
56
+ return 'Auto-detected: all contrast checks PASS, no FAIL';
57
+ }
58
+ }
59
+ // 3. Plan mode — grade meets target
60
+ if (config.mode === 'plan-iteration') {
61
+ const gradeMatch = /"grade"\s*:\s*"([A-F][+-]?)"/.exec(text);
62
+ if (gradeMatch) {
63
+ const grade = gradeMatch[1];
64
+ const target = config.targetGrade ?? 'A';
65
+ const gradeIdx = GRADE_ORDER.indexOf(grade);
66
+ const targetIdx = GRADE_ORDER.indexOf(target);
67
+ if (gradeIdx >= 0 && targetIdx >= 0 && gradeIdx <= targetIdx) {
68
+ return `Auto-detected: plan grade ${grade} >= target ${target}`;
69
+ }
70
+ }
71
+ }
72
+ // 4. Completion language + file modification signals (requires BOTH)
73
+ const completionPhrases = /\b(task complete|implementation finished|implementation complete|work complete|all done|changes complete|finished implementing|done implementing)\b/i;
74
+ const fileModPhrases = /\b(file modified|wrote to|saved to|created file|updated file|edited file|changes written)\b/i;
75
+ if (completionPhrases.test(text) && fileModPhrases.test(text)) {
76
+ return 'Auto-detected: completion language with file modification evidence';
77
+ }
78
+ // 5. Test pass signals
79
+ const testPass = /(\d+)\s+tests?\s+passed[,\s]+0\s+fail/i.test(text) ||
80
+ /\ball\s+tests?\s+(passing|passed)\b/i.test(text);
81
+ if (testPass) {
82
+ return 'Auto-detected: test suite passing';
83
+ }
84
+ return null;
85
+ }
86
+ // ─── Anomaly Detection ──────────────────────────────────────────
87
+ /**
88
+ * Minimum expected duration (ms) per mode.
89
+ * Iterations finishing faster than this with low scores are flagged as anomalous.
90
+ */
91
+ const MIN_DURATION_MS = {
92
+ 'token-migration': 5000,
93
+ 'contrast-fix': 2000,
94
+ 'component-build': 5000,
95
+ 'plan-iteration': 3000,
96
+ custom: 0,
97
+ };
98
+ /**
99
+ * Detect anomalous loop iteration patterns.
100
+ * Flags fast + low-score combos that suggest the agent is spinning without making progress.
101
+ */
102
+ export function detectAnomaly(iteration, mode) {
103
+ const minDuration = MIN_DURATION_MS[mode];
104
+ if (minDuration === 0)
105
+ return null; // custom mode — no threshold
106
+ const duration = iteration.durationMs ?? 0;
107
+ const score = iteration.validationScore ?? 0;
108
+ // Flag: iteration completed very fast with a low score
109
+ if (duration > 0 && duration < minDuration && !iteration.passed && score < 50) {
110
+ return `Anomaly: iteration ${iteration.iteration} completed in ${duration}ms (min expected: ${minDuration}ms) with score ${score} — possible no-op loop`;
111
+ }
112
+ return null;
113
+ }
114
+ // ─── LoopManager ─────────────────────────────────────────────────
115
+ export class LoopManager {
116
+ activeLoop = null;
117
+ completedLoops = [];
118
+ historyEntries = [];
119
+ /**
120
+ * Start a new validation loop.
121
+ * Throws if a loop is already active.
122
+ */
123
+ startLoop(config) {
124
+ if (this.activeLoop) {
125
+ throw new Error(`Loop already active: ${this.activeLoop.id} (mode: ${this.activeLoop.config.mode}). ` +
126
+ 'Cancel or complete it first.');
127
+ }
128
+ const loop = {
129
+ id: `loop-${Date.now()}`,
130
+ config,
131
+ iterations: [],
132
+ status: 'active',
133
+ startedAt: new Date().toISOString(),
134
+ };
135
+ this.activeLoop = loop;
136
+ return loop;
137
+ }
138
+ /**
139
+ * Record an iteration result on the active loop (simple API).
140
+ * If the iteration passes, the loop status is NOT automatically changed —
141
+ * call completeLoop() explicitly when validation is confirmed.
142
+ * If max iterations reached and this iteration fails, status becomes 'max-iterations'.
143
+ */
144
+ iterate(result) {
145
+ if (!this.activeLoop) {
146
+ throw new Error('No active loop. Start one first.');
147
+ }
148
+ const iteration = {
149
+ iteration: this.activeLoop.iterations.length + 1,
150
+ timestamp: new Date().toISOString(),
151
+ validationScore: result.validationScore,
152
+ validationResult: result.validationResult,
153
+ passed: result.passed,
154
+ };
155
+ this.activeLoop.iterations.push(iteration);
156
+ // Auto-transition to max-iterations if limit reached and not passing
157
+ if (!result.passed &&
158
+ this.activeLoop.iterations.length >= this.activeLoop.config.maxIterations) {
159
+ this.activeLoop.status = 'max-iterations';
160
+ this.activeLoop.completedAt = new Date().toISOString();
161
+ this.moveToHistory('max_iterations');
162
+ this.completedLoops.push(this.activeLoop);
163
+ this.activeLoop = null;
164
+ }
165
+ return iteration;
166
+ }
167
+ /**
168
+ * Gate-based iteration — accepts LLM output and returns allow/block decision.
169
+ * Ported from Salvador's loop_iterate handler.
170
+ *
171
+ * This is the primary method for Stop hook integration:
172
+ * - Scans output for completion promise tags
173
+ * - Runs 5-tier heuristic completion detection
174
+ * - Returns 'block' with injected prompt/systemMessage for next iteration
175
+ * - Returns 'allow' when loop ends (completed, max_iterations)
176
+ *
177
+ * @param lastOutput - The LLM's last response to scan for completion signals
178
+ * @param knowledge - Optional knowledge items discovered during this iteration
179
+ */
180
+ iterateWithGate(lastOutput, knowledge, durationMs) {
181
+ if (!this.activeLoop) {
182
+ return { decision: 'allow', reason: 'No active loop' };
183
+ }
184
+ const config = this.activeLoop.config;
185
+ // Accumulate knowledge
186
+ if (knowledge) {
187
+ if (!this.activeLoop.knowledge) {
188
+ this.activeLoop.knowledge = {};
189
+ }
190
+ if (knowledge.items) {
191
+ this.activeLoop.knowledge.items = [
192
+ ...(this.activeLoop.knowledge.items ?? []),
193
+ ...knowledge.items,
194
+ ];
195
+ }
196
+ if (knowledge.patternsApplied) {
197
+ this.activeLoop.knowledge.patternsApplied = [
198
+ ...(this.activeLoop.knowledge.patternsApplied ?? []),
199
+ ...knowledge.patternsApplied,
200
+ ];
201
+ }
202
+ if (knowledge.antiPatternsAvoided) {
203
+ this.activeLoop.knowledge.antiPatternsAvoided = [
204
+ ...(this.activeLoop.knowledge.antiPatternsAvoided ?? []),
205
+ ...knowledge.antiPatternsAvoided,
206
+ ];
207
+ }
208
+ }
209
+ // Check for completion promise
210
+ if (config.completionPromise) {
211
+ const promiseText = extractPromise(lastOutput);
212
+ if (promiseText && promiseText === config.completionPromise) {
213
+ const iterCount = this.activeLoop.iterations.length;
214
+ this.activeLoop.status = 'completed';
215
+ this.activeLoop.completedAt = new Date().toISOString();
216
+ this.moveToHistory('completed');
217
+ this.completedLoops.push(this.activeLoop);
218
+ this.activeLoop = null;
219
+ return {
220
+ decision: 'allow',
221
+ reason: `Loop completed — promise "${config.completionPromise}" detected`,
222
+ outcome: 'completed',
223
+ iteration: iterCount,
224
+ };
225
+ }
226
+ }
227
+ // Heuristic completion detection
228
+ const autoReason = detectImplicitCompletion(lastOutput, config);
229
+ if (autoReason) {
230
+ const iterCount = this.activeLoop.iterations.length;
231
+ this.activeLoop.status = 'completed';
232
+ this.activeLoop.completedAt = new Date().toISOString();
233
+ this.moveToHistory('completed');
234
+ this.completedLoops.push(this.activeLoop);
235
+ this.activeLoop = null;
236
+ return {
237
+ decision: 'allow',
238
+ reason: autoReason,
239
+ outcome: 'completed',
240
+ iteration: iterCount,
241
+ autoCompleted: true,
242
+ };
243
+ }
244
+ // Check max iterations
245
+ if (config.maxIterations > 0 && this.activeLoop.iterations.length >= config.maxIterations) {
246
+ const iterCount = this.activeLoop.iterations.length;
247
+ this.activeLoop.status = 'max-iterations';
248
+ this.activeLoop.completedAt = new Date().toISOString();
249
+ this.moveToHistory('max_iterations');
250
+ this.completedLoops.push(this.activeLoop);
251
+ this.activeLoop = null;
252
+ return {
253
+ decision: 'allow',
254
+ reason: `Max iterations (${config.maxIterations}) reached`,
255
+ outcome: 'max_iterations',
256
+ iteration: iterCount,
257
+ };
258
+ }
259
+ // Continue loop — increment iteration
260
+ const nextIteration = this.activeLoop.iterations.length + 1;
261
+ const iterationEntry = {
262
+ iteration: nextIteration,
263
+ timestamp: new Date().toISOString(),
264
+ passed: false,
265
+ ...(durationMs !== undefined && { durationMs }),
266
+ };
267
+ this.activeLoop.iterations.push(iterationEntry);
268
+ // Run anomaly detection
269
+ const anomalyWarning = detectAnomaly(iterationEntry, config.mode);
270
+ // Build validation hint for system message
271
+ let validationHint = '';
272
+ switch (config.mode) {
273
+ case 'token-migration':
274
+ validationHint = `Run validation (tokens check). Target: score >= ${config.targetScore ?? 95}`;
275
+ break;
276
+ case 'contrast-fix':
277
+ validationHint = 'Run contrast check on all color pairs. Target: all PASS';
278
+ break;
279
+ case 'component-build':
280
+ validationHint = `Run validation (full check). Target: score >= ${config.targetScore ?? 90}`;
281
+ break;
282
+ case 'plan-iteration':
283
+ validationHint = `Run plan grading. Target: grade >= ${config.targetGrade ?? 'A'}`;
284
+ break;
285
+ case 'custom':
286
+ validationHint = 'Complete the task and validate your work';
287
+ break;
288
+ }
289
+ const maxDisplay = config.maxIterations > 0 ? String(config.maxIterations) : 'unlimited';
290
+ const systemMessage = `[Loop — Iteration ${nextIteration}/${maxDisplay} | Mode: ${config.mode}] ${validationHint}` +
291
+ (config.completionPromise
292
+ ? ` | Output <promise>${config.completionPromise}</promise> ONLY when validation passes`
293
+ : '');
294
+ // Build full prompt with validation instructions
295
+ const fullPrompt = config.validationInstructions
296
+ ? `${config.prompt}\n\n${config.validationInstructions}`
297
+ : config.prompt;
298
+ return {
299
+ decision: 'block',
300
+ reason: fullPrompt,
301
+ prompt: fullPrompt,
302
+ systemMessage,
303
+ iteration: nextIteration,
304
+ ...(anomalyWarning && { anomalyWarning }),
305
+ };
306
+ }
307
+ /**
308
+ * Mark the active loop as completed (validation passed).
309
+ */
310
+ completeLoop() {
311
+ if (!this.activeLoop) {
312
+ throw new Error('No active loop to complete.');
313
+ }
314
+ this.activeLoop.status = 'completed';
315
+ this.activeLoop.completedAt = new Date().toISOString();
316
+ const completed = this.activeLoop;
317
+ this.moveToHistory('completed');
318
+ this.completedLoops.push(completed);
319
+ this.activeLoop = null;
320
+ return completed;
321
+ }
322
+ /**
323
+ * Cancel the active loop.
324
+ */
325
+ cancelLoop() {
326
+ if (!this.activeLoop) {
327
+ throw new Error('No active loop to cancel.');
328
+ }
329
+ this.activeLoop.status = 'cancelled';
330
+ this.activeLoop.completedAt = new Date().toISOString();
331
+ const cancelled = this.activeLoop;
332
+ this.moveToHistory('cancelled');
333
+ this.completedLoops.push(cancelled);
334
+ this.activeLoop = null;
335
+ return cancelled;
336
+ }
337
+ /**
338
+ * Get current loop status. Returns null if no active loop.
339
+ */
340
+ getStatus() {
341
+ return this.activeLoop;
342
+ }
343
+ /**
344
+ * Get history of all completed/cancelled/max-iterations loops.
345
+ */
346
+ getHistory() {
347
+ return [...this.completedLoops];
348
+ }
349
+ /**
350
+ * Get structured history entries (for brain session recording).
351
+ */
352
+ getHistoryEntries() {
353
+ return [...this.historyEntries];
354
+ }
355
+ /**
356
+ * Check if a loop is currently active.
357
+ */
358
+ isActive() {
359
+ return this.activeLoop !== null;
360
+ }
361
+ /**
362
+ * Move active loop to history entries.
363
+ */
364
+ moveToHistory(outcome) {
365
+ if (!this.activeLoop)
366
+ return;
367
+ this.historyEntries.push({
368
+ id: this.activeLoop.id,
369
+ mode: this.activeLoop.config.mode,
370
+ intent: this.activeLoop.config.intent,
371
+ prompt: this.activeLoop.config.prompt,
372
+ iterations: this.activeLoop.iterations.length,
373
+ outcome,
374
+ startedAt: this.activeLoop.startedAt,
375
+ completedAt: this.activeLoop.completedAt ?? new Date().toISOString(),
376
+ });
377
+ }
378
+ }
379
+ //# sourceMappingURL=loop-manager.js.map