digital-tools 2.1.3 → 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 (294) hide show
  1. package/CHANGELOG.md +9 -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 +21 -11
  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 +31 -14
  132. package/src/client.ts +136 -0
  133. package/src/define.ts +30 -24
  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 -4
  199. package/LICENSE +0 -21
  200. package/dist/providers/voice/vapi.d.ts +0 -27
  201. package/dist/providers/voice/vapi.d.ts.map +0 -1
  202. package/dist/providers/voice/vapi.js +0 -440
  203. package/dist/providers/voice/vapi.js.map +0 -1
  204. package/src/define.js +0 -259
  205. package/src/entities/advertising.js +0 -999
  206. package/src/entities/ai.js +0 -756
  207. package/src/entities/analytics.js +0 -1588
  208. package/src/entities/automation.js +0 -601
  209. package/src/entities/communication.js +0 -1150
  210. package/src/entities/crm.js +0 -1386
  211. package/src/entities/design.js +0 -546
  212. package/src/entities/development.js +0 -2212
  213. package/src/entities/document.js +0 -874
  214. package/src/entities/ecommerce.js +0 -1429
  215. package/src/entities/experiment.js +0 -1039
  216. package/src/entities/finance.js +0 -3478
  217. package/src/entities/forms.js +0 -1892
  218. package/src/entities/hr.js +0 -661
  219. package/src/entities/identity.js +0 -997
  220. package/src/entities/index.js +0 -282
  221. package/src/entities/infrastructure.js +0 -1153
  222. package/src/entities/knowledge.js +0 -1438
  223. package/src/entities/marketing.js +0 -1610
  224. package/src/entities/media.js +0 -1634
  225. package/src/entities/notification.js +0 -1199
  226. package/src/entities/presentation.js +0 -1274
  227. package/src/entities/productivity.js +0 -1317
  228. package/src/entities/project-management.js +0 -1136
  229. package/src/entities/recruiting.js +0 -736
  230. package/src/entities/shipping.js +0 -509
  231. package/src/entities/signature.js +0 -1102
  232. package/src/entities/site.js +0 -222
  233. package/src/entities/spreadsheet.js +0 -1341
  234. package/src/entities/storage.js +0 -1198
  235. package/src/entities/support.js +0 -1166
  236. package/src/entities/video-conferencing.js +0 -1750
  237. package/src/entities/video.js +0 -950
  238. package/src/entities.js +0 -1663
  239. package/src/index.js +0 -74
  240. package/src/providers/analytics/index.js +0 -17
  241. package/src/providers/analytics/mixpanel.js +0 -255
  242. package/src/providers/calendar/cal-com.js +0 -303
  243. package/src/providers/calendar/google-calendar.js +0 -335
  244. package/src/providers/calendar/index.js +0 -20
  245. package/src/providers/crm/hubspot.js +0 -566
  246. package/src/providers/crm/index.js +0 -17
  247. package/src/providers/development/github.js +0 -472
  248. package/src/providers/development/index.js +0 -17
  249. package/src/providers/ecommerce/index.js +0 -17
  250. package/src/providers/ecommerce/shopify.js +0 -378
  251. package/src/providers/email/index.js +0 -20
  252. package/src/providers/email/resend.js +0 -258
  253. package/src/providers/email/sendgrid.js +0 -161
  254. package/src/providers/finance/index.js +0 -17
  255. package/src/providers/finance/stripe.js +0 -549
  256. package/src/providers/forms/index.js +0 -17
  257. package/src/providers/forms/typeform.js +0 -500
  258. package/src/providers/index.js +0 -123
  259. package/src/providers/knowledge/index.js +0 -17
  260. package/src/providers/knowledge/notion.js +0 -389
  261. package/src/providers/marketing/index.js +0 -17
  262. package/src/providers/marketing/mailchimp.js +0 -443
  263. package/src/providers/media/cloudinary.js +0 -318
  264. package/src/providers/media/index.js +0 -17
  265. package/src/providers/messaging/index.js +0 -20
  266. package/src/providers/messaging/slack.js +0 -393
  267. package/src/providers/messaging/twilio-sms.js +0 -249
  268. package/src/providers/project-management/index.js +0 -17
  269. package/src/providers/project-management/linear.js +0 -575
  270. package/src/providers/registry.js +0 -86
  271. package/src/providers/spreadsheet/google-sheets.js +0 -375
  272. package/src/providers/spreadsheet/index.js +0 -20
  273. package/src/providers/spreadsheet/xlsx.js +0 -423
  274. package/src/providers/storage/index.js +0 -24
  275. package/src/providers/storage/s3.js +0 -419
  276. package/src/providers/support/index.js +0 -17
  277. package/src/providers/support/zendesk.js +0 -373
  278. package/src/providers/tasks/index.js +0 -17
  279. package/src/providers/tasks/todoist.js +0 -286
  280. package/src/providers/types.js +0 -9
  281. package/src/providers/video-conferencing/google-meet.js +0 -286
  282. package/src/providers/video-conferencing/index.js +0 -31
  283. package/src/providers/video-conferencing/jitsi.js +0 -254
  284. package/src/providers/video-conferencing/teams.js +0 -270
  285. package/src/providers/video-conferencing/zoom.js +0 -332
  286. package/src/registry.js +0 -128
  287. package/src/tools/communication.js +0 -184
  288. package/src/tools/data.js +0 -205
  289. package/src/tools/index.js +0 -11
  290. package/src/tools/web.js +0 -137
  291. package/src/types.js +0 -10
  292. package/test/define.test.js +0 -306
  293. package/test/registry.test.js +0 -357
  294. 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
+ })