@monoes/monomindcli 1.14.7 → 1.15.1
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.
- package/.claude/agents/reengineer-squad/boss.md +113 -0
- package/.claude/agents/reengineer-squad/critic-architect.md +132 -0
- package/.claude/agents/reengineer-squad/git-manager.md +145 -0
- package/.claude/agents/reengineer-squad/idea-generator.md +95 -0
- package/.claude/agents/reengineer-squad/implementer.md +112 -0
- package/.claude/agents/reengineer-squad/integration-planner.md +112 -0
- package/.claude/agents/reengineer-squad/source-analyst.md +103 -0
- package/.claude/agents/reengineer-squad/target-analyst.md +118 -0
- package/.claude/agents/reengineer-squad/tester.md +105 -0
- package/.claude/commands/mastermind/master.md +35 -14
- package/.claude/helpers/handlers/capture-handler.cjs +155 -18
- package/.claude/helpers/monolean-activate.cjs +20 -0
- package/.claude/helpers/monolean-config.cjs +76 -0
- package/.claude/helpers/monolean-instructions.cjs +109 -0
- package/.claude/helpers/monolean-propagate.cjs +9 -0
- package/.claude/helpers/monolean-tracker.cjs +18 -0
- package/.claude/helpers/skill-registry.json +2 -2
- package/.claude/skills/agent-browser-testing/SKILL.md +301 -18
- package/.claude/skills/mastermind/runorg.md +69 -23
- package/.claude/skills/monodesign/SKILL.md +32 -1
- package/.claude/skills/monodesign/adapt.md +53 -0
- package/.claude/skills/monodesign/agents/monodesign-asset-producer.md +100 -0
- package/.claude/skills/monodesign/animate.md +65 -0
- package/.claude/skills/monodesign/audit.md +89 -0
- package/.claude/skills/monodesign/bolder.md +50 -0
- package/.claude/skills/monodesign/clarify.md +64 -0
- package/.claude/skills/monodesign/colorize.md +68 -0
- package/.claude/skills/monodesign/craft.md +51 -0
- package/.claude/skills/monodesign/critique.md +66 -0
- package/.claude/skills/monodesign/delight.md +47 -0
- package/.claude/skills/monodesign/distill.md +56 -0
- package/.claude/skills/monodesign/document.md +80 -0
- package/.claude/skills/monodesign/extract.md +74 -0
- package/.claude/skills/monodesign/harden.md +65 -0
- package/.claude/skills/monodesign/live.md +59 -0
- package/.claude/skills/monodesign/onboard.md +50 -0
- package/.claude/skills/monodesign/optimize.md +64 -0
- package/.claude/skills/monodesign/overdrive.md +56 -0
- package/.claude/skills/monodesign/polish.md +68 -0
- package/.claude/skills/monodesign/quieter.md +57 -0
- package/.claude/skills/monodesign/reference/antipatterns-catalog.md +248 -76
- package/.claude/skills/monodesign/reference/codex.md +107 -0
- package/.claude/skills/monodesign/reference/craft.md +3 -0
- package/.claude/skills/monodesign/reference/hooks.md +99 -0
- package/.claude/skills/monodesign/reference/image-prompts.md +12 -0
- package/.claude/skills/monodesign/shape.md +71 -0
- package/.claude/skills/monodesign/teach.md +69 -0
- package/.claude/skills/monodesign/typeset.md +59 -0
- package/.claude/skills/monolean/SKILL.md +118 -0
- package/.claude/skills/monolean-audit/SKILL.md +41 -0
- package/.claude/skills/monolean-debt/SKILL.md +46 -0
- package/.claude/skills/monolean-help/SKILL.md +60 -0
- package/.claude/skills/monolean-review/SKILL.md +57 -0
- package/bin/cli.js +3 -1
- package/dist/dashboard/server.js +137 -0
- package/dist/src/__tests__/browse-adapters.test.d.ts +2 -0
- package/dist/src/__tests__/browse-adapters.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-adapters.test.js +51 -0
- package/dist/src/__tests__/browse-adapters.test.js.map +1 -0
- package/dist/src/__tests__/browse-analyzer.test.d.ts +2 -0
- package/dist/src/__tests__/browse-analyzer.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-analyzer.test.js +68 -0
- package/dist/src/__tests__/browse-analyzer.test.js.map +1 -0
- package/dist/src/__tests__/browse-builtin-handlers.test.d.ts +2 -0
- package/dist/src/__tests__/browse-builtin-handlers.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-builtin-handlers.test.js +139 -0
- package/dist/src/__tests__/browse-builtin-handlers.test.js.map +1 -0
- package/dist/src/__tests__/browse-cdp.test.d.ts +2 -0
- package/dist/src/__tests__/browse-cdp.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-cdp.test.js +169 -0
- package/dist/src/__tests__/browse-cdp.test.js.map +1 -0
- package/dist/src/__tests__/browse-dashboard.test.d.ts +2 -0
- package/dist/src/__tests__/browse-dashboard.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-dashboard.test.js +179 -0
- package/dist/src/__tests__/browse-dashboard.test.js.map +1 -0
- package/dist/src/__tests__/browse-engine.test.d.ts +2 -0
- package/dist/src/__tests__/browse-engine.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-engine.test.js +122 -0
- package/dist/src/__tests__/browse-engine.test.js.map +1 -0
- package/dist/src/__tests__/browse-expression.test.d.ts +2 -0
- package/dist/src/__tests__/browse-expression.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-expression.test.js +54 -0
- package/dist/src/__tests__/browse-expression.test.js.map +1 -0
- package/dist/src/__tests__/browse-store.test.d.ts +2 -0
- package/dist/src/__tests__/browse-store.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-store.test.js +99 -0
- package/dist/src/__tests__/browse-store.test.js.map +1 -0
- package/dist/src/__tests__/browse-workflow-types.test.d.ts +2 -0
- package/dist/src/__tests__/browse-workflow-types.test.d.ts.map +1 -0
- package/dist/src/__tests__/browse-workflow-types.test.js +33 -0
- package/dist/src/__tests__/browse-workflow-types.test.js.map +1 -0
- package/dist/src/browser/action-builder/analyzer.d.ts +11 -0
- package/dist/src/browser/action-builder/analyzer.d.ts.map +1 -0
- package/dist/src/browser/action-builder/analyzer.js +71 -0
- package/dist/src/browser/action-builder/analyzer.js.map +1 -0
- package/dist/src/browser/action-builder/types.d.ts +47 -0
- package/dist/src/browser/action-builder/types.d.ts.map +1 -0
- package/dist/src/browser/action-builder/types.js +2 -0
- package/dist/src/browser/action-builder/types.js.map +1 -0
- package/dist/src/browser/adapters/gemini.d.ts +3 -0
- package/dist/src/browser/adapters/gemini.d.ts.map +1 -0
- package/dist/src/browser/adapters/gemini.js +16 -0
- package/dist/src/browser/adapters/gemini.js.map +1 -0
- package/dist/src/browser/adapters/google.d.ts +3 -0
- package/dist/src/browser/adapters/google.d.ts.map +1 -0
- package/dist/src/browser/adapters/google.js +17 -0
- package/dist/src/browser/adapters/google.js.map +1 -0
- package/dist/src/browser/adapters/index.d.ts +19 -0
- package/dist/src/browser/adapters/index.d.ts.map +1 -0
- package/dist/src/browser/adapters/index.js +23 -0
- package/dist/src/browser/adapters/index.js.map +1 -0
- package/dist/src/browser/adapters/instagram.d.ts +3 -0
- package/dist/src/browser/adapters/instagram.d.ts.map +1 -0
- package/dist/src/browser/adapters/instagram.js +17 -0
- package/dist/src/browser/adapters/instagram.js.map +1 -0
- package/dist/src/browser/adapters/linkedin.d.ts +3 -0
- package/dist/src/browser/adapters/linkedin.d.ts.map +1 -0
- package/dist/src/browser/adapters/linkedin.js +19 -0
- package/dist/src/browser/adapters/linkedin.js.map +1 -0
- package/dist/src/browser/adapters/microsoft.d.ts +3 -0
- package/dist/src/browser/adapters/microsoft.d.ts.map +1 -0
- package/dist/src/browser/adapters/microsoft.js +16 -0
- package/dist/src/browser/adapters/microsoft.js.map +1 -0
- package/dist/src/browser/adapters/x.d.ts +3 -0
- package/dist/src/browser/adapters/x.d.ts.map +1 -0
- package/dist/src/browser/adapters/x.js +19 -0
- package/dist/src/browser/adapters/x.js.map +1 -0
- package/dist/src/browser/dashboard/api-types.d.ts +50 -0
- package/dist/src/browser/dashboard/api-types.d.ts.map +1 -0
- package/dist/src/browser/dashboard/api-types.js +14 -0
- package/dist/src/browser/dashboard/api-types.js.map +1 -0
- package/dist/src/browser/dashboard/server.d.ts +9 -0
- package/dist/src/browser/dashboard/server.d.ts.map +1 -0
- package/dist/src/browser/dashboard/server.js +62 -0
- package/dist/src/browser/dashboard/server.js.map +1 -0
- package/dist/src/browser/dashboard/ui.html +1811 -0
- package/dist/src/browser/workflow/builtin-handlers.d.ts +3 -0
- package/dist/src/browser/workflow/builtin-handlers.d.ts.map +1 -0
- package/dist/src/browser/workflow/builtin-handlers.js +343 -0
- package/dist/src/browser/workflow/builtin-handlers.js.map +1 -0
- package/dist/src/browser/workflow/engine.d.ts +15 -0
- package/dist/src/browser/workflow/engine.d.ts.map +1 -0
- package/dist/src/browser/workflow/engine.js +127 -0
- package/dist/src/browser/workflow/engine.js.map +1 -0
- package/dist/src/browser/workflow/expression.d.ts +4 -0
- package/dist/src/browser/workflow/expression.d.ts.map +1 -0
- package/dist/src/browser/workflow/expression.js +64 -0
- package/dist/src/browser/workflow/expression.js.map +1 -0
- package/dist/src/browser/workflow/store.d.ts +24 -0
- package/dist/src/browser/workflow/store.d.ts.map +1 -0
- package/dist/src/browser/workflow/store.js +145 -0
- package/dist/src/browser/workflow/store.js.map +1 -0
- package/dist/src/browser/workflow/types.d.ts +48 -0
- package/dist/src/browser/workflow/types.d.ts.map +1 -0
- package/dist/src/browser/workflow/types.js +2 -0
- package/dist/src/browser/workflow/types.js.map +1 -0
- package/dist/src/commands/browse-action.d.ts +4 -0
- package/dist/src/commands/browse-action.d.ts.map +1 -0
- package/dist/src/commands/browse-action.js +151 -0
- package/dist/src/commands/browse-action.js.map +1 -0
- package/dist/src/commands/browse-platform.d.ts +4 -0
- package/dist/src/commands/browse-platform.d.ts.map +1 -0
- package/dist/src/commands/browse-platform.js +117 -0
- package/dist/src/commands/browse-platform.js.map +1 -0
- package/dist/src/commands/browse-workflow.d.ts +4 -0
- package/dist/src/commands/browse-workflow.d.ts.map +1 -0
- package/dist/src/commands/browse-workflow.js +153 -0
- package/dist/src/commands/browse-workflow.js.map +1 -0
- package/dist/src/commands/browse.d.ts +10 -6
- package/dist/src/commands/browse.d.ts.map +1 -1
- package/dist/src/commands/browse.js +11 -2154
- package/dist/src/commands/browse.js.map +1 -1
- package/dist/src/commands/design-detect.d.ts +21 -0
- package/dist/src/commands/design-detect.d.ts.map +1 -0
- package/dist/src/commands/design-detect.js +127 -0
- package/dist/src/commands/design-detect.js.map +1 -0
- package/dist/src/commands/design-palette.d.ts +22 -0
- package/dist/src/commands/design-palette.d.ts.map +1 -0
- package/dist/src/commands/design-palette.js +539 -0
- package/dist/src/commands/design-palette.js.map +1 -0
- package/dist/src/commands/hooks-core-commands.d.ts +10 -0
- package/dist/src/commands/hooks-core-commands.d.ts.map +1 -0
- package/dist/src/commands/hooks-core-commands.js +377 -0
- package/dist/src/commands/hooks-core-commands.js.map +1 -0
- package/dist/src/commands/hooks-coverage-commands.d.ts +12 -0
- package/dist/src/commands/hooks-coverage-commands.d.ts.map +1 -0
- package/dist/src/commands/hooks-coverage-commands.js +1217 -0
- package/dist/src/commands/hooks-coverage-commands.js.map +1 -0
- package/dist/src/commands/hooks-coverage-utils.d.ts +42 -0
- package/dist/src/commands/hooks-coverage-utils.d.ts.map +1 -0
- package/dist/src/commands/hooks-coverage-utils.js +220 -0
- package/dist/src/commands/hooks-coverage-utils.js.map +1 -0
- package/dist/src/commands/hooks-extended-commands.d.ts +14 -0
- package/dist/src/commands/hooks-extended-commands.d.ts.map +1 -0
- package/dist/src/commands/hooks-extended-commands.js +579 -0
- package/dist/src/commands/hooks-extended-commands.js.map +1 -0
- package/dist/src/commands/hooks-formatting.d.ts +13 -0
- package/dist/src/commands/hooks-formatting.d.ts.map +1 -0
- package/dist/src/commands/hooks-formatting.js +42 -0
- package/dist/src/commands/hooks-formatting.js.map +1 -0
- package/dist/src/commands/hooks-routing-commands.d.ts +15 -0
- package/dist/src/commands/hooks-routing-commands.d.ts.map +1 -0
- package/dist/src/commands/hooks-routing-commands.js +723 -0
- package/dist/src/commands/hooks-routing-commands.js.map +1 -0
- package/dist/src/commands/hooks-workers.d.ts +9 -0
- package/dist/src/commands/hooks-workers.d.ts.map +1 -0
- package/dist/src/commands/hooks-workers.js +782 -0
- package/dist/src/commands/hooks-workers.js.map +1 -0
- package/dist/src/commands/hooks.d.ts +8 -0
- package/dist/src/commands/hooks.d.ts.map +1 -1
- package/dist/src/commands/hooks.js +179 -4103
- package/dist/src/commands/hooks.js.map +1 -1
- package/dist/src/commands/index.d.ts +1 -0
- package/dist/src/commands/index.d.ts.map +1 -1
- package/dist/src/commands/index.js +6 -0
- package/dist/src/commands/index.js.map +1 -1
- package/dist/src/commands/org.d.ts.map +1 -1
- package/dist/src/commands/org.js +14 -15
- package/dist/src/commands/org.js.map +1 -1
- package/dist/src/commands/tokens.d.ts.map +1 -1
- package/dist/src/commands/tokens.js +77 -1
- package/dist/src/commands/tokens.js.map +1 -1
- package/dist/src/init/executor.d.ts.map +1 -1
- package/dist/src/init/executor.js +18 -8
- package/dist/src/init/executor.js.map +1 -1
- package/dist/src/init/settings-generator.d.ts.map +1 -1
- package/dist/src/init/settings-generator.js +39 -5
- package/dist/src/init/settings-generator.js.map +1 -1
- package/dist/src/init/statusline-generator.d.ts.map +1 -1
- package/dist/src/init/statusline-generator.js +25 -5
- package/dist/src/init/statusline-generator.js.map +1 -1
- package/dist/src/mcp-tools/browser-tools.d.ts +3 -5
- package/dist/src/mcp-tools/browser-tools.d.ts.map +1 -1
- package/dist/src/mcp-tools/browser-tools.js +619 -326
- package/dist/src/mcp-tools/browser-tools.js.map +1 -1
- package/dist/src/mcp-tools/hooks-embedding.d.ts +161 -0
- package/dist/src/mcp-tools/hooks-embedding.d.ts.map +1 -0
- package/dist/src/mcp-tools/hooks-embedding.js +506 -0
- package/dist/src/mcp-tools/hooks-embedding.js.map +1 -0
- package/dist/src/mcp-tools/hooks-intelligence.d.ts +26 -0
- package/dist/src/mcp-tools/hooks-intelligence.d.ts.map +1 -0
- package/dist/src/mcp-tools/hooks-intelligence.js +1328 -0
- package/dist/src/mcp-tools/hooks-intelligence.js.map +1 -0
- package/dist/src/mcp-tools/hooks-routing.d.ts +27 -0
- package/dist/src/mcp-tools/hooks-routing.d.ts.map +1 -0
- package/dist/src/mcp-tools/hooks-routing.js +1591 -0
- package/dist/src/mcp-tools/hooks-routing.js.map +1 -0
- package/dist/src/mcp-tools/hooks-tools.d.ts +3 -38
- package/dist/src/mcp-tools/hooks-tools.d.ts.map +1 -1
- package/dist/src/mcp-tools/hooks-tools.js +5 -3393
- package/dist/src/mcp-tools/hooks-tools.js.map +1 -1
- package/dist/src/mcp-tools/monograph-tools.d.ts.map +1 -1
- package/dist/src/mcp-tools/monograph-tools.js +24 -14
- package/dist/src/mcp-tools/monograph-tools.js.map +1 -1
- package/dist/src/mcp-tools/workflow-tools.d.ts.map +1 -1
- package/dist/src/mcp-tools/workflow-tools.js +54 -1
- package/dist/src/mcp-tools/workflow-tools.js.map +1 -1
- package/dist/src/memory/embedding-operations.d.ts +58 -0
- package/dist/src/memory/embedding-operations.d.ts.map +1 -0
- package/dist/src/memory/embedding-operations.js +299 -0
- package/dist/src/memory/embedding-operations.js.map +1 -0
- package/dist/src/memory/ewc-consolidation.d.ts.map +1 -1
- package/dist/src/memory/ewc-consolidation.js +37 -3
- package/dist/src/memory/ewc-consolidation.js.map +1 -1
- package/dist/src/memory/hnsw-operations.d.ts +130 -0
- package/dist/src/memory/hnsw-operations.d.ts.map +1 -0
- package/dist/src/memory/hnsw-operations.js +400 -0
- package/dist/src/memory/hnsw-operations.js.map +1 -0
- package/dist/src/memory/intelligence.d.ts.map +1 -1
- package/dist/src/memory/intelligence.js +42 -23
- package/dist/src/memory/intelligence.js.map +1 -1
- package/dist/src/memory/memory-bridge.d.ts.map +1 -1
- package/dist/src/memory/memory-bridge.js +52 -8
- package/dist/src/memory/memory-bridge.js.map +1 -1
- package/dist/src/memory/memory-crud.d.ts +67 -0
- package/dist/src/memory/memory-crud.d.ts.map +1 -0
- package/dist/src/memory/memory-crud.js +415 -0
- package/dist/src/memory/memory-crud.js.map +1 -0
- package/dist/src/memory/memory-initializer.d.ts +9 -322
- package/dist/src/memory/memory-initializer.d.ts.map +1 -1
- package/dist/src/memory/memory-initializer.js +17 -1794
- package/dist/src/memory/memory-initializer.js.map +1 -1
- package/dist/src/memory/memory-migrations.d.ts +30 -0
- package/dist/src/memory/memory-migrations.d.ts.map +1 -0
- package/dist/src/memory/memory-migrations.js +134 -0
- package/dist/src/memory/memory-migrations.js.map +1 -0
- package/dist/src/memory/memory-read.d.ts +78 -0
- package/dist/src/memory/memory-read.d.ts.map +1 -0
- package/dist/src/memory/memory-read.js +331 -0
- package/dist/src/memory/memory-read.js.map +1 -0
- package/dist/src/memory/memory-schema.d.ts +13 -0
- package/dist/src/memory/memory-schema.d.ts.map +1 -0
- package/dist/src/memory/memory-schema.js +167 -0
- package/dist/src/memory/memory-schema.js.map +1 -0
- package/dist/src/memory/sona-optimizer.d.ts.map +1 -1
- package/dist/src/memory/sona-optimizer.js +37 -4
- package/dist/src/memory/sona-optimizer.js.map +1 -1
- package/dist/src/monovector/route-outcomes.d.ts.map +1 -1
- package/dist/src/monovector/route-outcomes.js +16 -6
- package/dist/src/monovector/route-outcomes.js.map +1 -1
- package/dist/src/pricing/model-pricing.d.ts +41 -0
- package/dist/src/pricing/model-pricing.d.ts.map +1 -0
- package/dist/src/pricing/model-pricing.js +61 -0
- package/dist/src/pricing/model-pricing.js.map +1 -0
- package/dist/src/ui/.monomind/capture/active-run.json +1 -0
- package/dist/src/ui/.monomind/orgs/system-trial-qa/runs/real-events-1782290897.convs.jsonl +3 -0
- package/dist/src/ui/.monomind/orgs/system-trial-qa/runs/real-events-1782290897.jsonl +11 -0
- package/dist/src/ui/.monomind/orgs/system-trial-qa/runs/rigid-qa-restart-1782288201.jsonl +540 -0
- package/dist/src/ui/.monomind/orgs/system-trial-qa-threads.jsonl +3 -0
- package/dist/src/ui/.monomind/orgs/test-event-fix/runs/rigid-qa-restart-1782288201.jsonl +2 -0
- package/dist/src/ui/MODULARIZATION_PLAN.md +79 -0
- package/dist/src/ui/collector.mjs +23 -13
- package/dist/src/ui/dashboard.html +1653 -14
- package/dist/src/ui/data/known-projects.json +1 -0
- package/dist/src/ui/data/mastermind-events.jsonl +553 -0
- package/dist/src/ui/data/sessions/_index.json +1 -0
- package/dist/src/ui/data/sessions/final-sess-001.jsonl +542 -0
- package/dist/src/ui/data/unknown-events.jsonl +1 -0
- package/dist/src/ui/orgs.html +154 -10
- package/dist/src/ui/server.mjs +1162 -168
- package/dist/src/ui/sse-manager.mjs +119 -0
- package/dist/src/update/checker.js +1 -1
- package/dist/src/update/checker.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/dist/workflow/builtin-handlers.js +321 -0
- package/dist/workflow/engine.js +253 -0
- package/dist/workflow/expression.js +98 -0
- package/dist/workflow/types.js +2 -0
- package/package.json +8 -6
|
@@ -0,0 +1,1591 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Hooks Routing MCP Tools
|
|
3
|
+
* MCP tool implementations for pre/post edit/command, route, explain, pretrain,
|
|
4
|
+
* build-agents, transfer, session, list, metrics, pre-task, post-task, intelligence.
|
|
5
|
+
* Extracted from hooks-tools.ts.
|
|
6
|
+
*/
|
|
7
|
+
import { mkdirSync, writeFileSync, renameSync, existsSync, readFileSync } from 'fs';
|
|
8
|
+
import { join, resolve, sep } from 'path';
|
|
9
|
+
import { getProjectCwd } from './types.js';
|
|
10
|
+
import { randomUUID } from 'node:crypto';
|
|
11
|
+
import { recordRoute, joinOutcome, joinLatestUnresolved } from '../monovector/route-outcomes.js';
|
|
12
|
+
import { recordCommand, deriveRecentSuccess } from '../monovector/command-outcomes.js';
|
|
13
|
+
import { extractKeywords, loadRoutingOutcomes, saveRoutingOutcomes, loadMemoryStore, getIntelligenceStatsFromMemory, suggestAgentsForFile, suggestAgentsForTask, suggestAgentsFromIntelligence, assessCommandRisk, activeTrajectories, getMemoryPath, getRouteOutcomesBaseDir, getRoutingOutcomesPath, getRealSearchFunction, getRealStoreFunction, getSONAOptimizer, getFileExtension, TASK_PATTERNS, MEMORY_DIR, MEMORY_FILE, } from './hooks-embedding.js';
|
|
14
|
+
// MCP Tool implementations - return raw data for direct CLI use
|
|
15
|
+
export const hooksPreEdit = {
|
|
16
|
+
name: 'hooks_pre-edit',
|
|
17
|
+
description: 'Get context and agent suggestions before editing a file',
|
|
18
|
+
inputSchema: {
|
|
19
|
+
type: 'object',
|
|
20
|
+
properties: {
|
|
21
|
+
filePath: { type: 'string', description: 'Path to the file being edited' },
|
|
22
|
+
operation: { type: 'string', description: 'Type of operation (create, update, delete, refactor)' },
|
|
23
|
+
context: { type: 'string', description: 'Additional context' },
|
|
24
|
+
},
|
|
25
|
+
required: ['filePath'],
|
|
26
|
+
},
|
|
27
|
+
handler: async (params) => {
|
|
28
|
+
// Cap filePath: passed to suggestAgentsForFile (O(n) regex) and reflected in
|
|
29
|
+
// response. Cap operation to prevent oversized strings in recommendations.
|
|
30
|
+
const MAX_PRE_EDIT_PATH_LEN = 4 * 1024;
|
|
31
|
+
const MAX_PRE_EDIT_OP_LEN = 64;
|
|
32
|
+
const rawFilePath = params.filePath;
|
|
33
|
+
const filePath = typeof rawFilePath === 'string' && rawFilePath.length > MAX_PRE_EDIT_PATH_LEN
|
|
34
|
+
? rawFilePath.slice(0, MAX_PRE_EDIT_PATH_LEN)
|
|
35
|
+
: rawFilePath;
|
|
36
|
+
const rawOperation = params.operation || 'update';
|
|
37
|
+
const operation = typeof rawOperation === 'string' && rawOperation.length > MAX_PRE_EDIT_OP_LEN
|
|
38
|
+
? rawOperation.slice(0, MAX_PRE_EDIT_OP_LEN)
|
|
39
|
+
: rawOperation;
|
|
40
|
+
const suggestedAgents = suggestAgentsForFile(filePath);
|
|
41
|
+
const ext = getFileExtension(filePath);
|
|
42
|
+
return {
|
|
43
|
+
filePath,
|
|
44
|
+
operation,
|
|
45
|
+
context: {
|
|
46
|
+
fileExists: true,
|
|
47
|
+
fileType: ext || 'unknown',
|
|
48
|
+
relatedFiles: [],
|
|
49
|
+
suggestedAgents,
|
|
50
|
+
patterns: [
|
|
51
|
+
{ pattern: `${ext} file editing`, confidence: 0.85 },
|
|
52
|
+
],
|
|
53
|
+
risks: operation === 'delete' ? ['File deletion is irreversible'] : [],
|
|
54
|
+
},
|
|
55
|
+
recommendations: [
|
|
56
|
+
`Recommended agents: ${suggestedAgents.join(', ')}`,
|
|
57
|
+
'Run tests after changes',
|
|
58
|
+
],
|
|
59
|
+
};
|
|
60
|
+
},
|
|
61
|
+
};
|
|
62
|
+
export const hooksPostEdit = {
|
|
63
|
+
name: 'hooks_post-edit',
|
|
64
|
+
description: 'Record editing outcome for learning',
|
|
65
|
+
inputSchema: {
|
|
66
|
+
type: 'object',
|
|
67
|
+
properties: {
|
|
68
|
+
filePath: { type: 'string', description: 'Path to the edited file' },
|
|
69
|
+
success: { type: 'boolean', description: 'Whether the edit was successful' },
|
|
70
|
+
agent: { type: 'string', description: 'Agent that performed the edit' },
|
|
71
|
+
},
|
|
72
|
+
required: ['filePath'],
|
|
73
|
+
},
|
|
74
|
+
handler: async (params) => {
|
|
75
|
+
// Cap filePath: interpolated into taskId and task text forwarded to
|
|
76
|
+
// bridgeRecordFeedback (which calls generateEmbedding — O(n) hash fallback).
|
|
77
|
+
// Cap agent: stored in feedback record and forwarded to bridge.
|
|
78
|
+
const MAX_POST_EDIT_PATH_LEN = 4 * 1024;
|
|
79
|
+
const MAX_POST_EDIT_AGENT_LEN = 256;
|
|
80
|
+
const rawFilePath = params.filePath;
|
|
81
|
+
const filePath = typeof rawFilePath === 'string' && rawFilePath.length > MAX_POST_EDIT_PATH_LEN
|
|
82
|
+
? rawFilePath.slice(0, MAX_POST_EDIT_PATH_LEN)
|
|
83
|
+
: rawFilePath;
|
|
84
|
+
const success = params.success !== false;
|
|
85
|
+
const rawAgent = params.agent;
|
|
86
|
+
const agent = typeof rawAgent === 'string' && rawAgent.length > MAX_POST_EDIT_AGENT_LEN
|
|
87
|
+
? rawAgent.slice(0, MAX_POST_EDIT_AGENT_LEN)
|
|
88
|
+
: rawAgent;
|
|
89
|
+
// Wire recordFeedback through bridge (issue #1209)
|
|
90
|
+
let feedbackResult = null;
|
|
91
|
+
try {
|
|
92
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
93
|
+
feedbackResult = await bridge.bridgeRecordFeedback({
|
|
94
|
+
taskId: `edit-${filePath}-${Date.now()}`,
|
|
95
|
+
success,
|
|
96
|
+
quality: success ? 0.85 : 0.3,
|
|
97
|
+
agent,
|
|
98
|
+
// B1.2: give the SONA embedder real semantics (the edited file) instead of
|
|
99
|
+
// the opaque task ID.
|
|
100
|
+
task: `edit ${filePath}`,
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
catch {
|
|
104
|
+
// Bridge not available — continue with basic response
|
|
105
|
+
}
|
|
106
|
+
return {
|
|
107
|
+
recorded: true,
|
|
108
|
+
filePath,
|
|
109
|
+
success,
|
|
110
|
+
timestamp: new Date().toISOString(),
|
|
111
|
+
learningUpdate: success ? 'pattern_reinforced' : 'pattern_adjusted',
|
|
112
|
+
feedback: feedbackResult ? {
|
|
113
|
+
recorded: feedbackResult.success,
|
|
114
|
+
controller: feedbackResult.controller,
|
|
115
|
+
updates: feedbackResult.updated,
|
|
116
|
+
} : { recorded: false, controller: 'unavailable', updates: 0 },
|
|
117
|
+
};
|
|
118
|
+
},
|
|
119
|
+
};
|
|
120
|
+
export const hooksPreCommand = {
|
|
121
|
+
name: 'hooks_pre-command',
|
|
122
|
+
description: 'Assess risk before executing a command',
|
|
123
|
+
inputSchema: {
|
|
124
|
+
type: 'object',
|
|
125
|
+
properties: {
|
|
126
|
+
command: { type: 'string', description: 'Command to execute' },
|
|
127
|
+
},
|
|
128
|
+
required: ['command'],
|
|
129
|
+
},
|
|
130
|
+
handler: async (params) => {
|
|
131
|
+
// Cap command length: assessCommandRisk runs O(n) string searches, and the
|
|
132
|
+
// raw command is reflected verbatim in the response. Limit to 4 KB which
|
|
133
|
+
// is far beyond any realistic shell command.
|
|
134
|
+
const MAX_CMD_LEN = 4 * 1024;
|
|
135
|
+
const rawCommand = params.command;
|
|
136
|
+
const command = typeof rawCommand === 'string' && rawCommand.length > MAX_CMD_LEN
|
|
137
|
+
? rawCommand.slice(0, MAX_CMD_LEN)
|
|
138
|
+
: rawCommand;
|
|
139
|
+
const assessment = assessCommandRisk(command);
|
|
140
|
+
const riskLevel = assessment.level >= 0.8 ? 'critical'
|
|
141
|
+
: assessment.level >= 0.6 ? 'high'
|
|
142
|
+
: assessment.level >= 0.3 ? 'medium'
|
|
143
|
+
: 'low';
|
|
144
|
+
return {
|
|
145
|
+
command,
|
|
146
|
+
riskLevel,
|
|
147
|
+
risks: assessment.warnings.map((warning, i) => ({
|
|
148
|
+
type: `risk-${i + 1}`,
|
|
149
|
+
severity: assessment.level >= 0.6 ? 'high' : 'medium',
|
|
150
|
+
description: warning,
|
|
151
|
+
})),
|
|
152
|
+
recommendations: assessment.warnings.length > 0
|
|
153
|
+
? ['Review warnings before proceeding', 'Consider using safer alternative']
|
|
154
|
+
: ['Command appears safe to execute'],
|
|
155
|
+
safeAlternatives: [],
|
|
156
|
+
shouldProceed: assessment.level < 0.7,
|
|
157
|
+
};
|
|
158
|
+
},
|
|
159
|
+
};
|
|
160
|
+
export const hooksPostCommand = {
|
|
161
|
+
name: 'hooks_post-command',
|
|
162
|
+
description: 'Record command execution outcome',
|
|
163
|
+
inputSchema: {
|
|
164
|
+
type: 'object',
|
|
165
|
+
properties: {
|
|
166
|
+
command: { type: 'string', description: 'Executed command' },
|
|
167
|
+
exitCode: { type: 'number', description: 'Command exit code' },
|
|
168
|
+
},
|
|
169
|
+
required: ['command'],
|
|
170
|
+
},
|
|
171
|
+
handler: async (params) => {
|
|
172
|
+
// Cap command: it is stored in JSON memory store (line 824), forwarded to
|
|
173
|
+
// bridgeStoreEntry which calls generateEmbedding by default — O(n) hash
|
|
174
|
+
// fallback, and reflected verbatim in the response. The recordCommand path
|
|
175
|
+
// already caps to 200 chars; apply a consistent 4 KB cap here that still
|
|
176
|
+
// covers any realistic shell command.
|
|
177
|
+
const MAX_POST_CMD_LEN = 4 * 1024;
|
|
178
|
+
const rawPostCommand = params.command;
|
|
179
|
+
const command = typeof rawPostCommand === 'string' && rawPostCommand.length > MAX_POST_CMD_LEN
|
|
180
|
+
? rawPostCommand.slice(0, MAX_POST_CMD_LEN)
|
|
181
|
+
: rawPostCommand;
|
|
182
|
+
const exitCode = params.exitCode || 0;
|
|
183
|
+
const success = exitCode === 0;
|
|
184
|
+
// Record the real exit code in the time-windowed command-outcome store so
|
|
185
|
+
// post-task can derive a MEASURED success signal (grounded in actual exit
|
|
186
|
+
// codes) when the caller does not explicitly assert --success. Non-fatal.
|
|
187
|
+
await recordCommand(getRouteOutcomesBaseDir(), {
|
|
188
|
+
ts: Date.now(),
|
|
189
|
+
command: typeof command === 'string' ? command.slice(0, 200) : String(command).slice(0, 200),
|
|
190
|
+
exitCode,
|
|
191
|
+
});
|
|
192
|
+
// Persist command outcome via AgentDB
|
|
193
|
+
let _storedIn = 'none';
|
|
194
|
+
try {
|
|
195
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
196
|
+
await bridge.bridgeStoreEntry({
|
|
197
|
+
key: `cmd-${Date.now()}`,
|
|
198
|
+
value: JSON.stringify({ command, exitCode, success }),
|
|
199
|
+
namespace: 'commands',
|
|
200
|
+
tags: [success ? 'success' : 'error'],
|
|
201
|
+
});
|
|
202
|
+
_storedIn = 'agentdb';
|
|
203
|
+
}
|
|
204
|
+
catch {
|
|
205
|
+
// AgentDB not available — store in JSON
|
|
206
|
+
try {
|
|
207
|
+
const store = loadMemoryStore();
|
|
208
|
+
const key = `cmd-${Date.now()}`;
|
|
209
|
+
store.entries[key] = { key, value: JSON.stringify({ command, exitCode, success }), namespace: 'commands', createdAt: new Date().toISOString() };
|
|
210
|
+
const memDir = join(getProjectCwd(), MEMORY_DIR);
|
|
211
|
+
if (!existsSync(memDir))
|
|
212
|
+
mkdirSync(memDir, { recursive: true });
|
|
213
|
+
const _mp = getMemoryPath();
|
|
214
|
+
const _mptmp = _mp + '.tmp';
|
|
215
|
+
writeFileSync(_mptmp, JSON.stringify(store, null, 2), 'utf-8');
|
|
216
|
+
renameSync(_mptmp, _mp);
|
|
217
|
+
_storedIn = 'json-store';
|
|
218
|
+
}
|
|
219
|
+
catch { /* non-critical */ }
|
|
220
|
+
}
|
|
221
|
+
return {
|
|
222
|
+
recorded: _storedIn !== 'none',
|
|
223
|
+
command,
|
|
224
|
+
exitCode,
|
|
225
|
+
success,
|
|
226
|
+
timestamp: new Date().toISOString(),
|
|
227
|
+
_storedIn,
|
|
228
|
+
};
|
|
229
|
+
},
|
|
230
|
+
};
|
|
231
|
+
export const hooksRoute = {
|
|
232
|
+
name: 'hooks_route',
|
|
233
|
+
description: 'Route task to optimal agent using semantic similarity (native HNSW or pure JS)',
|
|
234
|
+
inputSchema: {
|
|
235
|
+
type: 'object',
|
|
236
|
+
properties: {
|
|
237
|
+
task: { type: 'string', description: 'Task description' },
|
|
238
|
+
context: { type: 'string', description: 'Additional context' },
|
|
239
|
+
useSemanticRouter: { type: 'boolean', description: 'Use semantic similarity routing (default: true)' },
|
|
240
|
+
},
|
|
241
|
+
required: ['task'],
|
|
242
|
+
},
|
|
243
|
+
handler: async (params) => {
|
|
244
|
+
// Cap task and context lengths: both are forwarded to generateEmbedding
|
|
245
|
+
// via bridgeRouteTask, and task is used in extractKeywords + stored in
|
|
246
|
+
// route-outcomes.jsonl. 16 KB matches the cap in hooksPatternSearch.
|
|
247
|
+
const MAX_ROUTE_TASK_LEN = 16 * 1024;
|
|
248
|
+
const MAX_ROUTE_CTX_LEN = 4 * 1024;
|
|
249
|
+
const rawTask = params.task;
|
|
250
|
+
const task = typeof rawTask === 'string' && rawTask.length > MAX_ROUTE_TASK_LEN
|
|
251
|
+
? rawTask.slice(0, MAX_ROUTE_TASK_LEN)
|
|
252
|
+
: rawTask;
|
|
253
|
+
const rawContext = params.context;
|
|
254
|
+
const context = typeof rawContext === 'string' && rawContext.length > MAX_ROUTE_CTX_LEN
|
|
255
|
+
? rawContext.slice(0, MAX_ROUTE_CTX_LEN)
|
|
256
|
+
: rawContext;
|
|
257
|
+
const useSemanticRouter = params.useSemanticRouter !== false;
|
|
258
|
+
// Phase 5: Try AgentDB's SemanticRouter / LearningSystem first
|
|
259
|
+
if (useSemanticRouter) {
|
|
260
|
+
try {
|
|
261
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
262
|
+
const agentdbRoute = await bridge.bridgeRouteTask({ task, context });
|
|
263
|
+
if (agentdbRoute && agentdbRoute.confidence > 0.5) {
|
|
264
|
+
const agents = agentdbRoute.agents.length > 0 ? agentdbRoute.agents : ['coder', 'researcher'];
|
|
265
|
+
const complexity = task.length > 200 ? 'high' : task.length < 50 ? 'low' : 'medium';
|
|
266
|
+
const agentdbMethod = `agentdb-${agentdbRoute.controller}`;
|
|
267
|
+
const agentdbConfidence = Math.round(agentdbRoute.confidence * 100) / 100;
|
|
268
|
+
// Record the route recommendation so post-task can join the actual outcome
|
|
269
|
+
const routeId = randomUUID();
|
|
270
|
+
await recordRoute(getRouteOutcomesBaseDir(), {
|
|
271
|
+
routeId,
|
|
272
|
+
ts: Date.now(),
|
|
273
|
+
task,
|
|
274
|
+
recommendedAgent: agents[0],
|
|
275
|
+
routingMethod: agentdbMethod,
|
|
276
|
+
confidence: agentdbConfidence,
|
|
277
|
+
learningMode: 'js',
|
|
278
|
+
});
|
|
279
|
+
return {
|
|
280
|
+
routeId,
|
|
281
|
+
task,
|
|
282
|
+
routing: {
|
|
283
|
+
method: agentdbMethod,
|
|
284
|
+
backend: agentdbRoute.controller,
|
|
285
|
+
latencyMs: 0,
|
|
286
|
+
throughput: 'N/A',
|
|
287
|
+
},
|
|
288
|
+
matchedPattern: agentdbRoute.route,
|
|
289
|
+
semanticMatches: [{ pattern: agentdbRoute.route, score: agentdbRoute.confidence }],
|
|
290
|
+
primaryAgent: {
|
|
291
|
+
type: agents[0],
|
|
292
|
+
confidence: Math.round(agentdbRoute.confidence * 100) / 100,
|
|
293
|
+
reason: `AgentDB ${agentdbRoute.controller}: "${agentdbRoute.route}" (${Math.round(agentdbRoute.confidence * 100)}%)`,
|
|
294
|
+
},
|
|
295
|
+
alternativeAgents: agents.slice(1).map((agent, i) => ({
|
|
296
|
+
type: agent,
|
|
297
|
+
confidence: Math.round((agentdbRoute.confidence - (0.1 * (i + 1))) * 100) / 100,
|
|
298
|
+
reason: `Alternative from ${agentdbRoute.controller}`,
|
|
299
|
+
})),
|
|
300
|
+
estimatedMetrics: {
|
|
301
|
+
successProbability: Math.round(agentdbRoute.confidence * 100) / 100,
|
|
302
|
+
estimatedDuration: complexity === 'high' ? '2-4 hours' : complexity === 'medium' ? '30-60 min' : '10-30 min',
|
|
303
|
+
complexity,
|
|
304
|
+
},
|
|
305
|
+
swarmRecommendation: agents.length > 2 ? { topology: 'hierarchical', agents, coordination: 'queen-led' } : null,
|
|
306
|
+
};
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
catch {
|
|
310
|
+
// AgentDB router not available — fall through to local routing
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
// Deterministic keyword routing is the baseline (and only) local path.
|
|
314
|
+
const semanticResult = [];
|
|
315
|
+
let routingMethod = 'keyword';
|
|
316
|
+
const routingLatencyMs = 0;
|
|
317
|
+
let backendInfo = '';
|
|
318
|
+
// Get agents from keyword routing
|
|
319
|
+
let agents;
|
|
320
|
+
let confidence;
|
|
321
|
+
let matchedPattern = '';
|
|
322
|
+
{
|
|
323
|
+
// Keyword fallback is the baseline
|
|
324
|
+
const keywordSuggestion = suggestAgentsForTask(task);
|
|
325
|
+
agents = keywordSuggestion.agents;
|
|
326
|
+
confidence = keywordSuggestion.confidence;
|
|
327
|
+
matchedPattern = 'keyword-fallback';
|
|
328
|
+
routingMethod = 'keyword';
|
|
329
|
+
backendInfo = 'keyword matching';
|
|
330
|
+
// V3: augment with neural ReasoningBank patterns — merge into agent list
|
|
331
|
+
// rather than replacing, so keyword precision is preserved while neural
|
|
332
|
+
// adds learned agents from past sessions.
|
|
333
|
+
const intelSuggestion = await suggestAgentsFromIntelligence(task).catch(() => null);
|
|
334
|
+
if (intelSuggestion && intelSuggestion.confidence > 0.5) {
|
|
335
|
+
// Prepend neural agents (deduped) and boost confidence
|
|
336
|
+
const existingSet = new Set(agents);
|
|
337
|
+
const neuralOnly = intelSuggestion.agents.filter(a => !existingSet.has(a));
|
|
338
|
+
agents = [...intelSuggestion.agents, ...agents.filter(a => !new Set(intelSuggestion.agents).has(a))];
|
|
339
|
+
const neuralWeight = intelSuggestion.confidence > 0.7 ? 0.65 : 0.5;
|
|
340
|
+
const keywordWeight = 1 - neuralWeight;
|
|
341
|
+
confidence = Math.min(0.95, intelSuggestion.confidence * neuralWeight +
|
|
342
|
+
confidence * keywordWeight +
|
|
343
|
+
(neuralOnly.length > 0 ? 0.03 : 0));
|
|
344
|
+
matchedPattern = 'neural+keyword';
|
|
345
|
+
routingMethod = 'neural-augmented';
|
|
346
|
+
backendInfo = 'intelligence ReasoningBank + keyword matching';
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
// Determine complexity
|
|
350
|
+
const taskLower = task.toLowerCase();
|
|
351
|
+
const complexity = taskLower.includes('complex') || taskLower.includes('architecture') || task.length > 200
|
|
352
|
+
? 'high'
|
|
353
|
+
: taskLower.includes('simple') || taskLower.includes('fix') || task.length < 50
|
|
354
|
+
? 'low'
|
|
355
|
+
: 'medium';
|
|
356
|
+
const primaryConfidence = Math.round(confidence * 100) / 100;
|
|
357
|
+
// Record the route recommendation so post-task can join the actual outcome
|
|
358
|
+
const routeId = randomUUID();
|
|
359
|
+
await recordRoute(getRouteOutcomesBaseDir(), {
|
|
360
|
+
routeId,
|
|
361
|
+
ts: Date.now(),
|
|
362
|
+
task,
|
|
363
|
+
recommendedAgent: agents[0],
|
|
364
|
+
routingMethod,
|
|
365
|
+
confidence: primaryConfidence,
|
|
366
|
+
learningMode: 'js',
|
|
367
|
+
});
|
|
368
|
+
return {
|
|
369
|
+
routeId,
|
|
370
|
+
task,
|
|
371
|
+
routing: {
|
|
372
|
+
method: routingMethod,
|
|
373
|
+
backend: backendInfo,
|
|
374
|
+
latencyMs: routingLatencyMs,
|
|
375
|
+
throughput: routingLatencyMs > 0 ? `${Math.round(1000 / routingLatencyMs)} routes/s` : 'N/A',
|
|
376
|
+
},
|
|
377
|
+
matchedPattern,
|
|
378
|
+
semanticMatches: semanticResult.slice(0, 3).map(r => ({
|
|
379
|
+
pattern: r.intent,
|
|
380
|
+
score: Math.round(r.score * 100) / 100,
|
|
381
|
+
})),
|
|
382
|
+
primaryAgent: {
|
|
383
|
+
type: agents[0],
|
|
384
|
+
confidence: Math.round(confidence * 100) / 100,
|
|
385
|
+
reason: routingMethod.startsWith('semantic')
|
|
386
|
+
? `Semantic similarity to "${matchedPattern}" pattern (${Math.round(confidence * 100)}%)`
|
|
387
|
+
: `Task contains keywords matching ${agents[0]} specialization`,
|
|
388
|
+
},
|
|
389
|
+
alternativeAgents: agents.slice(1).map((agent, i) => ({
|
|
390
|
+
type: agent,
|
|
391
|
+
confidence: Math.round((confidence - (0.1 * (i + 1))) * 100) / 100,
|
|
392
|
+
reason: `Alternative agent for ${agent} capabilities`,
|
|
393
|
+
})),
|
|
394
|
+
estimatedMetrics: {
|
|
395
|
+
successProbability: Math.round(confidence * 100) / 100,
|
|
396
|
+
estimatedDuration: complexity === 'high' ? '2-4 hours' : complexity === 'medium' ? '30-60 min' : '10-30 min',
|
|
397
|
+
complexity,
|
|
398
|
+
},
|
|
399
|
+
swarmRecommendation: agents.length > 2 ? {
|
|
400
|
+
topology: 'hierarchical',
|
|
401
|
+
agents,
|
|
402
|
+
coordination: 'queen-led',
|
|
403
|
+
} : null,
|
|
404
|
+
};
|
|
405
|
+
},
|
|
406
|
+
};
|
|
407
|
+
export const hooksMetrics = {
|
|
408
|
+
name: 'hooks_metrics',
|
|
409
|
+
description: 'View learning metrics dashboard',
|
|
410
|
+
inputSchema: {
|
|
411
|
+
type: 'object',
|
|
412
|
+
properties: {
|
|
413
|
+
period: { type: 'string', description: 'Metrics period (1h, 24h, 7d, 30d)' },
|
|
414
|
+
includeV1: { type: 'boolean', description: 'Include v1 performance metrics' },
|
|
415
|
+
},
|
|
416
|
+
},
|
|
417
|
+
handler: async (params) => {
|
|
418
|
+
const period = params.period || '24h';
|
|
419
|
+
// Try to read real counts from memory store
|
|
420
|
+
const store = loadMemoryStore();
|
|
421
|
+
const entries = Object.values(store.entries);
|
|
422
|
+
// Count patterns by looking at stored pattern entries
|
|
423
|
+
const patternEntries = entries.filter(e => e.key.includes('pattern'));
|
|
424
|
+
const routingEntries = entries.filter(e => e.key.includes('route') || e.key.includes('routing'));
|
|
425
|
+
const taskEntries = entries.filter(e => e.key.includes('task'));
|
|
426
|
+
if (entries.length === 0) {
|
|
427
|
+
return {
|
|
428
|
+
_real: true,
|
|
429
|
+
_note: 'No metrics data collected yet. Data populates from hooks_post-task, hooks_post-edit, hooks_post-command, and hooks_route calls.',
|
|
430
|
+
period,
|
|
431
|
+
patterns: { total: 0, successful: 0, failed: 0, avgConfidence: null },
|
|
432
|
+
agents: { routingAccuracy: null, totalRoutes: 0, topAgent: null },
|
|
433
|
+
commands: { totalExecuted: 0, successRate: null, avgRiskScore: null },
|
|
434
|
+
lastUpdated: new Date().toISOString(),
|
|
435
|
+
};
|
|
436
|
+
}
|
|
437
|
+
return {
|
|
438
|
+
period,
|
|
439
|
+
patterns: {
|
|
440
|
+
total: patternEntries.length,
|
|
441
|
+
successful: null,
|
|
442
|
+
failed: null,
|
|
443
|
+
avgConfidence: null,
|
|
444
|
+
},
|
|
445
|
+
agents: {
|
|
446
|
+
routingAccuracy: null,
|
|
447
|
+
totalRoutes: routingEntries.length,
|
|
448
|
+
topAgent: null,
|
|
449
|
+
},
|
|
450
|
+
commands: {
|
|
451
|
+
totalExecuted: taskEntries.length,
|
|
452
|
+
successRate: null,
|
|
453
|
+
avgRiskScore: null,
|
|
454
|
+
},
|
|
455
|
+
dataSource: 'memory-store',
|
|
456
|
+
entriesFound: entries.length,
|
|
457
|
+
lastUpdated: new Date().toISOString(),
|
|
458
|
+
};
|
|
459
|
+
},
|
|
460
|
+
};
|
|
461
|
+
export const hooksList = {
|
|
462
|
+
name: 'hooks_list',
|
|
463
|
+
description: 'List all registered hooks',
|
|
464
|
+
inputSchema: {
|
|
465
|
+
type: 'object',
|
|
466
|
+
properties: {},
|
|
467
|
+
},
|
|
468
|
+
handler: async () => {
|
|
469
|
+
return {
|
|
470
|
+
hooks: [
|
|
471
|
+
// Core hooks
|
|
472
|
+
{ name: 'pre-edit', type: 'PreToolUse', status: 'active' },
|
|
473
|
+
{ name: 'post-edit', type: 'PostToolUse', status: 'active' },
|
|
474
|
+
{ name: 'pre-command', type: 'PreToolUse', status: 'active' },
|
|
475
|
+
{ name: 'post-command', type: 'PostToolUse', status: 'active' },
|
|
476
|
+
{ name: 'pre-task', type: 'PreToolUse', status: 'active' },
|
|
477
|
+
{ name: 'post-task', type: 'PostToolUse', status: 'active' },
|
|
478
|
+
// Routing hooks
|
|
479
|
+
{ name: 'route', type: 'intelligence', status: 'active' },
|
|
480
|
+
{ name: 'explain', type: 'intelligence', status: 'active' },
|
|
481
|
+
// Session hooks
|
|
482
|
+
{ name: 'session-start', type: 'SessionStart', status: 'active' },
|
|
483
|
+
{ name: 'session-end', type: 'SessionEnd', status: 'active' },
|
|
484
|
+
{ name: 'session-restore', type: 'SessionStart', status: 'active' },
|
|
485
|
+
// Learning hooks
|
|
486
|
+
{ name: 'pretrain', type: 'intelligence', status: 'active' },
|
|
487
|
+
{ name: 'build-agents', type: 'intelligence', status: 'active' },
|
|
488
|
+
{ name: 'transfer', type: 'intelligence', status: 'active' },
|
|
489
|
+
{ name: 'metrics', type: 'analytics', status: 'active' },
|
|
490
|
+
// System hooks
|
|
491
|
+
{ name: 'init', type: 'system', status: 'active' },
|
|
492
|
+
{ name: 'notify', type: 'coordination', status: 'active' },
|
|
493
|
+
// Intelligence subcommands
|
|
494
|
+
{ name: 'intelligence', type: 'intelligence', status: 'active' },
|
|
495
|
+
{ name: 'intelligence_trajectory-start', type: 'intelligence', status: 'active' },
|
|
496
|
+
{ name: 'intelligence_trajectory-step', type: 'intelligence', status: 'active' },
|
|
497
|
+
{ name: 'intelligence_trajectory-end', type: 'intelligence', status: 'active' },
|
|
498
|
+
{ name: 'intelligence_pattern-store', type: 'intelligence', status: 'active' },
|
|
499
|
+
{ name: 'intelligence_pattern-search', type: 'intelligence', status: 'active' },
|
|
500
|
+
{ name: 'intelligence_stats', type: 'analytics', status: 'active' },
|
|
501
|
+
{ name: 'intelligence_learn', type: 'intelligence', status: 'active' },
|
|
502
|
+
{ name: 'intelligence_attention', type: 'intelligence', status: 'active' },
|
|
503
|
+
],
|
|
504
|
+
total: 26,
|
|
505
|
+
};
|
|
506
|
+
},
|
|
507
|
+
};
|
|
508
|
+
export const hooksPreTask = {
|
|
509
|
+
name: 'hooks_pre-task',
|
|
510
|
+
description: 'Record task start and get agent suggestions with intelligent model routing (ADR-026)',
|
|
511
|
+
inputSchema: {
|
|
512
|
+
type: 'object',
|
|
513
|
+
properties: {
|
|
514
|
+
taskId: { type: 'string', description: 'Task identifier' },
|
|
515
|
+
description: { type: 'string', description: 'Task description' },
|
|
516
|
+
filePath: { type: 'string', description: 'Optional file path for AST analysis' },
|
|
517
|
+
},
|
|
518
|
+
required: ['taskId', 'description'],
|
|
519
|
+
},
|
|
520
|
+
handler: async (params) => {
|
|
521
|
+
// Cap taskId: it is used as a suffix in SQLite memory keys (heuristic:${taskId},
|
|
522
|
+
// routing-decision:${taskId}, textual_gradient:${taskId}) and as sourceId/targetId
|
|
523
|
+
// in causal-graph edges persisted to SQLite. An uncapped ID can inflate the DB key
|
|
524
|
+
// column and every JSON payload that includes the ID.
|
|
525
|
+
const MAX_TASK_ID_LEN = 256;
|
|
526
|
+
const rawTaskId = params.taskId;
|
|
527
|
+
const taskId = typeof rawTaskId === 'string' && rawTaskId.length > MAX_TASK_ID_LEN
|
|
528
|
+
? rawTaskId.slice(0, MAX_TASK_ID_LEN)
|
|
529
|
+
: rawTaskId;
|
|
530
|
+
// Cap description: it is forwarded to generateEmbedding twice (ERL heuristics
|
|
531
|
+
// + TextGrad gradient queries) and used in O(n) keyword extraction.
|
|
532
|
+
// 16 KB matches the cap applied in hooks_route and hooksPatternSearch.
|
|
533
|
+
const MAX_PRE_TASK_DESC_LEN = 16 * 1024;
|
|
534
|
+
const rawDescription = params.description;
|
|
535
|
+
const description = typeof rawDescription === 'string' && rawDescription.length > MAX_PRE_TASK_DESC_LEN
|
|
536
|
+
? rawDescription.slice(0, MAX_PRE_TASK_DESC_LEN)
|
|
537
|
+
: rawDescription;
|
|
538
|
+
const filePath = params.filePath;
|
|
539
|
+
const suggestion = suggestAgentsForTask(description);
|
|
540
|
+
// Determine complexity
|
|
541
|
+
const descLower = description.toLowerCase();
|
|
542
|
+
const complexity = descLower.includes('complex') || descLower.includes('architecture') || description.length > 200
|
|
543
|
+
? 'high'
|
|
544
|
+
: descLower.includes('simple') || descLower.includes('fix') || description.length < 50
|
|
545
|
+
? 'low'
|
|
546
|
+
: 'medium';
|
|
547
|
+
// Enhanced model routing module was never shipped — modelRouting stays undefined.
|
|
548
|
+
const modelRouting = undefined;
|
|
549
|
+
// ERL: Retrieve past heuristics to inject into recommendations
|
|
550
|
+
// Source: https://arxiv.org/abs/2603.24639
|
|
551
|
+
const erlHints = [];
|
|
552
|
+
try {
|
|
553
|
+
const searchFn = await getRealSearchFunction();
|
|
554
|
+
if (searchFn) {
|
|
555
|
+
const heuristicResults = await searchFn({
|
|
556
|
+
query: description,
|
|
557
|
+
namespace: 'heuristics',
|
|
558
|
+
limit: 3,
|
|
559
|
+
threshold: 0.6,
|
|
560
|
+
});
|
|
561
|
+
for (const r of (heuristicResults?.results ?? [])) {
|
|
562
|
+
try {
|
|
563
|
+
const h = JSON.parse(r.content ?? '{}');
|
|
564
|
+
if (h.action && h.confidence !== undefined && h.confidence >= 0.6) {
|
|
565
|
+
erlHints.push(`ERL hint (conf=${h.confidence.toFixed(2)}): use "${h.action}" for tasks involving "${h.condition ?? 'similar context'}"`);
|
|
566
|
+
}
|
|
567
|
+
}
|
|
568
|
+
catch { /* skip malformed */ }
|
|
569
|
+
}
|
|
570
|
+
// TextGrad: also inject relevant past failure gradients to guide away from known pitfalls
|
|
571
|
+
// Source: https://arxiv.org/abs/2406.07496
|
|
572
|
+
const gradientResults = await searchFn({
|
|
573
|
+
query: description,
|
|
574
|
+
namespace: 'gradients',
|
|
575
|
+
limit: 2,
|
|
576
|
+
threshold: 0.55,
|
|
577
|
+
});
|
|
578
|
+
for (const r of (gradientResults?.results ?? [])) {
|
|
579
|
+
const critique = r.content ?? '';
|
|
580
|
+
if (critique && critique.length > 10) {
|
|
581
|
+
erlHints.push(`TextGrad warning: ${critique.slice(0, 120)}`);
|
|
582
|
+
}
|
|
583
|
+
}
|
|
584
|
+
}
|
|
585
|
+
}
|
|
586
|
+
catch { /* non-critical */ }
|
|
587
|
+
return {
|
|
588
|
+
taskId,
|
|
589
|
+
description,
|
|
590
|
+
suggestedAgents: suggestion.agents.map((agent, i) => ({
|
|
591
|
+
type: agent,
|
|
592
|
+
confidence: suggestion.confidence - (0.05 * i),
|
|
593
|
+
reason: i === 0
|
|
594
|
+
? `Primary agent for ${agent} tasks based on learned patterns`
|
|
595
|
+
: `Alternative agent with ${agent} capabilities`,
|
|
596
|
+
})),
|
|
597
|
+
complexity,
|
|
598
|
+
estimatedDuration: complexity === 'high' ? '2-4 hours' : complexity === 'medium' ? '30-60 min' : '10-30 min',
|
|
599
|
+
risks: complexity === 'high' ? ['Complex task may require multiple iterations'] : [],
|
|
600
|
+
recommendations: [
|
|
601
|
+
`Use ${suggestion.agents[0]} as primary agent`,
|
|
602
|
+
suggestion.agents.length > 2 ? 'Consider using swarm coordination' : 'Single agent recommended',
|
|
603
|
+
...erlHints,
|
|
604
|
+
],
|
|
605
|
+
modelRouting,
|
|
606
|
+
timestamp: new Date().toISOString(),
|
|
607
|
+
};
|
|
608
|
+
},
|
|
609
|
+
};
|
|
610
|
+
export const hooksPostTask = {
|
|
611
|
+
name: 'hooks_post-task',
|
|
612
|
+
description: 'Record task completion for learning',
|
|
613
|
+
inputSchema: {
|
|
614
|
+
type: 'object',
|
|
615
|
+
properties: {
|
|
616
|
+
taskId: { type: 'string', description: 'Task identifier' },
|
|
617
|
+
success: { type: 'boolean', description: 'Whether task was successful' },
|
|
618
|
+
agent: { type: 'string', description: 'Agent that completed the task' },
|
|
619
|
+
quality: { type: 'number', description: 'Quality score (0-1)' },
|
|
620
|
+
task: { type: 'string', description: 'Task description text (used for learning keyword extraction)' },
|
|
621
|
+
storeDecisions: { type: 'boolean', description: 'Also store routing decision in memory DB' },
|
|
622
|
+
routeId: { type: 'string', description: 'Route ID from a prior hooks_route call — joins the recommendation to this outcome' },
|
|
623
|
+
},
|
|
624
|
+
required: ['taskId'],
|
|
625
|
+
},
|
|
626
|
+
handler: async (params) => {
|
|
627
|
+
// Cap taskId for the same reason as hooks_pre_task: it flows into SQLite memory keys
|
|
628
|
+
// (heuristic:${taskId}, routing-decision:${taskId}, textual_gradient:${taskId}) and
|
|
629
|
+
// into causal-graph edge IDs persisted to the DB. Without a cap an attacker can
|
|
630
|
+
// inflate every row that stores the raw ID.
|
|
631
|
+
const MAX_POST_TASK_ID_LEN = 256;
|
|
632
|
+
const rawPostTaskId = params.taskId;
|
|
633
|
+
const taskId = typeof rawPostTaskId === 'string' && rawPostTaskId.length > MAX_POST_TASK_ID_LEN
|
|
634
|
+
? rawPostTaskId.slice(0, MAX_POST_TASK_ID_LEN)
|
|
635
|
+
: rawPostTaskId;
|
|
636
|
+
// The success flag, when the caller asserts it (--success true), is taken as
|
|
637
|
+
// ground truth. But callers usually do NOT pass it. Rather than treating every
|
|
638
|
+
// unverified task as "unknown" (and thus excluding it from learning), we now
|
|
639
|
+
// derive a MEASURED success signal from the real command exit codes recorded by
|
|
640
|
+
// post-command within a recent time window. post-command appends each exit code
|
|
641
|
+
// to the command-outcome store keyed by timestamp; deriveRecentSuccess returns:
|
|
642
|
+
// true → recent commands exist and the LAST command exited 0 (final-state heuristic)
|
|
643
|
+
// false → recent commands exist and the LAST command exited non-zero
|
|
644
|
+
// null → no recent commands (genuinely no signal → stays unknown)
|
|
645
|
+
// Note: "final-state" not "all must pass" — intermediate failures (e.g. grep no-match,
|
|
646
|
+
// test-then-fix cycles) are intentionally ignored; the last exit code decides.
|
|
647
|
+
// Precedence: an explicit --success ALWAYS wins; the derived signal only fills
|
|
648
|
+
// in when no explicit flag is given; only when there is also no recent command
|
|
649
|
+
// signal does the outcome stay unknown (and excluded from SONA + route join,
|
|
650
|
+
// per the existing "unknown ≠ success" principle).
|
|
651
|
+
const explicitSuccess = typeof params.success === 'boolean';
|
|
652
|
+
let outcomeKnown = explicitSuccess;
|
|
653
|
+
let success = params.success !== false;
|
|
654
|
+
let successSource = explicitSuccess ? 'explicit' : 'unknown';
|
|
655
|
+
if (!explicitSuccess) {
|
|
656
|
+
const derived = await deriveRecentSuccess(getRouteOutcomesBaseDir());
|
|
657
|
+
if (derived !== null) {
|
|
658
|
+
outcomeKnown = true;
|
|
659
|
+
success = derived;
|
|
660
|
+
successSource = 'derived-commands';
|
|
661
|
+
}
|
|
662
|
+
}
|
|
663
|
+
// Cap agent: forwarded to bridgeRecordFeedback where it is stored in the
|
|
664
|
+
// feedback record and used as a tag string in the JSON store. An uncapped
|
|
665
|
+
// agent value inflates the on-disk store entry.
|
|
666
|
+
const MAX_POST_TASK_AGENT_LEN = 256;
|
|
667
|
+
const rawPostTaskAgent = params.agent;
|
|
668
|
+
const agent = typeof rawPostTaskAgent === 'string' && rawPostTaskAgent.length > MAX_POST_TASK_AGENT_LEN
|
|
669
|
+
? rawPostTaskAgent.slice(0, MAX_POST_TASK_AGENT_LEN)
|
|
670
|
+
: rawPostTaskAgent;
|
|
671
|
+
const quality = params.quality || (success ? 0.85 : 0.3);
|
|
672
|
+
const startTime = Date.now();
|
|
673
|
+
// Cap task description: passed to generateEmbedding via bridgeRecordFeedback
|
|
674
|
+
// and persisted to route-outcomes.jsonl. 16 KB matches hooks_route cap.
|
|
675
|
+
const MAX_POST_TASK_LEN = 16 * 1024;
|
|
676
|
+
const rawPostTask = params.task;
|
|
677
|
+
const cappedPostTask = typeof rawPostTask === 'string' && rawPostTask.length > MAX_POST_TASK_LEN
|
|
678
|
+
? rawPostTask.slice(0, MAX_POST_TASK_LEN)
|
|
679
|
+
: rawPostTask;
|
|
680
|
+
// Phase 3: Wire recordFeedback through bridge → LearningSystem + ReasoningBank
|
|
681
|
+
let feedbackResult = null;
|
|
682
|
+
try {
|
|
683
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
684
|
+
feedbackResult = await bridge.bridgeRecordFeedback({
|
|
685
|
+
taskId,
|
|
686
|
+
success,
|
|
687
|
+
quality,
|
|
688
|
+
agent,
|
|
689
|
+
// B1.2: thread the real task description into the SONA trajectory so the
|
|
690
|
+
// embedder encodes meaning, not the opaque task ID.
|
|
691
|
+
task: cappedPostTask || undefined,
|
|
692
|
+
// B1.3: only feed the SONA LoRA update when the outcome is actually known.
|
|
693
|
+
outcomeKnown,
|
|
694
|
+
duration: params.duration || undefined,
|
|
695
|
+
patterns: params.patterns || undefined,
|
|
696
|
+
});
|
|
697
|
+
}
|
|
698
|
+
catch {
|
|
699
|
+
// Bridge not available — continue with basic response
|
|
700
|
+
}
|
|
701
|
+
// Phase 3: Record causal edge (task → outcome)
|
|
702
|
+
try {
|
|
703
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
704
|
+
await bridge.bridgeRecordCausalEdge({
|
|
705
|
+
sourceId: taskId,
|
|
706
|
+
targetId: `outcome-${taskId}`,
|
|
707
|
+
relation: success ? 'succeeded' : 'failed',
|
|
708
|
+
weight: quality,
|
|
709
|
+
});
|
|
710
|
+
}
|
|
711
|
+
catch {
|
|
712
|
+
// Non-fatal
|
|
713
|
+
}
|
|
714
|
+
// Persist routing outcome for runtime learning (file-based, always reliable).
|
|
715
|
+
// B1.3: also gate this sibling learning sink on a known outcome — an unverified
|
|
716
|
+
// task must not train the router as a success either. When the caller did not
|
|
717
|
+
// assert success, the outcome is unknown and we skip persisting a labeled sample.
|
|
718
|
+
const taskText = cappedPostTask || '';
|
|
719
|
+
const outcomeKeywords = extractKeywords(taskText);
|
|
720
|
+
let outcomePersisted = false;
|
|
721
|
+
if (outcomeKnown && taskText && agent && agent.length <= 100 && /^[a-zA-Z0-9_-]+$/.test(agent)) {
|
|
722
|
+
try {
|
|
723
|
+
const outcomes = loadRoutingOutcomes();
|
|
724
|
+
outcomes.push({
|
|
725
|
+
task: taskText,
|
|
726
|
+
agent,
|
|
727
|
+
success,
|
|
728
|
+
quality,
|
|
729
|
+
keywords: outcomeKeywords,
|
|
730
|
+
timestamp: new Date().toISOString(),
|
|
731
|
+
});
|
|
732
|
+
saveRoutingOutcomes(outcomes);
|
|
733
|
+
outcomePersisted = true;
|
|
734
|
+
}
|
|
735
|
+
catch { /* non-critical */ }
|
|
736
|
+
}
|
|
737
|
+
// Join this outcome back onto the original route recommendation. This is the
|
|
738
|
+
// recommendation→actual→success link that routing-accuracy metrics and SONA
|
|
739
|
+
// labels depend on. When the caller threads an explicit routeId we join that
|
|
740
|
+
// record; otherwise we auto-correlate to the most recent unresolved route
|
|
741
|
+
// (within a 10-min window) so the loop closes without the LLM manually
|
|
742
|
+
// threading the routeId. Only join when the outcome is actually measured —
|
|
743
|
+
// per "unknown ≠ success", an unverified task must not pollute the metric.
|
|
744
|
+
if (outcomeKnown) {
|
|
745
|
+
const outcome = {
|
|
746
|
+
agentActuallyUsed: agent,
|
|
747
|
+
measuredSuccess: success,
|
|
748
|
+
quality: typeof params.quality === 'number' ? params.quality : undefined,
|
|
749
|
+
};
|
|
750
|
+
if (params.routeId) {
|
|
751
|
+
await joinOutcome(getRouteOutcomesBaseDir(), params.routeId, outcome);
|
|
752
|
+
}
|
|
753
|
+
else {
|
|
754
|
+
await joinLatestUnresolved(getRouteOutcomesBaseDir(), outcome);
|
|
755
|
+
}
|
|
756
|
+
}
|
|
757
|
+
// ERL: Extract and persist structured heuristic for future pre-task injection
|
|
758
|
+
// Source: https://arxiv.org/abs/2603.24639
|
|
759
|
+
if (taskText && agent && success !== undefined) {
|
|
760
|
+
try {
|
|
761
|
+
const storeFn = await getRealStoreFunction();
|
|
762
|
+
if (storeFn) {
|
|
763
|
+
const heuristic = {
|
|
764
|
+
condition: outcomeKeywords.slice(0, 3).join(', ') || taskText.slice(0, 60),
|
|
765
|
+
action: agent,
|
|
766
|
+
confidence: success ? (quality ?? 0.8) : 0.2,
|
|
767
|
+
};
|
|
768
|
+
await storeFn({
|
|
769
|
+
key: `heuristic:${taskId}`,
|
|
770
|
+
value: JSON.stringify(heuristic),
|
|
771
|
+
namespace: 'heuristics',
|
|
772
|
+
tags: ['erl', agent, success ? 'success' : 'failure'],
|
|
773
|
+
});
|
|
774
|
+
}
|
|
775
|
+
}
|
|
776
|
+
catch { /* non-critical */ }
|
|
777
|
+
}
|
|
778
|
+
// Optionally store in memory DB for cross-session vector retrieval
|
|
779
|
+
if (params.storeDecisions && taskText && agent) {
|
|
780
|
+
try {
|
|
781
|
+
const storeFn = await getRealStoreFunction();
|
|
782
|
+
if (storeFn) {
|
|
783
|
+
await storeFn({
|
|
784
|
+
key: `routing-decision:${taskId}`,
|
|
785
|
+
namespace: 'patterns',
|
|
786
|
+
value: JSON.stringify({ task: taskText, agent, success, quality, keywords: outcomeKeywords }),
|
|
787
|
+
tags: ['routing-decision'],
|
|
788
|
+
});
|
|
789
|
+
}
|
|
790
|
+
}
|
|
791
|
+
catch { /* non-critical */ }
|
|
792
|
+
}
|
|
793
|
+
const duration = Date.now() - startTime;
|
|
794
|
+
// TextGrad: Store textual gradient critique for failed tasks
|
|
795
|
+
// Source: https://arxiv.org/abs/2406.07496 (TextGrad — Nature)
|
|
796
|
+
if (!success && taskText) {
|
|
797
|
+
try {
|
|
798
|
+
const storeFn = await getRealStoreFunction();
|
|
799
|
+
if (storeFn) {
|
|
800
|
+
const critique = `Task "${taskText.slice(0, 80)}" failed with agent "${agent}". ` +
|
|
801
|
+
`Quality score: ${quality ?? 'unknown'}. ` +
|
|
802
|
+
`Improvement direction: review agent selection, consider more capable agent or task decomposition.`;
|
|
803
|
+
await storeFn({
|
|
804
|
+
key: `textual_gradient:${taskId}`,
|
|
805
|
+
value: critique,
|
|
806
|
+
namespace: 'gradients',
|
|
807
|
+
tags: ['textual_gradient', agent ?? 'unknown', 'failure'],
|
|
808
|
+
});
|
|
809
|
+
}
|
|
810
|
+
}
|
|
811
|
+
catch { /* non-critical */ }
|
|
812
|
+
}
|
|
813
|
+
// MAR: Structured multi-agent reflection on failure
|
|
814
|
+
// Source: https://arxiv.org/html/2512.20845 (MAR — December 2025)
|
|
815
|
+
const marReflection = !success ? {
|
|
816
|
+
needed: true,
|
|
817
|
+
suggestedAgents: [
|
|
818
|
+
{ role: 'diagnoser', description: 'Analyze root cause of task failure' },
|
|
819
|
+
{ role: 'critic-1', description: 'Critique from correctness angle (temperature 0.3)' },
|
|
820
|
+
{ role: 'critic-2', description: 'Critique from efficiency angle (temperature 0.8)' },
|
|
821
|
+
{ role: 'aggregator', description: 'Synthesize critiques into actionable reflection heuristic' },
|
|
822
|
+
],
|
|
823
|
+
storeAs: 'heuristics',
|
|
824
|
+
note: 'Spawn agents sequentially: Diagnoser → Critics in parallel → Aggregator',
|
|
825
|
+
} : { needed: false };
|
|
826
|
+
return {
|
|
827
|
+
taskId,
|
|
828
|
+
success,
|
|
829
|
+
outcomeKnown,
|
|
830
|
+
successSource,
|
|
831
|
+
duration,
|
|
832
|
+
learningUpdates: {
|
|
833
|
+
patternsUpdated: feedbackResult?.updated || (success ? 2 : 1),
|
|
834
|
+
newPatterns: success ? 1 : 0,
|
|
835
|
+
trajectoryId: `traj-${Date.now()}`,
|
|
836
|
+
controller: feedbackResult?.controller || 'none',
|
|
837
|
+
outcomePersisted,
|
|
838
|
+
},
|
|
839
|
+
quality,
|
|
840
|
+
feedback: feedbackResult ? {
|
|
841
|
+
recorded: feedbackResult.success,
|
|
842
|
+
controller: feedbackResult.controller,
|
|
843
|
+
updates: feedbackResult.updated,
|
|
844
|
+
} : { recorded: false, controller: 'unavailable', updates: 0 },
|
|
845
|
+
marReflection,
|
|
846
|
+
timestamp: new Date().toISOString(),
|
|
847
|
+
};
|
|
848
|
+
},
|
|
849
|
+
};
|
|
850
|
+
// Explain hook - transparent routing explanation
|
|
851
|
+
export const hooksExplain = {
|
|
852
|
+
name: 'hooks_explain',
|
|
853
|
+
description: 'Explain routing decision with full transparency',
|
|
854
|
+
inputSchema: {
|
|
855
|
+
type: 'object',
|
|
856
|
+
properties: {
|
|
857
|
+
task: { type: 'string', description: 'Task description' },
|
|
858
|
+
agent: { type: 'string', description: 'Specific agent to explain' },
|
|
859
|
+
verbose: { type: 'boolean', description: 'Verbose explanation' },
|
|
860
|
+
},
|
|
861
|
+
required: ['task'],
|
|
862
|
+
},
|
|
863
|
+
handler: async (params) => {
|
|
864
|
+
// Cap task: forwarded to suggestAgentsForTask (O(n) keyword loop + extractKeywords),
|
|
865
|
+
// .toLowerCase() (O(n)), and reflected verbatim in the response.
|
|
866
|
+
const MAX_EXPLAIN_TASK_LEN = 16 * 1024;
|
|
867
|
+
const rawExplainTask = params.task;
|
|
868
|
+
const task = typeof rawExplainTask === 'string' && rawExplainTask.length > MAX_EXPLAIN_TASK_LEN
|
|
869
|
+
? rawExplainTask.slice(0, MAX_EXPLAIN_TASK_LEN)
|
|
870
|
+
: rawExplainTask;
|
|
871
|
+
const suggestion = suggestAgentsForTask(task);
|
|
872
|
+
const taskLower = task.toLowerCase();
|
|
873
|
+
// Determine matched patterns
|
|
874
|
+
const matchedPatterns = [];
|
|
875
|
+
for (const [pattern, _result] of Object.entries(TASK_PATTERNS)) {
|
|
876
|
+
if (taskLower.includes(pattern)) {
|
|
877
|
+
matchedPatterns.push({
|
|
878
|
+
pattern,
|
|
879
|
+
matchScore: pattern.length / Math.max(taskLower.length, 1), // real ratio: pattern length vs task length
|
|
880
|
+
examples: [`Keyword "${pattern}" matched in task description`],
|
|
881
|
+
});
|
|
882
|
+
}
|
|
883
|
+
}
|
|
884
|
+
// Calculate real historical success rate from routing outcomes file
|
|
885
|
+
let historicalSuccess = null;
|
|
886
|
+
let historicalNote = 'No historical data yet';
|
|
887
|
+
try {
|
|
888
|
+
const outcomesPath = getRoutingOutcomesPath();
|
|
889
|
+
if (existsSync(outcomesPath)) {
|
|
890
|
+
const data = JSON.parse(readFileSync(outcomesPath, 'utf-8'));
|
|
891
|
+
const outcomes = data.outcomes || [];
|
|
892
|
+
if (outcomes.length > 0) {
|
|
893
|
+
historicalSuccess = outcomes.filter(o => o.success).length / outcomes.length;
|
|
894
|
+
historicalNote = `Calculated from ${outcomes.length} recorded outcomes`;
|
|
895
|
+
}
|
|
896
|
+
}
|
|
897
|
+
}
|
|
898
|
+
catch {
|
|
899
|
+
// File unreadable; leave as null
|
|
900
|
+
}
|
|
901
|
+
return {
|
|
902
|
+
task,
|
|
903
|
+
explanation: `The routing decision was made based on keyword analysis of the task description. ` +
|
|
904
|
+
`The task contains keywords that match the "${suggestion.agents[0]}" specialization with ${(suggestion.confidence * 100).toFixed(0)}% confidence.`,
|
|
905
|
+
factors: [
|
|
906
|
+
{ factor: 'Keyword Match', weight: 0.4, value: suggestion.confidence, impact: 'Primary routing signal' },
|
|
907
|
+
{ factor: 'Historical Success', weight: 0.3, value: historicalSuccess, impact: historicalNote },
|
|
908
|
+
{ factor: 'Agent Availability', weight: 0.2, value: null, impact: 'Agent availability tracking not implemented' },
|
|
909
|
+
{ factor: 'Task Complexity', weight: 0.1, value: task.length > 100 ? 0.8 : 0.3, impact: 'Complexity assessment' },
|
|
910
|
+
],
|
|
911
|
+
patterns: matchedPatterns.length > 0 ? matchedPatterns : [
|
|
912
|
+
{ pattern: 'general-task', matchScore: 0.7, examples: ['Default pattern for unclassified tasks'] }
|
|
913
|
+
],
|
|
914
|
+
decision: {
|
|
915
|
+
agent: suggestion.agents[0],
|
|
916
|
+
confidence: suggestion.confidence,
|
|
917
|
+
reasoning: [
|
|
918
|
+
`Task analysis identified ${matchedPatterns.length || 1} relevant patterns`,
|
|
919
|
+
`"${suggestion.agents[0]}" has highest capability match for this task type`,
|
|
920
|
+
historicalSuccess !== null
|
|
921
|
+
? `Historical success rate for similar tasks: ${(historicalSuccess * 100).toFixed(0)}%`
|
|
922
|
+
: `No historical outcome data available yet`,
|
|
923
|
+
`Confidence threshold met (${(suggestion.confidence * 100).toFixed(0)}% >= 70%)`,
|
|
924
|
+
],
|
|
925
|
+
},
|
|
926
|
+
};
|
|
927
|
+
},
|
|
928
|
+
};
|
|
929
|
+
// Pretrain hook - repository analysis for intelligence bootstrap
|
|
930
|
+
export const hooksPretrain = {
|
|
931
|
+
name: 'hooks_pretrain',
|
|
932
|
+
description: 'Analyze repository to bootstrap intelligence (4-step pipeline)',
|
|
933
|
+
inputSchema: {
|
|
934
|
+
type: 'object',
|
|
935
|
+
properties: {
|
|
936
|
+
path: { type: 'string', description: 'Repository path' },
|
|
937
|
+
depth: { type: 'string', description: 'Analysis depth (shallow, medium, deep)' },
|
|
938
|
+
skipCache: { type: 'boolean', description: 'Skip cached analysis' },
|
|
939
|
+
},
|
|
940
|
+
},
|
|
941
|
+
handler: async (params) => {
|
|
942
|
+
const repoPath = resolve(params.path || '.');
|
|
943
|
+
const projectRoot = getProjectCwd();
|
|
944
|
+
if (repoPath !== projectRoot && !repoPath.startsWith(projectRoot + sep)) {
|
|
945
|
+
return { error: 'Invalid path: must be within the project directory.' };
|
|
946
|
+
}
|
|
947
|
+
const depth = params.depth || 'medium';
|
|
948
|
+
const startTime = performance.now();
|
|
949
|
+
// Real file scanning — count files by extension, extract patterns
|
|
950
|
+
const { readdirSync, statSync } = await import('node:fs');
|
|
951
|
+
const extCounts = {};
|
|
952
|
+
let filesAnalyzed = 0;
|
|
953
|
+
let totalLines = 0;
|
|
954
|
+
const maxDepth = depth === 'shallow' ? 2 : depth === 'deep' ? 6 : 4;
|
|
955
|
+
const patterns = [];
|
|
956
|
+
const scan = (dir, currentDepth) => {
|
|
957
|
+
if (currentDepth > maxDepth)
|
|
958
|
+
return;
|
|
959
|
+
try {
|
|
960
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
961
|
+
for (const entry of entries) {
|
|
962
|
+
if (entry.name.startsWith('.') || entry.name === 'node_modules' || entry.name === 'dist')
|
|
963
|
+
continue;
|
|
964
|
+
const full = join(dir, entry.name);
|
|
965
|
+
if (entry.isDirectory()) {
|
|
966
|
+
scan(full, currentDepth + 1);
|
|
967
|
+
}
|
|
968
|
+
else if (entry.isFile()) {
|
|
969
|
+
const ext = entry.name.includes('.') ? entry.name.slice(entry.name.lastIndexOf('.')) : '';
|
|
970
|
+
if (ext)
|
|
971
|
+
extCounts[ext] = (extCounts[ext] || 0) + 1;
|
|
972
|
+
filesAnalyzed++;
|
|
973
|
+
// For code files, count lines and extract imports
|
|
974
|
+
if (['.ts', '.js', '.py', '.go', '.rs', '.java'].includes(ext)) {
|
|
975
|
+
try {
|
|
976
|
+
// Skip very large files (minified bundles, generated code) to prevent OOM.
|
|
977
|
+
// 1 MB is generous for a source file; anything larger is unlikely to have
|
|
978
|
+
// useful import patterns in the first 30 lines anyway.
|
|
979
|
+
const MAX_CODE_FILE_BYTES = 1 * 1024 * 1024;
|
|
980
|
+
if (statSync(full).size > MAX_CODE_FILE_BYTES)
|
|
981
|
+
continue;
|
|
982
|
+
const content = readFileSync(full, 'utf-8');
|
|
983
|
+
const lines = content.split('\n');
|
|
984
|
+
totalLines += lines.length;
|
|
985
|
+
// Extract import patterns (first 50 files max for performance)
|
|
986
|
+
if (filesAnalyzed <= 50) {
|
|
987
|
+
for (const line of lines.slice(0, 30)) {
|
|
988
|
+
if (line.startsWith('import ') || line.startsWith('from ') || line.startsWith('const ') && line.includes('require(')) {
|
|
989
|
+
const trimmed = line.trim();
|
|
990
|
+
if (trimmed.length < 120 && !patterns.includes(trimmed))
|
|
991
|
+
patterns.push(trimmed);
|
|
992
|
+
if (patterns.length >= 100)
|
|
993
|
+
break;
|
|
994
|
+
}
|
|
995
|
+
}
|
|
996
|
+
}
|
|
997
|
+
}
|
|
998
|
+
catch { /* skip unreadable */ }
|
|
999
|
+
}
|
|
1000
|
+
}
|
|
1001
|
+
}
|
|
1002
|
+
}
|
|
1003
|
+
catch { /* skip inaccessible dirs */ }
|
|
1004
|
+
};
|
|
1005
|
+
scan(repoPath, 0);
|
|
1006
|
+
const elapsed = Math.round(performance.now() - startTime);
|
|
1007
|
+
// Store extracted patterns in AgentDB
|
|
1008
|
+
let patternsStored = 0;
|
|
1009
|
+
try {
|
|
1010
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
1011
|
+
await bridge.bridgeStoreEntry({
|
|
1012
|
+
key: `pretrain-${Date.now()}`,
|
|
1013
|
+
value: JSON.stringify({ filesAnalyzed, totalLines, topExtensions: Object.entries(extCounts).sort((a, b) => b[1] - a[1]).slice(0, 10), importPatterns: patterns.slice(0, 20) }),
|
|
1014
|
+
namespace: 'pretrain',
|
|
1015
|
+
tags: ['pretrain', depth],
|
|
1016
|
+
});
|
|
1017
|
+
patternsStored = patterns.length;
|
|
1018
|
+
}
|
|
1019
|
+
catch { /* AgentDB not available */ }
|
|
1020
|
+
// Feed extracted import patterns into the neural training system so
|
|
1021
|
+
// pretrain actually trains, not just scans.
|
|
1022
|
+
let neuralPatternsLearned = 0;
|
|
1023
|
+
if (patterns.length > 0) {
|
|
1024
|
+
try {
|
|
1025
|
+
const intel = await import('../memory/intelligence.js');
|
|
1026
|
+
await intel.initializeIntelligence({ loraLearningRate: 0.002, maxTrajectorySize: patterns.length });
|
|
1027
|
+
// Record each extracted pattern as an action step
|
|
1028
|
+
for (const pat of patterns.slice(0, 50)) {
|
|
1029
|
+
await intel.recordStep({ type: 'action', content: pat, metadata: { source: 'pretrain', depth } });
|
|
1030
|
+
}
|
|
1031
|
+
// Record the entire scan as a completed trajectory
|
|
1032
|
+
const steps = patterns.slice(0, 50).map(p => ({ type: 'action', content: p }));
|
|
1033
|
+
await intel.recordTrajectory(steps, 'success');
|
|
1034
|
+
intel.flushPatterns();
|
|
1035
|
+
neuralPatternsLearned = steps.length;
|
|
1036
|
+
}
|
|
1037
|
+
catch { /* intelligence not available */ }
|
|
1038
|
+
}
|
|
1039
|
+
return {
|
|
1040
|
+
success: true,
|
|
1041
|
+
_real: true,
|
|
1042
|
+
path: repoPath,
|
|
1043
|
+
depth,
|
|
1044
|
+
durationMs: elapsed,
|
|
1045
|
+
stats: {
|
|
1046
|
+
filesAnalyzed,
|
|
1047
|
+
totalLines,
|
|
1048
|
+
patternsExtracted: patterns.length,
|
|
1049
|
+
patternsStored,
|
|
1050
|
+
neuralPatternsLearned,
|
|
1051
|
+
fileTypes: Object.entries(extCounts).sort((a, b) => b[1] - a[1]).slice(0, 15).map(([ext, count]) => ({ ext, count })),
|
|
1052
|
+
},
|
|
1053
|
+
};
|
|
1054
|
+
},
|
|
1055
|
+
};
|
|
1056
|
+
// Build agents hook - generate optimized agent configs
|
|
1057
|
+
export const hooksBuildAgents = {
|
|
1058
|
+
name: 'hooks_build-agents',
|
|
1059
|
+
description: 'Generate optimized agent configurations from pretrain data',
|
|
1060
|
+
inputSchema: {
|
|
1061
|
+
type: 'object',
|
|
1062
|
+
properties: {
|
|
1063
|
+
outputDir: { type: 'string', description: 'Output directory for configs' },
|
|
1064
|
+
focus: { type: 'string', description: 'Focus area (v1-implementation, security, performance, all)' },
|
|
1065
|
+
format: { type: 'string', description: 'Config format (yaml, json)' },
|
|
1066
|
+
persist: { type: 'boolean', description: 'Write configs to disk' },
|
|
1067
|
+
},
|
|
1068
|
+
},
|
|
1069
|
+
handler: async (params) => {
|
|
1070
|
+
const rawOutputDir = resolve(params.outputDir || './agents');
|
|
1071
|
+
const outputDir = rawOutputDir;
|
|
1072
|
+
if (!outputDir.startsWith(getProjectCwd() + sep) && outputDir !== getProjectCwd()) {
|
|
1073
|
+
return { error: 'Invalid outputDir: must be within the project directory.' };
|
|
1074
|
+
}
|
|
1075
|
+
const focus = params.focus || 'all';
|
|
1076
|
+
// Strict allowlist on `format` — without this, `format = "yaml/../../../etc/cron.d/x"`
|
|
1077
|
+
// collapses through `join` and lets writes escape the validated outputDir.
|
|
1078
|
+
const ALLOWED_FORMATS = new Set(['yaml', 'json']);
|
|
1079
|
+
const formatRaw = params.format || 'yaml';
|
|
1080
|
+
if (!ALLOWED_FORMATS.has(formatRaw)) {
|
|
1081
|
+
return { error: 'Invalid format: must be yaml or json' };
|
|
1082
|
+
}
|
|
1083
|
+
const format = formatRaw;
|
|
1084
|
+
const persist = params.persist !== false; // Default to true
|
|
1085
|
+
const agents = [
|
|
1086
|
+
{ type: 'coder', configFile: join(outputDir, `coder.${format}`), capabilities: ['code-generation', 'refactoring', 'debugging'], optimizations: ['token-reduction', 'context-caching'] },
|
|
1087
|
+
{ type: 'architect', configFile: join(outputDir, `architect.${format}`), capabilities: ['system-design', 'api-design', 'documentation'], optimizations: ['context-caching', 'memory-persistence'] },
|
|
1088
|
+
{ type: 'tester', configFile: join(outputDir, `tester.${format}`), capabilities: ['unit-testing', 'integration-testing', 'coverage'], optimizations: ['parallel-execution'] },
|
|
1089
|
+
{ type: 'security-architect', configFile: join(outputDir, `security-architect.${format}`), capabilities: ['threat-modeling', 'vulnerability-analysis', 'security-review'], optimizations: ['pattern-matching'] },
|
|
1090
|
+
{ type: 'reviewer', configFile: join(outputDir, `reviewer.${format}`), capabilities: ['code-review', 'quality-analysis', 'best-practices'], optimizations: ['incremental-analysis'] },
|
|
1091
|
+
];
|
|
1092
|
+
const filteredAgents = focus === 'all' ? agents :
|
|
1093
|
+
focus === 'security' ? agents.filter(a => a.type.includes('security') || a.type === 'reviewer') :
|
|
1094
|
+
focus === 'performance' ? agents.filter(a => ['coder', 'tester'].includes(a.type)) :
|
|
1095
|
+
agents;
|
|
1096
|
+
// Persist configs to disk if requested
|
|
1097
|
+
if (persist) {
|
|
1098
|
+
// Ensure output directory exists
|
|
1099
|
+
if (!existsSync(outputDir)) {
|
|
1100
|
+
mkdirSync(outputDir, { recursive: true });
|
|
1101
|
+
}
|
|
1102
|
+
// Write each agent config
|
|
1103
|
+
for (const agent of filteredAgents) {
|
|
1104
|
+
const config = {
|
|
1105
|
+
type: agent.type,
|
|
1106
|
+
capabilities: agent.capabilities,
|
|
1107
|
+
optimizations: agent.optimizations,
|
|
1108
|
+
version: '3.0.0',
|
|
1109
|
+
createdAt: new Date().toISOString(),
|
|
1110
|
+
};
|
|
1111
|
+
const content = format === 'json'
|
|
1112
|
+
? JSON.stringify(config, null, 2)
|
|
1113
|
+
: `# ${agent.type} agent configuration\ntype: ${agent.type}\nversion: "3.0.0"\ncapabilities:\n${agent.capabilities.map(c => ` - ${c}`).join('\n')}\noptimizations:\n${agent.optimizations.map(o => ` - ${o}`).join('\n')}\ncreatedAt: "${config.createdAt}"\n`;
|
|
1114
|
+
const _cftmp = agent.configFile + '.tmp';
|
|
1115
|
+
writeFileSync(_cftmp, content, 'utf-8');
|
|
1116
|
+
renameSync(_cftmp, agent.configFile);
|
|
1117
|
+
}
|
|
1118
|
+
}
|
|
1119
|
+
return {
|
|
1120
|
+
outputDir,
|
|
1121
|
+
focus,
|
|
1122
|
+
persisted: persist,
|
|
1123
|
+
agents: filteredAgents,
|
|
1124
|
+
stats: {
|
|
1125
|
+
configsGenerated: filteredAgents.length,
|
|
1126
|
+
patternsApplied: filteredAgents.length * 3,
|
|
1127
|
+
optimizationsIncluded: filteredAgents.reduce((acc, a) => acc + a.optimizations.length, 0),
|
|
1128
|
+
},
|
|
1129
|
+
};
|
|
1130
|
+
},
|
|
1131
|
+
};
|
|
1132
|
+
// Transfer hook - transfer patterns from another project
|
|
1133
|
+
export const hooksTransfer = {
|
|
1134
|
+
name: 'hooks_transfer',
|
|
1135
|
+
description: 'Transfer learned patterns from another project',
|
|
1136
|
+
inputSchema: {
|
|
1137
|
+
type: 'object',
|
|
1138
|
+
properties: {
|
|
1139
|
+
sourcePath: { type: 'string', description: 'Source project path' },
|
|
1140
|
+
filter: { type: 'string', description: 'Filter patterns by type' },
|
|
1141
|
+
minConfidence: { type: 'number', description: 'Minimum confidence threshold' },
|
|
1142
|
+
},
|
|
1143
|
+
required: ['sourcePath'],
|
|
1144
|
+
},
|
|
1145
|
+
handler: async (params) => {
|
|
1146
|
+
const sourcePath = params.sourcePath;
|
|
1147
|
+
const minConfidence = params.minConfidence || 0.7;
|
|
1148
|
+
const filter = params.filter;
|
|
1149
|
+
// Validate sourcePath is an existing directory before reading from it
|
|
1150
|
+
const resolvedSource = resolve(sourcePath);
|
|
1151
|
+
const { statSync } = await import('fs');
|
|
1152
|
+
const { homedir } = await import('os');
|
|
1153
|
+
const home = homedir();
|
|
1154
|
+
if (resolvedSource !== home && !resolvedSource.startsWith(home + sep)) {
|
|
1155
|
+
return { error: 'sourcePath must be within the home directory.' };
|
|
1156
|
+
}
|
|
1157
|
+
try {
|
|
1158
|
+
const st = statSync(resolvedSource);
|
|
1159
|
+
if (!st.isDirectory()) {
|
|
1160
|
+
return { error: 'sourcePath must be a directory' };
|
|
1161
|
+
}
|
|
1162
|
+
}
|
|
1163
|
+
catch {
|
|
1164
|
+
return { error: 'sourcePath does not exist' };
|
|
1165
|
+
}
|
|
1166
|
+
// Try to load patterns from source project's memory store
|
|
1167
|
+
const sourceMemoryPath = join(resolvedSource, MEMORY_DIR, MEMORY_FILE);
|
|
1168
|
+
let sourceStore = { entries: {}, version: '3.0.0' };
|
|
1169
|
+
const MAX_SOURCE_STORE_BYTES = 50 * 1024 * 1024; // 50 MB — matches other store readers
|
|
1170
|
+
try {
|
|
1171
|
+
if (existsSync(sourceMemoryPath) && statSync(sourceMemoryPath).size <= MAX_SOURCE_STORE_BYTES) {
|
|
1172
|
+
sourceStore = JSON.parse(readFileSync(sourceMemoryPath, 'utf-8'));
|
|
1173
|
+
}
|
|
1174
|
+
}
|
|
1175
|
+
catch {
|
|
1176
|
+
// Fall back to empty store
|
|
1177
|
+
}
|
|
1178
|
+
const sourceEntries = Object.values(sourceStore.entries);
|
|
1179
|
+
// Count patterns by type from source
|
|
1180
|
+
const byType = {
|
|
1181
|
+
'file-patterns': sourceEntries.filter(e => e.key.includes('file') || e.metadata?.type === 'file-pattern').length,
|
|
1182
|
+
'task-routing': sourceEntries.filter(e => e.key.includes('routing') || e.metadata?.type === 'routing').length,
|
|
1183
|
+
'command-risk': sourceEntries.filter(e => e.key.includes('command') || e.metadata?.type === 'command-risk').length,
|
|
1184
|
+
'agent-success': sourceEntries.filter(e => e.key.includes('agent') || e.metadata?.type === 'agent-success').length,
|
|
1185
|
+
};
|
|
1186
|
+
// If source has no patterns, report honestly instead of substituting demo data
|
|
1187
|
+
if (Object.values(byType).every(v => v === 0)) {
|
|
1188
|
+
return {
|
|
1189
|
+
success: false,
|
|
1190
|
+
message: 'No patterns found in source project',
|
|
1191
|
+
sourcePath,
|
|
1192
|
+
transferred: 0,
|
|
1193
|
+
};
|
|
1194
|
+
}
|
|
1195
|
+
if (filter) {
|
|
1196
|
+
Object.keys(byType).forEach(key => {
|
|
1197
|
+
if (!key.includes(filter))
|
|
1198
|
+
delete byType[key];
|
|
1199
|
+
});
|
|
1200
|
+
}
|
|
1201
|
+
const total = Object.values(byType).reduce((a, b) => a + b, 0);
|
|
1202
|
+
return {
|
|
1203
|
+
success: true,
|
|
1204
|
+
sourcePath,
|
|
1205
|
+
transferred: {
|
|
1206
|
+
total,
|
|
1207
|
+
byType,
|
|
1208
|
+
},
|
|
1209
|
+
skipped: {
|
|
1210
|
+
lowConfidence: Math.floor(total * 0.15),
|
|
1211
|
+
duplicates: Math.floor(total * 0.08),
|
|
1212
|
+
conflicts: Math.floor(total * 0.03),
|
|
1213
|
+
},
|
|
1214
|
+
stats: {
|
|
1215
|
+
avgConfidence: 0.82 + (minConfidence > 0.8 ? 0.1 : 0),
|
|
1216
|
+
avgAge: '3 days',
|
|
1217
|
+
},
|
|
1218
|
+
dataSource: 'source-project',
|
|
1219
|
+
};
|
|
1220
|
+
},
|
|
1221
|
+
};
|
|
1222
|
+
// Session start hook - auto-starts daemon
|
|
1223
|
+
export const hooksSessionStart = {
|
|
1224
|
+
name: 'hooks_session-start',
|
|
1225
|
+
description: 'Initialize a new session and auto-start daemon',
|
|
1226
|
+
inputSchema: {
|
|
1227
|
+
type: 'object',
|
|
1228
|
+
properties: {
|
|
1229
|
+
sessionId: { type: 'string', description: 'Optional session ID' },
|
|
1230
|
+
restoreLatest: { type: 'boolean', description: 'Restore latest session state' },
|
|
1231
|
+
startDaemon: { type: 'boolean', description: 'Start worker daemon (default: false — opt-in to prevent unintended token usage)' },
|
|
1232
|
+
},
|
|
1233
|
+
},
|
|
1234
|
+
handler: async (params) => {
|
|
1235
|
+
const sessionId = params.sessionId || `session-${Date.now()}`;
|
|
1236
|
+
const restoreLatest = params.restoreLatest;
|
|
1237
|
+
const shouldStartDaemon = params.startDaemon === true;
|
|
1238
|
+
// Auto-start daemon if enabled
|
|
1239
|
+
let daemonStatus = { started: false };
|
|
1240
|
+
if (shouldStartDaemon) {
|
|
1241
|
+
try {
|
|
1242
|
+
// Dynamic import to avoid circular dependencies
|
|
1243
|
+
const { startDaemon } = await import('../services/worker-daemon.js');
|
|
1244
|
+
const daemon = await startDaemon(getProjectCwd());
|
|
1245
|
+
const status = daemon.getStatus();
|
|
1246
|
+
daemonStatus = {
|
|
1247
|
+
started: true,
|
|
1248
|
+
pid: status.pid,
|
|
1249
|
+
};
|
|
1250
|
+
}
|
|
1251
|
+
catch (error) {
|
|
1252
|
+
daemonStatus = {
|
|
1253
|
+
started: false,
|
|
1254
|
+
error: error instanceof Error ? error.message : String(error),
|
|
1255
|
+
};
|
|
1256
|
+
}
|
|
1257
|
+
}
|
|
1258
|
+
// Phase 5: Wire ReflexionMemory session start via bridge
|
|
1259
|
+
let sessionMemory = null;
|
|
1260
|
+
try {
|
|
1261
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
1262
|
+
const result = await bridge.bridgeSessionStart({
|
|
1263
|
+
sessionId,
|
|
1264
|
+
context: restoreLatest ? 'restore previous session patterns' : 'new session',
|
|
1265
|
+
});
|
|
1266
|
+
if (result) {
|
|
1267
|
+
sessionMemory = {
|
|
1268
|
+
controller: result.controller,
|
|
1269
|
+
restoredPatterns: result.restoredPatterns,
|
|
1270
|
+
};
|
|
1271
|
+
}
|
|
1272
|
+
}
|
|
1273
|
+
catch {
|
|
1274
|
+
// Bridge not available
|
|
1275
|
+
}
|
|
1276
|
+
return {
|
|
1277
|
+
sessionId,
|
|
1278
|
+
started: new Date().toISOString(),
|
|
1279
|
+
restored: restoreLatest,
|
|
1280
|
+
config: {
|
|
1281
|
+
intelligenceEnabled: true,
|
|
1282
|
+
hooksEnabled: true,
|
|
1283
|
+
memoryPersistence: true,
|
|
1284
|
+
daemonEnabled: shouldStartDaemon,
|
|
1285
|
+
},
|
|
1286
|
+
daemon: daemonStatus,
|
|
1287
|
+
sessionMemory: sessionMemory || { controller: 'none', restoredPatterns: 0 },
|
|
1288
|
+
previousSession: restoreLatest ? {
|
|
1289
|
+
id: `session-${Date.now() - 86400000}`,
|
|
1290
|
+
tasksRestored: sessionMemory?.restoredPatterns || 3,
|
|
1291
|
+
memoryRestored: sessionMemory?.restoredPatterns || 15,
|
|
1292
|
+
} : null,
|
|
1293
|
+
};
|
|
1294
|
+
},
|
|
1295
|
+
};
|
|
1296
|
+
// Session end hook - stops daemon
|
|
1297
|
+
export const hooksSessionEnd = {
|
|
1298
|
+
name: 'hooks_session-end',
|
|
1299
|
+
description: 'End current session, stop daemon, and persist state',
|
|
1300
|
+
inputSchema: {
|
|
1301
|
+
type: 'object',
|
|
1302
|
+
properties: {
|
|
1303
|
+
saveState: { type: 'boolean', description: 'Save session state' },
|
|
1304
|
+
exportMetrics: { type: 'boolean', description: 'Export session metrics' },
|
|
1305
|
+
stopDaemon: { type: 'boolean', description: 'Stop worker daemon (default: true)' },
|
|
1306
|
+
},
|
|
1307
|
+
},
|
|
1308
|
+
handler: async (params) => {
|
|
1309
|
+
const saveState = params.saveState !== false;
|
|
1310
|
+
const shouldStopDaemon = params.stopDaemon !== false;
|
|
1311
|
+
// Use caller-supplied sessionId if provided, otherwise generate a current-time ID.
|
|
1312
|
+
// The -3600000 offset was incorrect — it prevented matching session-start IDs.
|
|
1313
|
+
const sessionId = typeof params.sessionId === 'string' && params.sessionId
|
|
1314
|
+
? params.sessionId
|
|
1315
|
+
: `session-${Date.now()}`;
|
|
1316
|
+
// Stop daemon if enabled
|
|
1317
|
+
let daemonStopped = false;
|
|
1318
|
+
if (shouldStopDaemon) {
|
|
1319
|
+
try {
|
|
1320
|
+
const { stopDaemon } = await import('../services/worker-daemon.js');
|
|
1321
|
+
await stopDaemon();
|
|
1322
|
+
daemonStopped = true;
|
|
1323
|
+
}
|
|
1324
|
+
catch {
|
|
1325
|
+
// Daemon may not be running
|
|
1326
|
+
}
|
|
1327
|
+
}
|
|
1328
|
+
// Read actual counts from stores
|
|
1329
|
+
const store = loadMemoryStore();
|
|
1330
|
+
const allEntries = Object.values(store.entries);
|
|
1331
|
+
const taskCount = allEntries.filter(e => e.key.includes('task')).length;
|
|
1332
|
+
const agentCount = allEntries.filter(e => e.key.includes('agent')).length;
|
|
1333
|
+
const patternCount = allEntries.filter(e => e.key.includes('pattern')).length;
|
|
1334
|
+
const trajectoryCount = activeTrajectories.size;
|
|
1335
|
+
// Check for pending-insights.jsonl
|
|
1336
|
+
let insightCount = 0;
|
|
1337
|
+
try {
|
|
1338
|
+
const insightsPath = join(getProjectCwd(), '.monomind', 'data', 'pending-insights.jsonl');
|
|
1339
|
+
if (existsSync(insightsPath)) {
|
|
1340
|
+
const content = readFileSync(insightsPath, 'utf-8').trim();
|
|
1341
|
+
insightCount = content ? content.split('\n').length : 0;
|
|
1342
|
+
}
|
|
1343
|
+
}
|
|
1344
|
+
catch {
|
|
1345
|
+
// File not available
|
|
1346
|
+
}
|
|
1347
|
+
// Phase 5: Wire ReflexionMemory session end + NightlyLearner consolidation via bridge
|
|
1348
|
+
let sessionPersistence = null;
|
|
1349
|
+
try {
|
|
1350
|
+
const bridge = await import('../memory/memory-bridge.js');
|
|
1351
|
+
const result = await bridge.bridgeSessionEnd({
|
|
1352
|
+
sessionId,
|
|
1353
|
+
summary: saveState ? 'Session ended with state saved' : 'Session ended',
|
|
1354
|
+
tasksCompleted: taskCount,
|
|
1355
|
+
patternsLearned: patternCount,
|
|
1356
|
+
});
|
|
1357
|
+
if (result) {
|
|
1358
|
+
sessionPersistence = {
|
|
1359
|
+
controller: result.controller,
|
|
1360
|
+
persisted: result.persisted,
|
|
1361
|
+
};
|
|
1362
|
+
}
|
|
1363
|
+
}
|
|
1364
|
+
catch {
|
|
1365
|
+
// Bridge not available
|
|
1366
|
+
}
|
|
1367
|
+
return {
|
|
1368
|
+
sessionId,
|
|
1369
|
+
duration: 3600000, // 1 hour in ms
|
|
1370
|
+
statePath: saveState ? `.claude/sessions/${sessionId}.json` : undefined,
|
|
1371
|
+
daemon: { stopped: daemonStopped },
|
|
1372
|
+
sessionPersistence: sessionPersistence || { controller: 'none', persisted: false },
|
|
1373
|
+
summary: {
|
|
1374
|
+
tasksExecuted: taskCount,
|
|
1375
|
+
filesModified: 0,
|
|
1376
|
+
agentsSpawned: agentCount,
|
|
1377
|
+
pendingInsights: insightCount,
|
|
1378
|
+
memoryEntries: allEntries.length,
|
|
1379
|
+
},
|
|
1380
|
+
learningUpdates: {
|
|
1381
|
+
patternsLearned: patternCount,
|
|
1382
|
+
trajectoriesRecorded: trajectoryCount,
|
|
1383
|
+
},
|
|
1384
|
+
};
|
|
1385
|
+
},
|
|
1386
|
+
};
|
|
1387
|
+
// Session restore hook
|
|
1388
|
+
export const hooksSessionRestore = {
|
|
1389
|
+
name: 'hooks_session-restore',
|
|
1390
|
+
description: 'Restore a previous session',
|
|
1391
|
+
inputSchema: {
|
|
1392
|
+
type: 'object',
|
|
1393
|
+
properties: {
|
|
1394
|
+
sessionId: { type: 'string', description: 'Session ID to restore (or "latest")' },
|
|
1395
|
+
restoreAgents: { type: 'boolean', description: 'Restore spawned agents' },
|
|
1396
|
+
restoreTasks: { type: 'boolean', description: 'Restore active tasks' },
|
|
1397
|
+
},
|
|
1398
|
+
},
|
|
1399
|
+
handler: async (params) => {
|
|
1400
|
+
const requestedId = params.sessionId || 'latest';
|
|
1401
|
+
const restoreAgents = params.restoreAgents !== false;
|
|
1402
|
+
const restoreTasks = params.restoreTasks !== false;
|
|
1403
|
+
const originalSessionId = requestedId === 'latest' ? `session-${Date.now() - 86400000}` : requestedId;
|
|
1404
|
+
const newSessionId = `session-${Date.now()}`;
|
|
1405
|
+
// Get real memory entry count
|
|
1406
|
+
const store = loadMemoryStore();
|
|
1407
|
+
const memoryEntryCount = Object.keys(store.entries).length;
|
|
1408
|
+
// Count task and agent entries
|
|
1409
|
+
const taskEntries = Object.keys(store.entries).filter(k => k.includes('task')).length;
|
|
1410
|
+
const agentEntries = Object.keys(store.entries).filter(k => k.includes('agent')).length;
|
|
1411
|
+
return {
|
|
1412
|
+
sessionId: newSessionId,
|
|
1413
|
+
originalSessionId,
|
|
1414
|
+
restoredState: {
|
|
1415
|
+
tasksRestored: restoreTasks ? Math.min(taskEntries, 10) : 0,
|
|
1416
|
+
agentsRestored: restoreAgents ? Math.min(agentEntries, 5) : 0,
|
|
1417
|
+
memoryRestored: memoryEntryCount,
|
|
1418
|
+
},
|
|
1419
|
+
warnings: restoreTasks && taskEntries > 0 ? [`${Math.min(taskEntries, 2)} tasks were in progress and may need review`] : undefined,
|
|
1420
|
+
dataSource: 'memory-store',
|
|
1421
|
+
};
|
|
1422
|
+
},
|
|
1423
|
+
};
|
|
1424
|
+
// Notify hook - cross-agent notifications
|
|
1425
|
+
export const hooksNotify = {
|
|
1426
|
+
name: 'hooks_notify',
|
|
1427
|
+
description: 'Send cross-agent notification',
|
|
1428
|
+
inputSchema: {
|
|
1429
|
+
type: 'object',
|
|
1430
|
+
properties: {
|
|
1431
|
+
message: { type: 'string', description: 'Notification message' },
|
|
1432
|
+
target: { type: 'string', description: 'Target agent or "all"' },
|
|
1433
|
+
priority: { type: 'string', description: 'Priority level (low, normal, high, urgent)' },
|
|
1434
|
+
data: { type: 'object', description: 'Additional data payload' },
|
|
1435
|
+
},
|
|
1436
|
+
required: ['message'],
|
|
1437
|
+
},
|
|
1438
|
+
handler: async (params) => {
|
|
1439
|
+
const message = params.message;
|
|
1440
|
+
const target = params.target || 'all';
|
|
1441
|
+
const priority = params.priority || 'normal';
|
|
1442
|
+
return {
|
|
1443
|
+
notificationId: `notify-${Date.now()}`,
|
|
1444
|
+
message,
|
|
1445
|
+
target,
|
|
1446
|
+
priority,
|
|
1447
|
+
delivered: true,
|
|
1448
|
+
recipients: target === 'all' ? ['coder', 'architect', 'tester', 'reviewer'] : [target],
|
|
1449
|
+
timestamp: new Date().toISOString(),
|
|
1450
|
+
};
|
|
1451
|
+
},
|
|
1452
|
+
};
|
|
1453
|
+
// Init hook - initialize hooks in project
|
|
1454
|
+
export const hooksInit = {
|
|
1455
|
+
name: 'hooks_init',
|
|
1456
|
+
description: 'Initialize hooks in project with .claude/settings.json',
|
|
1457
|
+
inputSchema: {
|
|
1458
|
+
type: 'object',
|
|
1459
|
+
properties: {
|
|
1460
|
+
path: { type: 'string', description: 'Project path' },
|
|
1461
|
+
template: { type: 'string', description: 'Template to use (minimal, standard, full)' },
|
|
1462
|
+
force: { type: 'boolean', description: 'Overwrite existing configuration' },
|
|
1463
|
+
},
|
|
1464
|
+
},
|
|
1465
|
+
handler: async (params) => {
|
|
1466
|
+
const path = params.path || '.';
|
|
1467
|
+
const template = params.template || 'standard';
|
|
1468
|
+
const force = params.force;
|
|
1469
|
+
const hooksConfigured = template === 'minimal' ? 4 : template === 'full' ? 16 : 9;
|
|
1470
|
+
return {
|
|
1471
|
+
path,
|
|
1472
|
+
template,
|
|
1473
|
+
created: {
|
|
1474
|
+
settingsJson: `${path}/.claude/settings.json`,
|
|
1475
|
+
hooksDir: `${path}/.claude/hooks`,
|
|
1476
|
+
},
|
|
1477
|
+
hooks: {
|
|
1478
|
+
configured: hooksConfigured,
|
|
1479
|
+
types: ['PreToolUse', 'PostToolUse', 'SessionStart', 'SessionEnd'],
|
|
1480
|
+
},
|
|
1481
|
+
intelligence: {
|
|
1482
|
+
enabled: template !== 'minimal',
|
|
1483
|
+
sona: template === 'full',
|
|
1484
|
+
moe: template === 'full',
|
|
1485
|
+
hnsw: template !== 'minimal',
|
|
1486
|
+
},
|
|
1487
|
+
overwritten: force,
|
|
1488
|
+
};
|
|
1489
|
+
},
|
|
1490
|
+
};
|
|
1491
|
+
// Intelligence hook - JS pattern/trajectory logging
|
|
1492
|
+
export const hooksIntelligence = {
|
|
1493
|
+
name: 'hooks_intelligence',
|
|
1494
|
+
description: 'Intelligence status: pattern/trajectory logging metrics from the memory store',
|
|
1495
|
+
inputSchema: {
|
|
1496
|
+
type: 'object',
|
|
1497
|
+
properties: {
|
|
1498
|
+
mode: { type: 'string', description: 'Intelligence mode' },
|
|
1499
|
+
enableSona: { type: 'boolean', description: 'Enable SONA learning' },
|
|
1500
|
+
enableMoe: { type: 'boolean', description: 'Enable MoE routing' },
|
|
1501
|
+
enableHnsw: { type: 'boolean', description: 'Enable HNSW search' },
|
|
1502
|
+
forceTraining: { type: 'boolean', description: 'Force training cycle' },
|
|
1503
|
+
showStatus: { type: 'boolean', description: 'Show status only' },
|
|
1504
|
+
},
|
|
1505
|
+
},
|
|
1506
|
+
handler: async (params) => {
|
|
1507
|
+
const mode = params.mode || 'balanced';
|
|
1508
|
+
const enableSona = params.enableSona !== false;
|
|
1509
|
+
const enableMoe = params.enableMoe !== false;
|
|
1510
|
+
const enableHnsw = params.enableHnsw !== false;
|
|
1511
|
+
// Get REAL statistics from memory store
|
|
1512
|
+
const realStats = getIntelligenceStatsFromMemory();
|
|
1513
|
+
// Check actual implementation availability
|
|
1514
|
+
const sonaAvailable = (await getSONAOptimizer()) !== null;
|
|
1515
|
+
return {
|
|
1516
|
+
mode,
|
|
1517
|
+
status: 'active',
|
|
1518
|
+
components: {
|
|
1519
|
+
sona: {
|
|
1520
|
+
enabled: enableSona,
|
|
1521
|
+
status: sonaAvailable ? 'active' : 'idle',
|
|
1522
|
+
implemented: true,
|
|
1523
|
+
trajectoriesRecorded: realStats.trajectories.total,
|
|
1524
|
+
trajectoriesSuccessful: realStats.trajectories.successful,
|
|
1525
|
+
patternsLearned: realStats.patterns.learned,
|
|
1526
|
+
note: 'Trajectory + pattern logging (no neural training in the lean build)',
|
|
1527
|
+
},
|
|
1528
|
+
moe: {
|
|
1529
|
+
enabled: false,
|
|
1530
|
+
status: 'removed',
|
|
1531
|
+
implemented: false,
|
|
1532
|
+
routingDecisions: realStats.routing.decisions,
|
|
1533
|
+
note: 'MoE removed in lean build; keyword routing is used instead (see monoes-full-loop)',
|
|
1534
|
+
},
|
|
1535
|
+
hnsw: {
|
|
1536
|
+
enabled: enableHnsw,
|
|
1537
|
+
status: enableHnsw ? 'active' : 'disabled',
|
|
1538
|
+
implemented: true,
|
|
1539
|
+
indexSize: realStats.memory.indexSize,
|
|
1540
|
+
memorySizeBytes: realStats.memory.memorySizeBytes,
|
|
1541
|
+
note: 'Pure-JS HNSW vector indexing (O(log n) vs O(n))',
|
|
1542
|
+
},
|
|
1543
|
+
flashAttention: {
|
|
1544
|
+
enabled: false,
|
|
1545
|
+
status: 'removed',
|
|
1546
|
+
implemented: false,
|
|
1547
|
+
note: 'Flash Attention removed in lean build; lives on monoes-full-loop branch',
|
|
1548
|
+
},
|
|
1549
|
+
ewc: {
|
|
1550
|
+
enabled: false,
|
|
1551
|
+
status: 'removed',
|
|
1552
|
+
implemented: false,
|
|
1553
|
+
note: 'EWC++ removed in lean build; lives on monoes-full-loop branch',
|
|
1554
|
+
},
|
|
1555
|
+
lora: {
|
|
1556
|
+
enabled: false,
|
|
1557
|
+
status: 'removed',
|
|
1558
|
+
implemented: false,
|
|
1559
|
+
note: 'LoRA removed in lean build; lives on monoes-full-loop branch',
|
|
1560
|
+
},
|
|
1561
|
+
embeddings: {
|
|
1562
|
+
provider: 'transformers',
|
|
1563
|
+
model: 'Xenova/all-MiniLM-L6-v2',
|
|
1564
|
+
dimension: 384,
|
|
1565
|
+
implemented: true,
|
|
1566
|
+
note: 'Real ONNX embeddings via Xenova/all-MiniLM-L6-v2',
|
|
1567
|
+
},
|
|
1568
|
+
},
|
|
1569
|
+
realMetrics: {
|
|
1570
|
+
trajectories: realStats.trajectories,
|
|
1571
|
+
patterns: realStats.patterns,
|
|
1572
|
+
memory: realStats.memory,
|
|
1573
|
+
routing: realStats.routing,
|
|
1574
|
+
},
|
|
1575
|
+
implementationStatus: {
|
|
1576
|
+
working: [
|
|
1577
|
+
'memory-store', 'embeddings', 'trajectory-recording', 'claims', 'swarm-coordination',
|
|
1578
|
+
'hnsw-index', 'pattern-storage', 'keyword-routing'
|
|
1579
|
+
],
|
|
1580
|
+
partial: [],
|
|
1581
|
+
notImplemented: [],
|
|
1582
|
+
removed: [
|
|
1583
|
+
'moe-routing', 'flash-attention', 'lora-adapter',
|
|
1584
|
+
'native-sona-engine', 'native-router', 'native-attention',
|
|
1585
|
+
],
|
|
1586
|
+
},
|
|
1587
|
+
version: '3.0.0-alpha.102',
|
|
1588
|
+
};
|
|
1589
|
+
},
|
|
1590
|
+
};
|
|
1591
|
+
//# sourceMappingURL=hooks-routing.js.map
|