digital-tools 2.1.1 → 2.3.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 (293) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/README.md +2 -0
  3. package/dist/client.d.ts +109 -0
  4. package/dist/client.d.ts.map +1 -0
  5. package/dist/client.js +69 -0
  6. package/dist/client.js.map +1 -0
  7. package/dist/define.d.ts +2 -2
  8. package/dist/define.d.ts.map +1 -1
  9. package/dist/define.js +22 -20
  10. package/dist/define.js.map +1 -1
  11. package/dist/function-ref.d.ts +229 -0
  12. package/dist/function-ref.d.ts.map +1 -0
  13. package/dist/function-ref.js +28 -0
  14. package/dist/function-ref.js.map +1 -0
  15. package/dist/function-sugar.d.ts +57 -0
  16. package/dist/function-sugar.d.ts.map +1 -0
  17. package/dist/function-sugar.js +79 -0
  18. package/dist/function-sugar.js.map +1 -0
  19. package/dist/index.d.ts +10 -3
  20. package/dist/index.d.ts.map +1 -1
  21. package/dist/index.js +24 -4
  22. package/dist/index.js.map +1 -1
  23. package/dist/providers/analytics/mixpanel.d.ts.map +1 -1
  24. package/dist/providers/analytics/mixpanel.js +21 -18
  25. package/dist/providers/analytics/mixpanel.js.map +1 -1
  26. package/dist/providers/calendar/cal-com.d.ts.map +1 -1
  27. package/dist/providers/calendar/cal-com.js +10 -10
  28. package/dist/providers/calendar/cal-com.js.map +1 -1
  29. package/dist/providers/calendar/google-calendar.d.ts.map +1 -1
  30. package/dist/providers/calendar/google-calendar.js +4 -4
  31. package/dist/providers/calendar/google-calendar.js.map +1 -1
  32. package/dist/providers/crm/hubspot.d.ts.map +1 -1
  33. package/dist/providers/crm/hubspot.js +107 -85
  34. package/dist/providers/crm/hubspot.js.map +1 -1
  35. package/dist/providers/development/github.d.ts.map +1 -1
  36. package/dist/providers/development/github.js +40 -43
  37. package/dist/providers/development/github.js.map +1 -1
  38. package/dist/providers/ecommerce/shopify.d.ts.map +1 -1
  39. package/dist/providers/ecommerce/shopify.js +79 -62
  40. package/dist/providers/ecommerce/shopify.js.map +1 -1
  41. package/dist/providers/email/resend.d.ts.map +1 -1
  42. package/dist/providers/email/resend.js +20 -16
  43. package/dist/providers/email/resend.js.map +1 -1
  44. package/dist/providers/email/sendgrid.d.ts.map +1 -1
  45. package/dist/providers/email/sendgrid.js +12 -9
  46. package/dist/providers/email/sendgrid.js.map +1 -1
  47. package/dist/providers/finance/stripe.d.ts.map +1 -1
  48. package/dist/providers/finance/stripe.js +44 -42
  49. package/dist/providers/finance/stripe.js.map +1 -1
  50. package/dist/providers/forms/typeform.d.ts.map +1 -1
  51. package/dist/providers/forms/typeform.js +68 -58
  52. package/dist/providers/forms/typeform.js.map +1 -1
  53. package/dist/providers/knowledge/notion.d.ts.map +1 -1
  54. package/dist/providers/knowledge/notion.js +75 -41
  55. package/dist/providers/knowledge/notion.js.map +1 -1
  56. package/dist/providers/marketing/mailchimp.d.ts.map +1 -1
  57. package/dist/providers/marketing/mailchimp.js +74 -61
  58. package/dist/providers/marketing/mailchimp.js.map +1 -1
  59. package/dist/providers/media/cloudinary.d.ts.map +1 -1
  60. package/dist/providers/media/cloudinary.js +30 -28
  61. package/dist/providers/media/cloudinary.js.map +1 -1
  62. package/dist/providers/messaging/slack.d.ts.map +1 -1
  63. package/dist/providers/messaging/slack.js +75 -58
  64. package/dist/providers/messaging/slack.js.map +1 -1
  65. package/dist/providers/messaging/twilio-sms.d.ts.map +1 -1
  66. package/dist/providers/messaging/twilio-sms.js +33 -15
  67. package/dist/providers/messaging/twilio-sms.js.map +1 -1
  68. package/dist/providers/project-management/linear.d.ts.map +1 -1
  69. package/dist/providers/project-management/linear.js +31 -27
  70. package/dist/providers/project-management/linear.js.map +1 -1
  71. package/dist/providers/spreadsheet/google-sheets.d.ts.map +1 -1
  72. package/dist/providers/spreadsheet/google-sheets.js +21 -18
  73. package/dist/providers/spreadsheet/google-sheets.js.map +1 -1
  74. package/dist/providers/spreadsheet/xlsx.d.ts.map +1 -1
  75. package/dist/providers/spreadsheet/xlsx.js +4 -4
  76. package/dist/providers/spreadsheet/xlsx.js.map +1 -1
  77. package/dist/providers/storage/index.js +1 -0
  78. package/dist/providers/storage/index.js.map +1 -1
  79. package/dist/providers/storage/s3.d.ts.map +1 -1
  80. package/dist/providers/storage/s3.js +36 -27
  81. package/dist/providers/storage/s3.js.map +1 -1
  82. package/dist/providers/support/zendesk.d.ts.map +1 -1
  83. package/dist/providers/support/zendesk.js +24 -25
  84. package/dist/providers/support/zendesk.js.map +1 -1
  85. package/dist/providers/tasks/todoist.d.ts.map +1 -1
  86. package/dist/providers/tasks/todoist.js +18 -18
  87. package/dist/providers/tasks/todoist.js.map +1 -1
  88. package/dist/providers/video-conferencing/google-meet.d.ts.map +1 -1
  89. package/dist/providers/video-conferencing/google-meet.js +11 -11
  90. package/dist/providers/video-conferencing/google-meet.js.map +1 -1
  91. package/dist/providers/video-conferencing/jitsi.js +14 -14
  92. package/dist/providers/video-conferencing/jitsi.js.map +1 -1
  93. package/dist/providers/video-conferencing/teams.d.ts.map +1 -1
  94. package/dist/providers/video-conferencing/teams.js +9 -7
  95. package/dist/providers/video-conferencing/teams.js.map +1 -1
  96. package/dist/providers/video-conferencing/zoom.d.ts.map +1 -1
  97. package/dist/providers/video-conferencing/zoom.js +26 -24
  98. package/dist/providers/video-conferencing/zoom.js.map +1 -1
  99. package/dist/tools/data.d.ts.map +1 -1
  100. package/dist/tools/data.js +5 -12
  101. package/dist/tools/data.js.map +1 -1
  102. package/dist/tools/index.d.ts +1 -0
  103. package/dist/tools/index.d.ts.map +1 -1
  104. package/dist/tools/index.js +1 -0
  105. package/dist/tools/index.js.map +1 -1
  106. package/dist/tools/system.d.ts +289 -0
  107. package/dist/tools/system.d.ts.map +1 -0
  108. package/dist/tools/system.js +752 -0
  109. package/dist/tools/system.js.map +1 -0
  110. package/dist/tools/web.d.ts.map +1 -1
  111. package/dist/tools/web.js +22 -10
  112. package/dist/tools/web.js.map +1 -1
  113. package/dist/track-record.d.ts +101 -0
  114. package/dist/track-record.d.ts.map +1 -0
  115. package/dist/track-record.js +17 -0
  116. package/dist/track-record.js.map +1 -0
  117. package/dist/types.d.ts +210 -9
  118. package/dist/types.d.ts.map +1 -1
  119. package/dist/verb-registration.d.ts +122 -0
  120. package/dist/verb-registration.d.ts.map +1 -0
  121. package/dist/verb-registration.js +176 -0
  122. package/dist/verb-registration.js.map +1 -0
  123. package/dist/worker.d.ts +93 -0
  124. package/dist/worker.d.ts.map +1 -0
  125. package/dist/worker.js +315 -0
  126. package/dist/worker.js.map +1 -0
  127. package/dist/wrap.d.ts +89 -0
  128. package/dist/wrap.d.ts.map +1 -0
  129. package/dist/wrap.js +225 -0
  130. package/dist/wrap.js.map +1 -0
  131. package/package.json +21 -4
  132. package/src/client.ts +136 -0
  133. package/src/define.ts +31 -37
  134. package/src/function-ref.ts +264 -0
  135. package/src/function-sugar.ts +134 -0
  136. package/src/index.ts +132 -10
  137. package/src/providers/analytics/mixpanel.ts +19 -18
  138. package/src/providers/calendar/cal-com.ts +29 -18
  139. package/src/providers/calendar/google-calendar.ts +20 -14
  140. package/src/providers/crm/hubspot.ts +225 -99
  141. package/src/providers/development/github.ts +206 -135
  142. package/src/providers/ecommerce/shopify.ts +250 -89
  143. package/src/providers/email/resend.ts +101 -28
  144. package/src/providers/email/sendgrid.ts +12 -9
  145. package/src/providers/finance/stripe.ts +128 -49
  146. package/src/providers/forms/typeform.ts +74 -58
  147. package/src/providers/knowledge/notion.ts +340 -88
  148. package/src/providers/marketing/mailchimp.ts +86 -70
  149. package/src/providers/media/cloudinary.ts +99 -41
  150. package/src/providers/messaging/slack.ts +283 -85
  151. package/src/providers/messaging/twilio-sms.ts +35 -15
  152. package/src/providers/project-management/linear.ts +143 -55
  153. package/src/providers/spreadsheet/google-sheets.ts +222 -56
  154. package/src/providers/spreadsheet/xlsx.ts +47 -16
  155. package/src/providers/storage/s3.ts +119 -47
  156. package/src/providers/support/zendesk.ts +196 -46
  157. package/src/providers/tasks/todoist.ts +20 -26
  158. package/src/providers/video-conferencing/google-meet.ts +17 -20
  159. package/src/providers/video-conferencing/jitsi.ts +14 -14
  160. package/src/providers/video-conferencing/teams.ts +14 -13
  161. package/src/providers/video-conferencing/zoom.ts +54 -49
  162. package/src/tools/data.ts +6 -16
  163. package/src/tools/index.ts +1 -0
  164. package/src/tools/system.ts +887 -0
  165. package/src/tools/web.ts +22 -10
  166. package/src/track-record.ts +106 -0
  167. package/src/types.ts +241 -13
  168. package/src/verb-registration.ts +197 -0
  169. package/src/worker.ts +370 -0
  170. package/src/wrap.ts +260 -0
  171. package/test/client.test.ts +146 -0
  172. package/test/communication-tools-extended.test.ts +734 -0
  173. package/test/data-tools-extended.test.ts +743 -0
  174. package/test/define-extended.test.ts +819 -0
  175. package/test/define.test.ts +150 -41
  176. package/test/entities.test.ts +623 -0
  177. package/test/extended-entities.test.ts +1228 -0
  178. package/test/provider-implementations.test.ts +725 -0
  179. package/test/provider-registry-extended.test.ts +583 -0
  180. package/test/providers/google-sheets.test.ts +851 -0
  181. package/test/providers/helpers.ts +554 -0
  182. package/test/providers/hubspot.test.ts +576 -0
  183. package/test/providers/slack.test.ts +932 -0
  184. package/test/providers/stripe.test.ts +701 -0
  185. package/test/providers.test.ts +578 -0
  186. package/test/system-tools-extended.test.ts +632 -0
  187. package/test/system.test.ts +673 -0
  188. package/test/tools.test.ts +15 -11
  189. package/test/types.test.ts +402 -0
  190. package/test/verb-registration.test.ts +395 -0
  191. package/test/web-tools.test.ts +553 -0
  192. package/test/worker-extended.test.ts +699 -0
  193. package/test/worker.test.ts +576 -0
  194. package/test/wrap.test.ts +366 -0
  195. package/tsconfig.json +3 -13
  196. package/vitest.config.ts +37 -0
  197. package/wrangler.jsonc +9 -0
  198. package/.turbo/turbo-build.log +0 -5
  199. package/dist/providers/voice/vapi.d.ts +0 -27
  200. package/dist/providers/voice/vapi.d.ts.map +0 -1
  201. package/dist/providers/voice/vapi.js +0 -440
  202. package/dist/providers/voice/vapi.js.map +0 -1
  203. package/src/define.js +0 -267
  204. package/src/entities/advertising.js +0 -999
  205. package/src/entities/ai.js +0 -756
  206. package/src/entities/analytics.js +0 -1588
  207. package/src/entities/automation.js +0 -601
  208. package/src/entities/communication.js +0 -1150
  209. package/src/entities/crm.js +0 -1386
  210. package/src/entities/design.js +0 -546
  211. package/src/entities/development.js +0 -2212
  212. package/src/entities/document.js +0 -874
  213. package/src/entities/ecommerce.js +0 -1429
  214. package/src/entities/experiment.js +0 -1039
  215. package/src/entities/finance.js +0 -3478
  216. package/src/entities/forms.js +0 -1892
  217. package/src/entities/hr.js +0 -661
  218. package/src/entities/identity.js +0 -997
  219. package/src/entities/index.js +0 -282
  220. package/src/entities/infrastructure.js +0 -1153
  221. package/src/entities/knowledge.js +0 -1438
  222. package/src/entities/marketing.js +0 -1610
  223. package/src/entities/media.js +0 -1634
  224. package/src/entities/notification.js +0 -1199
  225. package/src/entities/presentation.js +0 -1274
  226. package/src/entities/productivity.js +0 -1317
  227. package/src/entities/project-management.js +0 -1136
  228. package/src/entities/recruiting.js +0 -736
  229. package/src/entities/shipping.js +0 -509
  230. package/src/entities/signature.js +0 -1102
  231. package/src/entities/site.js +0 -222
  232. package/src/entities/spreadsheet.js +0 -1341
  233. package/src/entities/storage.js +0 -1198
  234. package/src/entities/support.js +0 -1166
  235. package/src/entities/video-conferencing.js +0 -1750
  236. package/src/entities/video.js +0 -950
  237. package/src/entities.js +0 -1663
  238. package/src/index.js +0 -74
  239. package/src/providers/analytics/index.js +0 -17
  240. package/src/providers/analytics/mixpanel.js +0 -255
  241. package/src/providers/calendar/cal-com.js +0 -303
  242. package/src/providers/calendar/google-calendar.js +0 -335
  243. package/src/providers/calendar/index.js +0 -20
  244. package/src/providers/crm/hubspot.js +0 -566
  245. package/src/providers/crm/index.js +0 -17
  246. package/src/providers/development/github.js +0 -472
  247. package/src/providers/development/index.js +0 -17
  248. package/src/providers/ecommerce/index.js +0 -17
  249. package/src/providers/ecommerce/shopify.js +0 -378
  250. package/src/providers/email/index.js +0 -20
  251. package/src/providers/email/resend.js +0 -258
  252. package/src/providers/email/sendgrid.js +0 -161
  253. package/src/providers/finance/index.js +0 -17
  254. package/src/providers/finance/stripe.js +0 -549
  255. package/src/providers/forms/index.js +0 -17
  256. package/src/providers/forms/typeform.js +0 -500
  257. package/src/providers/index.js +0 -123
  258. package/src/providers/knowledge/index.js +0 -17
  259. package/src/providers/knowledge/notion.js +0 -389
  260. package/src/providers/marketing/index.js +0 -17
  261. package/src/providers/marketing/mailchimp.js +0 -443
  262. package/src/providers/media/cloudinary.js +0 -318
  263. package/src/providers/media/index.js +0 -17
  264. package/src/providers/messaging/index.js +0 -20
  265. package/src/providers/messaging/slack.js +0 -393
  266. package/src/providers/messaging/twilio-sms.js +0 -249
  267. package/src/providers/project-management/index.js +0 -17
  268. package/src/providers/project-management/linear.js +0 -575
  269. package/src/providers/registry.js +0 -86
  270. package/src/providers/spreadsheet/google-sheets.js +0 -375
  271. package/src/providers/spreadsheet/index.js +0 -20
  272. package/src/providers/spreadsheet/xlsx.js +0 -423
  273. package/src/providers/storage/index.js +0 -24
  274. package/src/providers/storage/s3.js +0 -419
  275. package/src/providers/support/index.js +0 -17
  276. package/src/providers/support/zendesk.js +0 -373
  277. package/src/providers/tasks/index.js +0 -17
  278. package/src/providers/tasks/todoist.js +0 -286
  279. package/src/providers/types.js +0 -9
  280. package/src/providers/video-conferencing/google-meet.js +0 -286
  281. package/src/providers/video-conferencing/index.js +0 -31
  282. package/src/providers/video-conferencing/jitsi.js +0 -254
  283. package/src/providers/video-conferencing/teams.js +0 -270
  284. package/src/providers/video-conferencing/zoom.js +0 -332
  285. package/src/registry.js +0 -128
  286. package/src/tools/communication.js +0 -184
  287. package/src/tools/data.js +0 -205
  288. package/src/tools/index.js +0 -11
  289. package/src/tools/web.js +0 -137
  290. package/src/types.js +0 -10
  291. package/test/define.test.js +0 -306
  292. package/test/registry.test.js +0 -357
  293. package/test/tools.test.js +0 -363
