@namzu/sdk 0.4.2 → 0.4.4

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 (310) hide show
  1. package/CHANGELOG.md +46 -0
  2. package/dist/advisory/context.test.d.ts +16 -0
  3. package/dist/advisory/context.test.d.ts.map +1 -0
  4. package/dist/advisory/context.test.js +92 -0
  5. package/dist/advisory/context.test.js.map +1 -0
  6. package/dist/advisory/evaluator.test.d.ts +34 -0
  7. package/dist/advisory/evaluator.test.d.ts.map +1 -0
  8. package/dist/advisory/evaluator.test.js +172 -0
  9. package/dist/advisory/evaluator.test.js.map +1 -0
  10. package/dist/advisory/executor.test.d.ts +35 -0
  11. package/dist/advisory/executor.test.d.ts.map +1 -0
  12. package/dist/advisory/executor.test.js +233 -0
  13. package/dist/advisory/executor.test.js.map +1 -0
  14. package/dist/advisory/registry.test.d.ts +16 -0
  15. package/dist/advisory/registry.test.d.ts.map +1 -0
  16. package/dist/advisory/registry.test.js +62 -0
  17. package/dist/advisory/registry.test.js.map +1 -0
  18. package/dist/bridge/a2a/agent-card.test.d.ts +24 -0
  19. package/dist/bridge/a2a/agent-card.test.d.ts.map +1 -0
  20. package/dist/bridge/a2a/agent-card.test.js +118 -0
  21. package/dist/bridge/a2a/agent-card.test.js.map +1 -0
  22. package/dist/bridge/a2a/mapper.test.d.ts +29 -0
  23. package/dist/bridge/a2a/mapper.test.d.ts.map +1 -0
  24. package/dist/bridge/a2a/mapper.test.js +265 -0
  25. package/dist/bridge/a2a/mapper.test.js.map +1 -0
  26. package/dist/bridge/a2a/message.test.d.ts +20 -0
  27. package/dist/bridge/a2a/message.test.d.ts.map +1 -0
  28. package/dist/bridge/a2a/message.test.js +116 -0
  29. package/dist/bridge/a2a/message.test.js.map +1 -0
  30. package/dist/bridge/a2a/task.test.d.ts +29 -0
  31. package/dist/bridge/a2a/task.test.d.ts.map +1 -0
  32. package/dist/bridge/a2a/task.test.js +198 -0
  33. package/dist/bridge/a2a/task.test.js.map +1 -0
  34. package/dist/bridge/mcp/connector/adapter.test.d.ts +27 -0
  35. package/dist/bridge/mcp/connector/adapter.test.d.ts.map +1 -0
  36. package/dist/bridge/mcp/connector/adapter.test.js +203 -0
  37. package/dist/bridge/mcp/connector/adapter.test.js.map +1 -0
  38. package/dist/bridge/sse/mapper.test.d.ts +27 -0
  39. package/dist/bridge/sse/mapper.test.d.ts.map +1 -0
  40. package/dist/bridge/sse/mapper.test.js +271 -0
  41. package/dist/bridge/sse/mapper.test.js.map +1 -0
  42. package/dist/bridge/tools/connector/adapter.d.ts +2 -2
  43. package/dist/bridge/tools/connector/adapter.test.d.ts +28 -0
  44. package/dist/bridge/tools/connector/adapter.test.d.ts.map +1 -0
  45. package/dist/bridge/tools/connector/adapter.test.js +182 -0
  46. package/dist/bridge/tools/connector/adapter.test.js.map +1 -0
  47. package/dist/bridge/tools/connector/definitions.test.d.ts +23 -0
  48. package/dist/bridge/tools/connector/definitions.test.d.ts.map +1 -0
  49. package/dist/bridge/tools/connector/definitions.test.js +158 -0
  50. package/dist/bridge/tools/connector/definitions.test.js.map +1 -0
  51. package/dist/bridge/tools/connector/router.test.d.ts +21 -0
  52. package/dist/bridge/tools/connector/router.test.d.ts.map +1 -0
  53. package/dist/bridge/tools/connector/router.test.js +139 -0
  54. package/dist/bridge/tools/connector/router.test.js.map +1 -0
  55. package/dist/bus/breaker.test.d.ts +41 -0
  56. package/dist/bus/breaker.test.d.ts.map +1 -0
  57. package/dist/bus/breaker.test.js +242 -0
  58. package/dist/bus/breaker.test.js.map +1 -0
  59. package/dist/bus/index.d.ts +3 -1
  60. package/dist/bus/index.d.ts.map +1 -1
  61. package/dist/bus/index.js +18 -11
  62. package/dist/bus/index.js.map +1 -1
  63. package/dist/bus/index.test.d.ts +25 -0
  64. package/dist/bus/index.test.d.ts.map +1 -0
  65. package/dist/bus/index.test.js +151 -0
  66. package/dist/bus/index.test.js.map +1 -0
  67. package/dist/bus/lock.test.d.ts +44 -0
  68. package/dist/bus/lock.test.d.ts.map +1 -0
  69. package/dist/bus/lock.test.js +226 -0
  70. package/dist/bus/lock.test.js.map +1 -0
  71. package/dist/bus/ownership.test.d.ts +26 -0
  72. package/dist/bus/ownership.test.d.ts.map +1 -0
  73. package/dist/bus/ownership.test.js +205 -0
  74. package/dist/bus/ownership.test.js.map +1 -0
  75. package/dist/config/runtime.d.ts +28 -28
  76. package/dist/connector/BaseConnector.test.d.ts +21 -0
  77. package/dist/connector/BaseConnector.test.d.ts.map +1 -0
  78. package/dist/connector/BaseConnector.test.js +108 -0
  79. package/dist/connector/BaseConnector.test.js.map +1 -0
  80. package/dist/connector/builtins/http.test.d.ts +30 -0
  81. package/dist/connector/builtins/http.test.d.ts.map +1 -0
  82. package/dist/connector/builtins/http.test.js +232 -0
  83. package/dist/connector/builtins/http.test.js.map +1 -0
  84. package/dist/connector/builtins/webhook.test.d.ts +20 -0
  85. package/dist/connector/builtins/webhook.test.d.ts.map +1 -0
  86. package/dist/connector/builtins/webhook.test.js +113 -0
  87. package/dist/connector/builtins/webhook.test.js.map +1 -0
  88. package/dist/connector/execution/factory.test.d.ts +16 -0
  89. package/dist/connector/execution/factory.test.d.ts.map +1 -0
  90. package/dist/connector/execution/factory.test.js +64 -0
  91. package/dist/connector/execution/factory.test.js.map +1 -0
  92. package/dist/connector/execution/remote.test.d.ts +16 -0
  93. package/dist/connector/execution/remote.test.d.ts.map +1 -0
  94. package/dist/connector/execution/remote.test.js +53 -0
  95. package/dist/connector/execution/remote.test.js.map +1 -0
  96. package/dist/connector/mcp/adapter.test.d.ts +34 -0
  97. package/dist/connector/mcp/adapter.test.d.ts.map +1 -0
  98. package/dist/connector/mcp/adapter.test.js +199 -0
  99. package/dist/connector/mcp/adapter.test.js.map +1 -0
  100. package/dist/probe/context.d.ts +8 -0
  101. package/dist/probe/context.d.ts.map +1 -0
  102. package/dist/probe/context.js +7 -0
  103. package/dist/probe/context.js.map +1 -0
  104. package/dist/probe/errors.d.ts +12 -0
  105. package/dist/probe/errors.d.ts.map +1 -0
  106. package/dist/probe/errors.js +21 -0
  107. package/dist/probe/errors.js.map +1 -0
  108. package/dist/probe/index.d.ts +5 -0
  109. package/dist/probe/index.d.ts.map +1 -0
  110. package/dist/probe/index.js +4 -0
  111. package/dist/probe/index.js.map +1 -0
  112. package/dist/probe/registry.d.ts +24 -0
  113. package/dist/probe/registry.d.ts.map +1 -0
  114. package/dist/probe/registry.js +228 -0
  115. package/dist/probe/registry.js.map +1 -0
  116. package/dist/probe/registry.test.d.ts +7 -0
  117. package/dist/probe/registry.test.d.ts.map +1 -0
  118. package/dist/probe/registry.test.js +310 -0
  119. package/dist/probe/registry.test.js.map +1 -0
  120. package/dist/provider/instrumentation.d.ts +9 -0
  121. package/dist/provider/instrumentation.d.ts.map +1 -0
  122. package/dist/provider/instrumentation.js +104 -0
  123. package/dist/provider/instrumentation.js.map +1 -0
  124. package/dist/provider/instrumentation.test.d.ts +2 -0
  125. package/dist/provider/instrumentation.test.d.ts.map +1 -0
  126. package/dist/provider/instrumentation.test.js +152 -0
  127. package/dist/provider/instrumentation.test.js.map +1 -0
  128. package/dist/public-runtime.d.ts +5 -0
  129. package/dist/public-runtime.d.ts.map +1 -1
  130. package/dist/public-runtime.js +4 -0
  131. package/dist/public-runtime.js.map +1 -1
  132. package/dist/public-types.d.ts +3 -0
  133. package/dist/public-types.d.ts.map +1 -1
  134. package/dist/rag/chunking.test.d.ts +20 -0
  135. package/dist/rag/chunking.test.d.ts.map +1 -0
  136. package/dist/rag/chunking.test.js +92 -0
  137. package/dist/rag/chunking.test.js.map +1 -0
  138. package/dist/rag/context-assembler.test.d.ts +19 -0
  139. package/dist/rag/context-assembler.test.d.ts.map +1 -0
  140. package/dist/rag/context-assembler.test.js +98 -0
  141. package/dist/rag/context-assembler.test.js.map +1 -0
  142. package/dist/rag/embedding.test.d.ts +19 -0
  143. package/dist/rag/embedding.test.d.ts.map +1 -0
  144. package/dist/rag/embedding.test.js +115 -0
  145. package/dist/rag/embedding.test.js.map +1 -0
  146. package/dist/rag/ingestion.test.d.ts +22 -0
  147. package/dist/rag/ingestion.test.d.ts.map +1 -0
  148. package/dist/rag/ingestion.test.js +99 -0
  149. package/dist/rag/ingestion.test.js.map +1 -0
  150. package/dist/rag/knowledge-base.test.d.ts +17 -0
  151. package/dist/rag/knowledge-base.test.d.ts.map +1 -0
  152. package/dist/rag/knowledge-base.test.js +77 -0
  153. package/dist/rag/knowledge-base.test.js.map +1 -0
  154. package/dist/rag/rag-tool.test.d.ts +21 -0
  155. package/dist/rag/rag-tool.test.d.ts.map +1 -0
  156. package/dist/rag/rag-tool.test.js +149 -0
  157. package/dist/rag/rag-tool.test.js.map +1 -0
  158. package/dist/rag/retriever.test.d.ts +26 -0
  159. package/dist/rag/retriever.test.d.ts.map +1 -0
  160. package/dist/rag/retriever.test.js +180 -0
  161. package/dist/rag/retriever.test.js.map +1 -0
  162. package/dist/rag/vector-store.test.d.ts +38 -0
  163. package/dist/rag/vector-store.test.d.ts.map +1 -0
  164. package/dist/rag/vector-store.test.js +175 -0
  165. package/dist/rag/vector-store.test.js.map +1 -0
  166. package/dist/registry/ManagedRegistry.test.d.ts +21 -0
  167. package/dist/registry/ManagedRegistry.test.d.ts.map +1 -0
  168. package/dist/registry/ManagedRegistry.test.js +98 -0
  169. package/dist/registry/ManagedRegistry.test.js.map +1 -0
  170. package/dist/registry/Registry.test.d.ts +18 -0
  171. package/dist/registry/Registry.test.d.ts.map +1 -0
  172. package/dist/registry/Registry.test.js +79 -0
  173. package/dist/registry/Registry.test.js.map +1 -0
  174. package/dist/registry/agent/definitions.test.d.ts +15 -0
  175. package/dist/registry/agent/definitions.test.d.ts.map +1 -0
  176. package/dist/registry/agent/definitions.test.js +84 -0
  177. package/dist/registry/agent/definitions.test.js.map +1 -0
  178. package/dist/registry/connector/definitions.test.d.ts +13 -0
  179. package/dist/registry/connector/definitions.test.d.ts.map +1 -0
  180. package/dist/registry/connector/definitions.test.js +41 -0
  181. package/dist/registry/connector/definitions.test.js.map +1 -0
  182. package/dist/registry/connector/scoped.test.d.ts +21 -0
  183. package/dist/registry/connector/scoped.test.d.ts.map +1 -0
  184. package/dist/registry/connector/scoped.test.js +115 -0
  185. package/dist/registry/connector/scoped.test.js.map +1 -0
  186. package/dist/registry/plugin/index.test.d.ts +12 -0
  187. package/dist/registry/plugin/index.test.d.ts.map +1 -0
  188. package/dist/registry/plugin/index.test.js +69 -0
  189. package/dist/registry/plugin/index.test.js.map +1 -0
  190. package/dist/registry/tool/execute.test.d.ts +42 -0
  191. package/dist/registry/tool/execute.test.d.ts.map +1 -0
  192. package/dist/registry/tool/execute.test.js +281 -0
  193. package/dist/registry/tool/execute.test.js.map +1 -0
  194. package/dist/runtime/query/events.d.ts +3 -1
  195. package/dist/runtime/query/events.d.ts.map +1 -1
  196. package/dist/runtime/query/events.js +6 -1
  197. package/dist/runtime/query/events.js.map +1 -1
  198. package/dist/runtime/query/executor.d.ts +3 -1
  199. package/dist/runtime/query/executor.d.ts.map +1 -1
  200. package/dist/runtime/query/executor.js +30 -1
  201. package/dist/runtime/query/executor.js.map +1 -1
  202. package/dist/runtime/query/iteration/phases/advisory.test.d.ts +42 -0
  203. package/dist/runtime/query/iteration/phases/advisory.test.d.ts.map +1 -0
  204. package/dist/runtime/query/iteration/phases/advisory.test.js +334 -0
  205. package/dist/runtime/query/iteration/phases/advisory.test.js.map +1 -0
  206. package/dist/test-setup.d.ts +22 -0
  207. package/dist/test-setup.d.ts.map +1 -0
  208. package/dist/test-setup.js +23 -0
  209. package/dist/test-setup.js.map +1 -0
  210. package/dist/types/bus/index.d.ts +46 -2
  211. package/dist/types/bus/index.d.ts.map +1 -1
  212. package/dist/types/doctor/check.d.ts +41 -0
  213. package/dist/types/doctor/check.d.ts.map +1 -0
  214. package/dist/types/doctor/check.js +2 -0
  215. package/dist/types/doctor/check.js.map +1 -0
  216. package/dist/types/doctor/index.d.ts +2 -0
  217. package/dist/types/doctor/index.d.ts.map +1 -0
  218. package/dist/types/doctor/index.js +2 -0
  219. package/dist/types/doctor/index.js.map +1 -0
  220. package/dist/types/probe/event-kind.d.ts +6 -0
  221. package/dist/types/probe/event-kind.d.ts.map +1 -0
  222. package/dist/types/probe/event-kind.js +2 -0
  223. package/dist/types/probe/event-kind.js.map +1 -0
  224. package/dist/types/probe/event-of.d.ts +5 -0
  225. package/dist/types/probe/event-of.d.ts.map +1 -0
  226. package/dist/types/probe/event-of.js +2 -0
  227. package/dist/types/probe/event-of.js.map +1 -0
  228. package/dist/types/probe/index.d.ts +4 -0
  229. package/dist/types/probe/index.d.ts.map +1 -0
  230. package/dist/types/probe/index.js +2 -0
  231. package/dist/types/probe/index.js.map +1 -0
  232. package/dist/types/probe/registry.d.ts +27 -0
  233. package/dist/types/probe/registry.d.ts.map +1 -0
  234. package/dist/types/probe/registry.js +2 -0
  235. package/dist/types/probe/registry.js.map +1 -0
  236. package/dist/utils/logger.d.ts +1 -1
  237. package/dist/utils/logger.d.ts.map +1 -1
  238. package/dist/utils/logger.js +5 -0
  239. package/dist/utils/logger.js.map +1 -1
  240. package/dist/vault/instrumentation.d.ts +11 -0
  241. package/dist/vault/instrumentation.d.ts.map +1 -0
  242. package/dist/vault/instrumentation.js +32 -0
  243. package/dist/vault/instrumentation.js.map +1 -0
  244. package/dist/vault/instrumentation.test.d.ts +2 -0
  245. package/dist/vault/instrumentation.test.d.ts.map +1 -0
  246. package/dist/vault/instrumentation.test.js +80 -0
  247. package/dist/vault/instrumentation.test.js.map +1 -0
  248. package/package.json +4 -1
  249. package/src/advisory/context.test.ts +109 -0
  250. package/src/advisory/evaluator.test.ts +192 -0
  251. package/src/advisory/executor.test.ts +272 -0
  252. package/src/advisory/registry.test.ts +75 -0
  253. package/src/bridge/a2a/agent-card.test.ts +140 -0
  254. package/src/bridge/a2a/mapper.test.ts +293 -0
  255. package/src/bridge/a2a/message.test.ts +138 -0
  256. package/src/bridge/a2a/task.test.ts +235 -0
  257. package/src/bridge/mcp/connector/adapter.test.ts +230 -0
  258. package/src/bridge/sse/mapper.test.ts +422 -0
  259. package/src/bridge/tools/connector/adapter.test.ts +224 -0
  260. package/src/bridge/tools/connector/definitions.test.ts +183 -0
  261. package/src/bridge/tools/connector/router.test.ts +159 -0
  262. package/src/bus/breaker.test.ts +274 -0
  263. package/src/bus/index.test.ts +183 -0
  264. package/src/bus/index.ts +21 -10
  265. package/src/bus/lock.test.ts +265 -0
  266. package/src/bus/ownership.test.ts +243 -0
  267. package/src/connector/BaseConnector.test.ts +130 -0
  268. package/src/connector/builtins/http.test.ts +290 -0
  269. package/src/connector/builtins/webhook.test.ts +138 -0
  270. package/src/connector/execution/factory.test.ts +75 -0
  271. package/src/connector/execution/remote.test.ts +63 -0
  272. package/src/connector/mcp/adapter.test.ts +249 -0
  273. package/src/probe/context.ts +14 -0
  274. package/src/probe/errors.ts +27 -0
  275. package/src/probe/index.ts +4 -0
  276. package/src/probe/registry.test.ts +480 -0
  277. package/src/probe/registry.ts +276 -0
  278. package/src/provider/instrumentation.test.ts +192 -0
  279. package/src/provider/instrumentation.ts +139 -0
  280. package/src/public-runtime.ts +17 -0
  281. package/src/public-types.ts +3 -0
  282. package/src/rag/chunking.test.ts +107 -0
  283. package/src/rag/context-assembler.test.ts +114 -0
  284. package/src/rag/embedding.test.ts +130 -0
  285. package/src/rag/ingestion.test.ts +114 -0
  286. package/src/rag/knowledge-base.test.ts +106 -0
  287. package/src/rag/rag-tool.test.ts +167 -0
  288. package/src/rag/retriever.test.ts +210 -0
  289. package/src/rag/vector-store.test.ts +196 -0
  290. package/src/registry/ManagedRegistry.test.ts +118 -0
  291. package/src/registry/Registry.test.ts +91 -0
  292. package/src/registry/agent/definitions.test.ts +100 -0
  293. package/src/registry/connector/definitions.test.ts +51 -0
  294. package/src/registry/connector/scoped.test.ts +129 -0
  295. package/src/registry/plugin/index.test.ts +85 -0
  296. package/src/registry/tool/execute.test.ts +330 -0
  297. package/src/runtime/query/events.ts +6 -1
  298. package/src/runtime/query/executor.ts +34 -0
  299. package/src/runtime/query/iteration/phases/advisory.test.ts +412 -0
  300. package/src/test-setup.ts +24 -0
  301. package/src/types/bus/index.ts +54 -2
  302. package/src/types/doctor/check.ts +53 -0
  303. package/src/types/doctor/index.ts +9 -0
  304. package/src/types/probe/event-kind.ts +8 -0
  305. package/src/types/probe/event-of.ts +3 -0
  306. package/src/types/probe/index.ts +11 -0
  307. package/src/types/probe/registry.ts +36 -0
  308. package/src/utils/logger.ts +6 -1
  309. package/src/vault/instrumentation.test.ts +98 -0
  310. package/src/vault/instrumentation.ts +56 -0
