moflo 4.0.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 (902) hide show
  1. package/.claude/agents/MIGRATION_SUMMARY.md +222 -0
  2. package/.claude/agents/analysis/analyze-code-quality.md +179 -0
  3. package/.claude/agents/analysis/code-analyzer.md +210 -0
  4. package/.claude/agents/analysis/code-review/analyze-code-quality.md +179 -0
  5. package/.claude/agents/architecture/system-design/arch-system-design.md +155 -0
  6. package/.claude/agents/base-template-generator.md +42 -0
  7. package/.claude/agents/consensus/byzantine-coordinator.md +63 -0
  8. package/.claude/agents/consensus/crdt-synchronizer.md +997 -0
  9. package/.claude/agents/consensus/gossip-coordinator.md +63 -0
  10. package/.claude/agents/consensus/performance-benchmarker.md +851 -0
  11. package/.claude/agents/consensus/quorum-manager.md +823 -0
  12. package/.claude/agents/consensus/raft-manager.md +63 -0
  13. package/.claude/agents/consensus/security-manager.md +622 -0
  14. package/.claude/agents/core/coder.md +266 -0
  15. package/.claude/agents/core/planner.md +168 -0
  16. package/.claude/agents/core/researcher.md +190 -0
  17. package/.claude/agents/core/reviewer.md +326 -0
  18. package/.claude/agents/core/tester.md +319 -0
  19. package/.claude/agents/custom/test-long-runner.md +44 -0
  20. package/.claude/agents/data/ml/data-ml-model.md +193 -0
  21. package/.claude/agents/database-specialist.yaml +21 -0
  22. package/.claude/agents/development/backend/dev-backend-api.md +142 -0
  23. package/.claude/agents/development/dev-backend-api.md +345 -0
  24. package/.claude/agents/devops/ci-cd/ops-cicd-github.md +164 -0
  25. package/.claude/agents/documentation/api-docs/docs-api-openapi.md +174 -0
  26. package/.claude/agents/dual-mode/codex-coordinator.md +224 -0
  27. package/.claude/agents/dual-mode/codex-worker.md +211 -0
  28. package/.claude/agents/dual-mode/dual-orchestrator.md +291 -0
  29. package/.claude/agents/flow-nexus/app-store.md +88 -0
  30. package/.claude/agents/flow-nexus/authentication.md +69 -0
  31. package/.claude/agents/flow-nexus/challenges.md +81 -0
  32. package/.claude/agents/flow-nexus/neural-network.md +88 -0
  33. package/.claude/agents/flow-nexus/payments.md +83 -0
  34. package/.claude/agents/flow-nexus/sandbox.md +76 -0
  35. package/.claude/agents/flow-nexus/swarm.md +76 -0
  36. package/.claude/agents/flow-nexus/user-tools.md +96 -0
  37. package/.claude/agents/flow-nexus/workflow.md +84 -0
  38. package/.claude/agents/github/code-review-swarm.md +538 -0
  39. package/.claude/agents/github/github-modes.md +173 -0
  40. package/.claude/agents/github/issue-tracker.md +319 -0
  41. package/.claude/agents/github/multi-repo-swarm.md +553 -0
  42. package/.claude/agents/github/pr-manager.md +191 -0
  43. package/.claude/agents/github/project-board-sync.md +509 -0
  44. package/.claude/agents/github/release-manager.md +367 -0
  45. package/.claude/agents/github/release-swarm.md +583 -0
  46. package/.claude/agents/github/repo-architect.md +398 -0
  47. package/.claude/agents/github/swarm-issue.md +573 -0
  48. package/.claude/agents/github/swarm-pr.md +428 -0
  49. package/.claude/agents/github/sync-coordinator.md +452 -0
  50. package/.claude/agents/github/workflow-automation.md +635 -0
  51. package/.claude/agents/goal/agent.md +816 -0
  52. package/.claude/agents/goal/code-goal-planner.md +446 -0
  53. package/.claude/agents/goal/goal-planner.md +168 -0
  54. package/.claude/agents/hive-mind/collective-intelligence-coordinator.md +130 -0
  55. package/.claude/agents/hive-mind/queen-coordinator.md +203 -0
  56. package/.claude/agents/hive-mind/scout-explorer.md +242 -0
  57. package/.claude/agents/hive-mind/swarm-memory-manager.md +193 -0
  58. package/.claude/agents/hive-mind/worker-specialist.md +217 -0
  59. package/.claude/agents/index.yaml +17 -0
  60. package/.claude/agents/neural/safla-neural.md +74 -0
  61. package/.claude/agents/optimization/benchmark-suite.md +665 -0
  62. package/.claude/agents/optimization/load-balancer.md +431 -0
  63. package/.claude/agents/optimization/performance-monitor.md +672 -0
  64. package/.claude/agents/optimization/resource-allocator.md +674 -0
  65. package/.claude/agents/optimization/topology-optimizer.md +808 -0
  66. package/.claude/agents/payments/agentic-payments.md +126 -0
  67. package/.claude/agents/project-coordinator.yaml +15 -0
  68. package/.claude/agents/python-specialist.yaml +21 -0
  69. package/.claude/agents/reasoning/agent.md +816 -0
  70. package/.claude/agents/reasoning/goal-planner.md +73 -0
  71. package/.claude/agents/security-auditor.yaml +20 -0
  72. package/.claude/agents/sona/sona-learning-optimizer.md +74 -0
  73. package/.claude/agents/sparc/architecture.md +472 -0
  74. package/.claude/agents/sparc/pseudocode.md +318 -0
  75. package/.claude/agents/sparc/refinement.md +525 -0
  76. package/.claude/agents/sparc/specification.md +276 -0
  77. package/.claude/agents/specialized/mobile/spec-mobile-react-native.md +225 -0
  78. package/.claude/agents/sublinear/consensus-coordinator.md +338 -0
  79. package/.claude/agents/sublinear/matrix-optimizer.md +185 -0
  80. package/.claude/agents/sublinear/pagerank-analyzer.md +299 -0
  81. package/.claude/agents/sublinear/performance-optimizer.md +368 -0
  82. package/.claude/agents/sublinear/trading-predictor.md +246 -0
  83. package/.claude/agents/swarm/adaptive-coordinator.md +396 -0
  84. package/.claude/agents/swarm/hierarchical-coordinator.md +327 -0
  85. package/.claude/agents/swarm/mesh-coordinator.md +392 -0
  86. package/.claude/agents/templates/automation-smart-agent.md +205 -0
  87. package/.claude/agents/templates/coordinator-swarm-init.md +105 -0
  88. package/.claude/agents/templates/github-pr-manager.md +177 -0
  89. package/.claude/agents/templates/implementer-sparc-coder.md +259 -0
  90. package/.claude/agents/templates/memory-coordinator.md +187 -0
  91. package/.claude/agents/templates/migration-plan.md +746 -0
  92. package/.claude/agents/templates/orchestrator-task.md +139 -0
  93. package/.claude/agents/templates/performance-analyzer.md +199 -0
  94. package/.claude/agents/templates/sparc-coordinator.md +183 -0
  95. package/.claude/agents/testing/production-validator.md +395 -0
  96. package/.claude/agents/testing/tdd-london-swarm.md +244 -0
  97. package/.claude/agents/testing/unit/tdd-london-swarm.md +244 -0
  98. package/.claude/agents/testing/validation/production-validator.md +395 -0
  99. package/.claude/agents/typescript-specialist.yaml +21 -0
  100. package/.claude/agents/v3/database-specialist.yaml +21 -0
  101. package/.claude/agents/v3/index.yaml +17 -0
  102. package/.claude/agents/v3/project-coordinator.yaml +15 -0
  103. package/.claude/agents/v3/python-specialist.yaml +21 -0
  104. package/.claude/agents/v3/test-architect.yaml +20 -0
  105. package/.claude/agents/v3/typescript-specialist.yaml +21 -0
  106. package/.claude/agents/v3/v3-integration-architect.md +346 -0
  107. package/.claude/agents/v3/v3-memory-specialist.md +318 -0
  108. package/.claude/agents/v3/v3-performance-engineer.md +397 -0
  109. package/.claude/agents/v3/v3-queen-coordinator.md +98 -0
  110. package/.claude/agents/v3/v3-security-architect.md +174 -0
  111. package/.claude/checkpoints/1767754460.json +8 -0
  112. package/.claude/commands/agents/README.md +10 -0
  113. package/.claude/commands/agents/agent-capabilities.md +21 -0
  114. package/.claude/commands/agents/agent-coordination.md +28 -0
  115. package/.claude/commands/agents/agent-spawning.md +28 -0
  116. package/.claude/commands/agents/agent-types.md +26 -0
  117. package/.claude/commands/analysis/COMMAND_COMPLIANCE_REPORT.md +54 -0
  118. package/.claude/commands/analysis/README.md +9 -0
  119. package/.claude/commands/analysis/bottleneck-detect.md +162 -0
  120. package/.claude/commands/analysis/performance-bottlenecks.md +59 -0
  121. package/.claude/commands/analysis/performance-report.md +25 -0
  122. package/.claude/commands/analysis/token-efficiency.md +45 -0
  123. package/.claude/commands/analysis/token-usage.md +25 -0
  124. package/.claude/commands/automation/README.md +9 -0
  125. package/.claude/commands/automation/auto-agent.md +122 -0
  126. package/.claude/commands/automation/self-healing.md +106 -0
  127. package/.claude/commands/automation/session-memory.md +90 -0
  128. package/.claude/commands/automation/smart-agents.md +73 -0
  129. package/.claude/commands/automation/smart-spawn.md +25 -0
  130. package/.claude/commands/automation/workflow-select.md +25 -0
  131. package/.claude/commands/claude-flow-help.md +103 -0
  132. package/.claude/commands/claude-flow-memory.md +107 -0
  133. package/.claude/commands/claude-flow-swarm.md +205 -0
  134. package/.claude/commands/coordination/README.md +9 -0
  135. package/.claude/commands/coordination/agent-spawn.md +25 -0
  136. package/.claude/commands/coordination/init.md +44 -0
  137. package/.claude/commands/coordination/orchestrate.md +43 -0
  138. package/.claude/commands/coordination/spawn.md +45 -0
  139. package/.claude/commands/coordination/swarm-init.md +85 -0
  140. package/.claude/commands/coordination/task-orchestrate.md +25 -0
  141. package/.claude/commands/flow-nexus/app-store.md +124 -0
  142. package/.claude/commands/flow-nexus/challenges.md +120 -0
  143. package/.claude/commands/flow-nexus/login-registration.md +65 -0
  144. package/.claude/commands/flow-nexus/neural-network.md +134 -0
  145. package/.claude/commands/flow-nexus/payments.md +116 -0
  146. package/.claude/commands/flow-nexus/sandbox.md +83 -0
  147. package/.claude/commands/flow-nexus/swarm.md +87 -0
  148. package/.claude/commands/flow-nexus/user-tools.md +152 -0
  149. package/.claude/commands/flow-nexus/workflow.md +115 -0
  150. package/.claude/commands/github/README.md +11 -0
  151. package/.claude/commands/github/code-review-swarm.md +514 -0
  152. package/.claude/commands/github/code-review.md +25 -0
  153. package/.claude/commands/github/github-modes.md +147 -0
  154. package/.claude/commands/github/github-swarm.md +121 -0
  155. package/.claude/commands/github/issue-tracker.md +292 -0
  156. package/.claude/commands/github/issue-triage.md +25 -0
  157. package/.claude/commands/github/multi-repo-swarm.md +519 -0
  158. package/.claude/commands/github/pr-enhance.md +26 -0
  159. package/.claude/commands/github/pr-manager.md +170 -0
  160. package/.claude/commands/github/project-board-sync.md +471 -0
  161. package/.claude/commands/github/release-manager.md +338 -0
  162. package/.claude/commands/github/release-swarm.md +544 -0
  163. package/.claude/commands/github/repo-analyze.md +25 -0
  164. package/.claude/commands/github/repo-architect.md +367 -0
  165. package/.claude/commands/github/swarm-issue.md +482 -0
  166. package/.claude/commands/github/swarm-pr.md +285 -0
  167. package/.claude/commands/github/sync-coordinator.md +301 -0
  168. package/.claude/commands/github/workflow-automation.md +442 -0
  169. package/.claude/commands/hive-mind/README.md +17 -0
  170. package/.claude/commands/hive-mind/hive-mind-consensus.md +8 -0
  171. package/.claude/commands/hive-mind/hive-mind-init.md +18 -0
  172. package/.claude/commands/hive-mind/hive-mind-memory.md +8 -0
  173. package/.claude/commands/hive-mind/hive-mind-metrics.md +8 -0
  174. package/.claude/commands/hive-mind/hive-mind-resume.md +8 -0
  175. package/.claude/commands/hive-mind/hive-mind-sessions.md +8 -0
  176. package/.claude/commands/hive-mind/hive-mind-spawn.md +21 -0
  177. package/.claude/commands/hive-mind/hive-mind-status.md +8 -0
  178. package/.claude/commands/hive-mind/hive-mind-stop.md +8 -0
  179. package/.claude/commands/hive-mind/hive-mind-wizard.md +8 -0
  180. package/.claude/commands/hive-mind/hive-mind.md +27 -0
  181. package/.claude/commands/hooks/README.md +11 -0
  182. package/.claude/commands/hooks/overview.md +58 -0
  183. package/.claude/commands/hooks/post-edit.md +117 -0
  184. package/.claude/commands/hooks/post-task.md +112 -0
  185. package/.claude/commands/hooks/pre-edit.md +113 -0
  186. package/.claude/commands/hooks/pre-task.md +111 -0
  187. package/.claude/commands/hooks/session-end.md +118 -0
  188. package/.claude/commands/hooks/setup.md +103 -0
  189. package/.claude/commands/memory/README.md +9 -0
  190. package/.claude/commands/memory/memory-persist.md +25 -0
  191. package/.claude/commands/memory/memory-search.md +25 -0
  192. package/.claude/commands/memory/memory-usage.md +25 -0
  193. package/.claude/commands/memory/neural.md +47 -0
  194. package/.claude/commands/monitoring/README.md +9 -0
  195. package/.claude/commands/monitoring/agent-metrics.md +25 -0
  196. package/.claude/commands/monitoring/agents.md +44 -0
  197. package/.claude/commands/monitoring/real-time-view.md +25 -0
  198. package/.claude/commands/monitoring/status.md +46 -0
  199. package/.claude/commands/monitoring/swarm-monitor.md +25 -0
  200. package/.claude/commands/optimization/README.md +9 -0
  201. package/.claude/commands/optimization/auto-topology.md +62 -0
  202. package/.claude/commands/optimization/cache-manage.md +25 -0
  203. package/.claude/commands/optimization/parallel-execute.md +25 -0
  204. package/.claude/commands/optimization/parallel-execution.md +50 -0
  205. package/.claude/commands/optimization/topology-optimize.md +25 -0
  206. package/.claude/commands/pair/README.md +261 -0
  207. package/.claude/commands/pair/commands.md +546 -0
  208. package/.claude/commands/pair/config.md +510 -0
  209. package/.claude/commands/pair/examples.md +512 -0
  210. package/.claude/commands/pair/modes.md +348 -0
  211. package/.claude/commands/pair/session.md +407 -0
  212. package/.claude/commands/pair/start.md +209 -0
  213. package/.claude/commands/sparc/analyzer.md +52 -0
  214. package/.claude/commands/sparc/architect.md +53 -0
  215. package/.claude/commands/sparc/ask.md +97 -0
  216. package/.claude/commands/sparc/batch-executor.md +54 -0
  217. package/.claude/commands/sparc/code.md +89 -0
  218. package/.claude/commands/sparc/coder.md +54 -0
  219. package/.claude/commands/sparc/debug.md +83 -0
  220. package/.claude/commands/sparc/debugger.md +54 -0
  221. package/.claude/commands/sparc/designer.md +53 -0
  222. package/.claude/commands/sparc/devops.md +109 -0
  223. package/.claude/commands/sparc/docs-writer.md +80 -0
  224. package/.claude/commands/sparc/documenter.md +54 -0
  225. package/.claude/commands/sparc/innovator.md +54 -0
  226. package/.claude/commands/sparc/integration.md +83 -0
  227. package/.claude/commands/sparc/mcp.md +117 -0
  228. package/.claude/commands/sparc/memory-manager.md +54 -0
  229. package/.claude/commands/sparc/optimizer.md +54 -0
  230. package/.claude/commands/sparc/orchestrator.md +132 -0
  231. package/.claude/commands/sparc/post-deployment-monitoring-mode.md +83 -0
  232. package/.claude/commands/sparc/refinement-optimization-mode.md +83 -0
  233. package/.claude/commands/sparc/researcher.md +54 -0
  234. package/.claude/commands/sparc/reviewer.md +54 -0
  235. package/.claude/commands/sparc/security-review.md +80 -0
  236. package/.claude/commands/sparc/sparc-modes.md +174 -0
  237. package/.claude/commands/sparc/sparc.md +111 -0
  238. package/.claude/commands/sparc/spec-pseudocode.md +80 -0
  239. package/.claude/commands/sparc/supabase-admin.md +348 -0
  240. package/.claude/commands/sparc/swarm-coordinator.md +54 -0
  241. package/.claude/commands/sparc/tdd.md +54 -0
  242. package/.claude/commands/sparc/tester.md +54 -0
  243. package/.claude/commands/sparc/tutorial.md +79 -0
  244. package/.claude/commands/sparc/workflow-manager.md +54 -0
  245. package/.claude/commands/sparc.md +166 -0
  246. package/.claude/commands/stream-chain/pipeline.md +121 -0
  247. package/.claude/commands/stream-chain/run.md +70 -0
  248. package/.claude/commands/swarm/README.md +15 -0
  249. package/.claude/commands/swarm/analysis.md +95 -0
  250. package/.claude/commands/swarm/development.md +96 -0
  251. package/.claude/commands/swarm/examples.md +168 -0
  252. package/.claude/commands/swarm/maintenance.md +102 -0
  253. package/.claude/commands/swarm/optimization.md +117 -0
  254. package/.claude/commands/swarm/research.md +136 -0
  255. package/.claude/commands/swarm/swarm-analysis.md +8 -0
  256. package/.claude/commands/swarm/swarm-background.md +8 -0
  257. package/.claude/commands/swarm/swarm-init.md +19 -0
  258. package/.claude/commands/swarm/swarm-modes.md +8 -0
  259. package/.claude/commands/swarm/swarm-monitor.md +8 -0
  260. package/.claude/commands/swarm/swarm-spawn.md +19 -0
  261. package/.claude/commands/swarm/swarm-status.md +8 -0
  262. package/.claude/commands/swarm/swarm-strategies.md +8 -0
  263. package/.claude/commands/swarm/swarm.md +27 -0
  264. package/.claude/commands/swarm/testing.md +131 -0
  265. package/.claude/commands/training/README.md +9 -0
  266. package/.claude/commands/training/model-update.md +25 -0
  267. package/.claude/commands/training/neural-patterns.md +74 -0
  268. package/.claude/commands/training/neural-train.md +25 -0
  269. package/.claude/commands/training/pattern-learn.md +25 -0
  270. package/.claude/commands/training/specialization.md +63 -0
  271. package/.claude/commands/truth/start.md +143 -0
  272. package/.claude/commands/verify/check.md +50 -0
  273. package/.claude/commands/verify/start.md +128 -0
  274. package/.claude/commands/workflows/README.md +9 -0
  275. package/.claude/commands/workflows/development.md +78 -0
  276. package/.claude/commands/workflows/research.md +63 -0
  277. package/.claude/commands/workflows/workflow-create.md +25 -0
  278. package/.claude/commands/workflows/workflow-execute.md +25 -0
  279. package/.claude/commands/workflows/workflow-export.md +25 -0
  280. package/.claude/config/v3-dependency-optimization.json +266 -0
  281. package/.claude/config/v3-performance-targets.json +251 -0
  282. package/.claude/helpers/README.md +97 -0
  283. package/.claude/helpers/adr-compliance.sh +186 -0
  284. package/.claude/helpers/aggressive-microcompact.mjs +36 -0
  285. package/.claude/helpers/auto-commit.sh +178 -0
  286. package/.claude/helpers/auto-memory-hook.mjs +363 -0
  287. package/.claude/helpers/checkpoint-manager.sh +251 -0
  288. package/.claude/helpers/context-persistence-hook.mjs +1979 -0
  289. package/.claude/helpers/daemon-manager.sh +252 -0
  290. package/.claude/helpers/ddd-tracker.sh +144 -0
  291. package/.claude/helpers/github-safe.js +106 -0
  292. package/.claude/helpers/github-setup.sh +28 -0
  293. package/.claude/helpers/guidance-hook.sh +13 -0
  294. package/.claude/helpers/guidance-hooks.sh +102 -0
  295. package/.claude/helpers/health-monitor.sh +108 -0
  296. package/.claude/helpers/hook-handler.cjs +229 -0
  297. package/.claude/helpers/intelligence.cjs +197 -0
  298. package/.claude/helpers/learning-hooks.sh +329 -0
  299. package/.claude/helpers/learning-optimizer.sh +127 -0
  300. package/.claude/helpers/learning-service.mjs +1144 -0
  301. package/.claude/helpers/memory.cjs +84 -0
  302. package/.claude/helpers/metrics-db.mjs +488 -0
  303. package/.claude/helpers/patch-aggressive-prune.mjs +184 -0
  304. package/.claude/helpers/pattern-consolidator.sh +86 -0
  305. package/.claude/helpers/perf-worker.sh +160 -0
  306. package/.claude/helpers/quick-start.sh +19 -0
  307. package/.claude/helpers/router.cjs +62 -0
  308. package/.claude/helpers/security-scanner.sh +127 -0
  309. package/.claude/helpers/session.cjs +125 -0
  310. package/.claude/helpers/setup-mcp.sh +18 -0
  311. package/.claude/helpers/standard-checkpoint-hooks.sh +189 -0
  312. package/.claude/helpers/statusline.cjs +742 -0
  313. package/.claude/helpers/swarm-comms.sh +353 -0
  314. package/.claude/helpers/swarm-hooks.sh +761 -0
  315. package/.claude/helpers/swarm-monitor.sh +211 -0
  316. package/.claude/helpers/sync-v3-metrics.sh +245 -0
  317. package/.claude/helpers/update-v3-progress.sh +166 -0
  318. package/.claude/helpers/v3-quick-status.sh +58 -0
  319. package/.claude/helpers/v3.sh +111 -0
  320. package/.claude/helpers/validate-v3-config.sh +216 -0
  321. package/.claude/helpers/worker-manager.sh +170 -0
  322. package/.claude/mcp.json +13 -0
  323. package/.claude/settings copy.json +526 -0
  324. package/.claude/settings.json +305 -0
  325. package/.claude/skills/agentdb-advanced/SKILL.md +550 -0
  326. package/.claude/skills/agentdb-learning/SKILL.md +545 -0
  327. package/.claude/skills/agentdb-memory-patterns/SKILL.md +339 -0
  328. package/.claude/skills/agentdb-optimization/SKILL.md +509 -0
  329. package/.claude/skills/agentdb-vector-search/SKILL.md +339 -0
  330. package/.claude/skills/agentic-jujutsu/SKILL.md +645 -0
  331. package/.claude/skills/dual-mode/README.md +71 -0
  332. package/.claude/skills/dual-mode/dual-collect.md +103 -0
  333. package/.claude/skills/dual-mode/dual-coordinate.md +85 -0
  334. package/.claude/skills/dual-mode/dual-spawn.md +81 -0
  335. package/.claude/skills/flow-nexus-neural/SKILL.md +738 -0
  336. package/.claude/skills/flow-nexus-platform/SKILL.md +1157 -0
  337. package/.claude/skills/flow-nexus-swarm/SKILL.md +610 -0
  338. package/.claude/skills/github-code-review/SKILL.md +1140 -0
  339. package/.claude/skills/github-multi-repo/SKILL.md +874 -0
  340. package/.claude/skills/github-project-management/SKILL.md +1277 -0
  341. package/.claude/skills/github-release-management/SKILL.md +1081 -0
  342. package/.claude/skills/github-workflow-automation/SKILL.md +1065 -0
  343. package/.claude/skills/hive-mind-advanced/SKILL.md +712 -0
  344. package/.claude/skills/hooks-automation/SKILL.md +1201 -0
  345. package/.claude/skills/pair-programming/SKILL.md +1202 -0
  346. package/.claude/skills/performance-analysis/SKILL.md +563 -0
  347. package/.claude/skills/reasoningbank-agentdb/SKILL.md +446 -0
  348. package/.claude/skills/reasoningbank-intelligence/SKILL.md +201 -0
  349. package/.claude/skills/skill-builder/SKILL.md +910 -0
  350. package/.claude/skills/sparc-methodology/SKILL.md +1115 -0
  351. package/.claude/skills/stream-chain/SKILL.md +563 -0
  352. package/.claude/skills/swarm-advanced/SKILL.md +973 -0
  353. package/.claude/skills/swarm-orchestration/SKILL.md +179 -0
  354. package/.claude/skills/v3-cli-modernization/SKILL.md +872 -0
  355. package/.claude/skills/v3-core-implementation/SKILL.md +797 -0
  356. package/.claude/skills/v3-ddd-architecture/SKILL.md +442 -0
  357. package/.claude/skills/v3-integration-deep/SKILL.md +241 -0
  358. package/.claude/skills/v3-mcp-optimization/SKILL.md +777 -0
  359. package/.claude/skills/v3-memory-unification/SKILL.md +174 -0
  360. package/.claude/skills/v3-performance-optimization/SKILL.md +390 -0
  361. package/.claude/skills/v3-security-overhaul/SKILL.md +82 -0
  362. package/.claude/skills/v3-swarm-coordination/SKILL.md +340 -0
  363. package/.claude/skills/verification-quality/SKILL.md +649 -0
  364. package/.claude/skills/worker-benchmarks/skill.md +135 -0
  365. package/.claude/skills/worker-integration/skill.md +154 -0
  366. package/.claude/statusline-command.sh +176 -0
  367. package/.claude/statusline.mjs +109 -0
  368. package/.claude/statusline.sh +431 -0
  369. package/.claude/workflow-state.json +8 -0
  370. package/.claude-plugin/README.md +720 -0
  371. package/.claude-plugin/docs/INSTALLATION.md +261 -0
  372. package/.claude-plugin/docs/PLUGIN_SUMMARY.md +361 -0
  373. package/.claude-plugin/docs/QUICKSTART.md +361 -0
  374. package/.claude-plugin/docs/STRUCTURE.md +128 -0
  375. package/.claude-plugin/hooks/hooks.json +74 -0
  376. package/.claude-plugin/marketplace.json +96 -0
  377. package/.claude-plugin/plugin.json +71 -0
  378. package/.claude-plugin/scripts/install.sh +234 -0
  379. package/.claude-plugin/scripts/uninstall.sh +36 -0
  380. package/.claude-plugin/scripts/verify.sh +108 -0
  381. package/LICENSE +21 -0
  382. package/README.md +148 -0
  383. package/bin/cli.js +12 -0
  384. package/bin/npx-repair.js +7 -0
  385. package/bin/npx-safe-launch.js +9 -0
  386. package/package.json +109 -0
  387. package/v3/@claude-flow/cli/README.md +7536 -0
  388. package/v3/@claude-flow/cli/bin/cli.js +156 -0
  389. package/v3/@claude-flow/cli/bin/mcp-server.js +189 -0
  390. package/v3/@claude-flow/cli/bin/preinstall.cjs +2 -0
  391. package/v3/@claude-flow/cli/dist/src/appliance/gguf-engine.d.ts +91 -0
  392. package/v3/@claude-flow/cli/dist/src/appliance/gguf-engine.js +425 -0
  393. package/v3/@claude-flow/cli/dist/src/appliance/ruvllm-bridge.d.ts +102 -0
  394. package/v3/@claude-flow/cli/dist/src/appliance/ruvllm-bridge.js +292 -0
  395. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-builder.d.ts +44 -0
  396. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-builder.js +329 -0
  397. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-distribution.d.ts +97 -0
  398. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-distribution.js +370 -0
  399. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-format.d.ts +111 -0
  400. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-format.js +393 -0
  401. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-runner.d.ts +69 -0
  402. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-runner.js +238 -0
  403. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-signing.d.ts +123 -0
  404. package/v3/@claude-flow/cli/dist/src/appliance/rvfa-signing.js +347 -0
  405. package/v3/@claude-flow/cli/dist/src/benchmarks/pretrain/index.d.ts +58 -0
  406. package/v3/@claude-flow/cli/dist/src/benchmarks/pretrain/index.js +404 -0
  407. package/v3/@claude-flow/cli/dist/src/commands/agent.d.ts +8 -0
  408. package/v3/@claude-flow/cli/dist/src/commands/agent.js +857 -0
  409. package/v3/@claude-flow/cli/dist/src/commands/analyze.d.ts +19 -0
  410. package/v3/@claude-flow/cli/dist/src/commands/analyze.js +1823 -0
  411. package/v3/@claude-flow/cli/dist/src/commands/appliance-advanced.d.ts +9 -0
  412. package/v3/@claude-flow/cli/dist/src/commands/appliance-advanced.js +215 -0
  413. package/v3/@claude-flow/cli/dist/src/commands/appliance.d.ts +8 -0
  414. package/v3/@claude-flow/cli/dist/src/commands/appliance.js +406 -0
  415. package/v3/@claude-flow/cli/dist/src/commands/benchmark.d.ts +10 -0
  416. package/v3/@claude-flow/cli/dist/src/commands/benchmark.js +459 -0
  417. package/v3/@claude-flow/cli/dist/src/commands/claims.d.ts +10 -0
  418. package/v3/@claude-flow/cli/dist/src/commands/claims.js +373 -0
  419. package/v3/@claude-flow/cli/dist/src/commands/completions.d.ts +10 -0
  420. package/v3/@claude-flow/cli/dist/src/commands/completions.js +539 -0
  421. package/v3/@claude-flow/cli/dist/src/commands/config.d.ts +8 -0
  422. package/v3/@claude-flow/cli/dist/src/commands/config.js +406 -0
  423. package/v3/@claude-flow/cli/dist/src/commands/daemon.d.ts +8 -0
  424. package/v3/@claude-flow/cli/dist/src/commands/daemon.js +609 -0
  425. package/v3/@claude-flow/cli/dist/src/commands/deployment.d.ts +10 -0
  426. package/v3/@claude-flow/cli/dist/src/commands/deployment.js +289 -0
  427. package/v3/@claude-flow/cli/dist/src/commands/doctor.d.ts +10 -0
  428. package/v3/@claude-flow/cli/dist/src/commands/doctor.js +603 -0
  429. package/v3/@claude-flow/cli/dist/src/commands/embeddings.d.ts +18 -0
  430. package/v3/@claude-flow/cli/dist/src/commands/embeddings.js +1576 -0
  431. package/v3/@claude-flow/cli/dist/src/commands/gate.d.ts +23 -0
  432. package/v3/@claude-flow/cli/dist/src/commands/gate.js +54 -0
  433. package/v3/@claude-flow/cli/dist/src/commands/guidance.d.ts +8 -0
  434. package/v3/@claude-flow/cli/dist/src/commands/guidance.js +560 -0
  435. package/v3/@claude-flow/cli/dist/src/commands/hive-mind.d.ts +11 -0
  436. package/v3/@claude-flow/cli/dist/src/commands/hive-mind.js +1230 -0
  437. package/v3/@claude-flow/cli/dist/src/commands/hooks.d.ts +8 -0
  438. package/v3/@claude-flow/cli/dist/src/commands/hooks.js +3945 -0
  439. package/v3/@claude-flow/cli/dist/src/commands/index.d.ts +113 -0
  440. package/v3/@claude-flow/cli/dist/src/commands/index.js +372 -0
  441. package/v3/@claude-flow/cli/dist/src/commands/init.d.ts +8 -0
  442. package/v3/@claude-flow/cli/dist/src/commands/init.js +990 -0
  443. package/v3/@claude-flow/cli/dist/src/commands/issues.d.ts +21 -0
  444. package/v3/@claude-flow/cli/dist/src/commands/issues.js +567 -0
  445. package/v3/@claude-flow/cli/dist/src/commands/mcp.d.ts +11 -0
  446. package/v3/@claude-flow/cli/dist/src/commands/mcp.js +715 -0
  447. package/v3/@claude-flow/cli/dist/src/commands/memory.d.ts +8 -0
  448. package/v3/@claude-flow/cli/dist/src/commands/memory.js +2425 -0
  449. package/v3/@claude-flow/cli/dist/src/commands/migrate.d.ts +8 -0
  450. package/v3/@claude-flow/cli/dist/src/commands/migrate.js +410 -0
  451. package/v3/@claude-flow/cli/dist/src/commands/neural.d.ts +10 -0
  452. package/v3/@claude-flow/cli/dist/src/commands/neural.js +1448 -0
  453. package/v3/@claude-flow/cli/dist/src/commands/orc.d.ts +17 -0
  454. package/v3/@claude-flow/cli/dist/src/commands/orc.js +629 -0
  455. package/v3/@claude-flow/cli/dist/src/commands/performance.d.ts +10 -0
  456. package/v3/@claude-flow/cli/dist/src/commands/performance.js +579 -0
  457. package/v3/@claude-flow/cli/dist/src/commands/plugins.d.ts +11 -0
  458. package/v3/@claude-flow/cli/dist/src/commands/plugins.js +820 -0
  459. package/v3/@claude-flow/cli/dist/src/commands/process.d.ts +10 -0
  460. package/v3/@claude-flow/cli/dist/src/commands/process.js +641 -0
  461. package/v3/@claude-flow/cli/dist/src/commands/progress.d.ts +11 -0
  462. package/v3/@claude-flow/cli/dist/src/commands/progress.js +259 -0
  463. package/v3/@claude-flow/cli/dist/src/commands/providers.d.ts +10 -0
  464. package/v3/@claude-flow/cli/dist/src/commands/providers.js +232 -0
  465. package/v3/@claude-flow/cli/dist/src/commands/route.d.ts +16 -0
  466. package/v3/@claude-flow/cli/dist/src/commands/route.js +813 -0
  467. package/v3/@claude-flow/cli/dist/src/commands/ruvector/backup.d.ts +11 -0
  468. package/v3/@claude-flow/cli/dist/src/commands/ruvector/backup.js +746 -0
  469. package/v3/@claude-flow/cli/dist/src/commands/ruvector/benchmark.d.ts +11 -0
  470. package/v3/@claude-flow/cli/dist/src/commands/ruvector/benchmark.js +480 -0
  471. package/v3/@claude-flow/cli/dist/src/commands/ruvector/import.d.ts +18 -0
  472. package/v3/@claude-flow/cli/dist/src/commands/ruvector/import.js +350 -0
  473. package/v3/@claude-flow/cli/dist/src/commands/ruvector/index.d.ts +29 -0
  474. package/v3/@claude-flow/cli/dist/src/commands/ruvector/index.js +129 -0
  475. package/v3/@claude-flow/cli/dist/src/commands/ruvector/init.d.ts +11 -0
  476. package/v3/@claude-flow/cli/dist/src/commands/ruvector/init.js +431 -0
  477. package/v3/@claude-flow/cli/dist/src/commands/ruvector/migrate.d.ts +11 -0
  478. package/v3/@claude-flow/cli/dist/src/commands/ruvector/migrate.js +481 -0
  479. package/v3/@claude-flow/cli/dist/src/commands/ruvector/optimize.d.ts +11 -0
  480. package/v3/@claude-flow/cli/dist/src/commands/ruvector/optimize.js +503 -0
  481. package/v3/@claude-flow/cli/dist/src/commands/ruvector/setup.d.ts +18 -0
  482. package/v3/@claude-flow/cli/dist/src/commands/ruvector/setup.js +765 -0
  483. package/v3/@claude-flow/cli/dist/src/commands/ruvector/status.d.ts +11 -0
  484. package/v3/@claude-flow/cli/dist/src/commands/ruvector/status.js +456 -0
  485. package/v3/@claude-flow/cli/dist/src/commands/security.d.ts +10 -0
  486. package/v3/@claude-flow/cli/dist/src/commands/security.js +576 -0
  487. package/v3/@claude-flow/cli/dist/src/commands/session.d.ts +8 -0
  488. package/v3/@claude-flow/cli/dist/src/commands/session.js +750 -0
  489. package/v3/@claude-flow/cli/dist/src/commands/start.d.ts +8 -0
  490. package/v3/@claude-flow/cli/dist/src/commands/start.js +418 -0
  491. package/v3/@claude-flow/cli/dist/src/commands/status.d.ts +8 -0
  492. package/v3/@claude-flow/cli/dist/src/commands/status.js +591 -0
  493. package/v3/@claude-flow/cli/dist/src/commands/swarm.d.ts +8 -0
  494. package/v3/@claude-flow/cli/dist/src/commands/swarm.js +748 -0
  495. package/v3/@claude-flow/cli/dist/src/commands/task.d.ts +8 -0
  496. package/v3/@claude-flow/cli/dist/src/commands/task.js +671 -0
  497. package/v3/@claude-flow/cli/dist/src/commands/transfer-store.d.ts +13 -0
  498. package/v3/@claude-flow/cli/dist/src/commands/transfer-store.js +428 -0
  499. package/v3/@claude-flow/cli/dist/src/commands/update.d.ts +8 -0
  500. package/v3/@claude-flow/cli/dist/src/commands/update.js +276 -0
  501. package/v3/@claude-flow/cli/dist/src/commands/workflow.d.ts +8 -0
  502. package/v3/@claude-flow/cli/dist/src/commands/workflow.js +617 -0
  503. package/v3/@claude-flow/cli/dist/src/config/moflo-config.d.ts +43 -0
  504. package/v3/@claude-flow/cli/dist/src/config/moflo-config.js +181 -0
  505. package/v3/@claude-flow/cli/dist/src/config-adapter.d.ts +15 -0
  506. package/v3/@claude-flow/cli/dist/src/config-adapter.js +186 -0
  507. package/v3/@claude-flow/cli/dist/src/index.d.ts +76 -0
  508. package/v3/@claude-flow/cli/dist/src/index.js +470 -0
  509. package/v3/@claude-flow/cli/dist/src/infrastructure/in-memory-repositories.d.ts +68 -0
  510. package/v3/@claude-flow/cli/dist/src/infrastructure/in-memory-repositories.js +264 -0
  511. package/v3/@claude-flow/cli/dist/src/init/claudemd-generator.d.ts +25 -0
  512. package/v3/@claude-flow/cli/dist/src/init/claudemd-generator.js +486 -0
  513. package/v3/@claude-flow/cli/dist/src/init/executor.d.ts +41 -0
  514. package/v3/@claude-flow/cli/dist/src/init/executor.js +1741 -0
  515. package/v3/@claude-flow/cli/dist/src/init/helpers-generator.d.ts +60 -0
  516. package/v3/@claude-flow/cli/dist/src/init/helpers-generator.js +1166 -0
  517. package/v3/@claude-flow/cli/dist/src/init/index.d.ts +13 -0
  518. package/v3/@claude-flow/cli/dist/src/init/index.js +15 -0
  519. package/v3/@claude-flow/cli/dist/src/init/mcp-generator.d.ts +26 -0
  520. package/v3/@claude-flow/cli/dist/src/init/mcp-generator.js +116 -0
  521. package/v3/@claude-flow/cli/dist/src/init/moflo-init.d.ts +14 -0
  522. package/v3/@claude-flow/cli/dist/src/init/moflo-init.js +392 -0
  523. package/v3/@claude-flow/cli/dist/src/init/settings-generator.d.ts +14 -0
  524. package/v3/@claude-flow/cli/dist/src/init/settings-generator.js +399 -0
  525. package/v3/@claude-flow/cli/dist/src/init/statusline-generator.d.ts +28 -0
  526. package/v3/@claude-flow/cli/dist/src/init/statusline-generator.js +818 -0
  527. package/v3/@claude-flow/cli/dist/src/init/types.d.ts +287 -0
  528. package/v3/@claude-flow/cli/dist/src/init/types.js +259 -0
  529. package/v3/@claude-flow/cli/dist/src/mcp-client.d.ts +92 -0
  530. package/v3/@claude-flow/cli/dist/src/mcp-client.js +237 -0
  531. package/v3/@claude-flow/cli/dist/src/mcp-server.d.ts +161 -0
  532. package/v3/@claude-flow/cli/dist/src/mcp-server.js +627 -0
  533. package/v3/@claude-flow/cli/dist/src/mcp-tools/agent-tools.d.ts +9 -0
  534. package/v3/@claude-flow/cli/dist/src/mcp-tools/agent-tools.js +549 -0
  535. package/v3/@claude-flow/cli/dist/src/mcp-tools/agentdb-tools.d.ts +30 -0
  536. package/v3/@claude-flow/cli/dist/src/mcp-tools/agentdb-tools.js +557 -0
  537. package/v3/@claude-flow/cli/dist/src/mcp-tools/analyze-tools.d.ts +38 -0
  538. package/v3/@claude-flow/cli/dist/src/mcp-tools/analyze-tools.js +317 -0
  539. package/v3/@claude-flow/cli/dist/src/mcp-tools/auto-install.d.ts +83 -0
  540. package/v3/@claude-flow/cli/dist/src/mcp-tools/auto-install.js +132 -0
  541. package/v3/@claude-flow/cli/dist/src/mcp-tools/browser-tools.d.ts +13 -0
  542. package/v3/@claude-flow/cli/dist/src/mcp-tools/browser-tools.js +551 -0
  543. package/v3/@claude-flow/cli/dist/src/mcp-tools/claims-tools.d.ts +12 -0
  544. package/v3/@claude-flow/cli/dist/src/mcp-tools/claims-tools.js +732 -0
  545. package/v3/@claude-flow/cli/dist/src/mcp-tools/config-tools.d.ts +8 -0
  546. package/v3/@claude-flow/cli/dist/src/mcp-tools/config-tools.js +343 -0
  547. package/v3/@claude-flow/cli/dist/src/mcp-tools/coordination-tools.d.ts +13 -0
  548. package/v3/@claude-flow/cli/dist/src/mcp-tools/coordination-tools.js +486 -0
  549. package/v3/@claude-flow/cli/dist/src/mcp-tools/daa-tools.d.ts +13 -0
  550. package/v3/@claude-flow/cli/dist/src/mcp-tools/daa-tools.js +426 -0
  551. package/v3/@claude-flow/cli/dist/src/mcp-tools/embeddings-tools.d.ts +9 -0
  552. package/v3/@claude-flow/cli/dist/src/mcp-tools/embeddings-tools.js +782 -0
  553. package/v3/@claude-flow/cli/dist/src/mcp-tools/github-tools.d.ts +13 -0
  554. package/v3/@claude-flow/cli/dist/src/mcp-tools/github-tools.js +373 -0
  555. package/v3/@claude-flow/cli/dist/src/mcp-tools/hive-mind-tools.d.ts +8 -0
  556. package/v3/@claude-flow/cli/dist/src/mcp-tools/hive-mind-tools.js +583 -0
  557. package/v3/@claude-flow/cli/dist/src/mcp-tools/hooks-tools.d.ts +44 -0
  558. package/v3/@claude-flow/cli/dist/src/mcp-tools/hooks-tools.js +3045 -0
  559. package/v3/@claude-flow/cli/dist/src/mcp-tools/index.d.ts +23 -0
  560. package/v3/@claude-flow/cli/dist/src/mcp-tools/index.js +22 -0
  561. package/v3/@claude-flow/cli/dist/src/mcp-tools/memory-tools.d.ts +14 -0
  562. package/v3/@claude-flow/cli/dist/src/mcp-tools/memory-tools.js +499 -0
  563. package/v3/@claude-flow/cli/dist/src/mcp-tools/neural-tools.d.ts +16 -0
  564. package/v3/@claude-flow/cli/dist/src/mcp-tools/neural-tools.js +461 -0
  565. package/v3/@claude-flow/cli/dist/src/mcp-tools/performance-tools.d.ts +16 -0
  566. package/v3/@claude-flow/cli/dist/src/mcp-tools/performance-tools.js +534 -0
  567. package/v3/@claude-flow/cli/dist/src/mcp-tools/progress-tools.d.ts +14 -0
  568. package/v3/@claude-flow/cli/dist/src/mcp-tools/progress-tools.js +348 -0
  569. package/v3/@claude-flow/cli/dist/src/mcp-tools/security-tools.d.ts +18 -0
  570. package/v3/@claude-flow/cli/dist/src/mcp-tools/security-tools.js +434 -0
  571. package/v3/@claude-flow/cli/dist/src/mcp-tools/session-tools.d.ts +8 -0
  572. package/v3/@claude-flow/cli/dist/src/mcp-tools/session-tools.js +315 -0
  573. package/v3/@claude-flow/cli/dist/src/mcp-tools/swarm-tools.d.ts +8 -0
  574. package/v3/@claude-flow/cli/dist/src/mcp-tools/swarm-tools.js +102 -0
  575. package/v3/@claude-flow/cli/dist/src/mcp-tools/system-tools.d.ts +13 -0
  576. package/v3/@claude-flow/cli/dist/src/mcp-tools/system-tools.js +417 -0
  577. package/v3/@claude-flow/cli/dist/src/mcp-tools/task-tools.d.ts +8 -0
  578. package/v3/@claude-flow/cli/dist/src/mcp-tools/task-tools.js +338 -0
  579. package/v3/@claude-flow/cli/dist/src/mcp-tools/terminal-tools.d.ts +13 -0
  580. package/v3/@claude-flow/cli/dist/src/mcp-tools/terminal-tools.js +246 -0
  581. package/v3/@claude-flow/cli/dist/src/mcp-tools/transfer-tools.d.ts +14 -0
  582. package/v3/@claude-flow/cli/dist/src/mcp-tools/transfer-tools.js +396 -0
  583. package/v3/@claude-flow/cli/dist/src/mcp-tools/types.d.ts +31 -0
  584. package/v3/@claude-flow/cli/dist/src/mcp-tools/types.js +7 -0
  585. package/v3/@claude-flow/cli/dist/src/mcp-tools/workflow-tools.d.ts +8 -0
  586. package/v3/@claude-flow/cli/dist/src/mcp-tools/workflow-tools.js +572 -0
  587. package/v3/@claude-flow/cli/dist/src/memory/ewc-consolidation.d.ts +271 -0
  588. package/v3/@claude-flow/cli/dist/src/memory/ewc-consolidation.js +542 -0
  589. package/v3/@claude-flow/cli/dist/src/memory/intelligence.d.ts +285 -0
  590. package/v3/@claude-flow/cli/dist/src/memory/intelligence.js +794 -0
  591. package/v3/@claude-flow/cli/dist/src/memory/memory-bridge.d.ts +407 -0
  592. package/v3/@claude-flow/cli/dist/src/memory/memory-bridge.js +1494 -0
  593. package/v3/@claude-flow/cli/dist/src/memory/memory-initializer.d.ts +405 -0
  594. package/v3/@claude-flow/cli/dist/src/memory/memory-initializer.js +2213 -0
  595. package/v3/@claude-flow/cli/dist/src/memory/sona-optimizer.d.ts +227 -0
  596. package/v3/@claude-flow/cli/dist/src/memory/sona-optimizer.js +633 -0
  597. package/v3/@claude-flow/cli/dist/src/output.d.ts +133 -0
  598. package/v3/@claude-flow/cli/dist/src/output.js +514 -0
  599. package/v3/@claude-flow/cli/dist/src/parser.d.ts +41 -0
  600. package/v3/@claude-flow/cli/dist/src/parser.js +377 -0
  601. package/v3/@claude-flow/cli/dist/src/plugins/manager.d.ts +133 -0
  602. package/v3/@claude-flow/cli/dist/src/plugins/manager.js +400 -0
  603. package/v3/@claude-flow/cli/dist/src/plugins/store/discovery.d.ts +88 -0
  604. package/v3/@claude-flow/cli/dist/src/plugins/store/discovery.js +1147 -0
  605. package/v3/@claude-flow/cli/dist/src/plugins/store/index.d.ts +76 -0
  606. package/v3/@claude-flow/cli/dist/src/plugins/store/index.js +141 -0
  607. package/v3/@claude-flow/cli/dist/src/plugins/store/search.d.ts +46 -0
  608. package/v3/@claude-flow/cli/dist/src/plugins/store/search.js +230 -0
  609. package/v3/@claude-flow/cli/dist/src/plugins/store/types.d.ts +274 -0
  610. package/v3/@claude-flow/cli/dist/src/plugins/store/types.js +7 -0
  611. package/v3/@claude-flow/cli/dist/src/plugins/tests/demo-plugin-store.d.ts +7 -0
  612. package/v3/@claude-flow/cli/dist/src/plugins/tests/demo-plugin-store.js +126 -0
  613. package/v3/@claude-flow/cli/dist/src/plugins/tests/standalone-test.d.ts +12 -0
  614. package/v3/@claude-flow/cli/dist/src/plugins/tests/standalone-test.js +188 -0
  615. package/v3/@claude-flow/cli/dist/src/plugins/tests/test-plugin-store.d.ts +7 -0
  616. package/v3/@claude-flow/cli/dist/src/plugins/tests/test-plugin-store.js +206 -0
  617. package/v3/@claude-flow/cli/dist/src/production/circuit-breaker.d.ts +101 -0
  618. package/v3/@claude-flow/cli/dist/src/production/circuit-breaker.js +241 -0
  619. package/v3/@claude-flow/cli/dist/src/production/error-handler.d.ts +92 -0
  620. package/v3/@claude-flow/cli/dist/src/production/error-handler.js +299 -0
  621. package/v3/@claude-flow/cli/dist/src/production/index.d.ts +23 -0
  622. package/v3/@claude-flow/cli/dist/src/production/index.js +18 -0
  623. package/v3/@claude-flow/cli/dist/src/production/monitoring.d.ts +161 -0
  624. package/v3/@claude-flow/cli/dist/src/production/monitoring.js +356 -0
  625. package/v3/@claude-flow/cli/dist/src/production/rate-limiter.d.ts +80 -0
  626. package/v3/@claude-flow/cli/dist/src/production/rate-limiter.js +201 -0
  627. package/v3/@claude-flow/cli/dist/src/production/retry.d.ts +48 -0
  628. package/v3/@claude-flow/cli/dist/src/production/retry.js +179 -0
  629. package/v3/@claude-flow/cli/dist/src/prompt.d.ts +44 -0
  630. package/v3/@claude-flow/cli/dist/src/prompt.js +501 -0
  631. package/v3/@claude-flow/cli/dist/src/runtime/headless.d.ts +60 -0
  632. package/v3/@claude-flow/cli/dist/src/runtime/headless.js +284 -0
  633. package/v3/@claude-flow/cli/dist/src/ruvector/ast-analyzer.d.ts +67 -0
  634. package/v3/@claude-flow/cli/dist/src/ruvector/ast-analyzer.js +277 -0
  635. package/v3/@claude-flow/cli/dist/src/ruvector/coverage-router.d.ts +160 -0
  636. package/v3/@claude-flow/cli/dist/src/ruvector/coverage-router.js +529 -0
  637. package/v3/@claude-flow/cli/dist/src/ruvector/coverage-tools.d.ts +33 -0
  638. package/v3/@claude-flow/cli/dist/src/ruvector/coverage-tools.js +157 -0
  639. package/v3/@claude-flow/cli/dist/src/ruvector/diff-classifier.d.ts +175 -0
  640. package/v3/@claude-flow/cli/dist/src/ruvector/diff-classifier.js +698 -0
  641. package/v3/@claude-flow/cli/dist/src/ruvector/enhanced-model-router.d.ts +146 -0
  642. package/v3/@claude-flow/cli/dist/src/ruvector/enhanced-model-router.js +530 -0
  643. package/v3/@claude-flow/cli/dist/src/ruvector/flash-attention.d.ts +195 -0
  644. package/v3/@claude-flow/cli/dist/src/ruvector/flash-attention.js +643 -0
  645. package/v3/@claude-flow/cli/dist/src/ruvector/graph-analyzer.d.ts +187 -0
  646. package/v3/@claude-flow/cli/dist/src/ruvector/graph-analyzer.js +929 -0
  647. package/v3/@claude-flow/cli/dist/src/ruvector/index.d.ts +34 -0
  648. package/v3/@claude-flow/cli/dist/src/ruvector/index.js +60 -0
  649. package/v3/@claude-flow/cli/dist/src/ruvector/lora-adapter.d.ts +218 -0
  650. package/v3/@claude-flow/cli/dist/src/ruvector/lora-adapter.js +455 -0
  651. package/v3/@claude-flow/cli/dist/src/ruvector/model-router.d.ts +220 -0
  652. package/v3/@claude-flow/cli/dist/src/ruvector/model-router.js +488 -0
  653. package/v3/@claude-flow/cli/dist/src/ruvector/moe-router.d.ts +206 -0
  654. package/v3/@claude-flow/cli/dist/src/ruvector/moe-router.js +626 -0
  655. package/v3/@claude-flow/cli/dist/src/ruvector/q-learning-router.d.ts +211 -0
  656. package/v3/@claude-flow/cli/dist/src/ruvector/q-learning-router.js +681 -0
  657. package/v3/@claude-flow/cli/dist/src/ruvector/semantic-router.d.ts +77 -0
  658. package/v3/@claude-flow/cli/dist/src/ruvector/semantic-router.js +178 -0
  659. package/v3/@claude-flow/cli/dist/src/ruvector/vector-db.d.ts +69 -0
  660. package/v3/@claude-flow/cli/dist/src/ruvector/vector-db.js +243 -0
  661. package/v3/@claude-flow/cli/dist/src/services/agent-router.d.ts +49 -0
  662. package/v3/@claude-flow/cli/dist/src/services/agent-router.js +161 -0
  663. package/v3/@claude-flow/cli/dist/src/services/agentic-flow-bridge.d.ts +50 -0
  664. package/v3/@claude-flow/cli/dist/src/services/agentic-flow-bridge.js +95 -0
  665. package/v3/@claude-flow/cli/dist/src/services/claim-service.d.ts +204 -0
  666. package/v3/@claude-flow/cli/dist/src/services/claim-service.js +818 -0
  667. package/v3/@claude-flow/cli/dist/src/services/container-worker-pool.d.ts +197 -0
  668. package/v3/@claude-flow/cli/dist/src/services/container-worker-pool.js +584 -0
  669. package/v3/@claude-flow/cli/dist/src/services/headless-worker-executor.d.ts +304 -0
  670. package/v3/@claude-flow/cli/dist/src/services/headless-worker-executor.js +999 -0
  671. package/v3/@claude-flow/cli/dist/src/services/index.d.ts +15 -0
  672. package/v3/@claude-flow/cli/dist/src/services/index.js +15 -0
  673. package/v3/@claude-flow/cli/dist/src/services/learning-service.d.ts +161 -0
  674. package/v3/@claude-flow/cli/dist/src/services/learning-service.js +679 -0
  675. package/v3/@claude-flow/cli/dist/src/services/registry-api.d.ts +58 -0
  676. package/v3/@claude-flow/cli/dist/src/services/registry-api.js +146 -0
  677. package/v3/@claude-flow/cli/dist/src/services/ruvector-training.d.ts +214 -0
  678. package/v3/@claude-flow/cli/dist/src/services/ruvector-training.js +497 -0
  679. package/v3/@claude-flow/cli/dist/src/services/worker-daemon.d.ts +203 -0
  680. package/v3/@claude-flow/cli/dist/src/services/worker-daemon.js +756 -0
  681. package/v3/@claude-flow/cli/dist/src/services/worker-queue.d.ts +194 -0
  682. package/v3/@claude-flow/cli/dist/src/services/worker-queue.js +513 -0
  683. package/v3/@claude-flow/cli/dist/src/services/workflow-gate.d.ts +77 -0
  684. package/v3/@claude-flow/cli/dist/src/services/workflow-gate.js +278 -0
  685. package/v3/@claude-flow/cli/dist/src/suggest.d.ts +53 -0
  686. package/v3/@claude-flow/cli/dist/src/suggest.js +200 -0
  687. package/v3/@claude-flow/cli/dist/src/transfer/anonymization/index.d.ts +25 -0
  688. package/v3/@claude-flow/cli/dist/src/transfer/anonymization/index.js +175 -0
  689. package/v3/@claude-flow/cli/dist/src/transfer/deploy-seraphine.d.ts +13 -0
  690. package/v3/@claude-flow/cli/dist/src/transfer/deploy-seraphine.js +205 -0
  691. package/v3/@claude-flow/cli/dist/src/transfer/export.d.ts +25 -0
  692. package/v3/@claude-flow/cli/dist/src/transfer/export.js +113 -0
  693. package/v3/@claude-flow/cli/dist/src/transfer/index.d.ts +12 -0
  694. package/v3/@claude-flow/cli/dist/src/transfer/index.js +31 -0
  695. package/v3/@claude-flow/cli/dist/src/transfer/ipfs/client.d.ts +109 -0
  696. package/v3/@claude-flow/cli/dist/src/transfer/ipfs/client.js +307 -0
  697. package/v3/@claude-flow/cli/dist/src/transfer/ipfs/upload.d.ts +95 -0
  698. package/v3/@claude-flow/cli/dist/src/transfer/ipfs/upload.js +411 -0
  699. package/v3/@claude-flow/cli/dist/src/transfer/models/seraphine.d.ts +72 -0
  700. package/v3/@claude-flow/cli/dist/src/transfer/models/seraphine.js +373 -0
  701. package/v3/@claude-flow/cli/dist/src/transfer/serialization/cfp.d.ts +49 -0
  702. package/v3/@claude-flow/cli/dist/src/transfer/serialization/cfp.js +183 -0
  703. package/v3/@claude-flow/cli/dist/src/transfer/storage/gcs.d.ts +82 -0
  704. package/v3/@claude-flow/cli/dist/src/transfer/storage/gcs.js +230 -0
  705. package/v3/@claude-flow/cli/dist/src/transfer/storage/index.d.ts +6 -0
  706. package/v3/@claude-flow/cli/dist/src/transfer/storage/index.js +6 -0
  707. package/v3/@claude-flow/cli/dist/src/transfer/store/discovery.d.ts +84 -0
  708. package/v3/@claude-flow/cli/dist/src/transfer/store/discovery.js +382 -0
  709. package/v3/@claude-flow/cli/dist/src/transfer/store/download.d.ts +70 -0
  710. package/v3/@claude-flow/cli/dist/src/transfer/store/download.js +334 -0
  711. package/v3/@claude-flow/cli/dist/src/transfer/store/index.d.ts +84 -0
  712. package/v3/@claude-flow/cli/dist/src/transfer/store/index.js +153 -0
  713. package/v3/@claude-flow/cli/dist/src/transfer/store/publish.d.ts +76 -0
  714. package/v3/@claude-flow/cli/dist/src/transfer/store/publish.js +294 -0
  715. package/v3/@claude-flow/cli/dist/src/transfer/store/registry.d.ts +58 -0
  716. package/v3/@claude-flow/cli/dist/src/transfer/store/registry.js +285 -0
  717. package/v3/@claude-flow/cli/dist/src/transfer/store/search.d.ts +54 -0
  718. package/v3/@claude-flow/cli/dist/src/transfer/store/search.js +232 -0
  719. package/v3/@claude-flow/cli/dist/src/transfer/store/tests/standalone-test.d.ts +12 -0
  720. package/v3/@claude-flow/cli/dist/src/transfer/store/tests/standalone-test.js +190 -0
  721. package/v3/@claude-flow/cli/dist/src/transfer/store/types.d.ts +193 -0
  722. package/v3/@claude-flow/cli/dist/src/transfer/store/types.js +6 -0
  723. package/v3/@claude-flow/cli/dist/src/transfer/test-seraphine.d.ts +6 -0
  724. package/v3/@claude-flow/cli/dist/src/transfer/test-seraphine.js +105 -0
  725. package/v3/@claude-flow/cli/dist/src/transfer/tests/test-store.d.ts +7 -0
  726. package/v3/@claude-flow/cli/dist/src/transfer/tests/test-store.js +214 -0
  727. package/v3/@claude-flow/cli/dist/src/transfer/types.d.ts +245 -0
  728. package/v3/@claude-flow/cli/dist/src/transfer/types.js +6 -0
  729. package/v3/@claude-flow/cli/dist/src/types.d.ts +198 -0
  730. package/v3/@claude-flow/cli/dist/src/types.js +38 -0
  731. package/v3/@claude-flow/cli/dist/src/update/checker.d.ts +34 -0
  732. package/v3/@claude-flow/cli/dist/src/update/checker.js +190 -0
  733. package/v3/@claude-flow/cli/dist/src/update/executor.d.ts +32 -0
  734. package/v3/@claude-flow/cli/dist/src/update/executor.js +183 -0
  735. package/v3/@claude-flow/cli/dist/src/update/index.d.ts +33 -0
  736. package/v3/@claude-flow/cli/dist/src/update/index.js +64 -0
  737. package/v3/@claude-flow/cli/dist/src/update/rate-limiter.d.ts +20 -0
  738. package/v3/@claude-flow/cli/dist/src/update/rate-limiter.js +96 -0
  739. package/v3/@claude-flow/cli/dist/src/update/validator.d.ts +17 -0
  740. package/v3/@claude-flow/cli/dist/src/update/validator.js +123 -0
  741. package/v3/@claude-flow/cli/package.json +110 -0
  742. package/v3/@claude-flow/guidance/README.md +1195 -0
  743. package/v3/@claude-flow/guidance/package.json +198 -0
  744. package/v3/@claude-flow/shared/README.md +323 -0
  745. package/v3/@claude-flow/shared/dist/core/config/defaults.d.ts +41 -0
  746. package/v3/@claude-flow/shared/dist/core/config/defaults.js +186 -0
  747. package/v3/@claude-flow/shared/dist/core/config/index.d.ts +8 -0
  748. package/v3/@claude-flow/shared/dist/core/config/index.js +12 -0
  749. package/v3/@claude-flow/shared/dist/core/config/loader.d.ts +45 -0
  750. package/v3/@claude-flow/shared/dist/core/config/loader.js +222 -0
  751. package/v3/@claude-flow/shared/dist/core/config/schema.d.ts +1134 -0
  752. package/v3/@claude-flow/shared/dist/core/config/schema.js +158 -0
  753. package/v3/@claude-flow/shared/dist/core/config/validator.d.ts +92 -0
  754. package/v3/@claude-flow/shared/dist/core/config/validator.js +147 -0
  755. package/v3/@claude-flow/shared/dist/core/event-bus.d.ts +31 -0
  756. package/v3/@claude-flow/shared/dist/core/event-bus.js +197 -0
  757. package/v3/@claude-flow/shared/dist/core/index.d.ts +15 -0
  758. package/v3/@claude-flow/shared/dist/core/index.js +19 -0
  759. package/v3/@claude-flow/shared/dist/core/interfaces/agent.interface.d.ts +200 -0
  760. package/v3/@claude-flow/shared/dist/core/interfaces/agent.interface.js +6 -0
  761. package/v3/@claude-flow/shared/dist/core/interfaces/coordinator.interface.d.ts +310 -0
  762. package/v3/@claude-flow/shared/dist/core/interfaces/coordinator.interface.js +7 -0
  763. package/v3/@claude-flow/shared/dist/core/interfaces/event.interface.d.ts +224 -0
  764. package/v3/@claude-flow/shared/dist/core/interfaces/event.interface.js +46 -0
  765. package/v3/@claude-flow/shared/dist/core/interfaces/index.d.ts +10 -0
  766. package/v3/@claude-flow/shared/dist/core/interfaces/index.js +15 -0
  767. package/v3/@claude-flow/shared/dist/core/interfaces/memory.interface.d.ts +298 -0
  768. package/v3/@claude-flow/shared/dist/core/interfaces/memory.interface.js +7 -0
  769. package/v3/@claude-flow/shared/dist/core/interfaces/task.interface.d.ts +185 -0
  770. package/v3/@claude-flow/shared/dist/core/interfaces/task.interface.js +6 -0
  771. package/v3/@claude-flow/shared/dist/core/orchestrator/event-coordinator.d.ts +35 -0
  772. package/v3/@claude-flow/shared/dist/core/orchestrator/event-coordinator.js +101 -0
  773. package/v3/@claude-flow/shared/dist/core/orchestrator/health-monitor.d.ts +60 -0
  774. package/v3/@claude-flow/shared/dist/core/orchestrator/health-monitor.js +166 -0
  775. package/v3/@claude-flow/shared/dist/core/orchestrator/index.d.ts +46 -0
  776. package/v3/@claude-flow/shared/dist/core/orchestrator/index.js +64 -0
  777. package/v3/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.d.ts +56 -0
  778. package/v3/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.js +195 -0
  779. package/v3/@claude-flow/shared/dist/core/orchestrator/session-manager.d.ts +83 -0
  780. package/v3/@claude-flow/shared/dist/core/orchestrator/session-manager.js +193 -0
  781. package/v3/@claude-flow/shared/dist/core/orchestrator/task-manager.d.ts +49 -0
  782. package/v3/@claude-flow/shared/dist/core/orchestrator/task-manager.js +253 -0
  783. package/v3/@claude-flow/shared/dist/events/domain-events.d.ts +282 -0
  784. package/v3/@claude-flow/shared/dist/events/domain-events.js +165 -0
  785. package/v3/@claude-flow/shared/dist/events/event-store.d.ts +126 -0
  786. package/v3/@claude-flow/shared/dist/events/event-store.js +416 -0
  787. package/v3/@claude-flow/shared/dist/events/event-store.test.d.ts +8 -0
  788. package/v3/@claude-flow/shared/dist/events/event-store.test.js +293 -0
  789. package/v3/@claude-flow/shared/dist/events/example-usage.d.ts +10 -0
  790. package/v3/@claude-flow/shared/dist/events/example-usage.js +193 -0
  791. package/v3/@claude-flow/shared/dist/events/index.d.ts +21 -0
  792. package/v3/@claude-flow/shared/dist/events/index.js +22 -0
  793. package/v3/@claude-flow/shared/dist/events/projections.d.ts +177 -0
  794. package/v3/@claude-flow/shared/dist/events/projections.js +421 -0
  795. package/v3/@claude-flow/shared/dist/events/rvf-event-log.d.ts +82 -0
  796. package/v3/@claude-flow/shared/dist/events/rvf-event-log.js +340 -0
  797. package/v3/@claude-flow/shared/dist/events/state-reconstructor.d.ts +101 -0
  798. package/v3/@claude-flow/shared/dist/events/state-reconstructor.js +263 -0
  799. package/v3/@claude-flow/shared/dist/events.d.ts +80 -0
  800. package/v3/@claude-flow/shared/dist/events.js +249 -0
  801. package/v3/@claude-flow/shared/dist/hooks/example-usage.d.ts +42 -0
  802. package/v3/@claude-flow/shared/dist/hooks/example-usage.js +351 -0
  803. package/v3/@claude-flow/shared/dist/hooks/executor.d.ts +100 -0
  804. package/v3/@claude-flow/shared/dist/hooks/executor.js +264 -0
  805. package/v3/@claude-flow/shared/dist/hooks/hooks.test.d.ts +9 -0
  806. package/v3/@claude-flow/shared/dist/hooks/hooks.test.js +322 -0
  807. package/v3/@claude-flow/shared/dist/hooks/index.d.ts +52 -0
  808. package/v3/@claude-flow/shared/dist/hooks/index.js +51 -0
  809. package/v3/@claude-flow/shared/dist/hooks/registry.d.ts +133 -0
  810. package/v3/@claude-flow/shared/dist/hooks/registry.js +277 -0
  811. package/v3/@claude-flow/shared/dist/hooks/safety/bash-safety.d.ts +105 -0
  812. package/v3/@claude-flow/shared/dist/hooks/safety/bash-safety.js +481 -0
  813. package/v3/@claude-flow/shared/dist/hooks/safety/file-organization.d.ts +144 -0
  814. package/v3/@claude-flow/shared/dist/hooks/safety/file-organization.js +328 -0
  815. package/v3/@claude-flow/shared/dist/hooks/safety/git-commit.d.ts +158 -0
  816. package/v3/@claude-flow/shared/dist/hooks/safety/git-commit.js +450 -0
  817. package/v3/@claude-flow/shared/dist/hooks/safety/index.d.ts +17 -0
  818. package/v3/@claude-flow/shared/dist/hooks/safety/index.js +17 -0
  819. package/v3/@claude-flow/shared/dist/hooks/session-hooks.d.ts +234 -0
  820. package/v3/@claude-flow/shared/dist/hooks/session-hooks.js +334 -0
  821. package/v3/@claude-flow/shared/dist/hooks/task-hooks.d.ts +163 -0
  822. package/v3/@claude-flow/shared/dist/hooks/task-hooks.js +326 -0
  823. package/v3/@claude-flow/shared/dist/hooks/types.d.ts +267 -0
  824. package/v3/@claude-flow/shared/dist/hooks/types.js +62 -0
  825. package/v3/@claude-flow/shared/dist/hooks/verify-exports.test.d.ts +9 -0
  826. package/v3/@claude-flow/shared/dist/hooks/verify-exports.test.js +93 -0
  827. package/v3/@claude-flow/shared/dist/index.d.ts +20 -0
  828. package/v3/@claude-flow/shared/dist/index.js +50 -0
  829. package/v3/@claude-flow/shared/dist/mcp/connection-pool.d.ts +98 -0
  830. package/v3/@claude-flow/shared/dist/mcp/connection-pool.js +364 -0
  831. package/v3/@claude-flow/shared/dist/mcp/index.d.ts +69 -0
  832. package/v3/@claude-flow/shared/dist/mcp/index.js +84 -0
  833. package/v3/@claude-flow/shared/dist/mcp/server.d.ts +166 -0
  834. package/v3/@claude-flow/shared/dist/mcp/server.js +593 -0
  835. package/v3/@claude-flow/shared/dist/mcp/session-manager.d.ts +136 -0
  836. package/v3/@claude-flow/shared/dist/mcp/session-manager.js +335 -0
  837. package/v3/@claude-flow/shared/dist/mcp/tool-registry.d.ts +178 -0
  838. package/v3/@claude-flow/shared/dist/mcp/tool-registry.js +439 -0
  839. package/v3/@claude-flow/shared/dist/mcp/transport/http.d.ts +104 -0
  840. package/v3/@claude-flow/shared/dist/mcp/transport/http.js +476 -0
  841. package/v3/@claude-flow/shared/dist/mcp/transport/index.d.ts +102 -0
  842. package/v3/@claude-flow/shared/dist/mcp/transport/index.js +238 -0
  843. package/v3/@claude-flow/shared/dist/mcp/transport/stdio.d.ts +104 -0
  844. package/v3/@claude-flow/shared/dist/mcp/transport/stdio.js +263 -0
  845. package/v3/@claude-flow/shared/dist/mcp/transport/websocket.d.ts +133 -0
  846. package/v3/@claude-flow/shared/dist/mcp/transport/websocket.js +396 -0
  847. package/v3/@claude-flow/shared/dist/mcp/types.d.ts +438 -0
  848. package/v3/@claude-flow/shared/dist/mcp/types.js +54 -0
  849. package/v3/@claude-flow/shared/dist/plugin-interface.d.ts +544 -0
  850. package/v3/@claude-flow/shared/dist/plugin-interface.js +23 -0
  851. package/v3/@claude-flow/shared/dist/plugin-loader.d.ts +139 -0
  852. package/v3/@claude-flow/shared/dist/plugin-loader.js +434 -0
  853. package/v3/@claude-flow/shared/dist/plugin-registry.d.ts +183 -0
  854. package/v3/@claude-flow/shared/dist/plugin-registry.js +457 -0
  855. package/v3/@claude-flow/shared/dist/plugins/index.d.ts +10 -0
  856. package/v3/@claude-flow/shared/dist/plugins/index.js +10 -0
  857. package/v3/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.d.ts +106 -0
  858. package/v3/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.js +241 -0
  859. package/v3/@claude-flow/shared/dist/plugins/official/index.d.ts +10 -0
  860. package/v3/@claude-flow/shared/dist/plugins/official/index.js +10 -0
  861. package/v3/@claude-flow/shared/dist/plugins/official/maestro-plugin.d.ts +121 -0
  862. package/v3/@claude-flow/shared/dist/plugins/official/maestro-plugin.js +355 -0
  863. package/v3/@claude-flow/shared/dist/plugins/types.d.ts +93 -0
  864. package/v3/@claude-flow/shared/dist/plugins/types.js +9 -0
  865. package/v3/@claude-flow/shared/dist/resilience/bulkhead.d.ts +105 -0
  866. package/v3/@claude-flow/shared/dist/resilience/bulkhead.js +206 -0
  867. package/v3/@claude-flow/shared/dist/resilience/circuit-breaker.d.ts +132 -0
  868. package/v3/@claude-flow/shared/dist/resilience/circuit-breaker.js +233 -0
  869. package/v3/@claude-flow/shared/dist/resilience/index.d.ts +19 -0
  870. package/v3/@claude-flow/shared/dist/resilience/index.js +19 -0
  871. package/v3/@claude-flow/shared/dist/resilience/rate-limiter.d.ts +168 -0
  872. package/v3/@claude-flow/shared/dist/resilience/rate-limiter.js +314 -0
  873. package/v3/@claude-flow/shared/dist/resilience/retry.d.ts +91 -0
  874. package/v3/@claude-flow/shared/dist/resilience/retry.js +159 -0
  875. package/v3/@claude-flow/shared/dist/security/index.d.ts +10 -0
  876. package/v3/@claude-flow/shared/dist/security/index.js +12 -0
  877. package/v3/@claude-flow/shared/dist/security/input-validation.d.ts +73 -0
  878. package/v3/@claude-flow/shared/dist/security/input-validation.js +201 -0
  879. package/v3/@claude-flow/shared/dist/security/secure-random.d.ts +92 -0
  880. package/v3/@claude-flow/shared/dist/security/secure-random.js +142 -0
  881. package/v3/@claude-flow/shared/dist/services/index.d.ts +7 -0
  882. package/v3/@claude-flow/shared/dist/services/index.js +7 -0
  883. package/v3/@claude-flow/shared/dist/services/v3-progress.service.d.ts +124 -0
  884. package/v3/@claude-flow/shared/dist/services/v3-progress.service.js +402 -0
  885. package/v3/@claude-flow/shared/dist/types/agent.types.d.ts +137 -0
  886. package/v3/@claude-flow/shared/dist/types/agent.types.js +6 -0
  887. package/v3/@claude-flow/shared/dist/types/index.d.ts +11 -0
  888. package/v3/@claude-flow/shared/dist/types/index.js +17 -0
  889. package/v3/@claude-flow/shared/dist/types/mcp.types.d.ts +266 -0
  890. package/v3/@claude-flow/shared/dist/types/mcp.types.js +7 -0
  891. package/v3/@claude-flow/shared/dist/types/memory.types.d.ts +236 -0
  892. package/v3/@claude-flow/shared/dist/types/memory.types.js +7 -0
  893. package/v3/@claude-flow/shared/dist/types/swarm.types.d.ts +186 -0
  894. package/v3/@claude-flow/shared/dist/types/swarm.types.js +65 -0
  895. package/v3/@claude-flow/shared/dist/types/task.types.d.ts +178 -0
  896. package/v3/@claude-flow/shared/dist/types/task.types.js +32 -0
  897. package/v3/@claude-flow/shared/dist/types.d.ts +197 -0
  898. package/v3/@claude-flow/shared/dist/types.js +21 -0
  899. package/v3/@claude-flow/shared/dist/utils/secure-logger.d.ts +69 -0
  900. package/v3/@claude-flow/shared/dist/utils/secure-logger.js +208 -0
  901. package/v3/@claude-flow/shared/package.json +42 -0
  902. package/v3/README.md +493 -0
