@soleri/core 2.11.0 → 7.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (255) hide show
  1. package/data/flows/build.flow.yaml +128 -0
  2. package/data/flows/deliver.flow.yaml +110 -0
  3. package/data/flows/design.flow.yaml +108 -0
  4. package/data/flows/enhance.flow.yaml +90 -0
  5. package/data/flows/explore.flow.yaml +84 -0
  6. package/data/flows/fix.flow.yaml +90 -0
  7. package/data/flows/plan.flow.yaml +87 -0
  8. package/data/flows/review.flow.yaml +90 -0
  9. package/dist/brain/brain.d.ts.map +1 -1
  10. package/dist/brain/brain.js +10 -0
  11. package/dist/brain/brain.js.map +1 -1
  12. package/dist/brain/intelligence.d.ts.map +1 -1
  13. package/dist/brain/intelligence.js +16 -2
  14. package/dist/brain/intelligence.js.map +1 -1
  15. package/dist/capabilities/chain-mapping.d.ts +21 -0
  16. package/dist/capabilities/chain-mapping.d.ts.map +1 -0
  17. package/dist/capabilities/chain-mapping.js +86 -0
  18. package/dist/capabilities/chain-mapping.js.map +1 -0
  19. package/dist/capabilities/index.d.ts +10 -0
  20. package/dist/capabilities/index.d.ts.map +1 -0
  21. package/dist/capabilities/index.js +8 -0
  22. package/dist/capabilities/index.js.map +1 -0
  23. package/dist/capabilities/registry.d.ts +95 -0
  24. package/dist/capabilities/registry.d.ts.map +1 -0
  25. package/dist/capabilities/registry.js +227 -0
  26. package/dist/capabilities/registry.js.map +1 -0
  27. package/dist/capabilities/types.d.ts +106 -0
  28. package/dist/capabilities/types.d.ts.map +1 -0
  29. package/dist/capabilities/types.js +12 -0
  30. package/dist/capabilities/types.js.map +1 -0
  31. package/dist/control/intent-router.d.ts.map +1 -1
  32. package/dist/control/intent-router.js +58 -2
  33. package/dist/control/intent-router.js.map +1 -1
  34. package/dist/domain-packs/index.d.ts +8 -0
  35. package/dist/domain-packs/index.d.ts.map +1 -0
  36. package/dist/domain-packs/index.js +8 -0
  37. package/dist/domain-packs/index.js.map +1 -0
  38. package/dist/domain-packs/inject-rules.d.ts +24 -0
  39. package/dist/domain-packs/inject-rules.d.ts.map +1 -0
  40. package/dist/domain-packs/inject-rules.js +65 -0
  41. package/dist/domain-packs/inject-rules.js.map +1 -0
  42. package/dist/domain-packs/knowledge-installer.d.ts +27 -0
  43. package/dist/domain-packs/knowledge-installer.d.ts.map +1 -0
  44. package/dist/domain-packs/knowledge-installer.js +89 -0
  45. package/dist/domain-packs/knowledge-installer.js.map +1 -0
  46. package/dist/domain-packs/loader.d.ts +28 -0
  47. package/dist/domain-packs/loader.d.ts.map +1 -0
  48. package/dist/domain-packs/loader.js +105 -0
  49. package/dist/domain-packs/loader.js.map +1 -0
  50. package/dist/domain-packs/pack-runtime.d.ts +80 -0
  51. package/dist/domain-packs/pack-runtime.d.ts.map +1 -0
  52. package/dist/domain-packs/pack-runtime.js +36 -0
  53. package/dist/domain-packs/pack-runtime.js.map +1 -0
  54. package/dist/domain-packs/skills-installer.d.ts +21 -0
  55. package/dist/domain-packs/skills-installer.d.ts.map +1 -0
  56. package/dist/domain-packs/skills-installer.js +38 -0
  57. package/dist/domain-packs/skills-installer.js.map +1 -0
  58. package/dist/domain-packs/token-resolver.d.ts +37 -0
  59. package/dist/domain-packs/token-resolver.d.ts.map +1 -0
  60. package/dist/domain-packs/token-resolver.js +109 -0
  61. package/dist/domain-packs/token-resolver.js.map +1 -0
  62. package/dist/domain-packs/types.d.ts +91 -0
  63. package/dist/domain-packs/types.d.ts.map +1 -0
  64. package/dist/domain-packs/types.js +122 -0
  65. package/dist/domain-packs/types.js.map +1 -0
  66. package/dist/engine/bin/soleri-engine.d.ts +12 -0
  67. package/dist/engine/bin/soleri-engine.d.ts.map +1 -0
  68. package/dist/engine/bin/soleri-engine.js +183 -0
  69. package/dist/engine/bin/soleri-engine.js.map +1 -0
  70. package/dist/engine/core-ops.d.ts +27 -0
  71. package/dist/engine/core-ops.d.ts.map +1 -0
  72. package/dist/engine/core-ops.js +159 -0
  73. package/dist/engine/core-ops.js.map +1 -0
  74. package/dist/engine/index.d.ts +19 -0
  75. package/dist/engine/index.d.ts.map +1 -0
  76. package/dist/engine/index.js +17 -0
  77. package/dist/engine/index.js.map +1 -0
  78. package/dist/engine/register-engine.d.ts +54 -0
  79. package/dist/engine/register-engine.d.ts.map +1 -0
  80. package/dist/engine/register-engine.js +270 -0
  81. package/dist/engine/register-engine.js.map +1 -0
  82. package/dist/engine/test-helpers.d.ts +30 -0
  83. package/dist/engine/test-helpers.d.ts.map +1 -0
  84. package/dist/engine/test-helpers.js +59 -0
  85. package/dist/engine/test-helpers.js.map +1 -0
  86. package/dist/flows/context-router.d.ts +39 -0
  87. package/dist/flows/context-router.d.ts.map +1 -0
  88. package/dist/flows/context-router.js +206 -0
  89. package/dist/flows/context-router.js.map +1 -0
  90. package/dist/flows/dispatch-registry.d.ts +24 -0
  91. package/dist/flows/dispatch-registry.d.ts.map +1 -0
  92. package/dist/flows/dispatch-registry.js +70 -0
  93. package/dist/flows/dispatch-registry.js.map +1 -0
  94. package/dist/flows/epilogue.d.ts +24 -0
  95. package/dist/flows/epilogue.d.ts.map +1 -0
  96. package/dist/flows/epilogue.js +52 -0
  97. package/dist/flows/epilogue.js.map +1 -0
  98. package/dist/flows/executor.d.ts +25 -0
  99. package/dist/flows/executor.d.ts.map +1 -0
  100. package/dist/flows/executor.js +153 -0
  101. package/dist/flows/executor.js.map +1 -0
  102. package/dist/flows/gate-evaluator.d.ts +26 -0
  103. package/dist/flows/gate-evaluator.d.ts.map +1 -0
  104. package/dist/flows/gate-evaluator.js +162 -0
  105. package/dist/flows/gate-evaluator.js.map +1 -0
  106. package/dist/flows/index.d.ts +14 -0
  107. package/dist/flows/index.d.ts.map +1 -0
  108. package/dist/flows/index.js +20 -0
  109. package/dist/flows/index.js.map +1 -0
  110. package/dist/flows/loader.d.ts +17 -0
  111. package/dist/flows/loader.d.ts.map +1 -0
  112. package/dist/flows/loader.js +61 -0
  113. package/dist/flows/loader.js.map +1 -0
  114. package/dist/flows/plan-builder.d.ts +40 -0
  115. package/dist/flows/plan-builder.d.ts.map +1 -0
  116. package/dist/flows/plan-builder.js +213 -0
  117. package/dist/flows/plan-builder.js.map +1 -0
  118. package/dist/flows/probes.d.ts +11 -0
  119. package/dist/flows/probes.d.ts.map +1 -0
  120. package/dist/flows/probes.js +62 -0
  121. package/dist/flows/probes.js.map +1 -0
  122. package/dist/flows/types.d.ts +950 -0
  123. package/dist/flows/types.d.ts.map +1 -0
  124. package/dist/flows/types.js +105 -0
  125. package/dist/flows/types.js.map +1 -0
  126. package/dist/index.d.ts +11 -1
  127. package/dist/index.d.ts.map +1 -1
  128. package/dist/index.js +10 -1
  129. package/dist/index.js.map +1 -1
  130. package/dist/intelligence/loader.d.ts +19 -0
  131. package/dist/intelligence/loader.d.ts.map +1 -1
  132. package/dist/intelligence/loader.js +86 -5
  133. package/dist/intelligence/loader.js.map +1 -1
  134. package/dist/intelligence/types.d.ts +1 -0
  135. package/dist/intelligence/types.d.ts.map +1 -1
  136. package/dist/packs/types.d.ts +58 -19
  137. package/dist/packs/types.d.ts.map +1 -1
  138. package/dist/packs/types.js +14 -0
  139. package/dist/packs/types.js.map +1 -1
  140. package/dist/playbooks/generic/onboarding.d.ts +9 -0
  141. package/dist/playbooks/generic/onboarding.d.ts.map +1 -0
  142. package/dist/playbooks/generic/onboarding.js +74 -0
  143. package/dist/playbooks/generic/onboarding.js.map +1 -0
  144. package/dist/playbooks/playbook-registry.d.ts.map +1 -1
  145. package/dist/playbooks/playbook-registry.js +2 -0
  146. package/dist/playbooks/playbook-registry.js.map +1 -1
  147. package/dist/runtime/admin-extra-ops.d.ts.map +1 -1
  148. package/dist/runtime/admin-extra-ops.js +15 -9
  149. package/dist/runtime/admin-extra-ops.js.map +1 -1
  150. package/dist/runtime/admin-ops.js +4 -4
  151. package/dist/runtime/admin-ops.js.map +1 -1
  152. package/dist/runtime/capture-ops.d.ts.map +1 -1
  153. package/dist/runtime/capture-ops.js +33 -1
  154. package/dist/runtime/capture-ops.js.map +1 -1
  155. package/dist/runtime/domain-ops.d.ts +21 -5
  156. package/dist/runtime/domain-ops.d.ts.map +1 -1
  157. package/dist/runtime/domain-ops.js +85 -8
  158. package/dist/runtime/domain-ops.js.map +1 -1
  159. package/dist/runtime/facades/cognee-facade.d.ts.map +1 -1
  160. package/dist/runtime/facades/cognee-facade.js +3 -1
  161. package/dist/runtime/facades/cognee-facade.js.map +1 -1
  162. package/dist/runtime/facades/index.d.ts.map +1 -1
  163. package/dist/runtime/facades/index.js +10 -6
  164. package/dist/runtime/facades/index.js.map +1 -1
  165. package/dist/runtime/facades/vault-facade.d.ts.map +1 -1
  166. package/dist/runtime/facades/vault-facade.js +2 -0
  167. package/dist/runtime/facades/vault-facade.js.map +1 -1
  168. package/dist/runtime/orchestrate-ops.d.ts +8 -7
  169. package/dist/runtime/orchestrate-ops.d.ts.map +1 -1
  170. package/dist/runtime/orchestrate-ops.js +227 -58
  171. package/dist/runtime/orchestrate-ops.js.map +1 -1
  172. package/dist/runtime/runtime.d.ts.map +1 -1
  173. package/dist/runtime/runtime.js +23 -17
  174. package/dist/runtime/runtime.js.map +1 -1
  175. package/dist/runtime/types.d.ts +6 -2
  176. package/dist/runtime/types.d.ts.map +1 -1
  177. package/dist/runtime/vault-linking-ops.d.ts +13 -0
  178. package/dist/runtime/vault-linking-ops.d.ts.map +1 -0
  179. package/dist/runtime/vault-linking-ops.js +367 -0
  180. package/dist/runtime/vault-linking-ops.js.map +1 -0
  181. package/dist/vault/linking.d.ts +46 -0
  182. package/dist/vault/linking.d.ts.map +1 -0
  183. package/dist/vault/linking.js +275 -0
  184. package/dist/vault/linking.js.map +1 -0
  185. package/dist/vault/vault-types.d.ts +37 -0
  186. package/dist/vault/vault-types.d.ts.map +1 -1
  187. package/dist/vault/vault.d.ts +12 -0
  188. package/dist/vault/vault.d.ts.map +1 -1
  189. package/dist/vault/vault.js +85 -6
  190. package/dist/vault/vault.js.map +1 -1
  191. package/package.json +4 -1
  192. package/src/__tests__/admin-extra-ops.test.ts +1 -1
  193. package/src/__tests__/admin-ops.test.ts +2 -1
  194. package/src/__tests__/cognee-client-gaps.test.ts +470 -0
  195. package/src/__tests__/cognee-hybrid-search.test.ts +478 -0
  196. package/src/__tests__/cognee-sync-manager-deep.test.ts +630 -0
  197. package/src/__tests__/cognee-sync-manager.test.ts +1 -0
  198. package/src/__tests__/core-ops.test.ts +9 -61
  199. package/src/__tests__/domain-packs.test.ts +421 -0
  200. package/src/__tests__/flows.test.ts +604 -0
  201. package/src/__tests__/playbook-registry.test.ts +2 -2
  202. package/src/__tests__/playbook-seeder.test.ts +8 -8
  203. package/src/__tests__/playbook.test.ts +5 -5
  204. package/src/__tests__/token-resolver.test.ts +79 -0
  205. package/src/brain/brain.ts +12 -0
  206. package/src/brain/intelligence.ts +21 -2
  207. package/src/capabilities/chain-mapping.ts +93 -0
  208. package/src/capabilities/index.ts +21 -0
  209. package/src/capabilities/registry.ts +290 -0
  210. package/src/capabilities/types.ts +143 -0
  211. package/src/control/intent-router.ts +46 -2
  212. package/src/domain-packs/index.ts +27 -0
  213. package/src/domain-packs/inject-rules.ts +74 -0
  214. package/src/domain-packs/knowledge-installer.ts +116 -0
  215. package/src/domain-packs/loader.ts +124 -0
  216. package/src/domain-packs/pack-runtime.ts +99 -0
  217. package/src/domain-packs/skills-installer.ts +56 -0
  218. package/src/domain-packs/token-resolver.ts +126 -0
  219. package/src/domain-packs/types.ts +229 -0
  220. package/src/engine/__tests__/register-engine.test.ts +104 -0
  221. package/src/engine/bin/soleri-engine.ts +217 -0
  222. package/src/engine/core-ops.ts +178 -0
  223. package/src/engine/index.ts +19 -0
  224. package/src/engine/register-engine.ts +385 -0
  225. package/src/engine/test-helpers.ts +83 -0
  226. package/src/flows/context-router.ts +257 -0
  227. package/src/flows/dispatch-registry.ts +80 -0
  228. package/src/flows/epilogue.ts +65 -0
  229. package/src/flows/executor.ts +182 -0
  230. package/src/flows/gate-evaluator.ts +171 -0
  231. package/src/flows/index.ts +52 -0
  232. package/src/flows/loader.ts +63 -0
  233. package/src/flows/plan-builder.ts +250 -0
  234. package/src/flows/probes.ts +70 -0
  235. package/src/flows/types.ts +217 -0
  236. package/src/index.ts +68 -1
  237. package/src/intelligence/loader.ts +96 -5
  238. package/src/intelligence/types.ts +1 -0
  239. package/src/packs/types.ts +19 -0
  240. package/src/playbooks/generic/onboarding.ts +79 -0
  241. package/src/playbooks/playbook-registry.ts +2 -0
  242. package/src/runtime/admin-extra-ops.ts +14 -8
  243. package/src/runtime/admin-ops.ts +4 -4
  244. package/src/runtime/capture-ops.ts +40 -1
  245. package/src/runtime/domain-ops.ts +92 -7
  246. package/src/runtime/facades/cognee-facade.ts +3 -1
  247. package/src/runtime/facades/index.ts +12 -6
  248. package/src/runtime/facades/vault-facade.ts +2 -0
  249. package/src/runtime/orchestrate-ops.ts +271 -62
  250. package/src/runtime/runtime.ts +27 -18
  251. package/src/runtime/types.ts +6 -2
  252. package/src/runtime/vault-linking-ops.ts +454 -0
  253. package/src/vault/linking.ts +333 -0
  254. package/src/vault/vault-types.ts +46 -0
  255. package/src/vault/vault.ts +94 -7
