digital-tools 2.0.2 → 2.1.1

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 (93) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/package.json +3 -4
  3. package/src/define.js +267 -0
  4. package/src/entities/advertising.js +999 -0
  5. package/src/entities/ai.js +756 -0
  6. package/src/entities/analytics.js +1588 -0
  7. package/src/entities/automation.js +601 -0
  8. package/src/entities/communication.js +1150 -0
  9. package/src/entities/crm.js +1386 -0
  10. package/src/entities/design.js +546 -0
  11. package/src/entities/development.js +2212 -0
  12. package/src/entities/document.js +874 -0
  13. package/src/entities/ecommerce.js +1429 -0
  14. package/src/entities/experiment.js +1039 -0
  15. package/src/entities/finance.js +3478 -0
  16. package/src/entities/forms.js +1892 -0
  17. package/src/entities/hr.js +661 -0
  18. package/src/entities/identity.js +997 -0
  19. package/src/entities/index.js +282 -0
  20. package/src/entities/infrastructure.js +1153 -0
  21. package/src/entities/knowledge.js +1438 -0
  22. package/src/entities/marketing.js +1610 -0
  23. package/src/entities/media.js +1634 -0
  24. package/src/entities/notification.js +1199 -0
  25. package/src/entities/presentation.js +1274 -0
  26. package/src/entities/productivity.js +1317 -0
  27. package/src/entities/project-management.js +1136 -0
  28. package/src/entities/recruiting.js +736 -0
  29. package/src/entities/shipping.js +509 -0
  30. package/src/entities/signature.js +1102 -0
  31. package/src/entities/site.js +222 -0
  32. package/src/entities/spreadsheet.js +1341 -0
  33. package/src/entities/storage.js +1198 -0
  34. package/src/entities/support.js +1166 -0
  35. package/src/entities/video-conferencing.js +1750 -0
  36. package/src/entities/video.js +950 -0
  37. package/src/entities.js +1663 -0
  38. package/src/index.js +74 -0
  39. package/src/providers/analytics/index.js +17 -0
  40. package/src/providers/analytics/mixpanel.js +255 -0
  41. package/src/providers/calendar/cal-com.js +303 -0
  42. package/src/providers/calendar/google-calendar.js +335 -0
  43. package/src/providers/calendar/index.js +20 -0
  44. package/src/providers/crm/hubspot.js +566 -0
  45. package/src/providers/crm/index.js +17 -0
  46. package/src/providers/development/github.js +472 -0
  47. package/src/providers/development/index.js +17 -0
  48. package/src/providers/ecommerce/index.js +17 -0
  49. package/src/providers/ecommerce/shopify.js +378 -0
  50. package/src/providers/email/index.js +20 -0
  51. package/src/providers/email/resend.js +258 -0
  52. package/src/providers/email/sendgrid.js +161 -0
  53. package/src/providers/finance/index.js +17 -0
  54. package/src/providers/finance/stripe.js +549 -0
  55. package/src/providers/forms/index.js +17 -0
  56. package/src/providers/forms/typeform.js +500 -0
  57. package/src/providers/index.js +123 -0
  58. package/src/providers/knowledge/index.js +17 -0
  59. package/src/providers/knowledge/notion.js +389 -0
  60. package/src/providers/marketing/index.js +17 -0
  61. package/src/providers/marketing/mailchimp.js +443 -0
  62. package/src/providers/media/cloudinary.js +318 -0
  63. package/src/providers/media/index.js +17 -0
  64. package/src/providers/messaging/index.js +20 -0
  65. package/src/providers/messaging/slack.js +393 -0
  66. package/src/providers/messaging/twilio-sms.js +249 -0
  67. package/src/providers/project-management/index.js +17 -0
  68. package/src/providers/project-management/linear.js +575 -0
  69. package/src/providers/registry.js +86 -0
  70. package/src/providers/spreadsheet/google-sheets.js +375 -0
  71. package/src/providers/spreadsheet/index.js +20 -0
  72. package/src/providers/spreadsheet/xlsx.js +423 -0
  73. package/src/providers/storage/index.js +24 -0
  74. package/src/providers/storage/s3.js +419 -0
  75. package/src/providers/support/index.js +17 -0
  76. package/src/providers/support/zendesk.js +373 -0
  77. package/src/providers/tasks/index.js +17 -0
  78. package/src/providers/tasks/todoist.js +286 -0
  79. package/src/providers/types.js +9 -0
  80. package/src/providers/video-conferencing/google-meet.js +286 -0
  81. package/src/providers/video-conferencing/index.js +31 -0
  82. package/src/providers/video-conferencing/jitsi.js +254 -0
  83. package/src/providers/video-conferencing/teams.js +270 -0
  84. package/src/providers/video-conferencing/zoom.js +332 -0
  85. package/src/registry.js +128 -0
  86. package/src/tools/communication.js +184 -0
  87. package/src/tools/data.js +205 -0
  88. package/src/tools/index.js +11 -0
  89. package/src/tools/web.js +137 -0
  90. package/src/types.js +10 -0
  91. package/test/define.test.js +306 -0
  92. package/test/registry.test.js +357 -0
  93. package/test/tools.test.js +363 -0
