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.
- package/CHANGELOG.md +17 -0
- package/package.json +3 -4
- package/src/define.js +267 -0
- package/src/entities/advertising.js +999 -0
- package/src/entities/ai.js +756 -0
- package/src/entities/analytics.js +1588 -0
- package/src/entities/automation.js +601 -0
- package/src/entities/communication.js +1150 -0
- package/src/entities/crm.js +1386 -0
- package/src/entities/design.js +546 -0
- package/src/entities/development.js +2212 -0
- package/src/entities/document.js +874 -0
- package/src/entities/ecommerce.js +1429 -0
- package/src/entities/experiment.js +1039 -0
- package/src/entities/finance.js +3478 -0
- package/src/entities/forms.js +1892 -0
- package/src/entities/hr.js +661 -0
- package/src/entities/identity.js +997 -0
- package/src/entities/index.js +282 -0
- package/src/entities/infrastructure.js +1153 -0
- package/src/entities/knowledge.js +1438 -0
- package/src/entities/marketing.js +1610 -0
- package/src/entities/media.js +1634 -0
- package/src/entities/notification.js +1199 -0
- package/src/entities/presentation.js +1274 -0
- package/src/entities/productivity.js +1317 -0
- package/src/entities/project-management.js +1136 -0
- package/src/entities/recruiting.js +736 -0
- package/src/entities/shipping.js +509 -0
- package/src/entities/signature.js +1102 -0
- package/src/entities/site.js +222 -0
- package/src/entities/spreadsheet.js +1341 -0
- package/src/entities/storage.js +1198 -0
- package/src/entities/support.js +1166 -0
- package/src/entities/video-conferencing.js +1750 -0
- package/src/entities/video.js +950 -0
- package/src/entities.js +1663 -0
- package/src/index.js +74 -0
- package/src/providers/analytics/index.js +17 -0
- package/src/providers/analytics/mixpanel.js +255 -0
- package/src/providers/calendar/cal-com.js +303 -0
- package/src/providers/calendar/google-calendar.js +335 -0
- package/src/providers/calendar/index.js +20 -0
- package/src/providers/crm/hubspot.js +566 -0
- package/src/providers/crm/index.js +17 -0
- package/src/providers/development/github.js +472 -0
- package/src/providers/development/index.js +17 -0
- package/src/providers/ecommerce/index.js +17 -0
- package/src/providers/ecommerce/shopify.js +378 -0
- package/src/providers/email/index.js +20 -0
- package/src/providers/email/resend.js +258 -0
- package/src/providers/email/sendgrid.js +161 -0
- package/src/providers/finance/index.js +17 -0
- package/src/providers/finance/stripe.js +549 -0
- package/src/providers/forms/index.js +17 -0
- package/src/providers/forms/typeform.js +500 -0
- package/src/providers/index.js +123 -0
- package/src/providers/knowledge/index.js +17 -0
- package/src/providers/knowledge/notion.js +389 -0
- package/src/providers/marketing/index.js +17 -0
- package/src/providers/marketing/mailchimp.js +443 -0
- package/src/providers/media/cloudinary.js +318 -0
- package/src/providers/media/index.js +17 -0
- package/src/providers/messaging/index.js +20 -0
- package/src/providers/messaging/slack.js +393 -0
- package/src/providers/messaging/twilio-sms.js +249 -0
- package/src/providers/project-management/index.js +17 -0
- package/src/providers/project-management/linear.js +575 -0
- package/src/providers/registry.js +86 -0
- package/src/providers/spreadsheet/google-sheets.js +375 -0
- package/src/providers/spreadsheet/index.js +20 -0
- package/src/providers/spreadsheet/xlsx.js +423 -0
- package/src/providers/storage/index.js +24 -0
- package/src/providers/storage/s3.js +419 -0
- package/src/providers/support/index.js +17 -0
- package/src/providers/support/zendesk.js +373 -0
- package/src/providers/tasks/index.js +17 -0
- package/src/providers/tasks/todoist.js +286 -0
- package/src/providers/types.js +9 -0
- package/src/providers/video-conferencing/google-meet.js +286 -0
- package/src/providers/video-conferencing/index.js +31 -0
- package/src/providers/video-conferencing/jitsi.js +254 -0
- package/src/providers/video-conferencing/teams.js +270 -0
- package/src/providers/video-conferencing/zoom.js +332 -0
- package/src/registry.js +128 -0
- package/src/tools/communication.js +184 -0
- package/src/tools/data.js +205 -0
- package/src/tools/index.js +11 -0
- package/src/tools/web.js +137 -0
- package/src/types.js +10 -0
- package/test/define.test.js +306 -0
- package/test/registry.test.js +357 -0
- 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));
|