@@ -0,0 +1,395 @@
1
+ /**
2
+ * Tests for `registerToolVerb` / `registerToolVerbs` (aip-47tm).
3
+ *
4
+ * Uses a structural-fake `VerbRegistrationProvider` rather than depending
5
+ * on `MemoryProvider` so the suite stays portable and we can observe
6
+ * exactly which provider methods get called.
7
+ */
8
+
9
+ import { describe, it, expect } from 'vitest'
10
+ import type { Frame, Verb, VerbDefinition } from 'digital-objects'
11
+ import {
12
+ defineTool,
13
+ registerToolVerb,
14
+ registerToolVerbs,
15
+ bootstrapTools,
16
+ VerbRegistrationConflictError,
17
+ type VerbRegistrationProvider,
18
+ } from '../src/index.js'
19
+
20
+ /**
21
+ * Minimal structural fake: stores Verbs in a Map and counts calls so
22
+ * tests can assert idempotency (one defineVerb on second registration
23
+ * with the same shape).
24
+ */
25
+ function fakeProvider(): VerbRegistrationProvider & {
26
+ defineCalls: number
27
+ getCalls: number
28
+ verbs: Map<string, Verb>
29
+ } {
30
+ const verbs = new Map<string, Verb>()
31
+ const provider = {
32
+ defineCalls: 0,
33
+ getCalls: 0,
34
+ verbs,
35
+ async defineVerb(def: VerbDefinition): Promise<Verb> {
36
+ provider.defineCalls++
37
+ const verb: Verb = {
38
+ name: def.name,
39
+ action: def.action ?? def.name,
40
+ act: def.act ?? `${def.name}s`,
41
+ activity: def.activity ?? `${def.name}ing`,
42
+ event: def.event ?? `${def.name}ed`,
43
+ reverseBy: def.reverseBy,
44
+ reverseAt: def.reverseAt,
45
+ reverseIn: def.reverseIn,
46
+ inverse: def.inverse,
47
+ description: def.description,
48
+ frame: def.frame,
49
+ source: def.source ?? 'domain',
50
+ canonical: def.canonical ?? false,
51
+ createdAt: new Date(),
52
+ }
53
+ verbs.set(verb.name, verb)
54
+ return verb
55
+ },
56
+ async getVerb(name: string): Promise<Verb | null> {
57
+ provider.getCalls++
58
+ return verbs.get(name) ?? null
59
+ },
60
+ }
61
+ return provider
62
+ }
63
+
64
+ const sendFrame: Frame = {
65
+ subject: 'Agent',
66
+ object: 'Email',
67
+ recipient: 'Person',
68
+ }
69
+
70
+ describe('registerToolVerb', () => {
71
+ it('registers a Verb when the tool declares one', async () => {
72
+ const provider = fakeProvider()
73
+ const tool = defineTool({
74
+ id: 'communication.email.send',
75
+ name: 'Send Email',
76
+ description: 'Sends an email',
77
+ category: 'communication',
78
+ verb: 'send',
79
+ frame: sendFrame,
80
+ input: { type: 'object', properties: {} },
81
+ handler: async () => ({ ok: true }),
82
+ })
83
+
84
+ const verb = await registerToolVerb(provider, tool)
85
+
86
+ expect(verb).not.toBeNull()
87
+ expect(verb?.name).toBe('send')
88
+ expect(verb?.frame).toEqual(sendFrame)
89
+ expect(provider.defineCalls).toBe(1)
90
+ })
91
+
92
+ it('returns null and skips registration for tools without a verb', async () => {
93
+ const provider = fakeProvider()
94
+ const tool = defineTool({
95
+ id: 'legacy.tool',
96
+ name: 'Legacy',
97
+ description: 'No SVO metadata',
98
+ category: 'data',
99
+ input: { type: 'object', properties: {} },
100
+ handler: async () => ({}),
101
+ })
102
+
103
+ const verb = await registerToolVerb(provider, tool)
104
+
105
+ expect(verb).toBeNull()
106
+ expect(provider.defineCalls).toBe(0)
107
+ expect(provider.getCalls).toBe(0)
108
+ })
109
+
110
+ it('is idempotent for the same verb + frame (no second defineVerb)', async () => {
111
+ const provider = fakeProvider()
112
+ const tool = defineTool({
113
+ id: 'communication.email.send',
114
+ name: 'Send Email',
115
+ description: 'Sends an email',
116
+ category: 'communication',
117
+ verb: 'send',
118
+ frame: sendFrame,
119
+ input: { type: 'object', properties: {} },
120
+ handler: async () => ({ ok: true }),
121
+ })
122
+
123
+ const first = await registerToolVerb(provider, tool)
124
+ const second = await registerToolVerb(provider, tool)
125
+
126
+ expect(first?.name).toBe('send')
127
+ expect(second?.name).toBe('send')
128
+ expect(provider.defineCalls).toBe(1) // only the first call writes
129
+ expect(provider.getCalls).toBe(2) // both calls do the existence check
130
+ })
131
+
132
+ it('treats both-undefined frames as equal (idempotent)', async () => {
133
+ const provider = fakeProvider()
134
+ const tool = defineTool({
135
+ id: 'communication.email.send',
136
+ name: 'Send Email',
137
+ description: 'Sends an email (no frame)',
138
+ category: 'communication',
139
+ verb: 'send',
140
+ input: { type: 'object', properties: {} },
141
+ handler: async () => ({ ok: true }),
142
+ })
143
+
144
+ await registerToolVerb(provider, tool)
145
+ await registerToolVerb(provider, tool)
146
+
147
+ expect(provider.defineCalls).toBe(1)
148
+ })
149
+
150
+ it('throws VerbRegistrationConflictError on conflicting frames', async () => {
151
+ const provider = fakeProvider()
152
+ const toolA = defineTool({
153
+ id: 'a.send',
154
+ name: 'A Send',
155
+ description: 'A',
156
+ category: 'communication',
157
+ verb: 'send',
158
+ frame: sendFrame,
159
+ input: { type: 'object', properties: {} },
160
+ handler: async () => ({}),
161
+ })
162
+ const toolB = defineTool({
163
+ id: 'b.send',
164
+ name: 'B Send',
165
+ description: 'B',
166
+ category: 'communication',
167
+ verb: 'send',
168
+ frame: { subject: 'Agent', object: 'Sms' }, // different shape
169
+ input: { type: 'object', properties: {} },
170
+ handler: async () => ({}),
171
+ })
172
+
173
+ await registerToolVerb(provider, toolA)
174
+
175
+ await expect(registerToolVerb(provider, toolB)).rejects.toBeInstanceOf(
176
+ VerbRegistrationConflictError
177
+ )
178
+ expect(provider.defineCalls).toBe(1) // second tool's defineVerb never called
179
+ })
180
+
181
+ it('treats incoming-undefined as a conflict when the existing verb has a frame', async () => {
182
+ const provider = fakeProvider()
183
+ const framed = defineTool({
184
+ id: 'a.send',
185
+ name: 'A',
186
+ description: 'with frame',
187
+ category: 'communication',
188
+ verb: 'send',
189
+ frame: sendFrame,
190
+ input: { type: 'object', properties: {} },
191
+ handler: async () => ({}),
192
+ })
193
+ const unframed = defineTool({
194
+ id: 'b.send',
195
+ name: 'B',
196
+ description: 'no frame',
197
+ category: 'communication',
198
+ verb: 'send',
199
+ input: { type: 'object', properties: {} },
200
+ handler: async () => ({}),
201
+ })
202
+
203
+ await registerToolVerb(provider, framed)
204
+ await expect(registerToolVerb(provider, unframed)).rejects.toBeInstanceOf(
205
+ VerbRegistrationConflictError
206
+ )
207
+ })
208
+
209
+ it('compares manner arrays as unordered sets (idempotent across reorder)', async () => {
210
+ const provider = fakeProvider()
211
+ const toolA = defineTool({
212
+ id: 'a.move',
213
+ name: 'A',
214
+ description: 'A',
215
+ category: 'system',
216
+ verb: 'move',
217
+ frame: { subject: 'Agent', manner: ['fast', 'quiet'] },
218
+ input: { type: 'object', properties: {} },
219
+ handler: async () => ({}),
220
+ })
221
+ const toolB = defineTool({
222
+ id: 'b.move',
223
+ name: 'B',
224
+ description: 'B',
225
+ category: 'system',
226
+ verb: 'move',
227
+ frame: { subject: 'Agent', manner: ['quiet', 'fast'] },
228
+ input: { type: 'object', properties: {} },
229
+ handler: async () => ({}),
230
+ })
231
+
232
+ await registerToolVerb(provider, toolA)
233
+ await registerToolVerb(provider, toolB) // different order, same set
234
+
235
+ expect(provider.defineCalls).toBe(1)
236
+ })
237
+
238
+ it('detects conflict when manner sets differ', async () => {
239
+ const provider = fakeProvider()
240
+ const toolA = defineTool({
241
+ id: 'a.move',
242
+ name: 'A',
243
+ description: 'A',
244
+ category: 'system',
245
+ verb: 'move',
246
+ frame: { subject: 'Agent', manner: ['fast'] },
247
+ input: { type: 'object', properties: {} },
248
+ handler: async () => ({}),
249
+ })
250
+ const toolB = defineTool({
251
+ id: 'b.move',
252
+ name: 'B',
253
+ description: 'B',
254
+ category: 'system',
255
+ verb: 'move',
256
+ frame: { subject: 'Agent', manner: ['fast', 'slow'] },
257
+ input: { type: 'object', properties: {} },
258
+ handler: async () => ({}),
259
+ })
260
+
261
+ await registerToolVerb(provider, toolA)
262
+ await expect(registerToolVerb(provider, toolB)).rejects.toBeInstanceOf(
263
+ VerbRegistrationConflictError
264
+ )
265
+ })
266
+
267
+ it('passes description through to the registered Verb', async () => {
268
+ const provider = fakeProvider()
269
+ const tool = defineTool({
270
+ id: 'communication.email.send',
271
+ name: 'Send Email',
272
+ description: 'Sends an email',
273
+ category: 'communication',
274
+ verb: 'send',
275
+ frame: sendFrame,
276
+ input: { type: 'object', properties: {} },
277
+ handler: async () => ({}),
278
+ })
279
+
280
+ await registerToolVerb(provider, tool)
281
+ const verb = provider.verbs.get('send')
282
+
283
+ expect(verb?.description).toBe('Sends an email')
284
+ })
285
+ })
286
+
287
+ describe('registerToolVerbs (bulk)', () => {
288
+ it('registers all SVO-aware tools and skips legacy tools', async () => {
289
+ const provider = fakeProvider()
290
+ const tools = [
291
+ defineTool({
292
+ id: 'communication.email.send',
293
+ name: 'Email',
294
+ description: 'Email',
295
+ category: 'communication',
296
+ verb: 'send',
297
+ frame: sendFrame,
298
+ input: { type: 'object', properties: {} },
299
+ handler: async () => ({}),
300
+ }),
301
+ defineTool({
302
+ id: 'legacy.tool',
303
+ name: 'Legacy',
304
+ description: 'No verb',
305
+ category: 'data',
306
+ input: { type: 'object', properties: {} },
307
+ handler: async () => ({}),
308
+ }),
309
+ defineTool({
310
+ id: 'media.transcribe',
311
+ name: 'Transcribe',
312
+ description: 'Transcribe',
313
+ category: 'media',
314
+ verb: 'transcribe',
315
+ frame: { subject: 'Agent', object: 'Audio' },
316
+ input: { type: 'object', properties: {} },
317
+ handler: async () => ({}),
318
+ }),
319
+ ]
320
+
321
+ const registered = await registerToolVerbs(provider, tools)
322
+
323
+ expect(registered).toHaveLength(2)
324
+ expect(registered.map((v) => v.name).sort()).toEqual(['send', 'transcribe'])
325
+ expect(provider.defineCalls).toBe(2)
326
+ })
327
+
328
+ it('is idempotent across re-runs (safe to call at startup repeatedly)', async () => {
329
+ const provider = fakeProvider()
330
+ const tools = [
331
+ defineTool({
332
+ id: 'communication.email.send',
333
+ name: 'Email',
334
+ description: 'Email',
335
+ category: 'communication',
336
+ verb: 'send',
337
+ frame: sendFrame,
338
+ input: { type: 'object', properties: {} },
339
+ handler: async () => ({}),
340
+ }),
341
+ ]
342
+
343
+ await registerToolVerbs(provider, tools)
344
+ await registerToolVerbs(provider, tools)
345
+
346
+ expect(provider.defineCalls).toBe(1)
347
+ })
348
+
349
+ it('bootstrapTools is an alias of registerToolVerbs', () => {
350
+ expect(bootstrapTools).toBe(registerToolVerbs)
351
+ })
352
+
353
+ it('throws on conflict and stops at the first conflicting tool', async () => {
354
+ const provider = fakeProvider()
355
+ const tools = [
356
+ defineTool({
357
+ id: 'a.send',
358
+ name: 'A',
359
+ description: 'A',
360
+ category: 'communication',
361
+ verb: 'send',
362
+ frame: sendFrame,
363
+ input: { type: 'object', properties: {} },
364
+ handler: async () => ({}),
365
+ }),
366
+ defineTool({
367
+ id: 'b.send',
368
+ name: 'B',
369
+ description: 'B',
370
+ category: 'communication',
371
+ verb: 'send',
372
+ frame: { subject: 'Agent', object: 'Sms' },
373
+ input: { type: 'object', properties: {} },
374
+ handler: async () => ({}),
375
+ }),
376
+ defineTool({
377
+ id: 'c.never',
378
+ name: 'C',
379
+ description: 'C',
380
+ category: 'data',
381
+ verb: 'never',
382
+ input: { type: 'object', properties: {} },
383
+ handler: async () => ({}),
384
+ }),
385
+ ]
386
+
387
+ await expect(registerToolVerbs(provider, tools)).rejects.toBeInstanceOf(
388
+ VerbRegistrationConflictError
389
+ )
390
+ // First tool succeeded, conflicting second halted the loop;
391
+ // third tool was never reached.
392
+ expect(provider.defineCalls).toBe(1)
393
+ expect(provider.verbs.has('never')).toBe(false)
394
+ })
395
+ })