@@ -0,0 +1,363 @@
1
+ /**
2
+ * Tests for Pre-built Tools
3
+ *
4
+ * Covers web, data, and communication tools.
5
+ */
6
+ import { describe, it, expect, beforeEach } from 'vitest';
7
+ import {
8
+ // Web tools
9
+ fetchUrl, parseHtml, readUrl, webTools,
10
+ // Data tools
11
+ parseJson, stringifyJson, parseCsv, transformData, filterData, dataTools,
12
+ // Communication tools
13
+ sendEmail, sendSlackMessage, sendNotification, sendSms, communicationTools,
14
+ // Utilities
15
+ getBuiltinTools, registerBuiltinTools, registry, } from '../src/index.js';
16
+ describe('Web Tools', () => {
17
+ describe('fetchUrl', () => {
18
+ it('has correct metadata', () => {
19
+ expect(fetchUrl.id).toBe('web.fetch');
20
+ expect(fetchUrl.name).toBe('Fetch URL');
21
+ expect(fetchUrl.category).toBe('web');
22
+ expect(fetchUrl.subcategory).toBe('fetch');
23
+ });
24
+ it('has required parameters', () => {
25
+ const urlParam = fetchUrl.parameters.find(p => p.name === 'url');
26
+ expect(urlParam).toBeDefined();
27
+ expect(urlParam?.required).toBe(true);
28
+ });
29
+ it('has optional parameters', () => {
30
+ const methodParam = fetchUrl.parameters.find(p => p.name === 'method');
31
+ expect(methodParam).toBeDefined();
32
+ expect(methodParam?.required).toBe(false);
33
+ });
34
+ it('has audience and tags', () => {
35
+ expect(fetchUrl.audience).toBe('both');
36
+ expect(fetchUrl.tags).toContain('http');
37
+ });
38
+ });
39
+ describe('parseHtml', () => {
40
+ it('has correct metadata', () => {
41
+ expect(parseHtml.id).toBe('web.parse-html');
42
+ expect(parseHtml.category).toBe('web');
43
+ expect(parseHtml.subcategory).toBe('scrape');
44
+ });
45
+ it('extracts text from HTML', async () => {
46
+ const result = await parseHtml.handler({
47
+ html: '<p>Hello World</p>',
48
+ });
49
+ expect(result.text).toContain('Hello World');
50
+ });
51
+ it('extracts links from HTML', async () => {
52
+ const result = await parseHtml.handler({
53
+ html: '<a href="https://example.com">Link</a>',
54
+ });
55
+ expect(result.links).toContain('https://example.com');
56
+ });
57
+ it('extracts images from HTML', async () => {
58
+ const result = await parseHtml.handler({
59
+ html: '<img src="image.jpg" />',
60
+ });
61
+ expect(result.images).toContain('image.jpg');
62
+ });
63
+ });
64
+ describe('readUrl', () => {
65
+ it('has correct metadata', () => {
66
+ expect(readUrl.id).toBe('web.read');
67
+ expect(readUrl.category).toBe('web');
68
+ expect(readUrl.idempotent).toBe(true);
69
+ });
70
+ });
71
+ describe('webTools array', () => {
72
+ it('contains all web tools', () => {
73
+ expect(webTools).toHaveLength(3);
74
+ expect(webTools.map(t => t.id)).toContain('web.fetch');
75
+ expect(webTools.map(t => t.id)).toContain('web.parse-html');
76
+ expect(webTools.map(t => t.id)).toContain('web.read');
77
+ });
78
+ });
79
+ });
80
+ describe('Data Tools', () => {
81
+ describe('parseJson', () => {
82
+ it('has correct metadata', () => {
83
+ expect(parseJson.id).toBe('data.json.parse');
84
+ expect(parseJson.category).toBe('data');
85
+ expect(parseJson.subcategory).toBe('transform');
86
+ });
87
+ it('parses valid JSON', async () => {
88
+ const result = await parseJson.handler({
89
+ text: '{"name": "John", "age": 30}',
90
+ });
91
+ expect(result.valid).toBe(true);
92
+ expect(result.data).toEqual({ name: 'John', age: 30 });
93
+ });
94
+ it('handles invalid JSON', async () => {
95
+ const result = await parseJson.handler({
96
+ text: '{invalid}',
97
+ });
98
+ expect(result.valid).toBe(false);
99
+ expect(result.error).toBeDefined();
100
+ });
101
+ it('is idempotent', () => {
102
+ expect(parseJson.idempotent).toBe(true);
103
+ });
104
+ });
105
+ describe('stringifyJson', () => {
106
+ it('has correct metadata', () => {
107
+ expect(stringifyJson.id).toBe('data.json.stringify');
108
+ expect(stringifyJson.category).toBe('data');
109
+ });
110
+ it('stringifies object', async () => {
111
+ const result = await stringifyJson.handler({
112
+ data: { name: 'John' },
113
+ });
114
+ expect(result.text).toBe('{"name":"John"}');
115
+ });
116
+ it('supports pretty printing', async () => {
117
+ const result = await stringifyJson.handler({
118
+ data: { name: 'John' },
119
+ pretty: true,
120
+ });
121
+ expect(result.text).toContain('\n');
122
+ expect(result.text).toContain(' ');
123
+ });
124
+ });
125
+ describe('parseCsv', () => {
126
+ it('has correct metadata', () => {
127
+ expect(parseCsv.id).toBe('data.csv.parse');
128
+ expect(parseCsv.category).toBe('data');
129
+ });
130
+ it('parses CSV with headers', async () => {
131
+ const result = await parseCsv.handler({
132
+ text: 'name,age\nJohn,30\nJane,25',
133
+ });
134
+ expect(result.headers).toEqual(['name', 'age']);
135
+ expect(result.rows).toHaveLength(2);
136
+ expect(result.rows[0]).toEqual({ name: 'John', age: '30' });
137
+ });
138
+ it('parses CSV without headers', async () => {
139
+ const result = await parseCsv.handler({
140
+ text: 'John,30\nJane,25',
141
+ hasHeaders: false,
142
+ });
143
+ expect(result.headers).toEqual(['column1', 'column2']);
144
+ expect(result.rows).toHaveLength(2);
145
+ });
146
+ it('supports custom delimiter', async () => {
147
+ const result = await parseCsv.handler({
148
+ text: 'name;age\nJohn;30',
149
+ delimiter: ';',
150
+ });
151
+ expect(result.rows[0]).toEqual({ name: 'John', age: '30' });
152
+ });
153
+ it('handles empty input', async () => {
154
+ const result = await parseCsv.handler({
155
+ text: '',
156
+ });
157
+ expect(result.rows).toHaveLength(0);
158
+ expect(result.rowCount).toBe(0);
159
+ });
160
+ });
161
+ describe('transformData', () => {
162
+ it('has correct metadata', () => {
163
+ expect(transformData.id).toBe('data.transform');
164
+ expect(transformData.category).toBe('data');
165
+ });
166
+ it('transforms data with field mapping', async () => {
167
+ const result = await transformData.handler({
168
+ data: {
169
+ user: {
170
+ firstName: 'John',
171
+ lastName: 'Doe',
172
+ },
173
+ email: 'john@example.com',
174
+ },
175
+ transform: {
176
+ name: 'user.firstName',
177
+ contact: 'email',
178
+ },
179
+ });
180
+ expect(result.result).toEqual({
181
+ name: 'John',
182
+ contact: 'john@example.com',
183
+ });
184
+ });
185
+ it('handles nested paths', async () => {
186
+ const result = await transformData.handler({
187
+ data: { a: { b: { c: 'deep' } } },
188
+ transform: { value: 'a.b.c' },
189
+ });
190
+ expect(result.result.value).toBe('deep');
191
+ });
192
+ it('handles missing paths', async () => {
193
+ const result = await transformData.handler({
194
+ data: { a: 1 },
195
+ transform: { value: 'b.c.d' },
196
+ });
197
+ expect(result.result.value).toBeUndefined();
198
+ });
199
+ });
200
+ describe('filterData', () => {
201
+ it('has correct metadata', () => {
202
+ expect(filterData.id).toBe('data.filter');
203
+ expect(filterData.category).toBe('data');
204
+ });
205
+ it('filters array by criteria', async () => {
206
+ const result = await filterData.handler({
207
+ data: [
208
+ { name: 'John', status: 'active' },
209
+ { name: 'Jane', status: 'inactive' },
210
+ { name: 'Bob', status: 'active' },
211
+ ],
212
+ filter: { status: 'active' },
213
+ });
214
+ expect(result.results).toHaveLength(2);
215
+ expect(result.count).toBe(2);
216
+ });
217
+ it('handles multiple filter criteria', async () => {
218
+ const result = await filterData.handler({
219
+ data: [
220
+ { name: 'John', status: 'active', role: 'admin' },
221
+ { name: 'Jane', status: 'active', role: 'user' },
222
+ { name: 'Bob', status: 'inactive', role: 'admin' },
223
+ ],
224
+ filter: { status: 'active', role: 'admin' },
225
+ });
226
+ expect(result.results).toHaveLength(1);
227
+ expect(result.results[0]).toEqual({ name: 'John', status: 'active', role: 'admin' });
228
+ });
229
+ it('returns empty for no matches', async () => {
230
+ const result = await filterData.handler({
231
+ data: [{ a: 1 }, { a: 2 }],
232
+ filter: { a: 3 },
233
+ });
234
+ expect(result.results).toHaveLength(0);
235
+ });
236
+ });
237
+ describe('dataTools array', () => {
238
+ it('contains all data tools', () => {
239
+ expect(dataTools).toHaveLength(5);
240
+ const ids = dataTools.map(t => t.id);
241
+ expect(ids).toContain('data.json.parse');
242
+ expect(ids).toContain('data.json.stringify');
243
+ expect(ids).toContain('data.csv.parse');
244
+ expect(ids).toContain('data.transform');
245
+ expect(ids).toContain('data.filter');
246
+ });
247
+ });
248
+ });
249
+ describe('Communication Tools', () => {
250
+ describe('sendEmail', () => {
251
+ it('has correct metadata', () => {
252
+ expect(sendEmail.id).toBe('communication.email.send');
253
+ expect(sendEmail.category).toBe('communication');
254
+ expect(sendEmail.subcategory).toBe('email');
255
+ });
256
+ it('has required parameters', () => {
257
+ const toParam = sendEmail.parameters.find(p => p.name === 'to');
258
+ const subjectParam = sendEmail.parameters.find(p => p.name === 'subject');
259
+ const bodyParam = sendEmail.parameters.find(p => p.name === 'body');
260
+ expect(toParam?.required).toBe(true);
261
+ expect(subjectParam?.required).toBe(true);
262
+ expect(bodyParam?.required).toBe(true);
263
+ });
264
+ it('requires confirmation', () => {
265
+ expect(sendEmail.requiresConfirmation).toBe(true);
266
+ });
267
+ it('has permissions', () => {
268
+ expect(sendEmail.permissions).toBeDefined();
269
+ expect(sendEmail.permissions?.[0]?.resource).toBe('email');
270
+ });
271
+ it('simulates sending email', async () => {
272
+ const result = await sendEmail.handler({
273
+ to: ['test@example.com'],
274
+ subject: 'Test',
275
+ body: 'Test body',
276
+ });
277
+ expect(result.success).toBe(true);
278
+ expect(result.messageId).toBeDefined();
279
+ });
280
+ });
281
+ describe('sendSlackMessage', () => {
282
+ it('has correct metadata', () => {
283
+ expect(sendSlackMessage.id).toBe('communication.slack.send');
284
+ expect(sendSlackMessage.category).toBe('communication');
285
+ expect(sendSlackMessage.subcategory).toBe('slack');
286
+ });
287
+ it('simulates sending message', async () => {
288
+ const result = await sendSlackMessage.handler({
289
+ channel: '#general',
290
+ text: 'Hello Slack!',
291
+ });
292
+ expect(result.success).toBe(true);
293
+ expect(result.ts).toBeDefined();
294
+ expect(result.channel).toBe('#general');
295
+ });
296
+ });
297
+ describe('sendNotification', () => {
298
+ it('has correct metadata', () => {
299
+ expect(sendNotification.id).toBe('communication.notify');
300
+ expect(sendNotification.category).toBe('communication');
301
+ expect(sendNotification.subcategory).toBe('notification');
302
+ });
303
+ it('simulates sending notification', async () => {
304
+ const result = await sendNotification.handler({
305
+ channel: 'email',
306
+ recipients: ['user@example.com'],
307
+ title: 'Alert',
308
+ message: 'This is an alert',
309
+ priority: 'high',
310
+ });
311
+ expect(result.success).toBe(true);
312
+ expect(result.notificationId).toBeDefined();
313
+ expect(result.delivered).toContain('user@example.com');
314
+ });
315
+ });
316
+ describe('sendSms', () => {
317
+ it('has correct metadata', () => {
318
+ expect(sendSms.id).toBe('communication.sms.send');
319
+ expect(sendSms.category).toBe('communication');
320
+ expect(sendSms.subcategory).toBe('sms');
321
+ });
322
+ it('requires confirmation', () => {
323
+ expect(sendSms.requiresConfirmation).toBe(true);
324
+ });
325
+ it('simulates sending SMS', async () => {
326
+ const result = await sendSms.handler({
327
+ to: '+15551234567',
328
+ message: 'Test SMS',
329
+ });
330
+ expect(result.success).toBe(true);
331
+ expect(result.messageId).toBeDefined();
332
+ });
333
+ });
334
+ describe('communicationTools array', () => {
335
+ it('contains all communication tools', () => {
336
+ expect(communicationTools).toHaveLength(4);
337
+ const ids = communicationTools.map(t => t.id);
338
+ expect(ids).toContain('communication.email.send');
339
+ expect(ids).toContain('communication.slack.send');
340
+ expect(ids).toContain('communication.notify');
341
+ expect(ids).toContain('communication.sms.send');
342
+ });
343
+ });
344
+ });
345
+ describe('Builtin Tools Utilities', () => {
346
+ beforeEach(() => {
347
+ registry.clear();
348
+ });
349
+ describe('getBuiltinTools', () => {
350
+ it('returns all builtin tools', () => {
351
+ const tools = getBuiltinTools();
352
+ expect(tools.length).toBe(webTools.length + dataTools.length + communicationTools.length);
353
+ });
354
+ });
355
+ describe('registerBuiltinTools', () => {
356
+ it('registers all builtin tools in registry', () => {
357
+ registerBuiltinTools();
358
+ expect(registry.has('web.fetch')).toBe(true);
359
+ expect(registry.has('data.json.parse')).toBe(true);
360
+ expect(registry.has('communication.email.send')).toBe(true);
361
+ });
362
+ });
363
+ });