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,583 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Extended Tests for Provider Registry
|
|
3
|
+
*
|
|
4
|
+
* Comprehensive tests for provider registration, creation, and discovery.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { describe, it, expect, beforeEach } from 'vitest'
|
|
8
|
+
import {
|
|
9
|
+
createProviderRegistry,
|
|
10
|
+
providerRegistry,
|
|
11
|
+
registerProvider,
|
|
12
|
+
getProvider,
|
|
13
|
+
listProviders,
|
|
14
|
+
defineProvider,
|
|
15
|
+
type ProviderInfo,
|
|
16
|
+
type BaseProvider,
|
|
17
|
+
type ProviderConfig,
|
|
18
|
+
} from '../src/providers/index.js'
|
|
19
|
+
|
|
20
|
+
describe('Provider Registry - createProviderRegistry', () => {
|
|
21
|
+
describe('register', () => {
|
|
22
|
+
it('registers a new provider', () => {
|
|
23
|
+
const registry = createProviderRegistry()
|
|
24
|
+
|
|
25
|
+
const info: ProviderInfo = {
|
|
26
|
+
id: 'test.provider',
|
|
27
|
+
name: 'Test Provider',
|
|
28
|
+
description: 'A test provider',
|
|
29
|
+
category: 'email',
|
|
30
|
+
requiredConfig: ['apiKey'],
|
|
31
|
+
optionalConfig: ['debug'],
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const factory = async () => ({
|
|
35
|
+
initialize: async () => {},
|
|
36
|
+
getName: () => 'Test',
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
registry.register(info, factory)
|
|
40
|
+
|
|
41
|
+
expect(registry.has('test.provider')).toBe(true)
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
it('throws on duplicate registration', () => {
|
|
45
|
+
const registry = createProviderRegistry()
|
|
46
|
+
|
|
47
|
+
const info: ProviderInfo = {
|
|
48
|
+
id: 'duplicate.provider',
|
|
49
|
+
name: 'Duplicate',
|
|
50
|
+
description: 'Duplicate provider',
|
|
51
|
+
category: 'email',
|
|
52
|
+
requiredConfig: [],
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const factory = async () => ({
|
|
56
|
+
initialize: async () => {},
|
|
57
|
+
getName: () => 'Duplicate',
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
registry.register(info, factory)
|
|
61
|
+
|
|
62
|
+
expect(() => registry.register(info, factory)).toThrow('already registered')
|
|
63
|
+
})
|
|
64
|
+
})
|
|
65
|
+
|
|
66
|
+
describe('get', () => {
|
|
67
|
+
it('returns registered provider', () => {
|
|
68
|
+
const registry = createProviderRegistry()
|
|
69
|
+
|
|
70
|
+
const info: ProviderInfo = {
|
|
71
|
+
id: 'get.test',
|
|
72
|
+
name: 'Get Test',
|
|
73
|
+
description: 'Test get',
|
|
74
|
+
category: 'messaging',
|
|
75
|
+
requiredConfig: [],
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const factory = async () => ({
|
|
79
|
+
initialize: async () => {},
|
|
80
|
+
getName: () => 'Get Test',
|
|
81
|
+
})
|
|
82
|
+
|
|
83
|
+
registry.register(info, factory)
|
|
84
|
+
|
|
85
|
+
const result = registry.get('get.test')
|
|
86
|
+
expect(result).toBeDefined()
|
|
87
|
+
expect(result?.info.id).toBe('get.test')
|
|
88
|
+
})
|
|
89
|
+
|
|
90
|
+
it('returns undefined for non-existent provider', () => {
|
|
91
|
+
const registry = createProviderRegistry()
|
|
92
|
+
|
|
93
|
+
const result = registry.get('nonexistent')
|
|
94
|
+
expect(result).toBeUndefined()
|
|
95
|
+
})
|
|
96
|
+
})
|
|
97
|
+
|
|
98
|
+
describe('has', () => {
|
|
99
|
+
it('returns true for registered provider', () => {
|
|
100
|
+
const registry = createProviderRegistry()
|
|
101
|
+
|
|
102
|
+
const info: ProviderInfo = {
|
|
103
|
+
id: 'has.test',
|
|
104
|
+
name: 'Has Test',
|
|
105
|
+
description: 'Test has',
|
|
106
|
+
category: 'email',
|
|
107
|
+
requiredConfig: [],
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
registry.register(info, async () => ({
|
|
111
|
+
initialize: async () => {},
|
|
112
|
+
getName: () => 'Has Test',
|
|
113
|
+
}))
|
|
114
|
+
|
|
115
|
+
expect(registry.has('has.test')).toBe(true)
|
|
116
|
+
})
|
|
117
|
+
|
|
118
|
+
it('returns false for unregistered provider', () => {
|
|
119
|
+
const registry = createProviderRegistry()
|
|
120
|
+
|
|
121
|
+
expect(registry.has('unregistered')).toBe(false)
|
|
122
|
+
})
|
|
123
|
+
})
|
|
124
|
+
|
|
125
|
+
describe('list', () => {
|
|
126
|
+
it('lists all providers', () => {
|
|
127
|
+
const registry = createProviderRegistry()
|
|
128
|
+
|
|
129
|
+
const info1: ProviderInfo = {
|
|
130
|
+
id: 'list.test.1',
|
|
131
|
+
name: 'List Test 1',
|
|
132
|
+
description: 'Test 1',
|
|
133
|
+
category: 'email',
|
|
134
|
+
requiredConfig: [],
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
const info2: ProviderInfo = {
|
|
138
|
+
id: 'list.test.2',
|
|
139
|
+
name: 'List Test 2',
|
|
140
|
+
description: 'Test 2',
|
|
141
|
+
category: 'messaging',
|
|
142
|
+
requiredConfig: [],
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
const factory = async () => ({
|
|
146
|
+
initialize: async () => {},
|
|
147
|
+
getName: () => 'Test',
|
|
148
|
+
})
|
|
149
|
+
|
|
150
|
+
registry.register(info1, factory)
|
|
151
|
+
registry.register(info2, factory)
|
|
152
|
+
|
|
153
|
+
const all = registry.list()
|
|
154
|
+
expect(all).toHaveLength(2)
|
|
155
|
+
})
|
|
156
|
+
|
|
157
|
+
it('lists providers by category', () => {
|
|
158
|
+
const registry = createProviderRegistry()
|
|
159
|
+
|
|
160
|
+
registry.register(
|
|
161
|
+
{
|
|
162
|
+
id: 'cat.email',
|
|
163
|
+
name: 'Email',
|
|
164
|
+
description: 'Email provider',
|
|
165
|
+
category: 'email',
|
|
166
|
+
requiredConfig: [],
|
|
167
|
+
},
|
|
168
|
+
async () => ({
|
|
169
|
+
initialize: async () => {},
|
|
170
|
+
getName: () => 'Email',
|
|
171
|
+
})
|
|
172
|
+
)
|
|
173
|
+
|
|
174
|
+
registry.register(
|
|
175
|
+
{
|
|
176
|
+
id: 'cat.messaging',
|
|
177
|
+
name: 'Messaging',
|
|
178
|
+
description: 'Messaging provider',
|
|
179
|
+
category: 'messaging',
|
|
180
|
+
requiredConfig: [],
|
|
181
|
+
},
|
|
182
|
+
async () => ({
|
|
183
|
+
initialize: async () => {},
|
|
184
|
+
getName: () => 'Messaging',
|
|
185
|
+
})
|
|
186
|
+
)
|
|
187
|
+
|
|
188
|
+
const emailProviders = registry.list('email')
|
|
189
|
+
expect(emailProviders).toHaveLength(1)
|
|
190
|
+
expect(emailProviders[0].info.id).toBe('cat.email')
|
|
191
|
+
})
|
|
192
|
+
|
|
193
|
+
it('returns empty array for non-existent category', () => {
|
|
194
|
+
const registry = createProviderRegistry()
|
|
195
|
+
|
|
196
|
+
const result = registry.list('email')
|
|
197
|
+
expect(result).toEqual([])
|
|
198
|
+
})
|
|
199
|
+
})
|
|
200
|
+
|
|
201
|
+
describe('create', () => {
|
|
202
|
+
it('creates provider instance', async () => {
|
|
203
|
+
const registry = createProviderRegistry()
|
|
204
|
+
|
|
205
|
+
let initialized = false
|
|
206
|
+
|
|
207
|
+
registry.register(
|
|
208
|
+
{
|
|
209
|
+
id: 'create.test',
|
|
210
|
+
name: 'Create Test',
|
|
211
|
+
description: 'Test creation',
|
|
212
|
+
category: 'email',
|
|
213
|
+
requiredConfig: ['apiKey'],
|
|
214
|
+
},
|
|
215
|
+
async (config) => ({
|
|
216
|
+
initialize: async () => {
|
|
217
|
+
initialized = true
|
|
218
|
+
},
|
|
219
|
+
getName: () => 'Create Test',
|
|
220
|
+
config,
|
|
221
|
+
})
|
|
222
|
+
)
|
|
223
|
+
|
|
224
|
+
const instance = await registry.create('create.test', { apiKey: 'test-key' })
|
|
225
|
+
|
|
226
|
+
expect(initialized).toBe(true)
|
|
227
|
+
expect(instance).toBeDefined()
|
|
228
|
+
expect(instance.getName()).toBe('Create Test')
|
|
229
|
+
})
|
|
230
|
+
|
|
231
|
+
it('throws for non-existent provider', async () => {
|
|
232
|
+
const registry = createProviderRegistry()
|
|
233
|
+
|
|
234
|
+
await expect(registry.create('nonexistent', {})).rejects.toThrow('not found')
|
|
235
|
+
})
|
|
236
|
+
|
|
237
|
+
it('throws for missing required config', async () => {
|
|
238
|
+
const registry = createProviderRegistry()
|
|
239
|
+
|
|
240
|
+
registry.register(
|
|
241
|
+
{
|
|
242
|
+
id: 'missing.config',
|
|
243
|
+
name: 'Missing Config',
|
|
244
|
+
description: 'Test missing config',
|
|
245
|
+
category: 'email',
|
|
246
|
+
requiredConfig: ['apiKey', 'secret'],
|
|
247
|
+
},
|
|
248
|
+
async () => ({
|
|
249
|
+
initialize: async () => {},
|
|
250
|
+
getName: () => 'Missing Config',
|
|
251
|
+
})
|
|
252
|
+
)
|
|
253
|
+
|
|
254
|
+
await expect(registry.create('missing.config', { apiKey: 'key' })).rejects.toThrow(
|
|
255
|
+
'missing required config'
|
|
256
|
+
)
|
|
257
|
+
})
|
|
258
|
+
|
|
259
|
+
it('passes config to factory', async () => {
|
|
260
|
+
const registry = createProviderRegistry()
|
|
261
|
+
|
|
262
|
+
let receivedConfig: ProviderConfig | undefined
|
|
263
|
+
|
|
264
|
+
registry.register(
|
|
265
|
+
{
|
|
266
|
+
id: 'config.pass',
|
|
267
|
+
name: 'Config Pass',
|
|
268
|
+
description: 'Test config passing',
|
|
269
|
+
category: 'email',
|
|
270
|
+
requiredConfig: ['apiKey'],
|
|
271
|
+
},
|
|
272
|
+
async (config) => {
|
|
273
|
+
receivedConfig = config
|
|
274
|
+
return {
|
|
275
|
+
initialize: async () => {},
|
|
276
|
+
getName: () => 'Config Pass',
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
)
|
|
280
|
+
|
|
281
|
+
await registry.create('config.pass', { apiKey: 'test-api-key', extra: 'value' })
|
|
282
|
+
|
|
283
|
+
expect(receivedConfig).toEqual({ apiKey: 'test-api-key', extra: 'value' })
|
|
284
|
+
})
|
|
285
|
+
|
|
286
|
+
it('calls initialize on provider', async () => {
|
|
287
|
+
const registry = createProviderRegistry()
|
|
288
|
+
|
|
289
|
+
let initializeCalledWith: ProviderConfig | undefined
|
|
290
|
+
|
|
291
|
+
registry.register(
|
|
292
|
+
{
|
|
293
|
+
id: 'init.test',
|
|
294
|
+
name: 'Init Test',
|
|
295
|
+
description: 'Test initialization',
|
|
296
|
+
category: 'email',
|
|
297
|
+
requiredConfig: [],
|
|
298
|
+
},
|
|
299
|
+
async () => ({
|
|
300
|
+
initialize: async (config) => {
|
|
301
|
+
initializeCalledWith = config
|
|
302
|
+
},
|
|
303
|
+
getName: () => 'Init Test',
|
|
304
|
+
})
|
|
305
|
+
)
|
|
306
|
+
|
|
307
|
+
await registry.create('init.test', { setting: 'value' })
|
|
308
|
+
|
|
309
|
+
expect(initializeCalledWith).toEqual({ setting: 'value' })
|
|
310
|
+
})
|
|
311
|
+
})
|
|
312
|
+
})
|
|
313
|
+
|
|
314
|
+
describe('defineProvider', () => {
|
|
315
|
+
it('returns info and factory', () => {
|
|
316
|
+
const info: ProviderInfo = {
|
|
317
|
+
id: 'defined.provider',
|
|
318
|
+
name: 'Defined Provider',
|
|
319
|
+
description: 'A defined provider',
|
|
320
|
+
category: 'email',
|
|
321
|
+
requiredConfig: [],
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
const factory = async () => ({
|
|
325
|
+
initialize: async () => {},
|
|
326
|
+
getName: () => 'Defined',
|
|
327
|
+
})
|
|
328
|
+
|
|
329
|
+
const defined = defineProvider(info, factory)
|
|
330
|
+
|
|
331
|
+
expect(defined.info).toBe(info)
|
|
332
|
+
expect(defined.factory).toBe(factory)
|
|
333
|
+
})
|
|
334
|
+
|
|
335
|
+
it('provides register method', () => {
|
|
336
|
+
const info: ProviderInfo = {
|
|
337
|
+
id: 'register.method.test',
|
|
338
|
+
name: 'Register Method',
|
|
339
|
+
description: 'Test register method',
|
|
340
|
+
category: 'messaging',
|
|
341
|
+
requiredConfig: [],
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
const defined = defineProvider(info, async () => ({
|
|
345
|
+
initialize: async () => {},
|
|
346
|
+
getName: () => 'Register Method',
|
|
347
|
+
}))
|
|
348
|
+
|
|
349
|
+
expect(typeof defined.register).toBe('function')
|
|
350
|
+
})
|
|
351
|
+
})
|
|
352
|
+
|
|
353
|
+
describe('Global Registry Functions', () => {
|
|
354
|
+
// Note: These tests use the global providerRegistry which may have state from other tests
|
|
355
|
+
// We test that the functions exist and work correctly
|
|
356
|
+
|
|
357
|
+
describe('registerProvider', () => {
|
|
358
|
+
it('is a function', () => {
|
|
359
|
+
expect(typeof registerProvider).toBe('function')
|
|
360
|
+
})
|
|
361
|
+
})
|
|
362
|
+
|
|
363
|
+
describe('getProvider', () => {
|
|
364
|
+
it('is a function', () => {
|
|
365
|
+
expect(typeof getProvider).toBe('function')
|
|
366
|
+
})
|
|
367
|
+
|
|
368
|
+
it('returns undefined for non-existent', () => {
|
|
369
|
+
const result = getProvider('definitely.not.registered.12345')
|
|
370
|
+
expect(result).toBeUndefined()
|
|
371
|
+
})
|
|
372
|
+
})
|
|
373
|
+
|
|
374
|
+
describe('listProviders', () => {
|
|
375
|
+
it('is a function', () => {
|
|
376
|
+
expect(typeof listProviders).toBe('function')
|
|
377
|
+
})
|
|
378
|
+
|
|
379
|
+
it('returns an array', () => {
|
|
380
|
+
const result = listProviders()
|
|
381
|
+
expect(Array.isArray(result)).toBe(true)
|
|
382
|
+
})
|
|
383
|
+
|
|
384
|
+
it('can filter by category', () => {
|
|
385
|
+
const result = listProviders('email')
|
|
386
|
+
expect(Array.isArray(result)).toBe(true)
|
|
387
|
+
})
|
|
388
|
+
})
|
|
389
|
+
})
|
|
390
|
+
|
|
391
|
+
describe('Provider Info Structure', () => {
|
|
392
|
+
it('supports all required fields', () => {
|
|
393
|
+
const info: ProviderInfo = {
|
|
394
|
+
id: 'structure.test',
|
|
395
|
+
name: 'Structure Test',
|
|
396
|
+
description: 'Testing structure',
|
|
397
|
+
category: 'email',
|
|
398
|
+
requiredConfig: ['apiKey'],
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
expect(info.id).toBe('structure.test')
|
|
402
|
+
expect(info.name).toBe('Structure Test')
|
|
403
|
+
expect(info.description).toBe('Testing structure')
|
|
404
|
+
expect(info.category).toBe('email')
|
|
405
|
+
expect(info.requiredConfig).toEqual(['apiKey'])
|
|
406
|
+
})
|
|
407
|
+
|
|
408
|
+
it('supports optional config', () => {
|
|
409
|
+
const info: ProviderInfo = {
|
|
410
|
+
id: 'optional.test',
|
|
411
|
+
name: 'Optional Test',
|
|
412
|
+
description: 'Testing optional',
|
|
413
|
+
category: 'messaging',
|
|
414
|
+
requiredConfig: ['token'],
|
|
415
|
+
optionalConfig: ['debug', 'timeout'],
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
expect(info.optionalConfig).toEqual(['debug', 'timeout'])
|
|
419
|
+
})
|
|
420
|
+
|
|
421
|
+
it('supports all categories', () => {
|
|
422
|
+
const categories = [
|
|
423
|
+
'email',
|
|
424
|
+
'messaging',
|
|
425
|
+
'spreadsheet',
|
|
426
|
+
'document',
|
|
427
|
+
'presentation',
|
|
428
|
+
'calendar',
|
|
429
|
+
'task',
|
|
430
|
+
'phone',
|
|
431
|
+
'crm',
|
|
432
|
+
'finance',
|
|
433
|
+
'ecommerce',
|
|
434
|
+
'support',
|
|
435
|
+
'knowledge',
|
|
436
|
+
'media',
|
|
437
|
+
'storage',
|
|
438
|
+
'analytics',
|
|
439
|
+
'development',
|
|
440
|
+
'project-management',
|
|
441
|
+
'marketing',
|
|
442
|
+
'forms',
|
|
443
|
+
'video-conferencing',
|
|
444
|
+
'signature',
|
|
445
|
+
] as const
|
|
446
|
+
|
|
447
|
+
for (const category of categories) {
|
|
448
|
+
const info: ProviderInfo = {
|
|
449
|
+
id: `category.${category}`,
|
|
450
|
+
name: `Category ${category}`,
|
|
451
|
+
description: `Testing ${category}`,
|
|
452
|
+
category,
|
|
453
|
+
requiredConfig: [],
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
expect(info.category).toBe(category)
|
|
457
|
+
}
|
|
458
|
+
})
|
|
459
|
+
})
|
|
460
|
+
|
|
461
|
+
describe('Provider Integration', () => {
|
|
462
|
+
it('factory receives config object', async () => {
|
|
463
|
+
const registry = createProviderRegistry()
|
|
464
|
+
|
|
465
|
+
let factoryConfig: ProviderConfig | undefined
|
|
466
|
+
|
|
467
|
+
registry.register(
|
|
468
|
+
{
|
|
469
|
+
id: 'factory.config',
|
|
470
|
+
name: 'Factory Config',
|
|
471
|
+
description: 'Test factory config',
|
|
472
|
+
category: 'email',
|
|
473
|
+
requiredConfig: [],
|
|
474
|
+
},
|
|
475
|
+
async (config) => {
|
|
476
|
+
factoryConfig = config
|
|
477
|
+
return {
|
|
478
|
+
initialize: async () => {},
|
|
479
|
+
getName: () => 'Factory Config',
|
|
480
|
+
}
|
|
481
|
+
}
|
|
482
|
+
)
|
|
483
|
+
|
|
484
|
+
await registry.create('factory.config', { key: 'value', number: 123 })
|
|
485
|
+
|
|
486
|
+
expect(factoryConfig).toEqual({ key: 'value', number: 123 })
|
|
487
|
+
})
|
|
488
|
+
|
|
489
|
+
it('provider can access config after creation', async () => {
|
|
490
|
+
const registry = createProviderRegistry()
|
|
491
|
+
|
|
492
|
+
registry.register(
|
|
493
|
+
{
|
|
494
|
+
id: 'access.config',
|
|
495
|
+
name: 'Access Config',
|
|
496
|
+
description: 'Test accessing config',
|
|
497
|
+
category: 'email',
|
|
498
|
+
requiredConfig: ['apiKey'],
|
|
499
|
+
},
|
|
500
|
+
async (config) => ({
|
|
501
|
+
initialize: async () => {},
|
|
502
|
+
getName: () => 'Access Config',
|
|
503
|
+
getApiKey: () => config.apiKey,
|
|
504
|
+
})
|
|
505
|
+
)
|
|
506
|
+
|
|
507
|
+
const instance = (await registry.create('access.config', { apiKey: 'my-api-key' })) as any
|
|
508
|
+
|
|
509
|
+
expect(instance.getApiKey()).toBe('my-api-key')
|
|
510
|
+
})
|
|
511
|
+
|
|
512
|
+
it('multiple registries are independent', () => {
|
|
513
|
+
const registry1 = createProviderRegistry()
|
|
514
|
+
const registry2 = createProviderRegistry()
|
|
515
|
+
|
|
516
|
+
registry1.register(
|
|
517
|
+
{
|
|
518
|
+
id: 'independent.test',
|
|
519
|
+
name: 'Independent',
|
|
520
|
+
description: 'Test independence',
|
|
521
|
+
category: 'email',
|
|
522
|
+
requiredConfig: [],
|
|
523
|
+
},
|
|
524
|
+
async () => ({
|
|
525
|
+
initialize: async () => {},
|
|
526
|
+
getName: () => 'Independent',
|
|
527
|
+
})
|
|
528
|
+
)
|
|
529
|
+
|
|
530
|
+
expect(registry1.has('independent.test')).toBe(true)
|
|
531
|
+
expect(registry2.has('independent.test')).toBe(false)
|
|
532
|
+
})
|
|
533
|
+
|
|
534
|
+
it('async factory works correctly', async () => {
|
|
535
|
+
const registry = createProviderRegistry()
|
|
536
|
+
|
|
537
|
+
registry.register(
|
|
538
|
+
{
|
|
539
|
+
id: 'async.factory',
|
|
540
|
+
name: 'Async Factory',
|
|
541
|
+
description: 'Test async factory',
|
|
542
|
+
category: 'email',
|
|
543
|
+
requiredConfig: [],
|
|
544
|
+
},
|
|
545
|
+
async () => {
|
|
546
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
547
|
+
return {
|
|
548
|
+
initialize: async () => {},
|
|
549
|
+
getName: () => 'Async Factory',
|
|
550
|
+
}
|
|
551
|
+
}
|
|
552
|
+
)
|
|
553
|
+
|
|
554
|
+
const instance = await registry.create('async.factory', {})
|
|
555
|
+
expect(instance.getName()).toBe('Async Factory')
|
|
556
|
+
})
|
|
557
|
+
|
|
558
|
+
it('async initialize works correctly', async () => {
|
|
559
|
+
const registry = createProviderRegistry()
|
|
560
|
+
|
|
561
|
+
let initComplete = false
|
|
562
|
+
|
|
563
|
+
registry.register(
|
|
564
|
+
{
|
|
565
|
+
id: 'async.init',
|
|
566
|
+
name: 'Async Init',
|
|
567
|
+
description: 'Test async init',
|
|
568
|
+
category: 'email',
|
|
569
|
+
requiredConfig: [],
|
|
570
|
+
},
|
|
571
|
+
async () => ({
|
|
572
|
+
initialize: async () => {
|
|
573
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
574
|
+
initComplete = true
|
|
575
|
+
},
|
|
576
|
+
getName: () => 'Async Init',
|
|
577
|
+
})
|
|
578
|
+
)
|
|
579
|
+
|
|
580
|
+
await registry.create('async.init', {})
|
|
581
|
+
expect(initComplete).toBe(true)
|
|
582
|
+
})
|
|
583
|
+
})
|