@@ -0,0 +1,1979 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * Context Persistence Hook (ADR-051)
4
+ *
5
+ * Intercepts Claude Code's PreCompact, SessionStart, and UserPromptSubmit
6
+ * lifecycle events to persist conversation history in SQLite (primary),
7
+ * RuVector PostgreSQL (optional), or JSON (fallback), enabling "infinite
8
+ * context" across compaction boundaries.
9
+ *
10
+ * Backend priority:
11
+ * 1. better-sqlite3 (native, WAL mode, indexed queries, ACID transactions)
12
+ * 2. RuVector PostgreSQL (if RUVECTOR_* env vars set - TB-scale, GNN search)
13
+ * 3. AgentDB from @claude-flow/memory (HNSW vector search)
14
+ * 4. JsonFileBackend (zero dependencies, always works)
15
+ *
16
+ * Proactive archiving:
17
+ * - UserPromptSubmit hook archives on every prompt, BEFORE context fills up
18
+ * - PreCompact hook is a safety net that catches any remaining unarchived turns
19
+ * - SessionStart hook restores context after compaction
20
+ * - Together, compaction becomes invisible — no information is ever lost
21
+ *
22
+ * Usage:
23
+ * node context-persistence-hook.mjs pre-compact # PreCompact: archive transcript
24
+ * node context-persistence-hook.mjs session-start # SessionStart: restore context
25
+ * node context-persistence-hook.mjs user-prompt-submit # UserPromptSubmit: proactive archive
26
+ * node context-persistence-hook.mjs status # Show archive stats
27
+ */
28
+
29
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';
30
+ import { createHash } from 'crypto';
31
+ import { join, dirname } from 'path';
32
+ import { fileURLToPath } from 'url';
33
+ import { createRequire } from 'module';
34
+
35
+ const __filename = fileURLToPath(import.meta.url);
36
+ const __dirname = dirname(__filename);
37
+ const PROJECT_ROOT = join(__dirname, '../..');
38
+ const DATA_DIR = join(PROJECT_ROOT, '.claude-flow', 'data');
39
+ const ARCHIVE_JSON_PATH = join(DATA_DIR, 'transcript-archive.json');
40
+ const ARCHIVE_DB_PATH = join(DATA_DIR, 'transcript-archive.db');
41
+
42
+ const NAMESPACE = 'transcript-archive';
43
+ const RESTORE_BUDGET = parseInt(process.env.CLAUDE_FLOW_COMPACT_RESTORE_BUDGET || '4000', 10);
44
+ const MAX_MESSAGES = 500;
45
+ const BLOCK_COMPACTION = process.env.CLAUDE_FLOW_BLOCK_COMPACTION === 'true';
46
+ const COMPACT_INSTRUCTION_BUDGET = parseInt(process.env.CLAUDE_FLOW_COMPACT_INSTRUCTION_BUDGET || '2000', 10);
47
+ const RETENTION_DAYS = parseInt(process.env.CLAUDE_FLOW_RETENTION_DAYS || '30', 10);
48
+ const AUTO_OPTIMIZE = process.env.CLAUDE_FLOW_AUTO_OPTIMIZE !== 'false'; // on by default
49
+
50
+ // ============================================================================
51
+ // Context Autopilot — prevent compaction by managing context size in real-time
52
+ // ============================================================================
53
+ const AUTOPILOT_ENABLED = process.env.CLAUDE_FLOW_CONTEXT_AUTOPILOT !== 'false'; // on by default
54
+ const CONTEXT_WINDOW_TOKENS = parseInt(process.env.CLAUDE_FLOW_CONTEXT_WINDOW || '200000', 10);
55
+ const AUTOPILOT_WARN_PCT = parseFloat(process.env.CLAUDE_FLOW_AUTOPILOT_WARN || '0.70');
56
+ const AUTOPILOT_PRUNE_PCT = parseFloat(process.env.CLAUDE_FLOW_AUTOPILOT_PRUNE || '0.85');
57
+ const AUTOPILOT_STATE_PATH = join(DATA_DIR, 'autopilot-state.json');
58
+
59
+ // Approximate tokens per character (Claude averages ~3.5 chars per token)
60
+ const CHARS_PER_TOKEN = 3.5;
61
+
62
+ // Ensure data dir
63
+ if (!existsSync(DATA_DIR)) mkdirSync(DATA_DIR, { recursive: true });
64
+
65
+ // ============================================================================
66
+ // SQLite Backend (better-sqlite3 — synchronous, fast, WAL mode)
67
+ // ============================================================================
68
+
69
+ class SQLiteBackend {
70
+ constructor(dbPath) {
71
+ this.dbPath = dbPath;
72
+ this.db = null;
73
+ }
74
+
75
+ async initialize() {
76
+ const require = createRequire(import.meta.url);
77
+ const Database = require('better-sqlite3');
78
+ this.db = new Database(this.dbPath);
79
+
80
+ // Performance optimizations
81
+ this.db.pragma('journal_mode = WAL');
82
+ this.db.pragma('synchronous = NORMAL');
83
+ this.db.pragma('cache_size = 5000');
84
+ this.db.pragma('temp_store = MEMORY');
85
+
86
+ // Create schema
87
+ this.db.exec(`
88
+ CREATE TABLE IF NOT EXISTS transcript_entries (
89
+ id TEXT PRIMARY KEY,
90
+ key TEXT NOT NULL,
91
+ content TEXT NOT NULL,
92
+ type TEXT NOT NULL DEFAULT 'episodic',
93
+ namespace TEXT NOT NULL DEFAULT 'transcript-archive',
94
+ tags TEXT NOT NULL DEFAULT '[]',
95
+ metadata TEXT NOT NULL DEFAULT '{}',
96
+ access_level TEXT NOT NULL DEFAULT 'private',
97
+ created_at INTEGER NOT NULL,
98
+ updated_at INTEGER NOT NULL,
99
+ version INTEGER NOT NULL DEFAULT 1,
100
+ access_count INTEGER NOT NULL DEFAULT 0,
101
+ last_accessed_at INTEGER NOT NULL,
102
+ content_hash TEXT,
103
+ session_id TEXT,
104
+ chunk_index INTEGER,
105
+ summary TEXT
106
+ );
107
+
108
+ CREATE INDEX IF NOT EXISTS idx_te_namespace ON transcript_entries(namespace);
109
+ CREATE INDEX IF NOT EXISTS idx_te_session ON transcript_entries(session_id);
110
+ CREATE INDEX IF NOT EXISTS idx_te_hash ON transcript_entries(content_hash);
111
+ CREATE INDEX IF NOT EXISTS idx_te_chunk ON transcript_entries(session_id, chunk_index);
112
+ CREATE INDEX IF NOT EXISTS idx_te_created ON transcript_entries(created_at);
113
+ `);
114
+
115
+ // Schema migration: add confidence + embedding columns (self-learning support)
116
+ try {
117
+ this.db.exec(`ALTER TABLE transcript_entries ADD COLUMN confidence REAL NOT NULL DEFAULT 0.8`);
118
+ } catch { /* column already exists */ }
119
+ try {
120
+ this.db.exec(`ALTER TABLE transcript_entries ADD COLUMN embedding BLOB`);
121
+ } catch { /* column already exists */ }
122
+ try {
123
+ this.db.exec(`CREATE INDEX IF NOT EXISTS idx_te_confidence ON transcript_entries(confidence)`);
124
+ } catch { /* index already exists */ }
125
+
126
+ // Prepare statements for reuse
127
+ this._stmts = {
128
+ insert: this.db.prepare(`
129
+ INSERT OR IGNORE INTO transcript_entries
130
+ (id, key, content, type, namespace, tags, metadata, access_level,
131
+ created_at, updated_at, version, access_count, last_accessed_at,
132
+ content_hash, session_id, chunk_index, summary)
133
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
134
+ `),
135
+ queryByNamespace: this.db.prepare(
136
+ 'SELECT * FROM transcript_entries WHERE namespace = ? ORDER BY created_at DESC'
137
+ ),
138
+ queryBySession: this.db.prepare(
139
+ 'SELECT * FROM transcript_entries WHERE namespace = ? AND session_id = ? ORDER BY chunk_index DESC'
140
+ ),
141
+ countAll: this.db.prepare('SELECT COUNT(*) as cnt FROM transcript_entries'),
142
+ countByNamespace: this.db.prepare(
143
+ 'SELECT COUNT(*) as cnt FROM transcript_entries WHERE namespace = ?'
144
+ ),
145
+ hashExists: this.db.prepare(
146
+ 'SELECT 1 FROM transcript_entries WHERE content_hash = ? LIMIT 1'
147
+ ),
148
+ listNamespaces: this.db.prepare(
149
+ 'SELECT DISTINCT namespace FROM transcript_entries'
150
+ ),
151
+ listSessions: this.db.prepare(
152
+ 'SELECT session_id, COUNT(*) as cnt FROM transcript_entries WHERE namespace = ? GROUP BY session_id ORDER BY MAX(created_at) DESC'
153
+ ),
154
+ };
155
+
156
+ this._bulkInsert = this.db.transaction((entries) => {
157
+ for (const e of entries) {
158
+ this._stmts.insert.run(
159
+ e.id, e.key, e.content, e.type, e.namespace,
160
+ JSON.stringify(e.tags), JSON.stringify(e.metadata), e.accessLevel,
161
+ e.createdAt, e.updatedAt, e.version, e.accessCount, e.lastAccessedAt,
162
+ e.metadata?.contentHash || null,
163
+ e.metadata?.sessionId || null,
164
+ e.metadata?.chunkIndex ?? null,
165
+ e.metadata?.summary || null
166
+ );
167
+ }
168
+ });
169
+
170
+ // Optimization statements
171
+ this._stmts.markAccessed = this.db.prepare(
172
+ 'UPDATE transcript_entries SET access_count = access_count + 1, last_accessed_at = ? WHERE id = ?'
173
+ );
174
+ this._stmts.pruneStale = this.db.prepare(
175
+ 'DELETE FROM transcript_entries WHERE namespace = ? AND access_count = 0 AND created_at < ?'
176
+ );
177
+ this._stmts.queryByImportance = this.db.prepare(`
178
+ SELECT *, (
179
+ (CAST(access_count AS REAL) + 1) *
180
+ (1.0 / (1.0 + (? - created_at) / 86400000.0)) *
181
+ (CASE WHEN json_array_length(json_extract(metadata, '$.toolNames')) > 0 THEN 1.5 ELSE 1.0 END) *
182
+ (CASE WHEN json_array_length(json_extract(metadata, '$.filePaths')) > 0 THEN 1.3 ELSE 1.0 END)
183
+ ) AS importance_score
184
+ FROM transcript_entries
185
+ WHERE namespace = ? AND session_id = ?
186
+ ORDER BY importance_score DESC
187
+ `);
188
+ this._stmts.allForSync = this.db.prepare(
189
+ 'SELECT * FROM transcript_entries WHERE namespace = ? ORDER BY created_at ASC'
190
+ );
191
+ }
192
+
193
+ async store(entry) {
194
+ this._stmts.insert.run(
195
+ entry.id, entry.key, entry.content, entry.type, entry.namespace,
196
+ JSON.stringify(entry.tags), JSON.stringify(entry.metadata), entry.accessLevel,
197
+ entry.createdAt, entry.updatedAt, entry.version, entry.accessCount, entry.lastAccessedAt,
198
+ entry.metadata?.contentHash || null,
199
+ entry.metadata?.sessionId || null,
200
+ entry.metadata?.chunkIndex ?? null,
201
+ entry.metadata?.summary || null
202
+ );
203
+ }
204
+
205
+ async bulkInsert(entries) {
206
+ this._bulkInsert(entries);
207
+ }
208
+
209
+ async query(opts) {
210
+ let rows;
211
+ if (opts?.namespace && opts?.sessionId) {
212
+ rows = this._stmts.queryBySession.all(opts.namespace, opts.sessionId);
213
+ } else if (opts?.namespace) {
214
+ rows = this._stmts.queryByNamespace.all(opts.namespace);
215
+ } else {
216
+ rows = this.db.prepare('SELECT * FROM transcript_entries ORDER BY created_at DESC').all();
217
+ }
218
+ return rows.map(r => this._rowToEntry(r));
219
+ }
220
+
221
+ async queryBySession(namespace, sessionId) {
222
+ const rows = this._stmts.queryBySession.all(namespace, sessionId);
223
+ return rows.map(r => this._rowToEntry(r));
224
+ }
225
+
226
+ hashExists(hash) {
227
+ return !!this._stmts.hashExists.get(hash);
228
+ }
229
+
230
+ async count(namespace) {
231
+ if (namespace) {
232
+ return this._stmts.countByNamespace.get(namespace).cnt;
233
+ }
234
+ return this._stmts.countAll.get().cnt;
235
+ }
236
+
237
+ async listNamespaces() {
238
+ return this._stmts.listNamespaces.all().map(r => r.namespace);
239
+ }
240
+
241
+ async listSessions(namespace) {
242
+ return this._stmts.listSessions.all(namespace || NAMESPACE);
243
+ }
244
+
245
+ markAccessed(ids) {
246
+ const now = Date.now();
247
+ const boostStmt = this.db.prepare(
248
+ 'UPDATE transcript_entries SET access_count = access_count + 1, last_accessed_at = ?, confidence = MIN(1.0, confidence + 0.03) WHERE id = ?'
249
+ );
250
+ for (const id of ids) {
251
+ boostStmt.run(now, id);
252
+ }
253
+ }
254
+
255
+ /**
256
+ * Confidence decay: reduce confidence for entries not accessed recently.
257
+ * Decay rate: 0.5% per hour (matches LearningBridge default).
258
+ * Entries with confidence below 0.1 are floor-clamped.
259
+ */
260
+ decayConfidence(namespace, hoursElapsed = 1) {
261
+ const decayRate = 0.005 * hoursElapsed;
262
+ const result = this.db.prepare(
263
+ 'UPDATE transcript_entries SET confidence = MAX(0.1, confidence - ?) WHERE namespace = ? AND confidence > 0.1'
264
+ ).run(decayRate, namespace || NAMESPACE);
265
+ return result.changes;
266
+ }
267
+
268
+ /**
269
+ * Store embedding blob for an entry (768-dim Float32Array → Buffer).
270
+ */
271
+ storeEmbedding(id, embedding) {
272
+ const buf = Buffer.from(embedding.buffer, embedding.byteOffset, embedding.byteLength);
273
+ this.db.prepare('UPDATE transcript_entries SET embedding = ? WHERE id = ?').run(buf, id);
274
+ }
275
+
276
+ /**
277
+ * Cosine similarity search across all entries with embeddings.
278
+ * Handles both 384-dim (ONNX) and 768-dim (legacy hash) embeddings.
279
+ * Returns top-k entries ranked by similarity to the query embedding.
280
+ */
281
+ semanticSearch(queryEmbedding, k = 10, namespace) {
282
+ const rows = this.db.prepare(
283
+ 'SELECT id, embedding, summary, session_id, chunk_index, confidence, access_count FROM transcript_entries WHERE namespace = ? AND embedding IS NOT NULL'
284
+ ).all(namespace || NAMESPACE);
285
+
286
+ const queryDim = queryEmbedding.length;
287
+ const scored = [];
288
+ for (const row of rows) {
289
+ if (!row.embedding) continue;
290
+ const stored = new Float32Array(row.embedding.buffer, row.embedding.byteOffset, row.embedding.byteLength / 4);
291
+ // Only compare if dimensions match
292
+ if (stored.length !== queryDim) continue;
293
+ let dot = 0;
294
+ for (let i = 0; i < queryDim; i++) {
295
+ dot += queryEmbedding[i] * stored[i];
296
+ }
297
+ // Boost by confidence (self-learning signal)
298
+ const score = dot * (row.confidence || 0.8);
299
+ scored.push({ id: row.id, score, summary: row.summary, sessionId: row.session_id, chunkIndex: row.chunk_index, confidence: row.confidence, accessCount: row.access_count });
300
+ }
301
+
302
+ scored.sort((a, b) => b.score - a.score);
303
+ return scored.slice(0, k);
304
+ }
305
+
306
+ /**
307
+ * Smart pruning: prune by confidence instead of just age.
308
+ * Removes entries with confidence <= threshold AND access_count = 0.
309
+ */
310
+ pruneByConfidence(namespace, threshold = 0.2) {
311
+ const result = this.db.prepare(
312
+ 'DELETE FROM transcript_entries WHERE namespace = ? AND confidence <= ? AND access_count = 0'
313
+ ).run(namespace || NAMESPACE, threshold);
314
+ return result.changes;
315
+ }
316
+
317
+ pruneStale(namespace, maxAgeDays) {
318
+ const cutoff = Date.now() - (maxAgeDays * 24 * 60 * 60 * 1000);
319
+ const result = this._stmts.pruneStale.run(namespace || NAMESPACE, cutoff);
320
+ return result.changes;
321
+ }
322
+
323
+ queryByImportance(namespace, sessionId) {
324
+ const now = Date.now();
325
+ const rows = this._stmts.queryByImportance.all(now, namespace, sessionId);
326
+ return rows.map(r => ({ ...this._rowToEntry(r), importanceScore: r.importance_score }));
327
+ }
328
+
329
+ allForSync(namespace) {
330
+ const rows = this._stmts.allForSync.all(namespace || NAMESPACE);
331
+ return rows.map(r => this._rowToEntry(r));
332
+ }
333
+
334
+ async shutdown() {
335
+ if (this.db) {
336
+ this.db.pragma('optimize');
337
+ this.db.close();
338
+ this.db = null;
339
+ }
340
+ }
341
+
342
+ _rowToEntry(row) {
343
+ return {
344
+ id: row.id,
345
+ key: row.key,
346
+ content: row.content,
347
+ type: row.type,
348
+ namespace: row.namespace,
349
+ tags: JSON.parse(row.tags),
350
+ metadata: JSON.parse(row.metadata),
351
+ accessLevel: row.access_level,
352
+ createdAt: row.created_at,
353
+ updatedAt: row.updated_at,
354
+ version: row.version,
355
+ accessCount: row.access_count,
356
+ lastAccessedAt: row.last_accessed_at,
357
+ references: [],
358
+ };
359
+ }
360
+ }
361
+
362
+ // ============================================================================
363
+ // JSON File Backend (fallback when better-sqlite3 unavailable)
364
+ // ============================================================================
365
+
366
+ class JsonFileBackend {
367
+ constructor(filePath) {
368
+ this.filePath = filePath;
369
+ this.entries = new Map();
370
+ }
371
+
372
+ async initialize() {
373
+ if (existsSync(this.filePath)) {
374
+ try {
375
+ const data = JSON.parse(readFileSync(this.filePath, 'utf-8'));
376
+ if (Array.isArray(data)) {
377
+ for (const entry of data) this.entries.set(entry.id, entry);
378
+ }
379
+ } catch { /* start fresh */ }
380
+ }
381
+ }
382
+
383
+ async store(entry) { this.entries.set(entry.id, entry); this._persist(); }
384
+
385
+ async bulkInsert(entries) {
386
+ for (const e of entries) this.entries.set(e.id, e);
387
+ this._persist();
388
+ }
389
+
390
+ async query(opts) {
391
+ let results = [...this.entries.values()];
392
+ if (opts?.namespace) results = results.filter(e => e.namespace === opts.namespace);
393
+ if (opts?.type) results = results.filter(e => e.type === opts.type);
394
+ if (opts?.limit) results = results.slice(0, opts.limit);
395
+ return results;
396
+ }
397
+
398
+ async queryBySession(namespace, sessionId) {
399
+ return [...this.entries.values()]
400
+ .filter(e => e.namespace === namespace && e.metadata?.sessionId === sessionId)
401
+ .sort((a, b) => (b.metadata?.chunkIndex ?? 0) - (a.metadata?.chunkIndex ?? 0));
402
+ }
403
+
404
+ hashExists(hash) {
405
+ for (const e of this.entries.values()) {
406
+ if (e.metadata?.contentHash === hash) return true;
407
+ }
408
+ return false;
409
+ }
410
+
411
+ async count(namespace) {
412
+ if (!namespace) return this.entries.size;
413
+ let n = 0;
414
+ for (const e of this.entries.values()) {
415
+ if (e.namespace === namespace) n++;
416
+ }
417
+ return n;
418
+ }
419
+
420
+ async listNamespaces() {
421
+ const ns = new Set();
422
+ for (const e of this.entries.values()) ns.add(e.namespace || 'default');
423
+ return [...ns];
424
+ }
425
+
426
+ async listSessions(namespace) {
427
+ const sessions = new Map();
428
+ for (const e of this.entries.values()) {
429
+ if (e.namespace === (namespace || NAMESPACE) && e.metadata?.sessionId) {
430
+ sessions.set(e.metadata.sessionId, (sessions.get(e.metadata.sessionId) || 0) + 1);
431
+ }
432
+ }
433
+ return [...sessions.entries()].map(([session_id, cnt]) => ({ session_id, cnt }));
434
+ }
435
+
436
+ async shutdown() { this._persist(); }
437
+
438
+ _persist() {
439
+ try {
440
+ writeFileSync(this.filePath, JSON.stringify([...this.entries.values()], null, 2), 'utf-8');
441
+ } catch { /* best effort */ }
442
+ }
443
+ }
444
+
445
+ // ============================================================================
446
+ // RuVector PostgreSQL Backend (optional, TB-scale, GNN-enhanced)
447
+ // ============================================================================
448
+
449
+ class RuVectorBackend {
450
+ constructor(config) {
451
+ this.config = config;
452
+ this.pool = null;
453
+ }
454
+
455
+ async initialize() {
456
+ const pg = await import('pg');
457
+ const Pool = pg.default?.Pool || pg.Pool;
458
+ this.pool = new Pool({
459
+ host: this.config.host,
460
+ port: this.config.port || 5432,
461
+ database: this.config.database,
462
+ user: this.config.user,
463
+ password: this.config.password,
464
+ ssl: this.config.ssl || false,
465
+ max: 3,
466
+ idleTimeoutMillis: 10000,
467
+ connectionTimeoutMillis: 3000,
468
+ application_name: 'claude-flow-context-persistence',
469
+ });
470
+
471
+ // Test connection and create schema
472
+ const client = await this.pool.connect();
473
+ try {
474
+ await client.query(`
475
+ CREATE TABLE IF NOT EXISTS transcript_entries (
476
+ id TEXT PRIMARY KEY,
477
+ key TEXT NOT NULL,
478
+ content TEXT NOT NULL,
479
+ type TEXT NOT NULL DEFAULT 'episodic',
480
+ namespace TEXT NOT NULL DEFAULT 'transcript-archive',
481
+ tags JSONB NOT NULL DEFAULT '[]',
482
+ metadata JSONB NOT NULL DEFAULT '{}',
483
+ access_level TEXT NOT NULL DEFAULT 'private',
484
+ created_at BIGINT NOT NULL,
485
+ updated_at BIGINT NOT NULL,
486
+ version INTEGER NOT NULL DEFAULT 1,
487
+ access_count INTEGER NOT NULL DEFAULT 0,
488
+ last_accessed_at BIGINT NOT NULL,
489
+ content_hash TEXT,
490
+ session_id TEXT,
491
+ chunk_index INTEGER,
492
+ summary TEXT,
493
+ embedding vector(768)
494
+ );
495
+
496
+ CREATE INDEX IF NOT EXISTS idx_te_namespace ON transcript_entries(namespace);
497
+ CREATE INDEX IF NOT EXISTS idx_te_session ON transcript_entries(session_id);
498
+ CREATE INDEX IF NOT EXISTS idx_te_hash ON transcript_entries(content_hash);
499
+ CREATE INDEX IF NOT EXISTS idx_te_chunk ON transcript_entries(session_id, chunk_index);
500
+ CREATE INDEX IF NOT EXISTS idx_te_created ON transcript_entries(created_at);
501
+ `);
502
+ } finally {
503
+ client.release();
504
+ }
505
+ }
506
+
507
+ async store(entry) {
508
+ const embeddingArr = entry._embedding
509
+ ? `[${Array.from(entry._embedding).join(',')}]`
510
+ : null;
511
+ await this.pool.query(
512
+ `INSERT INTO transcript_entries
513
+ (id, key, content, type, namespace, tags, metadata, access_level,
514
+ created_at, updated_at, version, access_count, last_accessed_at,
515
+ content_hash, session_id, chunk_index, summary, embedding)
516
+ VALUES ($1,$2,$3,$4,$5,$6,$7,$8,$9,$10,$11,$12,$13,$14,$15,$16,$17,$18)
517
+ ON CONFLICT (id) DO NOTHING`,
518
+ [
519
+ entry.id, entry.key, entry.content, entry.type, entry.namespace,
520
+ JSON.stringify(entry.tags), JSON.stringify(entry.metadata), entry.accessLevel,
521
+ entry.createdAt, entry.updatedAt, entry.version, entry.accessCount, entry.lastAccessedAt,
522
+ entry.metadata?.contentHash || null,
523
+ entry.metadata?.sessionId || null,
524
+ entry.metadata?.chunkIndex ?? null,
525
+ entry.metadata?.summary || null,
526
+ embeddingArr,
527
+ ]
528
+ );
529
+ }
530
+
531
+ async bulkInsert(entries) {
532
+ const client = await this.pool.connect();
533
+ try {
534
+ await client.query('BEGIN');
535
+ for (const entry of entries) {
536
+ const embeddingArr = entry._embedding
537
+ ? `[${Array.from(entry._embedding).join(',')}]`
538
+ : null;
539
+ await client.query(
540
+ `INSERT INTO transcript_entries
541
+ (id, key, content, type, namespace, tags, metadata, access_level,
542
+ created_at, updated_at, version, access_count, last_accessed_at,
543
+ content_hash, session_id, chunk_index, summary, embedding)
544
+ VALUES ($1,$2,$3,$4,$5,$6,$7,$8,$9,$10,$11,$12,$13,$14,$15,$16,$17,$18)
545
+ ON CONFLICT (id) DO NOTHING`,
546
+ [
547
+ entry.id, entry.key, entry.content, entry.type, entry.namespace,
548
+ JSON.stringify(entry.tags), JSON.stringify(entry.metadata), entry.accessLevel,
549
+ entry.createdAt, entry.updatedAt, entry.version, entry.accessCount, entry.lastAccessedAt,
550
+ entry.metadata?.contentHash || null,
551
+ entry.metadata?.sessionId || null,
552
+ entry.metadata?.chunkIndex ?? null,
553
+ entry.metadata?.summary || null,
554
+ embeddingArr,
555
+ ]
556
+ );
557
+ }
558
+ await client.query('COMMIT');
559
+ } catch (err) {
560
+ await client.query('ROLLBACK');
561
+ throw err;
562
+ } finally {
563
+ client.release();
564
+ }
565
+ }
566
+
567
+ async query(opts) {
568
+ let sql = 'SELECT * FROM transcript_entries';
569
+ const params = [];
570
+ const clauses = [];
571
+ if (opts?.namespace) { params.push(opts.namespace); clauses.push(`namespace = $${params.length}`); }
572
+ if (clauses.length) sql += ' WHERE ' + clauses.join(' AND ');
573
+ sql += ' ORDER BY created_at DESC';
574
+ if (opts?.limit) { params.push(opts.limit); sql += ` LIMIT $${params.length}`; }
575
+ const { rows } = await this.pool.query(sql, params);
576
+ return rows.map(r => this._rowToEntry(r));
577
+ }
578
+
579
+ async queryBySession(namespace, sessionId) {
580
+ const { rows } = await this.pool.query(
581
+ 'SELECT * FROM transcript_entries WHERE namespace = $1 AND session_id = $2 ORDER BY chunk_index DESC',
582
+ [namespace, sessionId]
583
+ );
584
+ return rows.map(r => this._rowToEntry(r));
585
+ }
586
+
587
+ hashExists(hash) {
588
+ // Synchronous check not possible with pg — use a cached check
589
+ // The bulkInsert uses ON CONFLICT DO NOTHING for dedup at DB level
590
+ return false;
591
+ }
592
+
593
+ async hashExistsAsync(hash) {
594
+ const { rows } = await this.pool.query(
595
+ 'SELECT 1 FROM transcript_entries WHERE content_hash = $1 LIMIT 1',
596
+ [hash]
597
+ );
598
+ return rows.length > 0;
599
+ }
600
+
601
+ async count(namespace) {
602
+ const sql = namespace
603
+ ? 'SELECT COUNT(*) as cnt FROM transcript_entries WHERE namespace = $1'
604
+ : 'SELECT COUNT(*) as cnt FROM transcript_entries';
605
+ const params = namespace ? [namespace] : [];
606
+ const { rows } = await this.pool.query(sql, params);
607
+ return parseInt(rows[0].cnt, 10);
608
+ }
609
+
610
+ async listNamespaces() {
611
+ const { rows } = await this.pool.query('SELECT DISTINCT namespace FROM transcript_entries');
612
+ return rows.map(r => r.namespace);
613
+ }
614
+
615
+ async listSessions(namespace) {
616
+ const { rows } = await this.pool.query(
617
+ `SELECT session_id, COUNT(*) as cnt FROM transcript_entries
618
+ WHERE namespace = $1 GROUP BY session_id ORDER BY MAX(created_at) DESC`,
619
+ [namespace || NAMESPACE]
620
+ );
621
+ return rows.map(r => ({ session_id: r.session_id, cnt: parseInt(r.cnt, 10) }));
622
+ }
623
+
624
+ async markAccessed(ids) {
625
+ const now = Date.now();
626
+ for (const id of ids) {
627
+ await this.pool.query(
628
+ 'UPDATE transcript_entries SET access_count = access_count + 1, last_accessed_at = $1 WHERE id = $2',
629
+ [now, id]
630
+ );
631
+ }
632
+ }
633
+
634
+ async pruneStale(namespace, maxAgeDays) {
635
+ const cutoff = Date.now() - (maxAgeDays * 24 * 60 * 60 * 1000);
636
+ const { rowCount } = await this.pool.query(
637
+ 'DELETE FROM transcript_entries WHERE namespace = $1 AND access_count = 0 AND created_at < $2',
638
+ [namespace || NAMESPACE, cutoff]
639
+ );
640
+ return rowCount;
641
+ }
642
+
643
+ async queryByImportance(namespace, sessionId) {
644
+ const now = Date.now();
645
+ const { rows } = await this.pool.query(`
646
+ SELECT *, (
647
+ (CAST(access_count AS REAL) + 1) *
648
+ (1.0 / (1.0 + ($1 - created_at) / 86400000.0)) *
649
+ (CASE WHEN jsonb_array_length(metadata->'toolNames') > 0 THEN 1.5 ELSE 1.0 END) *
650
+ (CASE WHEN jsonb_array_length(metadata->'filePaths') > 0 THEN 1.3 ELSE 1.0 END)
651
+ ) AS importance_score
652
+ FROM transcript_entries
653
+ WHERE namespace = $2 AND session_id = $3
654
+ ORDER BY importance_score DESC
655
+ `, [now, namespace, sessionId]);
656
+ return rows.map(r => ({ ...this._rowToEntry(r), importanceScore: r.importance_score }));
657
+ }
658
+
659
+ async shutdown() {
660
+ if (this.pool) {
661
+ await this.pool.end();
662
+ this.pool = null;
663
+ }
664
+ }
665
+
666
+ _rowToEntry(row) {
667
+ return {
668
+ id: row.id,
669
+ key: row.key,
670
+ content: row.content,
671
+ type: row.type,
672
+ namespace: row.namespace,
673
+ tags: typeof row.tags === 'string' ? JSON.parse(row.tags) : row.tags,
674
+ metadata: typeof row.metadata === 'string' ? JSON.parse(row.metadata) : row.metadata,
675
+ accessLevel: row.access_level,
676
+ createdAt: parseInt(row.created_at, 10),
677
+ updatedAt: parseInt(row.updated_at, 10),
678
+ version: row.version,
679
+ accessCount: row.access_count,
680
+ lastAccessedAt: parseInt(row.last_accessed_at, 10),
681
+ references: [],
682
+ };
683
+ }
684
+ }
685
+
686
+ /**
687
+ * Parse RuVector config from environment variables.
688
+ * Returns null if required vars are not set.
689
+ */
690
+ function getRuVectorConfig() {
691
+ const host = process.env.RUVECTOR_HOST || process.env.PGHOST;
692
+ const database = process.env.RUVECTOR_DATABASE || process.env.PGDATABASE;
693
+ const user = process.env.RUVECTOR_USER || process.env.PGUSER;
694
+ const password = process.env.RUVECTOR_PASSWORD || process.env.PGPASSWORD;
695
+
696
+ if (!host || !database || !user) return null;
697
+
698
+ return {
699
+ host,
700
+ port: parseInt(process.env.RUVECTOR_PORT || process.env.PGPORT || '5432', 10),
701
+ database,
702
+ user,
703
+ password: password || '',
704
+ ssl: process.env.RUVECTOR_SSL === 'true',
705
+ };
706
+ }
707
+
708
+ // ============================================================================
709
+ // Backend resolution: SQLite > RuVector PostgreSQL > AgentDB > JSON
710
+ // ============================================================================
711
+
712
+ async function resolveBackend() {
713
+ // Tier 1: better-sqlite3 (native, fastest, local)
714
+ try {
715
+ const backend = new SQLiteBackend(ARCHIVE_DB_PATH);
716
+ await backend.initialize();
717
+ return { backend, type: 'sqlite' };
718
+ } catch { /* fall through */ }
719
+
720
+ // Tier 2: RuVector PostgreSQL (TB-scale, vector search, GNN)
721
+ try {
722
+ const rvConfig = getRuVectorConfig();
723
+ if (rvConfig) {
724
+ const backend = new RuVectorBackend(rvConfig);
725
+ await backend.initialize();
726
+ return { backend, type: 'ruvector' };
727
+ }
728
+ } catch { /* fall through */ }
729
+
730
+ // Tier 3: AgentDB from @claude-flow/memory (HNSW)
731
+ try {
732
+ const localDist = join(PROJECT_ROOT, 'v3/@claude-flow/memory/dist/index.js');
733
+ let memPkg = null;
734
+ if (existsSync(localDist)) {
735
+ memPkg = await import(`file://${localDist}`);
736
+ } else {
737
+ memPkg = await import('@claude-flow/memory');
738
+ }
739
+ if (memPkg?.AgentDBBackend) {
740
+ const backend = new memPkg.AgentDBBackend();
741
+ await backend.initialize();
742
+ return { backend, type: 'agentdb' };
743
+ }
744
+ } catch { /* fall through */ }
745
+
746
+ // Tier 4: JSON file (always works)
747
+ const backend = new JsonFileBackend(ARCHIVE_JSON_PATH);
748
+ await backend.initialize();
749
+ return { backend, type: 'json' };
750
+ }
751
+
752
+ // ============================================================================
753
+ // ONNX Embedding (384-dim, all-MiniLM-L6-v2 via @xenova/transformers)
754
+ // ============================================================================
755
+
756
+ const EMBEDDING_DIM = 384; // ONNX all-MiniLM-L6-v2 output dimension
757
+ let _onnxPipeline = null;
758
+ let _onnxFailed = false;
759
+
760
+ /**
761
+ * Initialize ONNX embedding pipeline (lazy, cached).
762
+ * Returns null if @xenova/transformers is not available.
763
+ */
764
+ async function getOnnxPipeline() {
765
+ if (_onnxFailed) return null;
766
+ if (_onnxPipeline) return _onnxPipeline;
767
+ try {
768
+ const { pipeline } = await import('@xenova/transformers');
769
+ _onnxPipeline = await pipeline('feature-extraction', 'Xenova/all-MiniLM-L6-v2');
770
+ return _onnxPipeline;
771
+ } catch {
772
+ _onnxFailed = true;
773
+ return null;
774
+ }
775
+ }
776
+
777
+ /**
778
+ * Generate ONNX embedding (384-dim, high quality semantic vectors).
779
+ * Falls back to hash embedding if ONNX is unavailable.
780
+ */
781
+ async function createEmbedding(text) {
782
+ // Try ONNX first (384-dim, real semantic understanding)
783
+ const pipe = await getOnnxPipeline();
784
+ if (pipe) {
785
+ try {
786
+ const truncated = text.slice(0, 512); // MiniLM max ~512 tokens
787
+ const output = await pipe(truncated, { pooling: 'mean', normalize: true });
788
+ return { embedding: new Float32Array(output.data), dim: 384, method: 'onnx' };
789
+ } catch { /* fall through to hash */ }
790
+ }
791
+ // Fallback: hash embedding (384-dim to match ONNX dimension)
792
+ return { embedding: createHashEmbedding(text, 384), dim: 384, method: 'hash' };
793
+ }
794
+
795
+ // ============================================================================
796
+ // Hash embedding fallback (deterministic, sub-millisecond)
797
+ // ============================================================================
798
+
799
+ function createHashEmbedding(text, dimensions = 384) {
800
+ const embedding = new Float32Array(dimensions);
801
+ const normalized = text.toLowerCase().trim();
802
+ for (let i = 0; i < dimensions; i++) {
803
+ let hash = 0;
804
+ for (let j = 0; j < normalized.length; j++) {
805
+ hash = ((hash << 5) - hash + normalized.charCodeAt(j) * (i + 1)) | 0;
806
+ }
807
+ embedding[i] = (Math.sin(hash) + 1) / 2;
808
+ }
809
+ let norm = 0;
810
+ for (let i = 0; i < dimensions; i++) norm += embedding[i] * embedding[i];
811
+ norm = Math.sqrt(norm);
812
+ if (norm > 0) for (let i = 0; i < dimensions; i++) embedding[i] /= norm;
813
+ return embedding;
814
+ }
815
+
816
+ // ============================================================================
817
+ // Content hash for dedup
818
+ // ============================================================================
819
+
820
+ function hashContent(content) {
821
+ return createHash('sha256').update(content).digest('hex');
822
+ }
823
+
824
+ // ============================================================================
825
+ // Read stdin with timeout (hooks receive JSON input on stdin)
826
+ // ============================================================================
827
+
828
+ function readStdin(timeoutMs = 100) {
829
+ return new Promise((resolve) => {
830
+ let data = '';
831
+ const timer = setTimeout(() => {
832
+ process.stdin.removeAllListeners();
833
+ resolve(data ? JSON.parse(data) : null);
834
+ }, timeoutMs);
835
+
836
+ if (process.stdin.isTTY) {
837
+ clearTimeout(timer);
838
+ resolve(null);
839
+ return;
840
+ }
841
+
842
+ process.stdin.setEncoding('utf-8');
843
+ process.stdin.on('data', (chunk) => { data += chunk; });
844
+ process.stdin.on('end', () => {
845
+ clearTimeout(timer);
846
+ try { resolve(data ? JSON.parse(data) : null); }
847
+ catch { resolve(null); }
848
+ });
849
+ process.stdin.on('error', () => {
850
+ clearTimeout(timer);
851
+ resolve(null);
852
+ });
853
+ process.stdin.resume();
854
+ });
855
+ }
856
+
857
+ // ============================================================================
858
+ // Transcript parsing
859
+ // ============================================================================
860
+
861
+ function parseTranscript(transcriptPath) {
862
+ if (!existsSync(transcriptPath)) return [];
863
+ const content = readFileSync(transcriptPath, 'utf-8');
864
+ const lines = content.split('\n').filter(Boolean);
865
+ const messages = [];
866
+ for (const line of lines) {
867
+ try {
868
+ const parsed = JSON.parse(line);
869
+ // SDK transcript wraps messages: { type: "user"|"A", message: { role, content } }
870
+ // Unwrap to get the inner API message with role/content
871
+ if (parsed.message && parsed.message.role) {
872
+ messages.push(parsed.message);
873
+ } else if (parsed.role) {
874
+ // Already in API message format (e.g. from tests)
875
+ messages.push(parsed);
876
+ }
877
+ // Skip non-message entries (progress, file-history-snapshot, queue-operation)
878
+ } catch { /* skip malformed lines */ }
879
+ }
880
+ return messages;
881
+ }
882
+
883
+ // ============================================================================
884
+ // Extract text content from message content blocks
885
+ // ============================================================================
886
+
887
+ function extractTextContent(message) {
888
+ if (!message) return '';
889
+ if (typeof message.content === 'string') return message.content;
890
+ if (Array.isArray(message.content)) {
891
+ return message.content
892
+ .filter(b => b.type === 'text')
893
+ .map(b => b.text || '')
894
+ .join('\n');
895
+ }
896
+ if (typeof message.text === 'string') return message.text;
897
+ return '';
898
+ }
899
+
900
+ // ============================================================================
901
+ // Extract tool calls from assistant message
902
+ // ============================================================================
903
+
904
+ function extractToolCalls(message) {
905
+ if (!message || !Array.isArray(message.content)) return [];
906
+ return message.content
907
+ .filter(b => b.type === 'tool_use')
908
+ .map(b => ({
909
+ name: b.name || 'unknown',
910
+ input: b.input || {},
911
+ }));
912
+ }
913
+
914
+ // ============================================================================
915
+ // Extract file paths from tool calls
916
+ // ============================================================================
917
+
918
+ function extractFilePaths(toolCalls) {
919
+ const paths = new Set();
920
+ for (const tc of toolCalls) {
921
+ if (tc.input?.file_path) paths.add(tc.input.file_path);
922
+ if (tc.input?.path) paths.add(tc.input.path);
923
+ if (tc.input?.notebook_path) paths.add(tc.input.notebook_path);
924
+ }
925
+ return [...paths];
926
+ }
927
+
928
+ // ============================================================================
929
+ // Chunk transcript into conversation turns
930
+ // ============================================================================
931
+
932
+ function chunkTranscript(messages) {
933
+ const relevant = messages.filter(
934
+ m => m.role === 'user' || m.role === 'assistant'
935
+ );
936
+ const capped = relevant.slice(-MAX_MESSAGES);
937
+
938
+ const chunks = [];
939
+ let currentChunk = null;
940
+
941
+ for (const msg of capped) {
942
+ if (msg.role === 'user') {
943
+ const isSynthetic = Array.isArray(msg.content) &&
944
+ msg.content.every(b => b.type === 'tool_result');
945
+ if (isSynthetic && currentChunk) continue;
946
+ if (currentChunk) chunks.push(currentChunk);
947
+ currentChunk = {
948
+ userMessage: msg,
949
+ assistantMessage: null,
950
+ toolCalls: [],
951
+ turnIndex: chunks.length,
952
+ };
953
+ } else if (msg.role === 'assistant' && currentChunk) {
954
+ currentChunk.assistantMessage = msg;
955
+ currentChunk.toolCalls = extractToolCalls(msg);
956
+ }
957
+ }
958
+
959
+ if (currentChunk) chunks.push(currentChunk);
960
+ return chunks;
961
+ }
962
+
963
+ // ============================================================================
964
+ // Extract summary from chunk (no LLM, extractive only)
965
+ // ============================================================================
966
+
967
+ function extractSummary(chunk) {
968
+ const parts = [];
969
+
970
+ const userText = extractTextContent(chunk.userMessage);
971
+ const firstUserLine = userText.split('\n').find(l => l.trim()) || '';
972
+ if (firstUserLine) parts.push(firstUserLine.slice(0, 100));
973
+
974
+ const toolNames = [...new Set(chunk.toolCalls.map(tc => tc.name))];
975
+ if (toolNames.length) parts.push('Tools: ' + toolNames.join(', '));
976
+
977
+ const filePaths = extractFilePaths(chunk.toolCalls);
978
+ if (filePaths.length) {
979
+ const shortPaths = filePaths.slice(0, 5).map(p => {
980
+ const segs = p.split('/');
981
+ return segs.length > 2 ? '.../' + segs.slice(-2).join('/') : p;
982
+ });
983
+ parts.push('Files: ' + shortPaths.join(', '));
984
+ }
985
+
986
+ const assistantText = extractTextContent(chunk.assistantMessage);
987
+ const assistantLines = assistantText.split('\n').filter(l => l.trim()).slice(0, 2);
988
+ if (assistantLines.length) parts.push(assistantLines.join(' ').slice(0, 120));
989
+
990
+ return parts.join(' | ').slice(0, 300);
991
+ }
992
+
993
+ // ============================================================================
994
+ // Generate unique ID
995
+ // ============================================================================
996
+
997
+ let idCounter = 0;
998
+ function generateId() {
999
+ return `ctx-${Date.now()}-${++idCounter}-${Math.random().toString(36).slice(2, 8)}`;
1000
+ }
1001
+
1002
+ // ============================================================================
1003
+ // Build MemoryEntry from chunk
1004
+ // ============================================================================
1005
+
1006
+ function buildEntry(chunk, sessionId, trigger, timestamp) {
1007
+ const userText = extractTextContent(chunk.userMessage);
1008
+ const assistantText = extractTextContent(chunk.assistantMessage);
1009
+ const fullContent = `User: ${userText}\n\nAssistant: ${assistantText}`;
1010
+ const toolNames = [...new Set(chunk.toolCalls.map(tc => tc.name))];
1011
+ const filePaths = extractFilePaths(chunk.toolCalls);
1012
+ const summary = extractSummary(chunk);
1013
+ const contentHash = hashContent(fullContent);
1014
+
1015
+ const now = Date.now();
1016
+ return {
1017
+ id: generateId(),
1018
+ key: `transcript:${sessionId}:${chunk.turnIndex}:${timestamp}`,
1019
+ content: fullContent,
1020
+ type: 'episodic',
1021
+ namespace: NAMESPACE,
1022
+ tags: ['transcript', 'compaction', sessionId, ...toolNames],
1023
+ metadata: {
1024
+ sessionId,
1025
+ chunkIndex: chunk.turnIndex,
1026
+ trigger,
1027
+ timestamp,
1028
+ toolNames,
1029
+ filePaths,
1030
+ summary,
1031
+ contentHash,
1032
+ turnRange: [chunk.turnIndex, chunk.turnIndex],
1033
+ },
1034
+ accessLevel: 'private',
1035
+ createdAt: now,
1036
+ updatedAt: now,
1037
+ version: 1,
1038
+ references: [],
1039
+ accessCount: 0,
1040
+ lastAccessedAt: now,
1041
+ };
1042
+ }
1043
+
1044
+ // ============================================================================
1045
+ // Store chunks with dedup (uses indexed hash lookup for SQLite)
1046
+ // ============================================================================
1047
+
1048
+ async function storeChunks(backend, chunks, sessionId, trigger) {
1049
+ const timestamp = new Date().toISOString();
1050
+
1051
+ const entries = [];
1052
+ for (const chunk of chunks) {
1053
+ const entry = buildEntry(chunk, sessionId, trigger, timestamp);
1054
+ // Fast hash-based dedup (indexed lookup in SQLite, scan in JSON)
1055
+ if (!backend.hashExists(entry.metadata.contentHash)) {
1056
+ entries.push(entry);
1057
+ }
1058
+ }
1059
+
1060
+ if (entries.length > 0) {
1061
+ await backend.bulkInsert(entries);
1062
+ }
1063
+
1064
+ return { stored: entries.length, deduped: chunks.length - entries.length };
1065
+ }
1066
+
1067
+ // ============================================================================
1068
+ // Retrieve context for restoration (uses indexed session query for SQLite)
1069
+ // ============================================================================
1070
+
1071
+ async function retrieveContext(backend, sessionId, budget) {
1072
+ // Use optimized session query if available, otherwise filter manually
1073
+ const sessionEntries = backend.queryBySession
1074
+ ? await backend.queryBySession(NAMESPACE, sessionId)
1075
+ : (await backend.query({ namespace: NAMESPACE }))
1076
+ .filter(e => e.metadata?.sessionId === sessionId)
1077
+ .sort((a, b) => (b.metadata?.chunkIndex ?? 0) - (a.metadata?.chunkIndex ?? 0));
1078
+
1079
+ if (sessionEntries.length === 0) return '';
1080
+
1081
+ const lines = [];
1082
+ let charCount = 0;
1083
+ const header = `## Restored Context (from pre-compaction archive)\n\nPrevious conversation included ${sessionEntries.length} archived turns:\n\n`;
1084
+ charCount += header.length;
1085
+
1086
+ for (const entry of sessionEntries) {
1087
+ const meta = entry.metadata || {};
1088
+ const toolStr = meta.toolNames?.length ? ` Tools: ${meta.toolNames.join(', ')}.` : '';
1089
+ const fileStr = meta.filePaths?.length ? ` Files: ${meta.filePaths.slice(0, 3).join(', ')}.` : '';
1090
+ const line = `- [Turn ${meta.chunkIndex ?? '?'}] ${meta.summary || '(no summary)'}${toolStr}${fileStr}`;
1091
+
1092
+ if (charCount + line.length + 1 > budget) break;
1093
+ lines.push(line);
1094
+ charCount += line.length + 1;
1095
+ }
1096
+
1097
+ if (lines.length === 0) return '';
1098
+
1099
+ const footer = `\n\nFull archive: ${NAMESPACE} namespace in AgentDB (query with session ID: ${sessionId})`;
1100
+ return header + lines.join('\n') + footer;
1101
+ }
1102
+
1103
+ // ============================================================================
1104
+ // Build custom compact instructions (exit code 0 stdout)
1105
+ // Guides Claude on what to preserve during compaction summary
1106
+ // ============================================================================
1107
+
1108
+ function buildCompactInstructions(chunks, sessionId, archiveResult) {
1109
+ const parts = [];
1110
+
1111
+ parts.push('COMPACTION GUIDANCE (from context-persistence-hook):');
1112
+ parts.push('');
1113
+ parts.push(`All ${chunks.length} conversation turns have been archived to the transcript-archive database.`);
1114
+ parts.push(`Session: ${sessionId} | Stored: ${archiveResult.stored} new, ${archiveResult.deduped} deduped.`);
1115
+ parts.push('After compaction, archived context will be automatically restored via SessionStart hook.');
1116
+ parts.push('');
1117
+
1118
+ // Collect unique tools and files across all chunks for preservation hints
1119
+ const allTools = new Set();
1120
+ const allFiles = new Set();
1121
+ const decisions = [];
1122
+
1123
+ for (const chunk of chunks) {
1124
+ const toolNames = [...new Set(chunk.toolCalls.map(tc => tc.name))];
1125
+ for (const t of toolNames) allTools.add(t);
1126
+ const filePaths = extractFilePaths(chunk.toolCalls);
1127
+ for (const f of filePaths) allFiles.add(f);
1128
+
1129
+ // Look for decision indicators in assistant text
1130
+ const assistantText = extractTextContent(chunk.assistantMessage);
1131
+ if (assistantText) {
1132
+ const lower = assistantText.toLowerCase();
1133
+ if (lower.includes('decided') || lower.includes('choosing') || lower.includes('approach')
1134
+ || lower.includes('instead of') || lower.includes('rather than')) {
1135
+ const firstLine = assistantText.split('\n').find(l => l.trim()) || '';
1136
+ if (firstLine.length > 10) decisions.push(firstLine.slice(0, 120));
1137
+ }
1138
+ }
1139
+ }
1140
+
1141
+ parts.push('PRESERVE in compaction summary:');
1142
+
1143
+ if (allFiles.size > 0) {
1144
+ const fileList = [...allFiles].slice(0, 15).map(f => {
1145
+ const segs = f.split('/');
1146
+ return segs.length > 3 ? '.../' + segs.slice(-3).join('/') : f;
1147
+ });
1148
+ parts.push(`- Files modified/read: ${fileList.join(', ')}`);
1149
+ }
1150
+
1151
+ if (allTools.size > 0) {
1152
+ parts.push(`- Tools used: ${[...allTools].join(', ')}`);
1153
+ }
1154
+
1155
+ if (decisions.length > 0) {
1156
+ parts.push('- Key decisions:');
1157
+ for (const d of decisions.slice(0, 5)) {
1158
+ parts.push(` * ${d}`);
1159
+ }
1160
+ }
1161
+
1162
+ // Recent turns summary (most important context)
1163
+ const recentChunks = chunks.slice(-5);
1164
+ if (recentChunks.length > 0) {
1165
+ parts.push('');
1166
+ parts.push('MOST RECENT TURNS (prioritize preserving):');
1167
+ for (const chunk of recentChunks) {
1168
+ const userText = extractTextContent(chunk.userMessage);
1169
+ const firstLine = userText.split('\n').find(l => l.trim()) || '';
1170
+ const toolNames = [...new Set(chunk.toolCalls.map(tc => tc.name))];
1171
+ parts.push(`- [Turn ${chunk.turnIndex}] ${firstLine.slice(0, 80)}${toolNames.length ? ` (${toolNames.join(', ')})` : ''}`);
1172
+ }
1173
+ }
1174
+
1175
+ // Cap at budget
1176
+ let result = parts.join('\n');
1177
+ if (result.length > COMPACT_INSTRUCTION_BUDGET) {
1178
+ result = result.slice(0, COMPACT_INSTRUCTION_BUDGET - 3) + '...';
1179
+ }
1180
+ return result;
1181
+ }
1182
+
1183
+ // ============================================================================
1184
+ // Importance scoring for retrieval ranking
1185
+ // ============================================================================
1186
+
1187
+ function computeImportance(entry, now) {
1188
+ const meta = entry.metadata || {};
1189
+ const accessCount = entry.accessCount || 0;
1190
+ const createdAt = entry.createdAt || now;
1191
+ const ageMs = Math.max(1, now - createdAt);
1192
+ const ageDays = ageMs / 86400000;
1193
+
1194
+ // Recency: exponential decay, half-life of 7 days
1195
+ const recency = Math.exp(-0.693 * ageDays / 7);
1196
+
1197
+ // Frequency: log-scaled access count
1198
+ const frequency = Math.log2(accessCount + 1) + 1;
1199
+
1200
+ // Richness: tool calls and file paths indicate actionable context
1201
+ const toolCount = meta.toolNames?.length || 0;
1202
+ const fileCount = meta.filePaths?.length || 0;
1203
+ const richness = 1.0 + (toolCount > 0 ? 0.5 : 0) + (fileCount > 0 ? 0.3 : 0);
1204
+
1205
+ return recency * frequency * richness;
1206
+ }
1207
+
1208
+ // ============================================================================
1209
+ // Smart retrieval: importance-ranked instead of just recency
1210
+ // ============================================================================
1211
+
1212
+ async function retrieveContextSmart(backend, sessionId, budget) {
1213
+ let sessionEntries;
1214
+
1215
+ // Use importance-ranked query if backend supports it
1216
+ if (backend.queryByImportance) {
1217
+ try {
1218
+ sessionEntries = backend.queryByImportance(NAMESPACE, sessionId);
1219
+ } catch {
1220
+ // Fall back to standard query
1221
+ sessionEntries = null;
1222
+ }
1223
+ }
1224
+
1225
+ if (!sessionEntries) {
1226
+ // Fall back: fetch all, compute importance in JS
1227
+ const raw = backend.queryBySession
1228
+ ? await backend.queryBySession(NAMESPACE, sessionId)
1229
+ : (await backend.query({ namespace: NAMESPACE }))
1230
+ .filter(e => e.metadata?.sessionId === sessionId);
1231
+
1232
+ const now = Date.now();
1233
+ sessionEntries = raw
1234
+ .map(e => ({ ...e, importanceScore: computeImportance(e, now) }))
1235
+ .sort((a, b) => b.importanceScore - a.importanceScore);
1236
+ }
1237
+
1238
+ if (sessionEntries.length === 0) return { text: '', accessedIds: [] };
1239
+
1240
+ const lines = [];
1241
+ const accessedIds = [];
1242
+ let charCount = 0;
1243
+ const header = `## Restored Context (importance-ranked from archive)\n\nPrevious conversation: ${sessionEntries.length} archived turns, ranked by importance:\n\n`;
1244
+ charCount += header.length;
1245
+
1246
+ for (const entry of sessionEntries) {
1247
+ const meta = entry.metadata || {};
1248
+ const score = entry.importanceScore?.toFixed(2) || '?';
1249
+ const toolStr = meta.toolNames?.length ? ` Tools: ${meta.toolNames.join(', ')}.` : '';
1250
+ const fileStr = meta.filePaths?.length ? ` Files: ${meta.filePaths.slice(0, 3).join(', ')}.` : '';
1251
+ const line = `- [Turn ${meta.chunkIndex ?? '?'}, score:${score}] ${meta.summary || '(no summary)'}${toolStr}${fileStr}`;
1252
+
1253
+ if (charCount + line.length + 1 > budget) break;
1254
+ lines.push(line);
1255
+ accessedIds.push(entry.id);
1256
+ charCount += line.length + 1;
1257
+ }
1258
+
1259
+ if (lines.length === 0) return { text: '', accessedIds: [] };
1260
+
1261
+ // Cross-session semantic search: find related context from previous sessions
1262
+ let crossSessionText = '';
1263
+ if (backend.semanticSearch && sessionEntries.length > 0) {
1264
+ try {
1265
+ // Use the most recent turn's summary as the search query
1266
+ const recentSummary = sessionEntries[0]?.metadata?.summary || '';
1267
+ if (recentSummary) {
1268
+ const crossResults = await crossSessionSearch(backend, recentSummary, sessionId, 3);
1269
+ if (crossResults.length > 0) {
1270
+ const crossLines = crossResults.map(r =>
1271
+ `- [Session ${r.sessionId?.slice(0, 8)}..., turn ${r.chunkIndex ?? '?'}, conf:${(r.confidence || 0).toFixed(2)}] ${r.summary || '(no summary)'}`
1272
+ );
1273
+ crossSessionText = `\n\nRelated context from previous sessions:\n${crossLines.join('\n')}`;
1274
+ }
1275
+ }
1276
+ } catch { /* cross-session search is best-effort */ }
1277
+ }
1278
+
1279
+ const footer = `\n\nFull archive: ${NAMESPACE} namespace (session: ${sessionId}). ${sessionEntries.length - lines.length} additional turns available.`;
1280
+ return { text: header + lines.join('\n') + crossSessionText + footer, accessedIds };
1281
+ }
1282
+
1283
+ // ============================================================================
1284
+ // Auto-optimize: prune stale entries, run after archiving
1285
+ // ============================================================================
1286
+
1287
+ async function autoOptimize(backend, backendType) {
1288
+ if (!AUTO_OPTIMIZE) return { pruned: 0, synced: 0, decayed: 0, embedded: 0 };
1289
+
1290
+ let pruned = 0;
1291
+ let decayed = 0;
1292
+ let embedded = 0;
1293
+
1294
+ // Step 1: Confidence decay — reduce confidence for unaccessed entries
1295
+ if (backend.decayConfidence) {
1296
+ try {
1297
+ decayed = backend.decayConfidence(NAMESPACE, 1); // 1 hour worth of decay per optimize cycle
1298
+ } catch { /* non-critical */ }
1299
+ }
1300
+
1301
+ // Step 2: Smart pruning — remove low-confidence entries first
1302
+ if (backend.pruneByConfidence) {
1303
+ try {
1304
+ pruned += backend.pruneByConfidence(NAMESPACE, 0.15);
1305
+ } catch { /* non-critical */ }
1306
+ }
1307
+
1308
+ // Step 3: Age-based pruning as fallback
1309
+ if (backend.pruneStale) {
1310
+ try {
1311
+ pruned += backend.pruneStale(NAMESPACE, RETENTION_DAYS);
1312
+ } catch { /* non-critical */ }
1313
+ }
1314
+
1315
+ // Step 4: Generate ONNX embeddings (384-dim) for entries missing them
1316
+ if (backend.storeEmbedding) {
1317
+ try {
1318
+ const rows = backend.db?.prepare?.(
1319
+ 'SELECT id, content FROM transcript_entries WHERE namespace = ? AND embedding IS NULL LIMIT 20'
1320
+ )?.all(NAMESPACE);
1321
+ if (rows) {
1322
+ for (const row of rows) {
1323
+ const { embedding } = await createEmbedding(row.content);
1324
+ backend.storeEmbedding(row.id, embedding);
1325
+ embedded++;
1326
+ }
1327
+ }
1328
+ } catch { /* non-critical */ }
1329
+ }
1330
+
1331
+ // Step 5: Auto-sync to RuVector if available
1332
+ let synced = 0;
1333
+ if (backendType === 'sqlite' && backend.allForSync) {
1334
+ try {
1335
+ const rvConfig = getRuVectorConfig();
1336
+ if (rvConfig) {
1337
+ const rvBackend = new RuVectorBackend(rvConfig);
1338
+ await rvBackend.initialize();
1339
+
1340
+ const allEntries = backend.allForSync(NAMESPACE);
1341
+ if (allEntries.length > 0) {
1342
+ // Add hash embeddings for vector search in RuVector
1343
+ const entriesToSync = allEntries.map(e => ({
1344
+ ...e,
1345
+ _embedding: createHashEmbedding(e.content),
1346
+ }));
1347
+ await rvBackend.bulkInsert(entriesToSync);
1348
+ synced = entriesToSync.length;
1349
+ }
1350
+
1351
+ await rvBackend.shutdown();
1352
+ }
1353
+ } catch { /* RuVector sync is best-effort */ }
1354
+ }
1355
+
1356
+ return { pruned, synced, decayed, embedded };
1357
+ }
1358
+
1359
+ // ============================================================================
1360
+ // Cross-session semantic retrieval
1361
+ // ============================================================================
1362
+
1363
+ /**
1364
+ * Find relevant context from OTHER sessions using semantic similarity.
1365
+ * This enables "What did we discuss about auth?" across sessions.
1366
+ */
1367
+ async function crossSessionSearch(backend, queryText, currentSessionId, k = 5) {
1368
+ if (!backend.semanticSearch) return [];
1369
+ try {
1370
+ const { embedding: queryEmb } = await createEmbedding(queryText);
1371
+ const results = backend.semanticSearch(queryEmb, k * 2, NAMESPACE);
1372
+ // Filter out current session entries (we already have those)
1373
+ return results
1374
+ .filter(r => r.sessionId !== currentSessionId)
1375
+ .slice(0, k);
1376
+ } catch { return []; }
1377
+ }
1378
+
1379
+ // ============================================================================
1380
+ // Context Autopilot Engine
1381
+ // ============================================================================
1382
+
1383
+ /**
1384
+ * Estimate context token usage from transcript JSONL.
1385
+ *
1386
+ * Primary method: Read the most recent assistant message's `usage` field which
1387
+ * contains `input_tokens` + `cache_read_input_tokens` — this is the ACTUAL
1388
+ * context size as reported by the Claude API. This includes system prompt,
1389
+ * CLAUDE.md, tool definitions, all messages, and everything Claude sees.
1390
+ *
1391
+ * Fallback: Sum character lengths and divide by CHARS_PER_TOKEN.
1392
+ */
1393
+ function estimateContextTokens(transcriptPath) {
1394
+ if (!existsSync(transcriptPath)) return { tokens: 0, turns: 0, method: 'none' };
1395
+
1396
+ const content = readFileSync(transcriptPath, 'utf-8');
1397
+ const lines = content.split('\n').filter(Boolean);
1398
+
1399
+ // Track the most recent usage data (from the last assistant message)
1400
+ let lastInputTokens = 0;
1401
+ let lastCacheRead = 0;
1402
+ let lastCacheCreate = 0;
1403
+ let turns = 0;
1404
+ let lastPreTokens = 0;
1405
+ let totalChars = 0;
1406
+
1407
+ for (let i = 0; i < lines.length; i++) {
1408
+ try {
1409
+ const parsed = JSON.parse(lines[i]);
1410
+
1411
+ // Check for compact_boundary
1412
+ if (parsed.type === 'system' && parsed.subtype === 'compact_boundary') {
1413
+ lastPreTokens = parsed.compactMetadata?.preTokens
1414
+ || parsed.compact_metadata?.pre_tokens || 0;
1415
+ // Reset after compaction — new context starts here
1416
+ totalChars = 0;
1417
+ turns = 0;
1418
+ lastInputTokens = 0;
1419
+ lastCacheRead = 0;
1420
+ lastCacheCreate = 0;
1421
+ continue;
1422
+ }
1423
+
1424
+ // Extract ACTUAL token usage from assistant messages
1425
+ // The SDK transcript stores: { message: { role, content, usage: { input_tokens, cache_read_input_tokens, ... } } }
1426
+ const msg = parsed.message || parsed;
1427
+ const usage = msg.usage;
1428
+ if (usage && (msg.role === 'assistant' || parsed.type === 'assistant')) {
1429
+ const inputTokens = usage.input_tokens || 0;
1430
+ const cacheRead = usage.cache_read_input_tokens || 0;
1431
+ const cacheCreate = usage.cache_creation_input_tokens || 0;
1432
+
1433
+ // The total context sent to Claude = input_tokens + cache_read + cache_create
1434
+ // input_tokens: non-cached tokens actually processed
1435
+ // cache_read: tokens served from cache (still in context)
1436
+ // cache_create: tokens newly cached (still in context)
1437
+ const totalContext = inputTokens + cacheRead + cacheCreate;
1438
+
1439
+ if (totalContext > 0) {
1440
+ lastInputTokens = inputTokens;
1441
+ lastCacheRead = cacheRead;
1442
+ lastCacheCreate = cacheCreate;
1443
+ }
1444
+ }
1445
+
1446
+ // Count turns for display
1447
+ const role = msg.role || parsed.type;
1448
+ if (role === 'user') turns++;
1449
+
1450
+ // Char fallback accumulation
1451
+ if (role === 'user' || role === 'assistant') {
1452
+ const c = msg.content;
1453
+ if (typeof c === 'string') totalChars += c.length;
1454
+ else if (Array.isArray(c)) {
1455
+ for (const block of c) {
1456
+ if (block.text) totalChars += block.text.length;
1457
+ else if (block.input) totalChars += JSON.stringify(block.input).length;
1458
+ }
1459
+ }
1460
+ }
1461
+ } catch { /* skip */ }
1462
+ }
1463
+
1464
+ // Primary: use actual API usage data
1465
+ const actualTotal = lastInputTokens + lastCacheRead + lastCacheCreate;
1466
+ if (actualTotal > 0) {
1467
+ return {
1468
+ tokens: actualTotal,
1469
+ turns,
1470
+ method: 'api-usage',
1471
+ lastPreTokens,
1472
+ breakdown: {
1473
+ input: lastInputTokens,
1474
+ cacheRead: lastCacheRead,
1475
+ cacheCreate: lastCacheCreate,
1476
+ },
1477
+ };
1478
+ }
1479
+
1480
+ // Fallback: char-based estimate
1481
+ const estimatedTokens = Math.ceil(totalChars / CHARS_PER_TOKEN);
1482
+ if (lastPreTokens > 0) {
1483
+ const compactSummaryTokens = 3000;
1484
+ return {
1485
+ tokens: compactSummaryTokens + estimatedTokens,
1486
+ turns,
1487
+ method: 'post-compact-char-estimate',
1488
+ lastPreTokens,
1489
+ };
1490
+ }
1491
+
1492
+ return { tokens: estimatedTokens, turns, method: 'char-estimate' };
1493
+ }
1494
+
1495
+ /**
1496
+ * Load autopilot state (persisted across hook invocations).
1497
+ */
1498
+ function loadAutopilotState() {
1499
+ try {
1500
+ if (existsSync(AUTOPILOT_STATE_PATH)) {
1501
+ return JSON.parse(readFileSync(AUTOPILOT_STATE_PATH, 'utf-8'));
1502
+ }
1503
+ } catch { /* fresh state */ }
1504
+ return {
1505
+ sessionId: null,
1506
+ lastTokenEstimate: 0,
1507
+ lastPercentage: 0,
1508
+ pruneCount: 0,
1509
+ warningIssued: false,
1510
+ lastCheck: 0,
1511
+ history: [], // Track token growth over time
1512
+ };
1513
+ }
1514
+
1515
+ /**
1516
+ * Save autopilot state.
1517
+ */
1518
+ function saveAutopilotState(state) {
1519
+ try {
1520
+ writeFileSync(AUTOPILOT_STATE_PATH, JSON.stringify(state, null, 2), 'utf-8');
1521
+ } catch { /* best effort */ }
1522
+ }
1523
+
1524
+ /**
1525
+ * Build a context optimization report for additionalContext injection.
1526
+ */
1527
+ function buildAutopilotReport(percentage, tokens, windowSize, turns, state) {
1528
+ const bar = buildProgressBar(percentage);
1529
+ const status = percentage >= AUTOPILOT_PRUNE_PCT
1530
+ ? 'OPTIMIZING'
1531
+ : percentage >= AUTOPILOT_WARN_PCT
1532
+ ? 'WARNING'
1533
+ : 'OK';
1534
+
1535
+ const parts = [
1536
+ `[ContextAutopilot] ${bar} ${(percentage * 100).toFixed(1)}% context used`,
1537
+ `(~${formatTokens(tokens)}/${formatTokens(windowSize)} tokens, ${turns} turns)`,
1538
+ `Status: ${status}`,
1539
+ ];
1540
+
1541
+ if (state.pruneCount > 0) {
1542
+ parts.push(`| Optimizations: ${state.pruneCount} prune cycles`);
1543
+ }
1544
+
1545
+ // Add trend if we have history
1546
+ if (state.history.length >= 2) {
1547
+ const recent = state.history.slice(-3);
1548
+ const avgGrowth = recent.reduce((sum, h, i) => {
1549
+ if (i === 0) return 0;
1550
+ return sum + (h.pct - recent[i - 1].pct);
1551
+ }, 0) / (recent.length - 1);
1552
+
1553
+ if (avgGrowth > 0) {
1554
+ const turnsUntilFull = Math.ceil((1.0 - percentage) / avgGrowth);
1555
+ parts.push(`| ~${turnsUntilFull} turns until optimization needed`);
1556
+ }
1557
+ }
1558
+
1559
+ return parts.join(' ');
1560
+ }
1561
+
1562
+ /**
1563
+ * Visual progress bar for context usage.
1564
+ */
1565
+ function buildProgressBar(percentage) {
1566
+ const width = 20;
1567
+ const filled = Math.round(percentage * width);
1568
+ const empty = width - filled;
1569
+ const fillChar = percentage >= AUTOPILOT_PRUNE_PCT ? '!' : percentage >= AUTOPILOT_WARN_PCT ? '#' : '=';
1570
+ return `[${fillChar.repeat(filled)}${'-'.repeat(empty)}]`;
1571
+ }
1572
+
1573
+ /**
1574
+ * Format token count for display.
1575
+ */
1576
+ function formatTokens(n) {
1577
+ if (n >= 1000000) return (n / 1000000).toFixed(1) + 'M';
1578
+ if (n >= 1000) return (n / 1000).toFixed(1) + 'K';
1579
+ return String(n);
1580
+ }
1581
+
1582
+ /**
1583
+ * Context Autopilot: run on every UserPromptSubmit.
1584
+ * Returns { additionalContext, shouldBlock } for the hook output.
1585
+ */
1586
+ async function runAutopilot(transcriptPath, sessionId, backend, backendType) {
1587
+ const state = loadAutopilotState();
1588
+
1589
+ // Reset state if session changed
1590
+ if (state.sessionId !== sessionId) {
1591
+ state.sessionId = sessionId;
1592
+ state.lastTokenEstimate = 0;
1593
+ state.lastPercentage = 0;
1594
+ state.pruneCount = 0;
1595
+ state.warningIssued = false;
1596
+ state.history = [];
1597
+ }
1598
+
1599
+ // Estimate current context usage
1600
+ const { tokens, turns, method, lastPreTokens } = estimateContextTokens(transcriptPath);
1601
+ const percentage = Math.min(tokens / CONTEXT_WINDOW_TOKENS, 1.0);
1602
+
1603
+ // Track history (keep last 50 data points)
1604
+ state.history.push({ ts: Date.now(), tokens, pct: percentage, turns });
1605
+ if (state.history.length > 50) state.history.shift();
1606
+
1607
+ state.lastTokenEstimate = tokens;
1608
+ state.lastPercentage = percentage;
1609
+ state.lastCheck = Date.now();
1610
+
1611
+ let optimizationMessage = '';
1612
+
1613
+ // Phase 1: Warning zone (70-85%) — advise concise responses
1614
+ if (percentage >= AUTOPILOT_WARN_PCT && percentage < AUTOPILOT_PRUNE_PCT) {
1615
+ if (!state.warningIssued) {
1616
+ state.warningIssued = true;
1617
+ optimizationMessage = ` | Context at ${(percentage * 100).toFixed(0)}%. Keep responses concise to extend session.`;
1618
+ }
1619
+ }
1620
+
1621
+ // Phase 2: Critical zone (85%+) — session rotation needed
1622
+ if (percentage >= AUTOPILOT_PRUNE_PCT) {
1623
+ state.pruneCount++;
1624
+
1625
+ // Prune stale entries from archive to free up storage
1626
+ if (backend.pruneStale) {
1627
+ try {
1628
+ const pruned = backend.pruneStale(NAMESPACE, Math.min(RETENTION_DAYS, 7));
1629
+ if (pruned > 0) {
1630
+ optimizationMessage += ` | Pruned ${pruned} stale archive entries.`;
1631
+ }
1632
+ } catch { /* non-critical */ }
1633
+ }
1634
+
1635
+ const turnsLeft = Math.max(0, Math.ceil((1.0 - percentage) / 0.03));
1636
+ optimizationMessage += ` | CRITICAL: ${(percentage * 100).toFixed(0)}% context used (~${turnsLeft} turns left). All ${turns} turns archived. Start a new session with /clear — context will be fully restored via SessionStart hook.`;
1637
+ }
1638
+
1639
+ const report = buildAutopilotReport(percentage, tokens, CONTEXT_WINDOW_TOKENS, turns, state);
1640
+ saveAutopilotState(state);
1641
+
1642
+ return {
1643
+ additionalContext: report + optimizationMessage,
1644
+ percentage,
1645
+ tokens,
1646
+ turns,
1647
+ method,
1648
+ state,
1649
+ };
1650
+ }
1651
+
1652
+ // ============================================================================
1653
+ // Commands
1654
+ // ============================================================================
1655
+
1656
+ async function doPreCompact() {
1657
+ const input = await readStdin(200);
1658
+ if (!input) return;
1659
+
1660
+ const { session_id: sessionId, transcript_path: transcriptPath, trigger } = input;
1661
+ if (!transcriptPath || !sessionId) return;
1662
+
1663
+ const messages = parseTranscript(transcriptPath);
1664
+ if (messages.length === 0) return;
1665
+
1666
+ const chunks = chunkTranscript(messages);
1667
+ if (chunks.length === 0) return;
1668
+
1669
+ const { backend, type } = await resolveBackend();
1670
+
1671
+ const archiveResult = await storeChunks(backend, chunks, sessionId, trigger || 'auto');
1672
+
1673
+ // Auto-optimize: prune stale entries + sync to RuVector if available
1674
+ const optimizeResult = await autoOptimize(backend, type);
1675
+
1676
+ const total = await backend.count(NAMESPACE);
1677
+ await backend.shutdown();
1678
+
1679
+ const optParts = [];
1680
+ if (optimizeResult.pruned > 0) optParts.push(`${optimizeResult.pruned} pruned`);
1681
+ if (optimizeResult.decayed > 0) optParts.push(`${optimizeResult.decayed} decayed`);
1682
+ if (optimizeResult.embedded > 0) optParts.push(`${optimizeResult.embedded} embedded`);
1683
+ if (optimizeResult.synced > 0) optParts.push(`${optimizeResult.synced} synced`);
1684
+ const optimizeMsg = optParts.length > 0 ? ` Optimized: ${optParts.join(', ')}.` : '';
1685
+ process.stderr.write(
1686
+ `[ContextPersistence] Archived ${archiveResult.stored} turns (${archiveResult.deduped} deduped) via ${type}. Total: ${total}.${optimizeMsg}\n`
1687
+ );
1688
+
1689
+ // Exit code 0: stdout is appended as custom compact instructions
1690
+ // This guides Claude on what to preserve in the compaction summary
1691
+ const instructions = buildCompactInstructions(chunks, sessionId, archiveResult);
1692
+ process.stdout.write(instructions);
1693
+
1694
+ // Context Autopilot: track state and log archival status
1695
+ // NOTE: Claude Code 2.0.76 executePreCompactHooks uses executeHooksOutsideREPL
1696
+ // which does NOT support exit code 2 blocking. Compaction always proceeds.
1697
+ // Our "infinite context" comes from archive + restore, not blocking.
1698
+ if (AUTOPILOT_ENABLED) {
1699
+ const state = loadAutopilotState();
1700
+ const pct = state.lastPercentage || 0;
1701
+ const bar = buildProgressBar(pct);
1702
+
1703
+ process.stderr.write(
1704
+ `[ContextAutopilot] ${bar} ${(pct * 100).toFixed(1)}% | ${trigger} compact — ${chunks.length} turns archived. Context will be restored after compaction.\n`
1705
+ );
1706
+
1707
+ // Reset autopilot state for post-compaction fresh start
1708
+ state.lastTokenEstimate = 0;
1709
+ state.lastPercentage = 0;
1710
+ state.warningIssued = false;
1711
+ saveAutopilotState(state);
1712
+ }
1713
+ }
1714
+
1715
+ async function doSessionStart() {
1716
+ const input = await readStdin(200);
1717
+
1718
+ // Restore context after compaction OR after /clear (session rotation)
1719
+ // With DISABLE_COMPACT, /clear is the primary way to free context
1720
+ if (!input || (input.source !== 'compact' && input.source !== 'clear')) return;
1721
+
1722
+ const sessionId = input.session_id;
1723
+ if (!sessionId) return;
1724
+
1725
+ const { backend, type } = await resolveBackend();
1726
+
1727
+ // Use smart retrieval (importance-ranked) when auto-optimize is on
1728
+ let additionalContext;
1729
+ if (AUTO_OPTIMIZE) {
1730
+ const { text, accessedIds } = await retrieveContextSmart(backend, sessionId, RESTORE_BUDGET);
1731
+ additionalContext = text;
1732
+
1733
+ // Track which entries were actually restored (access pattern learning)
1734
+ if (accessedIds.length > 0 && backend.markAccessed) {
1735
+ try { backend.markAccessed(accessedIds); } catch { /* non-critical */ }
1736
+ }
1737
+
1738
+ if (accessedIds.length > 0) {
1739
+ process.stderr.write(
1740
+ `[ContextPersistence] Smart restore: ${accessedIds.length} turns (importance-ranked) via ${type}\n`
1741
+ );
1742
+ }
1743
+ } else {
1744
+ additionalContext = await retrieveContext(backend, sessionId, RESTORE_BUDGET);
1745
+ }
1746
+
1747
+ await backend.shutdown();
1748
+
1749
+ if (!additionalContext) return;
1750
+
1751
+ const output = {
1752
+ hookSpecificOutput: {
1753
+ hookEventName: 'SessionStart',
1754
+ additionalContext,
1755
+ },
1756
+ };
1757
+ process.stdout.write(JSON.stringify(output));
1758
+ }
1759
+
1760
+ // ============================================================================
1761
+ // Proactive archiving on every user prompt (prevents context cliff)
1762
+ // ============================================================================
1763
+
1764
+ async function doUserPromptSubmit() {
1765
+ const input = await readStdin(200);
1766
+ if (!input) return;
1767
+
1768
+ const { session_id: sessionId, transcript_path: transcriptPath } = input;
1769
+ if (!transcriptPath || !sessionId) return;
1770
+
1771
+ const messages = parseTranscript(transcriptPath);
1772
+ if (messages.length === 0) return;
1773
+
1774
+ const chunks = chunkTranscript(messages);
1775
+ if (chunks.length === 0) return;
1776
+
1777
+ const { backend, type } = await resolveBackend();
1778
+
1779
+ // Only archive new turns (dedup handles the rest, but we can skip early
1780
+ // by only processing the last N chunks since the previous archive)
1781
+ const existingCount = backend.queryBySession
1782
+ ? (await backend.queryBySession(NAMESPACE, sessionId)).length
1783
+ : 0;
1784
+
1785
+ // Skip if we've already archived most turns (within 2 turns tolerance)
1786
+ const skipArchive = existingCount > 0 && chunks.length - existingCount <= 2;
1787
+
1788
+ let archiveMsg = '';
1789
+ if (!skipArchive) {
1790
+ const result = await storeChunks(backend, chunks, sessionId, 'proactive');
1791
+ if (result.stored > 0) {
1792
+ const total = await backend.count(NAMESPACE);
1793
+ archiveMsg = `[ContextPersistence] Proactively archived ${result.stored} turns (total: ${total}).`;
1794
+ process.stderr.write(
1795
+ `[ContextPersistence] Proactive archive: ${result.stored} new, ${result.deduped} deduped via ${type}. Total: ${total}\n`
1796
+ );
1797
+ }
1798
+ }
1799
+
1800
+ // Context Autopilot: estimate usage and report percentage
1801
+ let autopilotMsg = '';
1802
+ if (AUTOPILOT_ENABLED && transcriptPath) {
1803
+ try {
1804
+ const autopilot = await runAutopilot(transcriptPath, sessionId, backend, type);
1805
+ autopilotMsg = autopilot.additionalContext;
1806
+
1807
+ process.stderr.write(
1808
+ `[ContextAutopilot] ${(autopilot.percentage * 100).toFixed(1)}% context used (~${formatTokens(autopilot.tokens)} tokens, ${autopilot.turns} turns, ${autopilot.method})\n`
1809
+ );
1810
+ } catch (err) {
1811
+ process.stderr.write(`[ContextAutopilot] Error: ${err.message}\n`);
1812
+ }
1813
+ }
1814
+
1815
+ await backend.shutdown();
1816
+
1817
+ // Combine archive message and autopilot report
1818
+ const additionalContext = [archiveMsg, autopilotMsg].filter(Boolean).join(' ');
1819
+
1820
+ if (additionalContext) {
1821
+ const output = {
1822
+ hookSpecificOutput: {
1823
+ hookEventName: 'UserPromptSubmit',
1824
+ additionalContext,
1825
+ },
1826
+ };
1827
+ process.stdout.write(JSON.stringify(output));
1828
+ }
1829
+ }
1830
+
1831
+ async function doStatus() {
1832
+ const { backend, type } = await resolveBackend();
1833
+
1834
+ const total = await backend.count();
1835
+ const archiveCount = await backend.count(NAMESPACE);
1836
+ const namespaces = await backend.listNamespaces();
1837
+ const sessions = await backend.listSessions(NAMESPACE);
1838
+
1839
+ console.log('\n=== Context Persistence Archive Status ===\n');
1840
+ const backendLabel = {
1841
+ sqlite: ARCHIVE_DB_PATH,
1842
+ ruvector: `${process.env.RUVECTOR_HOST || 'N/A'}:${process.env.RUVECTOR_PORT || '5432'}`,
1843
+ agentdb: 'in-memory HNSW',
1844
+ json: ARCHIVE_JSON_PATH,
1845
+ };
1846
+ console.log(` Backend: ${type} (${backendLabel[type] || type})`);
1847
+ console.log(` Total: ${total} entries`);
1848
+ console.log(` Transcripts: ${archiveCount} entries`);
1849
+ console.log(` Namespaces: ${namespaces.join(', ') || 'none'}`);
1850
+ console.log(` Budget: ${RESTORE_BUDGET} chars`);
1851
+ console.log(` Sessions: ${sessions.length}`);
1852
+ console.log(` Proactive: enabled (UserPromptSubmit hook)`);
1853
+ console.log(` Auto-opt: ${AUTO_OPTIMIZE ? 'enabled' : 'disabled'} (importance ranking, pruning, sync)`);
1854
+ console.log(` Retention: ${RETENTION_DAYS} days (prune never-accessed entries)`);
1855
+ const rvConfig = getRuVectorConfig();
1856
+ console.log(` RuVector: ${rvConfig ? `${rvConfig.host}:${rvConfig.port}/${rvConfig.database} (auto-sync enabled)` : 'not configured'}`);
1857
+
1858
+ // Self-learning stats
1859
+ if (type === 'sqlite' && backend.db) {
1860
+ try {
1861
+ const embCount = backend.db.prepare('SELECT COUNT(*) as cnt FROM transcript_entries WHERE embedding IS NOT NULL').get().cnt;
1862
+ const avgConf = backend.db.prepare('SELECT AVG(confidence) as avg FROM transcript_entries WHERE namespace = ?').get(NAMESPACE)?.avg || 0;
1863
+ const lowConf = backend.db.prepare('SELECT COUNT(*) as cnt FROM transcript_entries WHERE namespace = ? AND confidence < 0.3').get(NAMESPACE).cnt;
1864
+ console.log('');
1865
+ console.log(' --- Self-Learning ---');
1866
+ console.log(` Embeddings: ${embCount}/${archiveCount} entries have vector embeddings`);
1867
+ console.log(` Avg conf: ${(avgConf * 100).toFixed(1)}% (decay: -0.5%/hr, boost: +3%/access)`);
1868
+ console.log(` Low conf: ${lowConf} entries below 30% (pruned at 15%)`);
1869
+ console.log(` Semantic: ${embCount > 0 ? 'enabled (cross-session search)' : 'pending (embeddings generating)'}`);
1870
+ } catch { /* stats are non-critical */ }
1871
+ }
1872
+
1873
+ // Autopilot status
1874
+ console.log('');
1875
+ console.log(' --- Context Autopilot ---');
1876
+ console.log(` Enabled: ${AUTOPILOT_ENABLED}`);
1877
+ console.log(` Window: ${formatTokens(CONTEXT_WINDOW_TOKENS)} tokens`);
1878
+ console.log(` Warn at: ${(AUTOPILOT_WARN_PCT * 100).toFixed(0)}%`);
1879
+ console.log(` Prune at: ${(AUTOPILOT_PRUNE_PCT * 100).toFixed(0)}%`);
1880
+ console.log(` Compaction: LOSSLESS (archive before, restore after)`);
1881
+
1882
+ const apState = loadAutopilotState();
1883
+ if (apState.sessionId) {
1884
+ const pct = apState.lastPercentage || 0;
1885
+ const bar = buildProgressBar(pct);
1886
+ console.log(` Current: ${bar} ${(pct * 100).toFixed(1)}% (~${formatTokens(apState.lastTokenEstimate)} tokens)`);
1887
+ console.log(` Prune cycles: ${apState.pruneCount}`);
1888
+ if (apState.history.length >= 2) {
1889
+ const first = apState.history[0];
1890
+ const last = apState.history[apState.history.length - 1];
1891
+ const growthRate = (last.pct - first.pct) / apState.history.length;
1892
+ if (growthRate > 0) {
1893
+ const turnsLeft = Math.ceil((1.0 - pct) / growthRate);
1894
+ console.log(` Est. runway: ~${turnsLeft} turns until prune threshold`);
1895
+ }
1896
+ }
1897
+ }
1898
+
1899
+ if (sessions.length > 0) {
1900
+ console.log('\n Recent sessions:');
1901
+ for (const s of sessions.slice(0, 10)) {
1902
+ console.log(` - ${s.session_id}: ${s.cnt} turns`);
1903
+ }
1904
+ }
1905
+
1906
+ console.log('');
1907
+ await backend.shutdown();
1908
+ }
1909
+
1910
+ // ============================================================================
1911
+ // Exports for testing
1912
+ // ============================================================================
1913
+
1914
+ export {
1915
+ SQLiteBackend,
1916
+ RuVectorBackend,
1917
+ JsonFileBackend,
1918
+ resolveBackend,
1919
+ getRuVectorConfig,
1920
+ createEmbedding,
1921
+ createHashEmbedding,
1922
+ getOnnxPipeline,
1923
+ EMBEDDING_DIM,
1924
+ hashContent,
1925
+ parseTranscript,
1926
+ extractTextContent,
1927
+ extractToolCalls,
1928
+ extractFilePaths,
1929
+ chunkTranscript,
1930
+ extractSummary,
1931
+ buildEntry,
1932
+ buildCompactInstructions,
1933
+ computeImportance,
1934
+ retrieveContextSmart,
1935
+ autoOptimize,
1936
+ crossSessionSearch,
1937
+ storeChunks,
1938
+ retrieveContext,
1939
+ readStdin,
1940
+ // Autopilot
1941
+ estimateContextTokens,
1942
+ loadAutopilotState,
1943
+ saveAutopilotState,
1944
+ runAutopilot,
1945
+ buildProgressBar,
1946
+ formatTokens,
1947
+ buildAutopilotReport,
1948
+ NAMESPACE,
1949
+ ARCHIVE_DB_PATH,
1950
+ ARCHIVE_JSON_PATH,
1951
+ COMPACT_INSTRUCTION_BUDGET,
1952
+ RETENTION_DAYS,
1953
+ AUTO_OPTIMIZE,
1954
+ AUTOPILOT_ENABLED,
1955
+ CONTEXT_WINDOW_TOKENS,
1956
+ AUTOPILOT_WARN_PCT,
1957
+ AUTOPILOT_PRUNE_PCT,
1958
+ };
1959
+
1960
+ // ============================================================================
1961
+ // Main
1962
+ // ============================================================================
1963
+
1964
+ const command = process.argv[2] || 'status';
1965
+
1966
+ try {
1967
+ switch (command) {
1968
+ case 'pre-compact': await doPreCompact(); break;
1969
+ case 'session-start': await doSessionStart(); break;
1970
+ case 'user-prompt-submit': await doUserPromptSubmit(); break;
1971
+ case 'status': await doStatus(); break;
1972
+ default:
1973
+ console.log('Usage: context-persistence-hook.mjs <pre-compact|session-start|user-prompt-submit|status>');
1974
+ process.exit(1);
1975
+ }
1976
+ } catch (err) {
1977
+ // Hooks must never crash Claude Code - fail silently
1978
+ process.stderr.write(`[ContextPersistence] Error (non-critical): ${err.message}\n`);
1979
+ }