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.
- package/CHANGELOG.md +17 -0
- package/README.md +2 -0
- package/dist/client.d.ts +109 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +69 -0
- package/dist/client.js.map +1 -0
- package/dist/define.d.ts +2 -2
- package/dist/define.d.ts.map +1 -1
- package/dist/define.js +22 -20
- package/dist/define.js.map +1 -1
- package/dist/function-ref.d.ts +229 -0
- package/dist/function-ref.d.ts.map +1 -0
- package/dist/function-ref.js +28 -0
- package/dist/function-ref.js.map +1 -0
- package/dist/function-sugar.d.ts +57 -0
- package/dist/function-sugar.d.ts.map +1 -0
- package/dist/function-sugar.js +79 -0
- package/dist/function-sugar.js.map +1 -0
- package/dist/index.d.ts +10 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +24 -4
- package/dist/index.js.map +1 -1
- package/dist/providers/analytics/mixpanel.d.ts.map +1 -1
- package/dist/providers/analytics/mixpanel.js +21 -18
- package/dist/providers/analytics/mixpanel.js.map +1 -1
- package/dist/providers/calendar/cal-com.d.ts.map +1 -1
- package/dist/providers/calendar/cal-com.js +10 -10
- package/dist/providers/calendar/cal-com.js.map +1 -1
- package/dist/providers/calendar/google-calendar.d.ts.map +1 -1
- package/dist/providers/calendar/google-calendar.js +4 -4
- package/dist/providers/calendar/google-calendar.js.map +1 -1
- package/dist/providers/crm/hubspot.d.ts.map +1 -1
- package/dist/providers/crm/hubspot.js +107 -85
- package/dist/providers/crm/hubspot.js.map +1 -1
- package/dist/providers/development/github.d.ts.map +1 -1
- package/dist/providers/development/github.js +40 -43
- package/dist/providers/development/github.js.map +1 -1
- package/dist/providers/ecommerce/shopify.d.ts.map +1 -1
- package/dist/providers/ecommerce/shopify.js +79 -62
- package/dist/providers/ecommerce/shopify.js.map +1 -1
- package/dist/providers/email/resend.d.ts.map +1 -1
- package/dist/providers/email/resend.js +20 -16
- package/dist/providers/email/resend.js.map +1 -1
- package/dist/providers/email/sendgrid.d.ts.map +1 -1
- package/dist/providers/email/sendgrid.js +12 -9
- package/dist/providers/email/sendgrid.js.map +1 -1
- package/dist/providers/finance/stripe.d.ts.map +1 -1
- package/dist/providers/finance/stripe.js +44 -42
- package/dist/providers/finance/stripe.js.map +1 -1
- package/dist/providers/forms/typeform.d.ts.map +1 -1
- package/dist/providers/forms/typeform.js +68 -58
- package/dist/providers/forms/typeform.js.map +1 -1
- package/dist/providers/knowledge/notion.d.ts.map +1 -1
- package/dist/providers/knowledge/notion.js +75 -41
- package/dist/providers/knowledge/notion.js.map +1 -1
- package/dist/providers/marketing/mailchimp.d.ts.map +1 -1
- package/dist/providers/marketing/mailchimp.js +74 -61
- package/dist/providers/marketing/mailchimp.js.map +1 -1
- package/dist/providers/media/cloudinary.d.ts.map +1 -1
- package/dist/providers/media/cloudinary.js +30 -28
- package/dist/providers/media/cloudinary.js.map +1 -1
- package/dist/providers/messaging/slack.d.ts.map +1 -1
- package/dist/providers/messaging/slack.js +75 -58
- package/dist/providers/messaging/slack.js.map +1 -1
- package/dist/providers/messaging/twilio-sms.d.ts.map +1 -1
- package/dist/providers/messaging/twilio-sms.js +33 -15
- package/dist/providers/messaging/twilio-sms.js.map +1 -1
- package/dist/providers/project-management/linear.d.ts.map +1 -1
- package/dist/providers/project-management/linear.js +31 -27
- package/dist/providers/project-management/linear.js.map +1 -1
- package/dist/providers/spreadsheet/google-sheets.d.ts.map +1 -1
- package/dist/providers/spreadsheet/google-sheets.js +21 -18
- package/dist/providers/spreadsheet/google-sheets.js.map +1 -1
- package/dist/providers/spreadsheet/xlsx.d.ts.map +1 -1
- package/dist/providers/spreadsheet/xlsx.js +4 -4
- package/dist/providers/spreadsheet/xlsx.js.map +1 -1
- package/dist/providers/storage/index.js +1 -0
- package/dist/providers/storage/index.js.map +1 -1
- package/dist/providers/storage/s3.d.ts.map +1 -1
- package/dist/providers/storage/s3.js +36 -27
- package/dist/providers/storage/s3.js.map +1 -1
- package/dist/providers/support/zendesk.d.ts.map +1 -1
- package/dist/providers/support/zendesk.js +24 -25
- package/dist/providers/support/zendesk.js.map +1 -1
- package/dist/providers/tasks/todoist.d.ts.map +1 -1
- package/dist/providers/tasks/todoist.js +18 -18
- package/dist/providers/tasks/todoist.js.map +1 -1
- package/dist/providers/video-conferencing/google-meet.d.ts.map +1 -1
- package/dist/providers/video-conferencing/google-meet.js +11 -11
- package/dist/providers/video-conferencing/google-meet.js.map +1 -1
- package/dist/providers/video-conferencing/jitsi.js +14 -14
- package/dist/providers/video-conferencing/jitsi.js.map +1 -1
- package/dist/providers/video-conferencing/teams.d.ts.map +1 -1
- package/dist/providers/video-conferencing/teams.js +9 -7
- package/dist/providers/video-conferencing/teams.js.map +1 -1
- package/dist/providers/video-conferencing/zoom.d.ts.map +1 -1
- package/dist/providers/video-conferencing/zoom.js +26 -24
- package/dist/providers/video-conferencing/zoom.js.map +1 -1
- package/dist/tools/data.d.ts.map +1 -1
- package/dist/tools/data.js +5 -12
- package/dist/tools/data.js.map +1 -1
- package/dist/tools/index.d.ts +1 -0
- package/dist/tools/index.d.ts.map +1 -1
- package/dist/tools/index.js +1 -0
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/system.d.ts +289 -0
- package/dist/tools/system.d.ts.map +1 -0
- package/dist/tools/system.js +752 -0
- package/dist/tools/system.js.map +1 -0
- package/dist/tools/web.d.ts.map +1 -1
- package/dist/tools/web.js +22 -10
- package/dist/tools/web.js.map +1 -1
- package/dist/track-record.d.ts +101 -0
- package/dist/track-record.d.ts.map +1 -0
- package/dist/track-record.js +17 -0
- package/dist/track-record.js.map +1 -0
- package/dist/types.d.ts +210 -9
- package/dist/types.d.ts.map +1 -1
- package/dist/verb-registration.d.ts +122 -0
- package/dist/verb-registration.d.ts.map +1 -0
- package/dist/verb-registration.js +176 -0
- package/dist/verb-registration.js.map +1 -0
- package/dist/worker.d.ts +93 -0
- package/dist/worker.d.ts.map +1 -0
- package/dist/worker.js +315 -0
- package/dist/worker.js.map +1 -0
- package/dist/wrap.d.ts +89 -0
- package/dist/wrap.d.ts.map +1 -0
- package/dist/wrap.js +225 -0
- package/dist/wrap.js.map +1 -0
- package/package.json +21 -4
- package/src/client.ts +136 -0
- package/src/define.ts +31 -37
- package/src/function-ref.ts +264 -0
- package/src/function-sugar.ts +134 -0
- package/src/index.ts +132 -10
- package/src/providers/analytics/mixpanel.ts +19 -18
- package/src/providers/calendar/cal-com.ts +29 -18
- package/src/providers/calendar/google-calendar.ts +20 -14
- package/src/providers/crm/hubspot.ts +225 -99
- package/src/providers/development/github.ts +206 -135
- package/src/providers/ecommerce/shopify.ts +250 -89
- package/src/providers/email/resend.ts +101 -28
- package/src/providers/email/sendgrid.ts +12 -9
- package/src/providers/finance/stripe.ts +128 -49
- package/src/providers/forms/typeform.ts +74 -58
- package/src/providers/knowledge/notion.ts +340 -88
- package/src/providers/marketing/mailchimp.ts +86 -70
- package/src/providers/media/cloudinary.ts +99 -41
- package/src/providers/messaging/slack.ts +283 -85
- package/src/providers/messaging/twilio-sms.ts +35 -15
- package/src/providers/project-management/linear.ts +143 -55
- package/src/providers/spreadsheet/google-sheets.ts +222 -56
- package/src/providers/spreadsheet/xlsx.ts +47 -16
- package/src/providers/storage/s3.ts +119 -47
- package/src/providers/support/zendesk.ts +196 -46
- package/src/providers/tasks/todoist.ts +20 -26
- package/src/providers/video-conferencing/google-meet.ts +17 -20
- package/src/providers/video-conferencing/jitsi.ts +14 -14
- package/src/providers/video-conferencing/teams.ts +14 -13
- package/src/providers/video-conferencing/zoom.ts +54 -49
- package/src/tools/data.ts +6 -16
- package/src/tools/index.ts +1 -0
- package/src/tools/system.ts +887 -0
- package/src/tools/web.ts +22 -10
- package/src/track-record.ts +106 -0
- package/src/types.ts +241 -13
- package/src/verb-registration.ts +197 -0
- package/src/worker.ts +370 -0
- package/src/wrap.ts +260 -0
- package/test/client.test.ts +146 -0
- package/test/communication-tools-extended.test.ts +734 -0
- package/test/data-tools-extended.test.ts +743 -0
- package/test/define-extended.test.ts +819 -0
- package/test/define.test.ts +150 -41
- package/test/entities.test.ts +623 -0
- package/test/extended-entities.test.ts +1228 -0
- package/test/provider-implementations.test.ts +725 -0
- package/test/provider-registry-extended.test.ts +583 -0
- package/test/providers/google-sheets.test.ts +851 -0
- package/test/providers/helpers.ts +554 -0
- package/test/providers/hubspot.test.ts +576 -0
- package/test/providers/slack.test.ts +932 -0
- package/test/providers/stripe.test.ts +701 -0
- package/test/providers.test.ts +578 -0
- package/test/system-tools-extended.test.ts +632 -0
- package/test/system.test.ts +673 -0
- package/test/tools.test.ts +15 -11
- package/test/types.test.ts +402 -0
- package/test/verb-registration.test.ts +395 -0
- package/test/web-tools.test.ts +553 -0
- package/test/worker-extended.test.ts +699 -0
- package/test/worker.test.ts +576 -0
- package/test/wrap.test.ts +366 -0
- package/tsconfig.json +3 -13
- package/vitest.config.ts +37 -0
- package/wrangler.jsonc +9 -0
- package/.turbo/turbo-build.log +0 -5
- package/dist/providers/voice/vapi.d.ts +0 -27
- package/dist/providers/voice/vapi.d.ts.map +0 -1
- package/dist/providers/voice/vapi.js +0 -440
- package/dist/providers/voice/vapi.js.map +0 -1
- package/src/define.js +0 -267
- package/src/entities/advertising.js +0 -999
- package/src/entities/ai.js +0 -756
- package/src/entities/analytics.js +0 -1588
- package/src/entities/automation.js +0 -601
- package/src/entities/communication.js +0 -1150
- package/src/entities/crm.js +0 -1386
- package/src/entities/design.js +0 -546
- package/src/entities/development.js +0 -2212
- package/src/entities/document.js +0 -874
- package/src/entities/ecommerce.js +0 -1429
- package/src/entities/experiment.js +0 -1039
- package/src/entities/finance.js +0 -3478
- package/src/entities/forms.js +0 -1892
- package/src/entities/hr.js +0 -661
- package/src/entities/identity.js +0 -997
- package/src/entities/index.js +0 -282
- package/src/entities/infrastructure.js +0 -1153
- package/src/entities/knowledge.js +0 -1438
- package/src/entities/marketing.js +0 -1610
- package/src/entities/media.js +0 -1634
- package/src/entities/notification.js +0 -1199
- package/src/entities/presentation.js +0 -1274
- package/src/entities/productivity.js +0 -1317
- package/src/entities/project-management.js +0 -1136
- package/src/entities/recruiting.js +0 -736
- package/src/entities/shipping.js +0 -509
- package/src/entities/signature.js +0 -1102
- package/src/entities/site.js +0 -222
- package/src/entities/spreadsheet.js +0 -1341
- package/src/entities/storage.js +0 -1198
- package/src/entities/support.js +0 -1166
- package/src/entities/video-conferencing.js +0 -1750
- package/src/entities/video.js +0 -950
- package/src/entities.js +0 -1663
- package/src/index.js +0 -74
- package/src/providers/analytics/index.js +0 -17
- package/src/providers/analytics/mixpanel.js +0 -255
- package/src/providers/calendar/cal-com.js +0 -303
- package/src/providers/calendar/google-calendar.js +0 -335
- package/src/providers/calendar/index.js +0 -20
- package/src/providers/crm/hubspot.js +0 -566
- package/src/providers/crm/index.js +0 -17
- package/src/providers/development/github.js +0 -472
- package/src/providers/development/index.js +0 -17
- package/src/providers/ecommerce/index.js +0 -17
- package/src/providers/ecommerce/shopify.js +0 -378
- package/src/providers/email/index.js +0 -20
- package/src/providers/email/resend.js +0 -258
- package/src/providers/email/sendgrid.js +0 -161
- package/src/providers/finance/index.js +0 -17
- package/src/providers/finance/stripe.js +0 -549
- package/src/providers/forms/index.js +0 -17
- package/src/providers/forms/typeform.js +0 -500
- package/src/providers/index.js +0 -123
- package/src/providers/knowledge/index.js +0 -17
- package/src/providers/knowledge/notion.js +0 -389
- package/src/providers/marketing/index.js +0 -17
- package/src/providers/marketing/mailchimp.js +0 -443
- package/src/providers/media/cloudinary.js +0 -318
- package/src/providers/media/index.js +0 -17
- package/src/providers/messaging/index.js +0 -20
- package/src/providers/messaging/slack.js +0 -393
- package/src/providers/messaging/twilio-sms.js +0 -249
- package/src/providers/project-management/index.js +0 -17
- package/src/providers/project-management/linear.js +0 -575
- package/src/providers/registry.js +0 -86
- package/src/providers/spreadsheet/google-sheets.js +0 -375
- package/src/providers/spreadsheet/index.js +0 -20
- package/src/providers/spreadsheet/xlsx.js +0 -423
- package/src/providers/storage/index.js +0 -24
- package/src/providers/storage/s3.js +0 -419
- package/src/providers/support/index.js +0 -17
- package/src/providers/support/zendesk.js +0 -373
- package/src/providers/tasks/index.js +0 -17
- package/src/providers/tasks/todoist.js +0 -286
- package/src/providers/types.js +0 -9
- package/src/providers/video-conferencing/google-meet.js +0 -286
- package/src/providers/video-conferencing/index.js +0 -31
- package/src/providers/video-conferencing/jitsi.js +0 -254
- package/src/providers/video-conferencing/teams.js +0 -270
- package/src/providers/video-conferencing/zoom.js +0 -332
- package/src/registry.js +0 -128
- package/src/tools/communication.js +0 -184
- package/src/tools/data.js +0 -205
- package/src/tools/index.js +0 -11
- package/src/tools/web.js +0 -137
- package/src/types.js +0 -10
- package/test/define.test.js +0 -306
- package/test/registry.test.js +0 -357
- package/test/tools.test.js +0 -363
|
@@ -0,0 +1,699 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Extended Tests for Worker/ToolServiceCore
|
|
3
|
+
*
|
|
4
|
+
* Additional comprehensive tests for ToolServiceCore functionality,
|
|
5
|
+
* including fromMCP, clear, and edge cases.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it, expect, beforeEach } from 'vitest'
|
|
9
|
+
import { ToolServiceCore } from '../src/worker.js'
|
|
10
|
+
import type { MCPTool, AnyTool } from '../src/types.js'
|
|
11
|
+
|
|
12
|
+
describe('ToolServiceCore - fromMCP', () => {
|
|
13
|
+
let service: ToolServiceCore
|
|
14
|
+
|
|
15
|
+
beforeEach(() => {
|
|
16
|
+
service = new ToolServiceCore()
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
describe('basic conversion', () => {
|
|
20
|
+
it('converts MCP tool to internal format', () => {
|
|
21
|
+
const mcpTool: MCPTool = {
|
|
22
|
+
name: 'test.mcp.tool',
|
|
23
|
+
description: 'A tool from MCP',
|
|
24
|
+
inputSchema: {
|
|
25
|
+
type: 'object',
|
|
26
|
+
properties: {
|
|
27
|
+
input: { type: 'string', description: 'Input value' },
|
|
28
|
+
},
|
|
29
|
+
required: ['input'],
|
|
30
|
+
},
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const handler = async () => ({ result: 'done' })
|
|
34
|
+
const tool = service.fromMCP(mcpTool, handler)
|
|
35
|
+
|
|
36
|
+
expect(tool.id).toBe('test.mcp.tool')
|
|
37
|
+
expect(tool.name).toBe('test.mcp.tool')
|
|
38
|
+
expect(tool.description).toBe('A tool from MCP')
|
|
39
|
+
})
|
|
40
|
+
|
|
41
|
+
it('converts parameters from inputSchema', () => {
|
|
42
|
+
const mcpTool: MCPTool = {
|
|
43
|
+
name: 'params.test',
|
|
44
|
+
description: 'Test parameters',
|
|
45
|
+
inputSchema: {
|
|
46
|
+
type: 'object',
|
|
47
|
+
properties: {
|
|
48
|
+
name: { type: 'string', description: 'Name param' },
|
|
49
|
+
count: { type: 'number', description: 'Count param' },
|
|
50
|
+
},
|
|
51
|
+
required: ['name'],
|
|
52
|
+
},
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const tool = service.fromMCP(mcpTool, async () => ({}))
|
|
56
|
+
|
|
57
|
+
expect(tool.parameters).toHaveLength(2)
|
|
58
|
+
|
|
59
|
+
const nameParam = tool.parameters.find((p) => p.name === 'name')
|
|
60
|
+
const countParam = tool.parameters.find((p) => p.name === 'count')
|
|
61
|
+
|
|
62
|
+
expect(nameParam?.required).toBe(true)
|
|
63
|
+
expect(nameParam?.description).toBe('Name param')
|
|
64
|
+
expect(countParam?.required).toBe(false)
|
|
65
|
+
})
|
|
66
|
+
|
|
67
|
+
it('handles missing description in properties', () => {
|
|
68
|
+
const mcpTool: MCPTool = {
|
|
69
|
+
name: 'no.desc',
|
|
70
|
+
description: 'No param descriptions',
|
|
71
|
+
inputSchema: {
|
|
72
|
+
type: 'object',
|
|
73
|
+
properties: {
|
|
74
|
+
value: { type: 'string' },
|
|
75
|
+
},
|
|
76
|
+
},
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
const tool = service.fromMCP(mcpTool, async () => ({}))
|
|
80
|
+
const valueParam = tool.parameters.find((p) => p.name === 'value')
|
|
81
|
+
|
|
82
|
+
expect(valueParam?.description).toBe('Parameter: value')
|
|
83
|
+
})
|
|
84
|
+
|
|
85
|
+
it('defaults category to integration', () => {
|
|
86
|
+
const mcpTool: MCPTool = {
|
|
87
|
+
name: 'default.category',
|
|
88
|
+
description: 'Test default category',
|
|
89
|
+
inputSchema: { type: 'object' },
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
const tool = service.fromMCP(mcpTool, async () => ({}))
|
|
93
|
+
|
|
94
|
+
expect(tool.category).toBe('integration')
|
|
95
|
+
})
|
|
96
|
+
})
|
|
97
|
+
|
|
98
|
+
describe('options handling', () => {
|
|
99
|
+
it('sets custom category', () => {
|
|
100
|
+
const mcpTool: MCPTool = {
|
|
101
|
+
name: 'custom.category',
|
|
102
|
+
description: 'Custom category',
|
|
103
|
+
inputSchema: { type: 'object' },
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
const tool = service.fromMCP(mcpTool, async () => ({}), {
|
|
107
|
+
category: 'data',
|
|
108
|
+
})
|
|
109
|
+
|
|
110
|
+
expect(tool.category).toBe('data')
|
|
111
|
+
})
|
|
112
|
+
|
|
113
|
+
it('sets subcategory', () => {
|
|
114
|
+
const mcpTool: MCPTool = {
|
|
115
|
+
name: 'with.subcategory',
|
|
116
|
+
description: 'With subcategory',
|
|
117
|
+
inputSchema: { type: 'object' },
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
const tool = service.fromMCP(mcpTool, async () => ({}), {
|
|
121
|
+
category: 'data',
|
|
122
|
+
subcategory: 'transform',
|
|
123
|
+
})
|
|
124
|
+
|
|
125
|
+
expect(tool.subcategory).toBe('transform')
|
|
126
|
+
})
|
|
127
|
+
|
|
128
|
+
it('sets tags', () => {
|
|
129
|
+
const mcpTool: MCPTool = {
|
|
130
|
+
name: 'with.tags',
|
|
131
|
+
description: 'With tags',
|
|
132
|
+
inputSchema: { type: 'object' },
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
const tool = service.fromMCP(mcpTool, async () => ({}), {
|
|
136
|
+
tags: ['custom', 'mcp', 'imported'],
|
|
137
|
+
})
|
|
138
|
+
|
|
139
|
+
expect(tool.tags).toContain('custom')
|
|
140
|
+
expect(tool.tags).toContain('mcp')
|
|
141
|
+
expect(tool.tags).toContain('imported')
|
|
142
|
+
})
|
|
143
|
+
|
|
144
|
+
it('handles empty options', () => {
|
|
145
|
+
const mcpTool: MCPTool = {
|
|
146
|
+
name: 'empty.options',
|
|
147
|
+
description: 'Empty options',
|
|
148
|
+
inputSchema: { type: 'object' },
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
const tool = service.fromMCP(mcpTool, async () => ({}), {})
|
|
152
|
+
|
|
153
|
+
expect(tool.id).toBe('empty.options')
|
|
154
|
+
expect(tool.category).toBe('integration')
|
|
155
|
+
})
|
|
156
|
+
})
|
|
157
|
+
|
|
158
|
+
describe('handler preservation', () => {
|
|
159
|
+
it('preserves handler function', async () => {
|
|
160
|
+
const mcpTool: MCPTool = {
|
|
161
|
+
name: 'handler.test',
|
|
162
|
+
description: 'Handler test',
|
|
163
|
+
inputSchema: {
|
|
164
|
+
type: 'object',
|
|
165
|
+
properties: { x: { type: 'number' } },
|
|
166
|
+
},
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
const handler = async (input: { x: number }) => ({ doubled: input.x * 2 })
|
|
170
|
+
const tool = service.fromMCP(mcpTool, handler)
|
|
171
|
+
|
|
172
|
+
const result = await tool.handler({ x: 5 })
|
|
173
|
+
expect(result).toEqual({ doubled: 10 })
|
|
174
|
+
})
|
|
175
|
+
|
|
176
|
+
it('handler can be registered and executed', async () => {
|
|
177
|
+
const mcpTool: MCPTool = {
|
|
178
|
+
name: 'register.mcp',
|
|
179
|
+
description: 'Registered MCP tool',
|
|
180
|
+
inputSchema: {
|
|
181
|
+
type: 'object',
|
|
182
|
+
properties: { value: { type: 'string' } },
|
|
183
|
+
},
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
const tool = service.fromMCP(mcpTool, async (input: { value: string }) => ({
|
|
187
|
+
upper: input.value.toUpperCase(),
|
|
188
|
+
}))
|
|
189
|
+
|
|
190
|
+
service.register(tool)
|
|
191
|
+
const result = await service.executeTool('register.mcp', { value: 'hello' })
|
|
192
|
+
|
|
193
|
+
expect(result).toEqual({ upper: 'HELLO' })
|
|
194
|
+
})
|
|
195
|
+
})
|
|
196
|
+
|
|
197
|
+
describe('edge cases', () => {
|
|
198
|
+
it('handles empty properties', () => {
|
|
199
|
+
const mcpTool: MCPTool = {
|
|
200
|
+
name: 'empty.props',
|
|
201
|
+
description: 'Empty properties',
|
|
202
|
+
inputSchema: {
|
|
203
|
+
type: 'object',
|
|
204
|
+
properties: {},
|
|
205
|
+
},
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const tool = service.fromMCP(mcpTool, async () => ({}))
|
|
209
|
+
|
|
210
|
+
expect(tool.parameters).toHaveLength(0)
|
|
211
|
+
})
|
|
212
|
+
|
|
213
|
+
it('handles missing properties', () => {
|
|
214
|
+
const mcpTool: MCPTool = {
|
|
215
|
+
name: 'no.props',
|
|
216
|
+
description: 'No properties',
|
|
217
|
+
inputSchema: {
|
|
218
|
+
type: 'object',
|
|
219
|
+
},
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
const tool = service.fromMCP(mcpTool, async () => ({}))
|
|
223
|
+
|
|
224
|
+
expect(tool.parameters).toHaveLength(0)
|
|
225
|
+
})
|
|
226
|
+
|
|
227
|
+
it('handles empty required array', () => {
|
|
228
|
+
const mcpTool: MCPTool = {
|
|
229
|
+
name: 'empty.required',
|
|
230
|
+
description: 'Empty required',
|
|
231
|
+
inputSchema: {
|
|
232
|
+
type: 'object',
|
|
233
|
+
properties: {
|
|
234
|
+
value: { type: 'string' },
|
|
235
|
+
},
|
|
236
|
+
required: [],
|
|
237
|
+
},
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
const tool = service.fromMCP(mcpTool, async () => ({}))
|
|
241
|
+
const param = tool.parameters.find((p) => p.name === 'value')
|
|
242
|
+
|
|
243
|
+
expect(param?.required).toBe(false)
|
|
244
|
+
})
|
|
245
|
+
})
|
|
246
|
+
})
|
|
247
|
+
|
|
248
|
+
describe('ToolServiceCore - clear', () => {
|
|
249
|
+
let service: ToolServiceCore
|
|
250
|
+
|
|
251
|
+
beforeEach(() => {
|
|
252
|
+
service = new ToolServiceCore()
|
|
253
|
+
})
|
|
254
|
+
|
|
255
|
+
describe('full clear', () => {
|
|
256
|
+
it('removes all tools', () => {
|
|
257
|
+
service.register({
|
|
258
|
+
id: 'clear.test.1',
|
|
259
|
+
name: 'Clear Test 1',
|
|
260
|
+
description: 'Test 1',
|
|
261
|
+
category: 'data',
|
|
262
|
+
parameters: [],
|
|
263
|
+
handler: async () => ({}),
|
|
264
|
+
})
|
|
265
|
+
|
|
266
|
+
service.register({
|
|
267
|
+
id: 'clear.test.2',
|
|
268
|
+
name: 'Clear Test 2',
|
|
269
|
+
description: 'Test 2',
|
|
270
|
+
category: 'web',
|
|
271
|
+
parameters: [],
|
|
272
|
+
handler: async () => ({}),
|
|
273
|
+
})
|
|
274
|
+
|
|
275
|
+
expect(service.list().length).toBeGreaterThan(0)
|
|
276
|
+
|
|
277
|
+
service.clear()
|
|
278
|
+
|
|
279
|
+
expect(service.list()).toHaveLength(0)
|
|
280
|
+
})
|
|
281
|
+
|
|
282
|
+
it('clears built-in tools when not keeping', () => {
|
|
283
|
+
// Fresh service has built-in tools
|
|
284
|
+
expect(service.has('data.json.parse')).toBe(true)
|
|
285
|
+
|
|
286
|
+
service.clear(false)
|
|
287
|
+
|
|
288
|
+
expect(service.has('data.json.parse')).toBe(false)
|
|
289
|
+
})
|
|
290
|
+
})
|
|
291
|
+
|
|
292
|
+
describe('keeping built-ins', () => {
|
|
293
|
+
it('preserves data.json.parse', () => {
|
|
294
|
+
service.register({
|
|
295
|
+
id: 'custom.tool',
|
|
296
|
+
name: 'Custom Tool',
|
|
297
|
+
description: 'Custom',
|
|
298
|
+
category: 'data',
|
|
299
|
+
parameters: [],
|
|
300
|
+
handler: async () => ({}),
|
|
301
|
+
})
|
|
302
|
+
|
|
303
|
+
service.clear(true)
|
|
304
|
+
|
|
305
|
+
expect(service.has('data.json.parse')).toBe(true)
|
|
306
|
+
expect(service.has('custom.tool')).toBe(false)
|
|
307
|
+
})
|
|
308
|
+
|
|
309
|
+
it('preserves data.csv.parse', () => {
|
|
310
|
+
service.clear(true)
|
|
311
|
+
|
|
312
|
+
expect(service.has('data.csv.parse')).toBe(true)
|
|
313
|
+
})
|
|
314
|
+
|
|
315
|
+
it('preserves data.transform', () => {
|
|
316
|
+
service.clear(true)
|
|
317
|
+
|
|
318
|
+
expect(service.has('data.transform')).toBe(true)
|
|
319
|
+
})
|
|
320
|
+
|
|
321
|
+
it('removes custom tools while keeping built-ins', () => {
|
|
322
|
+
service.register({
|
|
323
|
+
id: 'custom.1',
|
|
324
|
+
name: 'Custom 1',
|
|
325
|
+
description: 'Custom 1',
|
|
326
|
+
category: 'data',
|
|
327
|
+
parameters: [],
|
|
328
|
+
handler: async () => ({}),
|
|
329
|
+
})
|
|
330
|
+
|
|
331
|
+
service.register({
|
|
332
|
+
id: 'custom.2',
|
|
333
|
+
name: 'Custom 2',
|
|
334
|
+
description: 'Custom 2',
|
|
335
|
+
category: 'web',
|
|
336
|
+
parameters: [],
|
|
337
|
+
handler: async () => ({}),
|
|
338
|
+
})
|
|
339
|
+
|
|
340
|
+
service.clear(true)
|
|
341
|
+
|
|
342
|
+
expect(service.has('custom.1')).toBe(false)
|
|
343
|
+
expect(service.has('custom.2')).toBe(false)
|
|
344
|
+
expect(service.has('data.json.parse')).toBe(true)
|
|
345
|
+
})
|
|
346
|
+
|
|
347
|
+
it('built-ins are functional after clear', async () => {
|
|
348
|
+
service.clear(true)
|
|
349
|
+
|
|
350
|
+
const result = await service.executeTool('data.json.parse', {
|
|
351
|
+
text: '{"test": 123}',
|
|
352
|
+
})
|
|
353
|
+
|
|
354
|
+
expect(result.valid).toBe(true)
|
|
355
|
+
expect(result.data).toEqual({ test: 123 })
|
|
356
|
+
})
|
|
357
|
+
})
|
|
358
|
+
|
|
359
|
+
describe('default behavior', () => {
|
|
360
|
+
it('defaults to full clear (keepBuiltins=false)', () => {
|
|
361
|
+
service.clear()
|
|
362
|
+
|
|
363
|
+
expect(service.list()).toHaveLength(0)
|
|
364
|
+
})
|
|
365
|
+
})
|
|
366
|
+
})
|
|
367
|
+
|
|
368
|
+
describe('ToolServiceCore - unregister', () => {
|
|
369
|
+
let service: ToolServiceCore
|
|
370
|
+
|
|
371
|
+
beforeEach(() => {
|
|
372
|
+
service = new ToolServiceCore()
|
|
373
|
+
})
|
|
374
|
+
|
|
375
|
+
it('removes registered tool', () => {
|
|
376
|
+
service.register({
|
|
377
|
+
id: 'unregister.test',
|
|
378
|
+
name: 'Unregister Test',
|
|
379
|
+
description: 'Test',
|
|
380
|
+
category: 'data',
|
|
381
|
+
parameters: [],
|
|
382
|
+
handler: async () => ({}),
|
|
383
|
+
})
|
|
384
|
+
|
|
385
|
+
expect(service.has('unregister.test')).toBe(true)
|
|
386
|
+
|
|
387
|
+
const result = service.unregister('unregister.test')
|
|
388
|
+
|
|
389
|
+
expect(result).toBe(true)
|
|
390
|
+
expect(service.has('unregister.test')).toBe(false)
|
|
391
|
+
})
|
|
392
|
+
|
|
393
|
+
it('returns false for non-existent tool', () => {
|
|
394
|
+
const result = service.unregister('nonexistent.tool')
|
|
395
|
+
|
|
396
|
+
expect(result).toBe(false)
|
|
397
|
+
})
|
|
398
|
+
|
|
399
|
+
it('can unregister built-in tools', () => {
|
|
400
|
+
expect(service.has('data.json.parse')).toBe(true)
|
|
401
|
+
|
|
402
|
+
service.unregister('data.json.parse')
|
|
403
|
+
|
|
404
|
+
expect(service.has('data.json.parse')).toBe(false)
|
|
405
|
+
})
|
|
406
|
+
})
|
|
407
|
+
|
|
408
|
+
describe('ToolServiceCore - query edge cases', () => {
|
|
409
|
+
let service: ToolServiceCore
|
|
410
|
+
|
|
411
|
+
beforeEach(() => {
|
|
412
|
+
service = new ToolServiceCore()
|
|
413
|
+
service.clear()
|
|
414
|
+
|
|
415
|
+
service.register({
|
|
416
|
+
id: 'query.test.1',
|
|
417
|
+
name: 'Query Test 1',
|
|
418
|
+
description: 'First test tool for queries',
|
|
419
|
+
category: 'data',
|
|
420
|
+
subcategory: 'transform',
|
|
421
|
+
parameters: [],
|
|
422
|
+
handler: async () => ({}),
|
|
423
|
+
tags: ['json', 'parse'],
|
|
424
|
+
audience: 'agent',
|
|
425
|
+
})
|
|
426
|
+
|
|
427
|
+
service.register({
|
|
428
|
+
id: 'query.test.2',
|
|
429
|
+
name: 'Query Test 2',
|
|
430
|
+
description: 'Second test tool',
|
|
431
|
+
category: 'web',
|
|
432
|
+
subcategory: 'fetch',
|
|
433
|
+
parameters: [],
|
|
434
|
+
handler: async () => ({}),
|
|
435
|
+
tags: ['http'],
|
|
436
|
+
audience: 'human',
|
|
437
|
+
})
|
|
438
|
+
|
|
439
|
+
service.register({
|
|
440
|
+
id: 'query.test.3',
|
|
441
|
+
name: 'Query Test 3',
|
|
442
|
+
description: 'Third test tool for both',
|
|
443
|
+
category: 'data',
|
|
444
|
+
parameters: [],
|
|
445
|
+
handler: async () => ({}),
|
|
446
|
+
tags: ['transform'],
|
|
447
|
+
audience: 'both',
|
|
448
|
+
})
|
|
449
|
+
})
|
|
450
|
+
|
|
451
|
+
describe('combined filters', () => {
|
|
452
|
+
it('filters by category and subcategory', () => {
|
|
453
|
+
const results = service.query({
|
|
454
|
+
category: 'data',
|
|
455
|
+
subcategory: 'transform',
|
|
456
|
+
})
|
|
457
|
+
|
|
458
|
+
expect(results).toHaveLength(1)
|
|
459
|
+
expect(results[0].id).toBe('query.test.1')
|
|
460
|
+
})
|
|
461
|
+
|
|
462
|
+
it('filters by category and tags', () => {
|
|
463
|
+
const results = service.query({
|
|
464
|
+
category: 'data',
|
|
465
|
+
tags: ['json'],
|
|
466
|
+
})
|
|
467
|
+
|
|
468
|
+
expect(results).toHaveLength(1)
|
|
469
|
+
expect(results[0].id).toBe('query.test.1')
|
|
470
|
+
})
|
|
471
|
+
|
|
472
|
+
it('filters by category and audience', () => {
|
|
473
|
+
const results = service.query({
|
|
474
|
+
category: 'data',
|
|
475
|
+
audience: 'agent',
|
|
476
|
+
})
|
|
477
|
+
|
|
478
|
+
// agent can see 'agent' and 'both' and undefined
|
|
479
|
+
expect(results.length).toBeGreaterThanOrEqual(1)
|
|
480
|
+
})
|
|
481
|
+
|
|
482
|
+
it('filters by search and category', () => {
|
|
483
|
+
const results = service.query({
|
|
484
|
+
search: 'first',
|
|
485
|
+
category: 'data',
|
|
486
|
+
})
|
|
487
|
+
|
|
488
|
+
expect(results).toHaveLength(1)
|
|
489
|
+
expect(results[0].id).toBe('query.test.1')
|
|
490
|
+
})
|
|
491
|
+
})
|
|
492
|
+
|
|
493
|
+
describe('pagination edge cases', () => {
|
|
494
|
+
it('returns all when limit exceeds count', () => {
|
|
495
|
+
const results = service.query({ limit: 100 })
|
|
496
|
+
|
|
497
|
+
expect(results.length).toBeLessThanOrEqual(100)
|
|
498
|
+
})
|
|
499
|
+
|
|
500
|
+
it('returns empty when offset exceeds count', () => {
|
|
501
|
+
const results = service.query({ offset: 1000 })
|
|
502
|
+
|
|
503
|
+
expect(results).toHaveLength(0)
|
|
504
|
+
})
|
|
505
|
+
|
|
506
|
+
it('handles offset with limit', () => {
|
|
507
|
+
const allResults = service.query({})
|
|
508
|
+
const offsetResults = service.query({ offset: 1, limit: 1 })
|
|
509
|
+
|
|
510
|
+
if (allResults.length > 1) {
|
|
511
|
+
expect(offsetResults).toHaveLength(1)
|
|
512
|
+
expect(offsetResults[0].id).toBe(allResults[1].id)
|
|
513
|
+
}
|
|
514
|
+
})
|
|
515
|
+
|
|
516
|
+
it('handles zero limit', () => {
|
|
517
|
+
const results = service.query({ limit: 0 })
|
|
518
|
+
|
|
519
|
+
expect(results).toHaveLength(0)
|
|
520
|
+
})
|
|
521
|
+
})
|
|
522
|
+
|
|
523
|
+
describe('search variations', () => {
|
|
524
|
+
it('searches in name', () => {
|
|
525
|
+
const results = service.query({ search: 'query test 2' })
|
|
526
|
+
|
|
527
|
+
expect(results.some((t) => t.id === 'query.test.2')).toBe(true)
|
|
528
|
+
})
|
|
529
|
+
|
|
530
|
+
it('searches in description', () => {
|
|
531
|
+
const results = service.query({ search: 'third' })
|
|
532
|
+
|
|
533
|
+
expect(results.some((t) => t.id === 'query.test.3')).toBe(true)
|
|
534
|
+
})
|
|
535
|
+
|
|
536
|
+
it('searches in id', () => {
|
|
537
|
+
const results = service.query({ search: 'query.test.1' })
|
|
538
|
+
|
|
539
|
+
expect(results.some((t) => t.id === 'query.test.1')).toBe(true)
|
|
540
|
+
})
|
|
541
|
+
|
|
542
|
+
it('search is case-insensitive', () => {
|
|
543
|
+
const results = service.query({ search: 'FIRST' })
|
|
544
|
+
|
|
545
|
+
expect(results.some((t) => t.id === 'query.test.1')).toBe(true)
|
|
546
|
+
})
|
|
547
|
+
})
|
|
548
|
+
|
|
549
|
+
describe('audience matching', () => {
|
|
550
|
+
it('agent sees both and agent tools', () => {
|
|
551
|
+
const results = service.query({ audience: 'agent' })
|
|
552
|
+
|
|
553
|
+
expect(results.some((t) => t.id === 'query.test.1')).toBe(true) // agent
|
|
554
|
+
expect(results.some((t) => t.id === 'query.test.3')).toBe(true) // both
|
|
555
|
+
})
|
|
556
|
+
|
|
557
|
+
it('human sees both and human tools', () => {
|
|
558
|
+
const results = service.query({ audience: 'human' })
|
|
559
|
+
|
|
560
|
+
expect(results.some((t) => t.id === 'query.test.2')).toBe(true) // human
|
|
561
|
+
expect(results.some((t) => t.id === 'query.test.3')).toBe(true) // both
|
|
562
|
+
})
|
|
563
|
+
})
|
|
564
|
+
})
|
|
565
|
+
|
|
566
|
+
describe('ToolServiceCore - byCategory', () => {
|
|
567
|
+
let service: ToolServiceCore
|
|
568
|
+
|
|
569
|
+
beforeEach(() => {
|
|
570
|
+
service = new ToolServiceCore()
|
|
571
|
+
service.clear()
|
|
572
|
+
|
|
573
|
+
service.register({
|
|
574
|
+
id: 'cat.data.1',
|
|
575
|
+
name: 'Data 1',
|
|
576
|
+
description: 'Data tool',
|
|
577
|
+
category: 'data',
|
|
578
|
+
parameters: [],
|
|
579
|
+
handler: async () => ({}),
|
|
580
|
+
})
|
|
581
|
+
|
|
582
|
+
service.register({
|
|
583
|
+
id: 'cat.data.2',
|
|
584
|
+
name: 'Data 2',
|
|
585
|
+
description: 'Another data tool',
|
|
586
|
+
category: 'data',
|
|
587
|
+
parameters: [],
|
|
588
|
+
handler: async () => ({}),
|
|
589
|
+
})
|
|
590
|
+
|
|
591
|
+
service.register({
|
|
592
|
+
id: 'cat.web.1',
|
|
593
|
+
name: 'Web 1',
|
|
594
|
+
description: 'Web tool',
|
|
595
|
+
category: 'web',
|
|
596
|
+
parameters: [],
|
|
597
|
+
handler: async () => ({}),
|
|
598
|
+
})
|
|
599
|
+
})
|
|
600
|
+
|
|
601
|
+
it('returns tools for specified category', () => {
|
|
602
|
+
const dataTools = service.byCategory('data')
|
|
603
|
+
|
|
604
|
+
expect(dataTools).toHaveLength(2)
|
|
605
|
+
expect(dataTools.every((t) => t.category === 'data')).toBe(true)
|
|
606
|
+
})
|
|
607
|
+
|
|
608
|
+
it('returns empty for non-existent category', () => {
|
|
609
|
+
const results = service.byCategory('security')
|
|
610
|
+
|
|
611
|
+
expect(results).toHaveLength(0)
|
|
612
|
+
})
|
|
613
|
+
|
|
614
|
+
it('uses query internally', () => {
|
|
615
|
+
// byCategory is shorthand for query({ category })
|
|
616
|
+
const viaByCategory = service.byCategory('web')
|
|
617
|
+
const viaQuery = service.query({ category: 'web' })
|
|
618
|
+
|
|
619
|
+
expect(viaByCategory.length).toBe(viaQuery.length)
|
|
620
|
+
expect(viaByCategory.map((t) => t.id).sort()).toEqual(viaQuery.map((t) => t.id).sort())
|
|
621
|
+
})
|
|
622
|
+
})
|
|
623
|
+
|
|
624
|
+
describe('ToolServiceCore - executeTool edge cases', () => {
|
|
625
|
+
let service: ToolServiceCore
|
|
626
|
+
|
|
627
|
+
beforeEach(() => {
|
|
628
|
+
service = new ToolServiceCore()
|
|
629
|
+
})
|
|
630
|
+
|
|
631
|
+
it('handles async handler', async () => {
|
|
632
|
+
service.register({
|
|
633
|
+
id: 'async.exec',
|
|
634
|
+
name: 'Async Exec',
|
|
635
|
+
description: 'Async handler',
|
|
636
|
+
category: 'data',
|
|
637
|
+
parameters: [],
|
|
638
|
+
handler: async () => {
|
|
639
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
640
|
+
return { delayed: true }
|
|
641
|
+
},
|
|
642
|
+
})
|
|
643
|
+
|
|
644
|
+
const result = await service.executeTool('async.exec', {})
|
|
645
|
+
|
|
646
|
+
expect(result).toEqual({ delayed: true })
|
|
647
|
+
})
|
|
648
|
+
|
|
649
|
+
it('handles handler that returns primitive', async () => {
|
|
650
|
+
service.register({
|
|
651
|
+
id: 'primitive.return',
|
|
652
|
+
name: 'Primitive Return',
|
|
653
|
+
description: 'Returns primitive',
|
|
654
|
+
category: 'data',
|
|
655
|
+
parameters: [],
|
|
656
|
+
handler: async () => 42,
|
|
657
|
+
})
|
|
658
|
+
|
|
659
|
+
const result = await service.executeTool('primitive.return', {})
|
|
660
|
+
|
|
661
|
+
expect(result).toBe(42)
|
|
662
|
+
})
|
|
663
|
+
|
|
664
|
+
it('handles handler that returns null', async () => {
|
|
665
|
+
service.register({
|
|
666
|
+
id: 'null.return',
|
|
667
|
+
name: 'Null Return',
|
|
668
|
+
description: 'Returns null',
|
|
669
|
+
category: 'data',
|
|
670
|
+
parameters: [],
|
|
671
|
+
handler: async () => null,
|
|
672
|
+
})
|
|
673
|
+
|
|
674
|
+
const result = await service.executeTool('null.return', {})
|
|
675
|
+
|
|
676
|
+
expect(result).toBeNull()
|
|
677
|
+
})
|
|
678
|
+
|
|
679
|
+
it('throws for non-existent tool', async () => {
|
|
680
|
+
await expect(service.executeTool('does.not.exist', {})).rejects.toThrow(
|
|
681
|
+
'Tool "does.not.exist" not found'
|
|
682
|
+
)
|
|
683
|
+
})
|
|
684
|
+
|
|
685
|
+
it('propagates handler errors', async () => {
|
|
686
|
+
service.register({
|
|
687
|
+
id: 'error.throw',
|
|
688
|
+
name: 'Error Throw',
|
|
689
|
+
description: 'Throws error',
|
|
690
|
+
category: 'data',
|
|
691
|
+
parameters: [],
|
|
692
|
+
handler: async () => {
|
|
693
|
+
throw new Error('Test error')
|
|
694
|
+
},
|
|
695
|
+
})
|
|
696
|
+
|
|
697
|
+
await expect(service.executeTool('error.throw', {})).rejects.toThrow('Test error')
|
|
698
|
+
})
|
|
699
|
+
})
|