@@ -0,0 +1,385 @@
1
+ /**
2
+ * Soleri v7 — Direct Engine Registration
3
+ *
4
+ * Registers all engine modules as MCP tools without the facade factory.
5
+ * Each module gets a single MCP tool with op-based dispatch.
6
+ *
7
+ * This replaces:
8
+ * - facade-factory.ts (generic dispatch layer)
9
+ * - registerAllFacades() + registerFacade() + dispatchOp()
10
+ * - FacadeConfig type (no longer needed)
11
+ *
12
+ * What stays:
13
+ * - createVaultFacadeOps(), createBrainFacadeOps(), etc. (op definitions)
14
+ * - createAgentRuntime() (module initialization)
15
+ * - OpDefinition type (handler + schema + auth)
16
+ * - Auth checking (same logic, inlined)
17
+ */
18
+
19
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
20
+ import { z } from 'zod';
21
+ import type { AgentRuntime } from '../runtime/types.js';
22
+ import type { OpDefinition, AuthPolicy } from '../facades/types.js';
23
+ import { AUTH_LEVEL_RANK } from '../facades/types.js';
24
+
25
+ // Re-export op creators (these are the source of truth for op definitions)
26
+ import { createVaultFacadeOps } from '../runtime/facades/vault-facade.js';
27
+ import { createPlanFacadeOps } from '../runtime/facades/plan-facade.js';
28
+ import { createBrainFacadeOps } from '../runtime/facades/brain-facade.js';
29
+ import { createMemoryFacadeOps } from '../runtime/facades/memory-facade.js';
30
+ import { createAdminFacadeOps } from '../runtime/facades/admin-facade.js';
31
+ import { createCuratorFacadeOps } from '../runtime/facades/curator-facade.js';
32
+ import { createLoopFacadeOps } from '../runtime/facades/loop-facade.js';
33
+ import { createOrchestrateFacadeOps } from '../runtime/facades/orchestrate-facade.js';
34
+ import { createControlFacadeOps } from '../runtime/facades/control-facade.js';
35
+ import { createCogneeFacadeOps } from '../runtime/facades/cognee-facade.js';
36
+ import { createContextFacadeOps } from '../runtime/facades/context-facade.js';
37
+ import { createAgencyFacadeOps } from '../runtime/facades/agency-facade.js';
38
+ import { createChatFacadeOps } from '../runtime/facades/chat-facade.js';
39
+ import { createDomainFacade } from '../runtime/domain-ops.js';
40
+
41
+ // ─── Types ────────────────────────────────────────────────────────────
42
+
43
+ export interface EngineRegistrationOptions {
44
+ /** Agent ID — used as tool name prefix (e.g., "gaudi" → "gaudi_vault") */
45
+ agentId: string;
46
+ /** Auth policy factory (optional — defaults to permissive) */
47
+ authPolicy?: () => AuthPolicy;
48
+ /** Additional ops to add to the core facade (agent-specific ops like health, activate) */
49
+ coreOps?: OpDefinition[];
50
+ /** Knowledge domains for domain-specific facades */
51
+ domains?: string[];
52
+ /** Domain packs (from @soleri/domain-*) */
53
+ domainPacks?: Array<{ name: string; facades?: Array<{ name: string; ops: OpDefinition[] }> }>;
54
+ /** Op names to ALSO register as standalone MCP tools (hot ops) */
55
+ hotOps?: string[];
56
+ }
57
+
58
+ export interface EngineRegistrationResult {
59
+ /** All registered MCP tool names */
60
+ tools: string[];
61
+ /** Total op count across all tools */
62
+ totalOps: number;
63
+ }
64
+
65
+ // ─── Module Definition ────────────────────────────────────────────────
66
+
67
+ interface ModuleDef {
68
+ /** Suffix for tool name: {agentId}_{suffix} */
69
+ suffix: string;
70
+ /** Tool description */
71
+ description: string;
72
+ /** Op creator function */
73
+ createOps: (runtime: AgentRuntime) => OpDefinition[];
74
+ /** Only register if this condition is true (default: always) */
75
+ condition?: (runtime: AgentRuntime) => boolean;
76
+ }
77
+
78
+ const ENGINE_MODULES: ModuleDef[] = [
79
+ {
80
+ suffix: 'vault',
81
+ description: 'Knowledge management — search, CRUD, import/export, intake, archival.',
82
+ createOps: createVaultFacadeOps,
83
+ },
84
+ {
85
+ suffix: 'plan',
86
+ description: 'Plan lifecycle — create, approve, execute, reconcile, complete, grading.',
87
+ createOps: createPlanFacadeOps,
88
+ },
89
+ {
90
+ suffix: 'brain',
91
+ description: 'Learning system — intelligence pipeline, strengths, feedback, sessions.',
92
+ createOps: createBrainFacadeOps,
93
+ },
94
+ {
95
+ suffix: 'memory',
96
+ description: 'Session & cross-project memory — capture, search, dedup, promote.',
97
+ createOps: createMemoryFacadeOps,
98
+ },
99
+ {
100
+ suffix: 'admin',
101
+ description: 'Infrastructure — health, config, telemetry, tokens, LLM, prompts.',
102
+ createOps: createAdminFacadeOps,
103
+ },
104
+ {
105
+ suffix: 'curator',
106
+ description: 'Quality — duplicate detection, contradictions, grooming, health audit.',
107
+ createOps: createCuratorFacadeOps,
108
+ },
109
+ {
110
+ suffix: 'loop',
111
+ description: 'Iterative validation loops — start, iterate, cancel, complete, history.',
112
+ createOps: createLoopFacadeOps,
113
+ },
114
+ {
115
+ suffix: 'orchestrate',
116
+ description:
117
+ 'Execution orchestration — project registration, playbooks, plan/execute/complete.',
118
+ createOps: createOrchestrateFacadeOps,
119
+ },
120
+ {
121
+ suffix: 'control',
122
+ description: 'Agent behavior — identity, intent routing, morphing, guidelines, governance.',
123
+ createOps: createControlFacadeOps,
124
+ },
125
+ {
126
+ suffix: 'context',
127
+ description: 'Context analysis — entity extraction, knowledge retrieval, confidence scoring.',
128
+ createOps: createContextFacadeOps,
129
+ },
130
+ {
131
+ suffix: 'agency',
132
+ description: 'Proactive intelligence — file watching, pattern surfacing, warnings.',
133
+ createOps: createAgencyFacadeOps,
134
+ },
135
+ {
136
+ suffix: 'chat',
137
+ description: 'Chat transport — session management, response chunking, authentication.',
138
+ createOps: createChatFacadeOps,
139
+ },
140
+ {
141
+ suffix: 'cognee',
142
+ description: 'Knowledge graph — Cognee search, sync, export, graph stats.',
143
+ createOps: createCogneeFacadeOps,
144
+ condition: (rt) => rt.cognee !== null && rt.cognee !== undefined,
145
+ },
146
+ ];
147
+
148
+ // ─── Core Registration ────────────────────────────────────────────────
149
+
150
+ /**
151
+ * Register all engine modules as MCP tools on the given server.
152
+ *
153
+ * Each module becomes one MCP tool: `{agentId}_{module}` with `op` + `params` arguments.
154
+ * Dispatch is a direct switch on op name — no generic factory, no FacadeConfig type.
155
+ */
156
+ export function registerEngine(
157
+ server: McpServer,
158
+ runtime: AgentRuntime,
159
+ options: EngineRegistrationOptions,
160
+ ): EngineRegistrationResult {
161
+ const { agentId, authPolicy, coreOps, domains, domainPacks, hotOps } = options;
162
+ const hotSet = new Set(hotOps ?? []);
163
+ const registeredTools: string[] = [];
164
+ let totalOps = 0;
165
+
166
+ // 1. Register semantic module tools
167
+ for (const mod of ENGINE_MODULES) {
168
+ if (mod.condition && !mod.condition(runtime)) continue;
169
+
170
+ const ops = mod.createOps(runtime);
171
+ const toolName = `${agentId}_${mod.suffix}`;
172
+
173
+ registerModuleTool(server, toolName, mod.description, ops, authPolicy);
174
+ registeredTools.push(toolName);
175
+ totalOps += ops.length;
176
+
177
+ // Hot ops: also register as standalone tools
178
+ for (const op of ops) {
179
+ if (op.hot || hotSet.has(op.name)) {
180
+ registerStandaloneTool(server, agentId, toolName, op, authPolicy);
181
+ registeredTools.push(`${agentId}_${op.name}`);
182
+ }
183
+ }
184
+ }
185
+
186
+ // 2. Register core facade (agent-specific ops: health, identity, activate, etc.)
187
+ if (coreOps && coreOps.length > 0) {
188
+ const coreName = `${agentId}_core`;
189
+ registerModuleTool(
190
+ server,
191
+ coreName,
192
+ 'Agent-specific operations — health, identity, activation.',
193
+ coreOps,
194
+ authPolicy,
195
+ );
196
+ registeredTools.push(coreName);
197
+ totalOps += coreOps.length;
198
+ }
199
+
200
+ // 3. Register domain facades
201
+ if (domains) {
202
+ for (const domain of domains) {
203
+ const domainConfig = createDomainFacade(runtime, agentId, domain);
204
+ registerModuleTool(
205
+ server,
206
+ domainConfig.name,
207
+ domainConfig.description,
208
+ domainConfig.ops,
209
+ authPolicy,
210
+ );
211
+ registeredTools.push(domainConfig.name);
212
+ totalOps += domainConfig.ops.length;
213
+ }
214
+ }
215
+
216
+ // 4. Register domain pack facades
217
+ if (domainPacks) {
218
+ for (const pack of domainPacks) {
219
+ if (pack.facades) {
220
+ for (const facade of pack.facades) {
221
+ const packToolName = `${agentId}_${facade.name}`;
222
+ registerModuleTool(
223
+ server,
224
+ packToolName,
225
+ `Domain pack: ${pack.name}`,
226
+ facade.ops,
227
+ authPolicy,
228
+ );
229
+ registeredTools.push(packToolName);
230
+ totalOps += facade.ops.length;
231
+ }
232
+ }
233
+ }
234
+ }
235
+
236
+ return { tools: registeredTools, totalOps };
237
+ }
238
+
239
+ // ─── Tool Registration (No Factory) ──────────────────────────────────
240
+
241
+ /**
242
+ * Register a single grouped tool with op dispatch.
243
+ * This is the replacement for registerFacade() — same behavior, no FacadeConfig type.
244
+ */
245
+ function registerModuleTool(
246
+ server: McpServer,
247
+ toolName: string,
248
+ description: string,
249
+ ops: OpDefinition[],
250
+ authPolicy?: () => AuthPolicy,
251
+ ): void {
252
+ const opNames = ops.map((o) => o.name);
253
+ const opMap = new Map(ops.map((o) => [o.name, o]));
254
+
255
+ server.tool(
256
+ toolName,
257
+ description,
258
+ {
259
+ op: z.string().describe(`Operation: ${opNames.join(' | ')}`),
260
+ params: z.record(z.unknown()).optional().default({}).describe('Operation parameters'),
261
+ },
262
+ async ({ op: opName, params }) => {
263
+ const op = opMap.get(opName);
264
+ if (!op) {
265
+ return jsonResponse({
266
+ success: false,
267
+ error: `Unknown operation "${opName}" on ${toolName}. Available: ${opNames.join(', ')}`,
268
+ op: opName,
269
+ facade: toolName,
270
+ });
271
+ }
272
+
273
+ // Auth check
274
+ const policy = authPolicy?.();
275
+ const authErr = checkAuth(opName, op.auth, toolName, policy);
276
+ if (authErr) return jsonResponse(authErr);
277
+
278
+ // Validate + execute
279
+ try {
280
+ let validatedParams = params;
281
+ if (op.schema) {
282
+ const result = op.schema.safeParse(params);
283
+ if (!result.success) {
284
+ return jsonResponse({
285
+ success: false,
286
+ error: `Invalid params for ${opName}: ${result.error.message}`,
287
+ op: opName,
288
+ facade: toolName,
289
+ });
290
+ }
291
+ validatedParams = result.data as Record<string, unknown>;
292
+ }
293
+
294
+ const data = await op.handler(validatedParams);
295
+ return jsonResponse({ success: true, data, op: opName, facade: toolName });
296
+ } catch (err) {
297
+ const message = err instanceof Error ? err.message : String(err);
298
+ return jsonResponse({ success: false, error: message, op: opName, facade: toolName });
299
+ }
300
+ },
301
+ );
302
+ }
303
+
304
+ /**
305
+ * Register a single op as a standalone MCP tool (hot op).
306
+ */
307
+ function registerStandaloneTool(
308
+ server: McpServer,
309
+ agentId: string,
310
+ parentTool: string,
311
+ op: OpDefinition,
312
+ authPolicy?: () => AuthPolicy,
313
+ ): void {
314
+ const toolName = `${agentId}_${op.name}`;
315
+ const schema = op.schema
316
+ ? (op.schema as z.ZodObject<z.ZodRawShape>).shape
317
+ ? (op.schema as z.ZodObject<z.ZodRawShape>)
318
+ : z.object({ params: op.schema })
319
+ : z.object({});
320
+
321
+ server.tool(
322
+ toolName,
323
+ op.description,
324
+ schema instanceof z.ZodObject ? schema.shape : {},
325
+ async (params) => {
326
+ const policy = authPolicy?.();
327
+ const authErr = checkAuth(op.name, op.auth, parentTool, policy);
328
+ if (authErr) return jsonResponse(authErr);
329
+
330
+ try {
331
+ let validatedParams = params as Record<string, unknown>;
332
+ if (op.schema) {
333
+ const result = op.schema.safeParse(params);
334
+ if (!result.success) {
335
+ return jsonResponse({
336
+ success: false,
337
+ error: `Invalid params: ${result.error.message}`,
338
+ op: op.name,
339
+ facade: parentTool,
340
+ });
341
+ }
342
+ validatedParams = result.data as Record<string, unknown>;
343
+ }
344
+
345
+ const data = await op.handler(validatedParams);
346
+ return jsonResponse({ success: true, data, op: op.name, facade: parentTool });
347
+ } catch (err) {
348
+ const message = err instanceof Error ? err.message : String(err);
349
+ return jsonResponse({ success: false, error: message, op: op.name, facade: parentTool });
350
+ }
351
+ },
352
+ );
353
+ }
354
+
355
+ // ─── Helpers ──────────────────────────────────────────────────────────
356
+
357
+ function checkAuth(
358
+ opName: string,
359
+ opAuth: string,
360
+ toolName: string,
361
+ policy: AuthPolicy | undefined,
362
+ ): { success: false; error: string; op: string; facade: string } | null {
363
+ if (!policy || policy.mode === 'permissive') return null;
364
+
365
+ const requiredLevel = policy.overrides?.[opName] ?? (opAuth as keyof typeof AUTH_LEVEL_RANK);
366
+ const callerRank = AUTH_LEVEL_RANK[policy.callerLevel] ?? 0;
367
+ const requiredRank = AUTH_LEVEL_RANK[requiredLevel as keyof typeof AUTH_LEVEL_RANK] ?? 0;
368
+
369
+ if (callerRank >= requiredRank) return null;
370
+
371
+ const message = `Auth denied: "${opName}" requires ${requiredLevel}, caller has ${policy.callerLevel}`;
372
+
373
+ if (policy.mode === 'warn') {
374
+ console.error(`[auth-warn] ${message}`);
375
+ return null;
376
+ }
377
+
378
+ return { success: false, error: message, op: opName, facade: toolName };
379
+ }
380
+
381
+ function jsonResponse(data: Record<string, unknown>): {
382
+ content: Array<{ type: 'text'; text: string }>;
383
+ } {
384
+ return { content: [{ type: 'text' as const, text: JSON.stringify(data, null, 2) }] };
385
+ }
@@ -0,0 +1,83 @@
1
+ /**
2
+ * Test helpers for the engine registration.
3
+ *
4
+ * Provides captureOps() — equivalent to the old captureHandler() pattern
5
+ * used across 18+ E2E tests — but for the new direct registration.
6
+ */
7
+
8
+ import type { OpDefinition } from '../facades/types.js';
9
+
10
+ export interface CapturedOp {
11
+ name: string;
12
+ handler: (params: Record<string, unknown>) => Promise<unknown>;
13
+ schema?: unknown;
14
+ auth: string;
15
+ }
16
+
17
+ /**
18
+ * Capture all op handlers from an op creator function for direct testing.
19
+ * Replaces the old pattern of mocking McpServer + registerFacade + captureHandler.
20
+ */
21
+ export function captureOps(ops: OpDefinition[]): Map<string, CapturedOp> {
22
+ const map = new Map<string, CapturedOp>();
23
+ for (const op of ops) {
24
+ map.set(op.name, {
25
+ name: op.name,
26
+ handler: op.handler,
27
+ schema: op.schema,
28
+ auth: op.auth,
29
+ });
30
+ }
31
+ return map;
32
+ }
33
+
34
+ /**
35
+ * Execute an op by name against captured ops, with response envelope.
36
+ * Matches the old dispatchOp() behavior for test compatibility.
37
+ */
38
+ export async function executeOp(
39
+ ops: Map<string, CapturedOp>,
40
+ opName: string,
41
+ params: Record<string, unknown> = {},
42
+ facadeName = 'test',
43
+ ): Promise<{ success: boolean; data?: unknown; error?: string; op: string; facade: string }> {
44
+ const op = ops.get(opName);
45
+ if (!op) {
46
+ return {
47
+ success: false,
48
+ error: `Unknown operation "${opName}". Available: ${[...ops.keys()].join(', ')}`,
49
+ op: opName,
50
+ facade: facadeName,
51
+ };
52
+ }
53
+
54
+ try {
55
+ let validatedParams = params;
56
+ if (op.schema && typeof (op.schema as { safeParse?: unknown }).safeParse === 'function') {
57
+ const result = (
58
+ op.schema as {
59
+ safeParse: (p: unknown) => {
60
+ success: boolean;
61
+ data?: unknown;
62
+ error?: { message: string };
63
+ };
64
+ }
65
+ ).safeParse(params);
66
+ if (!result.success) {
67
+ return {
68
+ success: false,
69
+ error: `Invalid params for ${opName}: ${result.error?.message}`,
70
+ op: opName,
71
+ facade: facadeName,
72
+ };
73
+ }
74
+ validatedParams = result.data as Record<string, unknown>;
75
+ }
76
+
77
+ const data = await op.handler(validatedParams);
78
+ return { success: true, data, op: opName, facade: facadeName };
79
+ } catch (err) {
80
+ const message = err instanceof Error ? err.message : String(err);
81
+ return { success: false, error: message, op: opName, facade: facadeName };
82
+ }
83
+ }