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,443 @@
1
+ /**
2
+ * Mailchimp Marketing Provider
3
+ *
4
+ * Concrete implementation of MarketingProvider using Mailchimp Marketing API v3.
5
+ *
6
+ * @packageDocumentation
7
+ */
8
+ import { defineProvider } from '../registry.js';
9
+ /**
10
+ * Mailchimp provider info
11
+ */
12
+ export const mailchimpInfo = {
13
+ id: 'marketing.mailchimp',
14
+ name: 'Mailchimp',
15
+ description: 'Mailchimp email marketing and automation platform',
16
+ category: 'marketing',
17
+ website: 'https://mailchimp.com',
18
+ docsUrl: 'https://mailchimp.com/developer/marketing/api/',
19
+ requiredConfig: ['apiKey', 'serverPrefix'],
20
+ optionalConfig: [],
21
+ };
22
+ /**
23
+ * Create Mailchimp marketing provider
24
+ */
25
+ export function createMailchimpProvider(config) {
26
+ let apiKey;
27
+ let serverPrefix;
28
+ let baseUrl;
29
+ /**
30
+ * Make authenticated API request
31
+ */
32
+ async function makeRequest(endpoint, options = {}) {
33
+ const url = `${baseUrl}${endpoint}`;
34
+ const response = await fetch(url, {
35
+ ...options,
36
+ headers: {
37
+ 'Content-Type': 'application/json',
38
+ Authorization: `Bearer ${apiKey}`,
39
+ ...options.headers,
40
+ },
41
+ });
42
+ if (!response.ok) {
43
+ const errorData = await response.json().catch(() => ({}));
44
+ throw new Error(`Mailchimp API error: ${response.status} - ${errorData?.title || errorData?.detail || response.statusText}`);
45
+ }
46
+ if (response.status === 204) {
47
+ return {};
48
+ }
49
+ return response.json();
50
+ }
51
+ /**
52
+ * Convert Mailchimp subscriber status to standard format
53
+ */
54
+ function mapStatus(status) {
55
+ switch (status) {
56
+ case 'subscribed':
57
+ return 'subscribed';
58
+ case 'unsubscribed':
59
+ return 'unsubscribed';
60
+ case 'cleaned':
61
+ case 'pending':
62
+ case 'transactional':
63
+ return status;
64
+ default:
65
+ return 'unknown';
66
+ }
67
+ }
68
+ return {
69
+ info: mailchimpInfo,
70
+ async initialize(cfg) {
71
+ apiKey = cfg.apiKey;
72
+ serverPrefix = cfg.serverPrefix;
73
+ if (!apiKey) {
74
+ throw new Error('Mailchimp API key is required');
75
+ }
76
+ if (!serverPrefix) {
77
+ throw new Error('Mailchimp server prefix is required (e.g., "us1")');
78
+ }
79
+ baseUrl = `https://${serverPrefix}.api.mailchimp.com/3.0`;
80
+ },
81
+ async healthCheck() {
82
+ const start = Date.now();
83
+ try {
84
+ await makeRequest('/ping');
85
+ return {
86
+ healthy: true,
87
+ latencyMs: Date.now() - start,
88
+ message: 'Connected',
89
+ checkedAt: new Date(),
90
+ };
91
+ }
92
+ catch (error) {
93
+ return {
94
+ healthy: false,
95
+ latencyMs: Date.now() - start,
96
+ message: error instanceof Error ? error.message : 'Unknown error',
97
+ checkedAt: new Date(),
98
+ };
99
+ }
100
+ },
101
+ async dispose() {
102
+ // No cleanup needed
103
+ },
104
+ async listAudiences() {
105
+ const response = await makeRequest('/lists?count=1000');
106
+ return response.lists.map((list) => ({
107
+ id: list.id,
108
+ name: list.name,
109
+ memberCount: list.stats.member_count,
110
+ createdAt: new Date(list.date_created),
111
+ }));
112
+ },
113
+ async getAudience(audienceId) {
114
+ try {
115
+ const response = await makeRequest(`/lists/${audienceId}`);
116
+ return {
117
+ id: response.id,
118
+ name: response.name,
119
+ memberCount: response.stats.member_count,
120
+ createdAt: new Date(response.date_created),
121
+ };
122
+ }
123
+ catch (error) {
124
+ if (error instanceof Error && error.message.includes('404')) {
125
+ return null;
126
+ }
127
+ throw error;
128
+ }
129
+ },
130
+ async addSubscriber(audienceId, subscriber) {
131
+ const body = {
132
+ email_address: subscriber.email,
133
+ status: subscriber.status || 'subscribed',
134
+ };
135
+ if (subscriber.firstName || subscriber.lastName) {
136
+ body.merge_fields = {
137
+ ...(subscriber.firstName && { FNAME: subscriber.firstName }),
138
+ ...(subscriber.lastName && { LNAME: subscriber.lastName }),
139
+ ...subscriber.mergeFields,
140
+ };
141
+ }
142
+ else if (subscriber.mergeFields) {
143
+ body.merge_fields = subscriber.mergeFields;
144
+ }
145
+ if (subscriber.tags?.length) {
146
+ body.tags = subscriber.tags;
147
+ }
148
+ const response = await makeRequest(`/lists/${audienceId}/members`, {
149
+ method: 'POST',
150
+ body: JSON.stringify(body),
151
+ });
152
+ return {
153
+ id: response.id,
154
+ email: response.email_address,
155
+ firstName: response.merge_fields?.FNAME,
156
+ lastName: response.merge_fields?.LNAME,
157
+ status: mapStatus(response.status),
158
+ tags: response.tags.map((t) => t.name),
159
+ subscribedAt: response.timestamp_signup
160
+ ? new Date(response.timestamp_signup)
161
+ : response.timestamp_opt
162
+ ? new Date(response.timestamp_opt)
163
+ : undefined,
164
+ };
165
+ },
166
+ async updateSubscriber(audienceId, email, updates) {
167
+ // Mailchimp uses MD5 hash of lowercase email as subscriber ID
168
+ const crypto = await import('crypto');
169
+ const subscriberId = crypto
170
+ .createHash('md5')
171
+ .update(email.toLowerCase())
172
+ .digest('hex');
173
+ const body = {};
174
+ if (updates.email) {
175
+ body.email_address = updates.email;
176
+ }
177
+ if (updates.status) {
178
+ body.status = updates.status;
179
+ }
180
+ if (updates.firstName || updates.lastName || updates.mergeFields) {
181
+ body.merge_fields = {
182
+ ...(updates.firstName && { FNAME: updates.firstName }),
183
+ ...(updates.lastName && { LNAME: updates.lastName }),
184
+ ...updates.mergeFields,
185
+ };
186
+ }
187
+ if (updates.tags?.length) {
188
+ body.tags = updates.tags;
189
+ }
190
+ const response = await makeRequest(`/lists/${audienceId}/members/${subscriberId}`, {
191
+ method: 'PATCH',
192
+ body: JSON.stringify(body),
193
+ });
194
+ return {
195
+ id: response.id,
196
+ email: response.email_address,
197
+ firstName: response.merge_fields?.FNAME,
198
+ lastName: response.merge_fields?.LNAME,
199
+ status: mapStatus(response.status),
200
+ tags: response.tags.map((t) => t.name),
201
+ subscribedAt: response.timestamp_signup
202
+ ? new Date(response.timestamp_signup)
203
+ : response.timestamp_opt
204
+ ? new Date(response.timestamp_opt)
205
+ : undefined,
206
+ };
207
+ },
208
+ async removeSubscriber(audienceId, email) {
209
+ try {
210
+ const crypto = await import('crypto');
211
+ const subscriberId = crypto
212
+ .createHash('md5')
213
+ .update(email.toLowerCase())
214
+ .digest('hex');
215
+ await makeRequest(`/lists/${audienceId}/members/${subscriberId}`, {
216
+ method: 'DELETE',
217
+ });
218
+ return true;
219
+ }
220
+ catch (error) {
221
+ if (error instanceof Error && error.message.includes('404')) {
222
+ return false;
223
+ }
224
+ throw error;
225
+ }
226
+ },
227
+ async listSubscribers(audienceId, options) {
228
+ const params = new URLSearchParams();
229
+ params.set('count', String(options?.limit || 10));
230
+ params.set('offset', String(options?.offset || 0));
231
+ if (options?.status) {
232
+ params.set('status', options.status);
233
+ }
234
+ if (options?.sinceSubscribed) {
235
+ params.set('since_timestamp_opt', options.sinceSubscribed.toISOString());
236
+ }
237
+ const response = await makeRequest(`/lists/${audienceId}/members?${params}`);
238
+ const items = response.members.map((member) => ({
239
+ id: member.id,
240
+ email: member.email_address,
241
+ firstName: member.merge_fields?.FNAME,
242
+ lastName: member.merge_fields?.LNAME,
243
+ status: mapStatus(member.status),
244
+ tags: member.tags.map((t) => t.name),
245
+ subscribedAt: member.timestamp_signup
246
+ ? new Date(member.timestamp_signup)
247
+ : member.timestamp_opt
248
+ ? new Date(member.timestamp_opt)
249
+ : undefined,
250
+ }));
251
+ const offset = options?.offset || 0;
252
+ const limit = options?.limit || 10;
253
+ return {
254
+ items,
255
+ total: response.total_items,
256
+ hasMore: offset + limit < response.total_items,
257
+ };
258
+ },
259
+ async createCampaign(campaign) {
260
+ const body = {
261
+ type: 'regular',
262
+ recipients: {
263
+ list_id: campaign.audienceId,
264
+ },
265
+ settings: {
266
+ subject_line: campaign.subject,
267
+ title: campaign.name,
268
+ from_name: campaign.fromName,
269
+ reply_to: campaign.fromEmail,
270
+ },
271
+ };
272
+ const response = await makeRequest('/campaigns', {
273
+ method: 'POST',
274
+ body: JSON.stringify(body),
275
+ });
276
+ // Set content if provided
277
+ if (campaign.content?.html || campaign.content?.text) {
278
+ const contentBody = {};
279
+ if (campaign.content.html) {
280
+ contentBody.html = campaign.content.html;
281
+ }
282
+ if (campaign.content.text) {
283
+ contentBody.plain_text = campaign.content.text;
284
+ }
285
+ await makeRequest(`/campaigns/${response.id}/content`, {
286
+ method: 'PUT',
287
+ body: JSON.stringify(contentBody),
288
+ });
289
+ }
290
+ return {
291
+ id: response.id,
292
+ name: response.settings.title,
293
+ status: response.status,
294
+ audienceId: response.recipients.list_id,
295
+ subject: response.settings.subject_line,
296
+ fromName: response.settings.from_name,
297
+ fromEmail: response.settings.reply_to,
298
+ sentAt: response.send_time ? new Date(response.send_time) : undefined,
299
+ createdAt: new Date(response.create_time),
300
+ };
301
+ },
302
+ async getCampaign(campaignId) {
303
+ try {
304
+ const response = await makeRequest(`/campaigns/${campaignId}`);
305
+ return {
306
+ id: response.id,
307
+ name: response.settings.title,
308
+ status: response.status,
309
+ audienceId: response.recipients.list_id,
310
+ subject: response.settings.subject_line,
311
+ fromName: response.settings.from_name,
312
+ fromEmail: response.settings.reply_to,
313
+ sentAt: response.send_time ? new Date(response.send_time) : undefined,
314
+ createdAt: new Date(response.create_time),
315
+ };
316
+ }
317
+ catch (error) {
318
+ if (error instanceof Error && error.message.includes('404')) {
319
+ return null;
320
+ }
321
+ throw error;
322
+ }
323
+ },
324
+ async updateCampaign(campaignId, updates) {
325
+ const body = {};
326
+ if (updates.name || updates.subject || updates.fromName || updates.fromEmail) {
327
+ body.settings = {
328
+ ...(updates.name && { title: updates.name }),
329
+ ...(updates.subject && { subject_line: updates.subject }),
330
+ ...(updates.fromName && { from_name: updates.fromName }),
331
+ ...(updates.fromEmail && { reply_to: updates.fromEmail }),
332
+ };
333
+ }
334
+ if (updates.audienceId) {
335
+ body.recipients = { list_id: updates.audienceId };
336
+ }
337
+ const response = await makeRequest(`/campaigns/${campaignId}`, {
338
+ method: 'PATCH',
339
+ body: JSON.stringify(body),
340
+ });
341
+ // Update content if provided
342
+ if (updates.content?.html || updates.content?.text) {
343
+ const contentBody = {};
344
+ if (updates.content.html) {
345
+ contentBody.html = updates.content.html;
346
+ }
347
+ if (updates.content.text) {
348
+ contentBody.plain_text = updates.content.text;
349
+ }
350
+ await makeRequest(`/campaigns/${response.id}/content`, {
351
+ method: 'PUT',
352
+ body: JSON.stringify(contentBody),
353
+ });
354
+ }
355
+ return {
356
+ id: response.id,
357
+ name: response.settings.title,
358
+ status: response.status,
359
+ audienceId: response.recipients.list_id,
360
+ subject: response.settings.subject_line,
361
+ fromName: response.settings.from_name,
362
+ fromEmail: response.settings.reply_to,
363
+ sentAt: response.send_time ? new Date(response.send_time) : undefined,
364
+ createdAt: new Date(response.create_time),
365
+ };
366
+ },
367
+ async listCampaigns(options) {
368
+ const params = new URLSearchParams();
369
+ params.set('count', String(options?.limit || 10));
370
+ params.set('offset', String(options?.offset || 0));
371
+ if (options?.status) {
372
+ params.set('status', options.status);
373
+ }
374
+ if (options?.since) {
375
+ params.set('since_create_time', options.since.toISOString());
376
+ }
377
+ const response = await makeRequest(`/campaigns?${params}`);
378
+ const items = response.campaigns.map((campaign) => ({
379
+ id: campaign.id,
380
+ name: campaign.settings.title,
381
+ status: campaign.status,
382
+ audienceId: campaign.recipients.list_id,
383
+ subject: campaign.settings.subject_line,
384
+ fromName: campaign.settings.from_name,
385
+ fromEmail: campaign.settings.reply_to,
386
+ sentAt: campaign.send_time ? new Date(campaign.send_time) : undefined,
387
+ createdAt: new Date(campaign.create_time),
388
+ }));
389
+ const offset = options?.offset || 0;
390
+ const limit = options?.limit || 10;
391
+ return {
392
+ items,
393
+ total: response.total_items,
394
+ hasMore: offset + limit < response.total_items,
395
+ };
396
+ },
397
+ async sendCampaign(campaignId) {
398
+ try {
399
+ await makeRequest(`/campaigns/${campaignId}/actions/send`, {
400
+ method: 'POST',
401
+ });
402
+ return true;
403
+ }
404
+ catch (error) {
405
+ return false;
406
+ }
407
+ },
408
+ async scheduleCampaign(campaignId, sendAt) {
409
+ try {
410
+ await makeRequest(`/campaigns/${campaignId}/actions/schedule`, {
411
+ method: 'POST',
412
+ body: JSON.stringify({
413
+ schedule_time: sendAt.toISOString(),
414
+ }),
415
+ });
416
+ return true;
417
+ }
418
+ catch (error) {
419
+ return false;
420
+ }
421
+ },
422
+ async getCampaignReport(campaignId) {
423
+ const response = await makeRequest(`/reports/${campaignId}`);
424
+ return {
425
+ campaignId: response.campaign_id,
426
+ sent: response.emails_sent,
427
+ delivered: response.emails_sent - (response.bounces.hard_bounces + response.bounces.soft_bounces),
428
+ opens: response.opens.opens_total,
429
+ uniqueOpens: response.opens.unique_opens,
430
+ clicks: response.clicks.clicks_total,
431
+ uniqueClicks: response.clicks.unique_clicks,
432
+ bounces: response.bounces.hard_bounces + response.bounces.soft_bounces,
433
+ unsubscribes: response.unsubscribed,
434
+ openRate: response.opens.open_rate,
435
+ clickRate: response.clicks.click_rate,
436
+ };
437
+ },
438
+ };
439
+ }
440
+ /**
441
+ * Mailchimp provider definition
442
+ */
443
+ export const mailchimpProvider = defineProvider(mailchimpInfo, async (config) => createMailchimpProvider(config));