@@ -0,0 +1,91 @@
1
+ /**
2
+ * Current-code invariants asserted (2026-04-21, ses_006 Phase 3):
3
+ *
4
+ * - `register(id, item)` silently overwrites an existing entry —
5
+ * NO duplicate-error is thrown (Codex #1 verified the design
6
+ * §2.3 "typed DuplicateRegistrationError" claim was fiction).
7
+ * - `get(id)` returns `undefined` for an unknown id (NOT a typed
8
+ * error). `getAll`, `listIds`, `has`, `size` reflect the current
9
+ * `Map` state; iteration order follows insertion order.
10
+ * - `unregister(id)` returns true if the entry existed (and was
11
+ * deleted), false otherwise.
12
+ * - `clear()` empties the map.
13
+ * - The generic `Registry<T>` has no component name, no logger, no
14
+ * lifecycle. All enrichment (logs, idField inference, lifecycle)
15
+ * happens in `ManagedRegistry`, not here.
16
+ */
17
+
18
+ import { describe, expect, it } from 'vitest'
19
+
20
+ import { Registry } from './Registry.js'
21
+
22
+ interface Item {
23
+ id: string
24
+ value: number
25
+ }
26
+
27
+ describe('Registry', () => {
28
+ it('register + get + has roundtrip', () => {
29
+ const r = new Registry<Item>()
30
+ const a: Item = { id: 'a', value: 1 }
31
+ r.register('a', a)
32
+ expect(r.get('a')).toBe(a)
33
+ expect(r.has('a')).toBe(true)
34
+ expect(r.get('b')).toBeUndefined()
35
+ expect(r.has('b')).toBe(false)
36
+ })
37
+
38
+ it('register silently overwrites an existing key', () => {
39
+ const r = new Registry<Item>()
40
+ r.register('a', { id: 'a', value: 1 })
41
+ r.register('a', { id: 'a', value: 2 })
42
+ expect(r.get('a')?.value).toBe(2)
43
+ expect(r.size()).toBe(1)
44
+ })
45
+
46
+ it('listIds reflects insertion order', () => {
47
+ const r = new Registry<Item>()
48
+ r.register('x', { id: 'x', value: 1 })
49
+ r.register('y', { id: 'y', value: 2 })
50
+ r.register('z', { id: 'z', value: 3 })
51
+ expect(r.listIds()).toEqual(['x', 'y', 'z'])
52
+ })
53
+
54
+ it('getAll returns every item in insertion order', () => {
55
+ const r = new Registry<Item>()
56
+ const items = [
57
+ { id: 'a', value: 1 },
58
+ { id: 'b', value: 2 },
59
+ ]
60
+ for (const item of items) r.register(item.id, item)
61
+ expect(r.getAll()).toEqual(items)
62
+ })
63
+
64
+ it('unregister returns true iff the key existed', () => {
65
+ const r = new Registry<Item>()
66
+ r.register('a', { id: 'a', value: 1 })
67
+ expect(r.unregister('a')).toBe(true)
68
+ expect(r.unregister('a')).toBe(false)
69
+ expect(r.get('a')).toBeUndefined()
70
+ })
71
+
72
+ it('clear empties the map', () => {
73
+ const r = new Registry<Item>()
74
+ r.register('a', { id: 'a', value: 1 })
75
+ r.register('b', { id: 'b', value: 2 })
76
+ r.clear()
77
+ expect(r.size()).toBe(0)
78
+ expect(r.getAll()).toEqual([])
79
+ })
80
+
81
+ it('size matches the map size', () => {
82
+ const r = new Registry<Item>()
83
+ expect(r.size()).toBe(0)
84
+ r.register('a', { id: 'a', value: 1 })
85
+ expect(r.size()).toBe(1)
86
+ r.register('a', { id: 'a', value: 2 }) // overwrite
87
+ expect(r.size()).toBe(1)
88
+ r.register('b', { id: 'b', value: 3 })
89
+ expect(r.size()).toBe(2)
90
+ })
91
+ })
@@ -0,0 +1,100 @@
1
+ /**
2
+ * Current-code invariants asserted (2026-04-21, ses_006 Phase 3):
3
+ *
4
+ * - `AgentRegistry` extends `ManagedRegistry<AgentDefinition>` and
5
+ * keys by `def.info.id` via `computeId` (NOT by the top-level id
6
+ * field — there isn't one on AgentDefinition).
7
+ * - `resolve(agentId)` returns `.typedAgent`; delegates to
8
+ * `getOrThrow` and therefore throws a plain `Error` when missing.
9
+ * - `listByType(type)` filters by `typedAgent.type`.
10
+ * - `search({category?, query?})` filters by info.category then by
11
+ * case-insensitive match against name OR description. An empty
12
+ * query matches all.
13
+ */
14
+
15
+ import { describe, expect, it } from 'vitest'
16
+
17
+ import type { AgentInfo } from '../../contracts/api.js'
18
+ import type { AgentType, BaseAgentConfig, BaseAgentResult } from '../../types/agent/base.js'
19
+ import type { Agent } from '../../types/agent/core.js'
20
+ import type { AgentDefinition } from '../../types/agent/factory.js'
21
+
22
+ import { AgentRegistry } from './definitions.js'
23
+
24
+ function makeDef(id: string, type: AgentType, overrides: Partial<AgentInfo> = {}): AgentDefinition {
25
+ const info: AgentInfo = {
26
+ id,
27
+ name: `Agent ${id}`,
28
+ version: '1.0.0',
29
+ category: 'general',
30
+ description: `agent ${id} does stuff`,
31
+ tools: [],
32
+ defaults: { model: 'opus', tokenBudget: 1000 },
33
+ ...overrides,
34
+ }
35
+ const typedAgent = { type } as unknown as Agent<BaseAgentConfig, BaseAgentResult>
36
+ return { info, typedAgent }
37
+ }
38
+
39
+ describe('AgentRegistry', () => {
40
+ it('keys by info.id (computeId), not by a top-level id field', () => {
41
+ const r = new AgentRegistry()
42
+ r.register(makeDef('coder', 'reactive'))
43
+ expect(r.get('coder')).toBeDefined()
44
+ })
45
+
46
+ it('resolve returns typedAgent', () => {
47
+ const r = new AgentRegistry()
48
+ const def = makeDef('coder', 'reactive')
49
+ r.register(def)
50
+ expect(r.resolve('coder')).toBe(def.typedAgent)
51
+ })
52
+
53
+ it('resolve throws (plain Error) on unknown agentId', () => {
54
+ const r = new AgentRegistry()
55
+ expect(() => r.resolve('nope')).toThrow(/Not found/)
56
+ })
57
+
58
+ it('listByType filters by typedAgent.type', () => {
59
+ const r = new AgentRegistry()
60
+ r.register(makeDef('a', 'reactive'))
61
+ r.register(makeDef('b', 'pipeline'))
62
+ r.register(makeDef('c', 'reactive'))
63
+ expect(r.listByType('reactive').map((d) => d.info.id)).toEqual(['a', 'c'])
64
+ expect(r.listByType('pipeline').map((d) => d.info.id)).toEqual(['b'])
65
+ })
66
+
67
+ describe('search', () => {
68
+ it('empty query returns all', () => {
69
+ const r = new AgentRegistry()
70
+ r.register(makeDef('a', 'reactive'))
71
+ r.register(makeDef('b', 'pipeline'))
72
+ expect(r.search({})).toHaveLength(2)
73
+ })
74
+
75
+ it('category filter is strict-equal', () => {
76
+ const r = new AgentRegistry()
77
+ r.register(makeDef('coder', 'reactive', { category: 'coding' }))
78
+ r.register(makeDef('writer', 'reactive', { category: 'writing' }))
79
+ expect(r.search({ category: 'coding' }).map((d) => d.info.id)).toEqual(['coder'])
80
+ })
81
+
82
+ it('query matches name OR description case-insensitively', () => {
83
+ const r = new AgentRegistry()
84
+ r.register(makeDef('alpha', 'reactive', { name: 'CoderBot', description: 'writes code' }))
85
+ r.register(makeDef('beta', 'reactive', { name: 'Other', description: 'tests CODE' }))
86
+ r.register(makeDef('gamma', 'reactive', { name: 'Third', description: 'documentation' }))
87
+
88
+ const hits = r.search({ query: 'code' }).map((d) => d.info.id)
89
+ expect(hits).toEqual(['alpha', 'beta'])
90
+ })
91
+
92
+ it('combines category + query', () => {
93
+ const r = new AgentRegistry()
94
+ r.register(makeDef('alpha', 'reactive', { category: 'coding', name: 'CoderBot' }))
95
+ r.register(makeDef('beta', 'reactive', { category: 'writing', name: 'CoderWriter' }))
96
+ const hits = r.search({ category: 'coding', query: 'coder' }).map((d) => d.info.id)
97
+ expect(hits).toEqual(['alpha'])
98
+ })
99
+ })
100
+ })
@@ -0,0 +1,51 @@
1
+ /**
2
+ * Current-code invariants asserted (2026-04-21, ses_006 Phase 3):
3
+ *
4
+ * - `ConnectorRegistry` extends `ManagedRegistry<ConnectorDefinition>`
5
+ * with `idField: 'id'` — it keys by the top-level `id` field, NOT
6
+ * a nested path.
7
+ * - `listByType(connectionType)` filters by `connectionType`.
8
+ * - As a global (non-tenant-scoped) registry, ConnectorRegistry is
9
+ * shared across tenants; tenant isolation lives in
10
+ * `TenantConnectorManager` per Codex #5.
11
+ */
12
+
13
+ import { describe, expect, it } from 'vitest'
14
+ import { z } from 'zod'
15
+
16
+ import type { ConnectorDefinition } from '../../types/connector/index.js'
17
+ import type { ConnectorId } from '../../types/ids/index.js'
18
+
19
+ import { ConnectorRegistry } from './definitions.js'
20
+
21
+ function makeDef(
22
+ id: string,
23
+ connectionType: ConnectorDefinition['connectionType'],
24
+ ): ConnectorDefinition {
25
+ return {
26
+ id: id as ConnectorId,
27
+ name: id,
28
+ description: `${id} connector`,
29
+ connectionType,
30
+ configSchema: z.object({}),
31
+ methods: [],
32
+ }
33
+ }
34
+
35
+ describe('ConnectorRegistry', () => {
36
+ it('keys by top-level id', () => {
37
+ const r = new ConnectorRegistry()
38
+ r.register(makeDef('conn_a', 'http'))
39
+ expect(r.get('conn_a')).toBeDefined()
40
+ })
41
+
42
+ it('listByType filters by connectionType', () => {
43
+ const r = new ConnectorRegistry()
44
+ r.register(makeDef('conn_a', 'http'))
45
+ r.register(makeDef('conn_b', 'webhook'))
46
+ r.register(makeDef('conn_c', 'http'))
47
+ expect(r.listByType('http').map((d) => d.id)).toEqual(['conn_a', 'conn_c'])
48
+ expect(r.listByType('webhook').map((d) => d.id)).toEqual(['conn_b'])
49
+ expect(r.listByType('custom')).toEqual([])
50
+ })
51
+ })
@@ -0,0 +1,129 @@
1
+ /**
2
+ * Current-code invariants asserted (2026-04-21, ses_006 Phase 3):
3
+ *
4
+ * - `ScopedConnectorRegistry` stores per-scope config by a composite
5
+ * key `<scope>:<scopeId>:<connectorId>`. Multiple configs at
6
+ * different scopes for the same connector coexist.
7
+ * - `set`, `remove`, `getAt` round-trip through the same key.
8
+ * - `resolve(connectorId, chain)` iterates in `CONNECTOR_SCOPE_ORDER`
9
+ * (org → environment → team → project → agent) and merges layers
10
+ * — later scopes override earlier ones. Specifically:
11
+ * - `options` are shallow-merged across layers.
12
+ * - `auth` is last-wins (any layer with an explicit auth replaces).
13
+ * - `enabled` is last-wins (defaulting to true when no layer sets it).
14
+ * - The final `ConnectorConfig.name` falls back to `connectorId`
15
+ * when no layer sets a name.
16
+ * - `resolve` returns undefined when no layer matches the chain.
17
+ * - `listForConnector` returns every config for a connector across
18
+ * all scopes; `listAtScope` returns every connector at a given scope.
19
+ */
20
+
21
+ import { describe, expect, it } from 'vitest'
22
+
23
+ import type { ConnectorId } from '../../types/ids/index.js'
24
+
25
+ import { ScopedConnectorRegistry } from './scoped.js'
26
+
27
+ const CID = 'conn_http' as ConnectorId
28
+
29
+ describe('ScopedConnectorRegistry', () => {
30
+ describe('set + getAt + remove', () => {
31
+ it('round-trips via the composite key', () => {
32
+ const r = new ScopedConnectorRegistry()
33
+ r.set({
34
+ scope: { scope: 'org', scopeId: 'org_1' },
35
+ connectorId: CID,
36
+ options: { k: 'v' },
37
+ })
38
+ expect(r.getAt({ scope: 'org', scopeId: 'org_1' }, CID)?.options).toEqual({ k: 'v' })
39
+ })
40
+
41
+ it('remove returns true iff an entry existed', () => {
42
+ const r = new ScopedConnectorRegistry()
43
+ r.set({ scope: { scope: 'org', scopeId: 'org_1' }, connectorId: CID })
44
+ expect(r.remove({ scope: 'org', scopeId: 'org_1' }, CID)).toBe(true)
45
+ expect(r.remove({ scope: 'org', scopeId: 'org_1' }, CID)).toBe(false)
46
+ })
47
+
48
+ it('different scopes coexist for the same connector', () => {
49
+ const r = new ScopedConnectorRegistry()
50
+ r.set({ scope: { scope: 'org', scopeId: 'org_1' }, connectorId: CID })
51
+ r.set({ scope: { scope: 'project', scopeId: 'proj_1' }, connectorId: CID })
52
+ expect(r.listForConnector(CID)).toHaveLength(2)
53
+ })
54
+ })
55
+
56
+ describe('resolve layering', () => {
57
+ it('merges options shallow; later scopes override', () => {
58
+ const r = new ScopedConnectorRegistry()
59
+ r.set({
60
+ scope: { scope: 'org', scopeId: 'org_1' },
61
+ connectorId: CID,
62
+ options: { host: 'api.org', timeout: 30 },
63
+ })
64
+ r.set({
65
+ scope: { scope: 'project', scopeId: 'proj_1' },
66
+ connectorId: CID,
67
+ options: { host: 'api.project' },
68
+ })
69
+ const resolved = r.resolve(CID, { org: 'org_1', project: 'proj_1' })
70
+ expect(resolved?.options).toEqual({ host: 'api.project', timeout: 30 })
71
+ })
72
+
73
+ it('auth is last-wins (CONNECTOR_SCOPE_ORDER)', () => {
74
+ const r = new ScopedConnectorRegistry()
75
+ r.set({
76
+ scope: { scope: 'org', scopeId: 'org_1' },
77
+ connectorId: CID,
78
+ auth: { type: 'none' },
79
+ })
80
+ r.set({
81
+ scope: { scope: 'project', scopeId: 'proj_1' },
82
+ connectorId: CID,
83
+ auth: { type: 'api_key', credentials: { apiKey: 'secret' } },
84
+ })
85
+ const resolved = r.resolve(CID, { org: 'org_1', project: 'proj_1' })
86
+ expect(resolved?.auth).toEqual({ type: 'api_key', credentials: { apiKey: 'secret' } })
87
+ })
88
+
89
+ it('enabled defaults to true and is last-wins', () => {
90
+ const r = new ScopedConnectorRegistry()
91
+ r.set({ scope: { scope: 'org', scopeId: 'org_1' }, connectorId: CID, enabled: false })
92
+ const resolved = r.resolve(CID, { org: 'org_1' })
93
+ expect(resolved?.enabled).toBe(false)
94
+ })
95
+
96
+ it('config.name falls back to connectorId when no layer sets a name', () => {
97
+ const r = new ScopedConnectorRegistry()
98
+ r.set({ scope: { scope: 'org', scopeId: 'org_1' }, connectorId: CID })
99
+ const resolved = r.resolve(CID, { org: 'org_1' })
100
+ expect(resolved?.config.name).toBe(CID)
101
+ })
102
+
103
+ it('resolvedFrom preserves layer order by CONNECTOR_SCOPE_ORDER', () => {
104
+ const r = new ScopedConnectorRegistry()
105
+ r.set({ scope: { scope: 'project', scopeId: 'proj_1' }, connectorId: CID })
106
+ r.set({ scope: { scope: 'org', scopeId: 'org_1' }, connectorId: CID })
107
+ const resolved = r.resolve(CID, { org: 'org_1', project: 'proj_1' })
108
+ expect(resolved?.resolvedFrom.map((s) => s.scope)).toEqual(['org', 'project'])
109
+ })
110
+
111
+ it('returns undefined when no layer matches the chain', () => {
112
+ const r = new ScopedConnectorRegistry()
113
+ expect(r.resolve(CID, { org: 'org_none' })).toBeUndefined()
114
+ })
115
+ })
116
+
117
+ describe('list operations', () => {
118
+ it('listAtScope filters by scope prefix', () => {
119
+ const r = new ScopedConnectorRegistry()
120
+ r.set({ scope: { scope: 'org', scopeId: 'org_1' }, connectorId: CID })
121
+ r.set({
122
+ scope: { scope: 'org', scopeId: 'org_1' },
123
+ connectorId: 'conn_other' as ConnectorId,
124
+ })
125
+ r.set({ scope: { scope: 'project', scopeId: 'proj_1' }, connectorId: CID })
126
+ expect(r.listAtScope({ scope: 'org', scopeId: 'org_1' })).toHaveLength(2)
127
+ })
128
+ })
129
+ })
@@ -0,0 +1,85 @@
1
+ /**
2
+ * Current-code invariants asserted (2026-04-21, ses_006 Phase 3):
3
+ *
4
+ * - `PluginRegistry` extends `ManagedRegistry<PluginDefinition>`
5
+ * with `idField: 'id'`.
6
+ * - `listByScope(scope)` filters by `def.scope`.
7
+ * - `listByStatus(status)` filters by `def.status`.
8
+ * - `findByName(name)` does a linear scan through `getAll()` and
9
+ * returns the first match by `manifest.name`, or undefined.
10
+ */
11
+
12
+ import { describe, expect, it } from 'vitest'
13
+
14
+ import type { PluginId } from '../../types/ids/index.js'
15
+ import type { PluginDefinition, PluginScope, PluginStatus } from '../../types/plugin/index.js'
16
+
17
+ import { PluginRegistry } from './index.js'
18
+
19
+ function makeDef(
20
+ id: string,
21
+ scope: PluginScope,
22
+ status: PluginStatus,
23
+ name = id,
24
+ ): PluginDefinition {
25
+ return {
26
+ id: id as PluginId,
27
+ manifest: {
28
+ name,
29
+ version: '1.0.0',
30
+ description: `${name} plugin`,
31
+ },
32
+ scope,
33
+ status,
34
+ rootDir: `/plugins/${id}`,
35
+ installedAt: Date.now(),
36
+ }
37
+ }
38
+
39
+ describe('PluginRegistry', () => {
40
+ it('keys by top-level id', () => {
41
+ const r = new PluginRegistry()
42
+ r.register(makeDef('plugin_a', 'project', 'installed'))
43
+ expect(r.get('plugin_a')).toBeDefined()
44
+ })
45
+
46
+ it('listByScope filters by scope', () => {
47
+ const r = new PluginRegistry()
48
+ r.register(makeDef('a', 'project', 'installed'))
49
+ r.register(makeDef('b', 'user', 'enabled'))
50
+ r.register(makeDef('c', 'project', 'disabled'))
51
+ expect(r.listByScope('project').map((d) => d.id)).toEqual(['a', 'c'])
52
+ expect(r.listByScope('user').map((d) => d.id)).toEqual(['b'])
53
+ })
54
+
55
+ it('listByStatus filters by status', () => {
56
+ const r = new PluginRegistry()
57
+ r.register(makeDef('a', 'project', 'installed'))
58
+ r.register(makeDef('b', 'project', 'enabled'))
59
+ r.register(makeDef('c', 'project', 'enabled'))
60
+ expect(r.listByStatus('enabled').map((d) => d.id)).toEqual(['b', 'c'])
61
+ expect(r.listByStatus('error')).toEqual([])
62
+ })
63
+
64
+ describe('findByName', () => {
65
+ it('returns the first def with matching manifest.name', () => {
66
+ const r = new PluginRegistry()
67
+ r.register(makeDef('id_1', 'project', 'installed', 'alpha'))
68
+ r.register(makeDef('id_2', 'project', 'installed', 'beta'))
69
+ expect(r.findByName('beta')?.id).toBe('id_2')
70
+ })
71
+
72
+ it('returns undefined when no match', () => {
73
+ const r = new PluginRegistry()
74
+ r.register(makeDef('id_1', 'project', 'installed', 'alpha'))
75
+ expect(r.findByName('missing')).toBeUndefined()
76
+ })
77
+
78
+ it('is case-sensitive (exact string match)', () => {
79
+ const r = new PluginRegistry()
80
+ r.register(makeDef('id_1', 'project', 'installed', 'Alpha'))
81
+ expect(r.findByName('alpha')).toBeUndefined()
82
+ expect(r.findByName('Alpha')?.id).toBe('id_1')
83
+ })
84
+ })
85
+ })