digital-tools 2.0.2 → 2.1.3
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/.turbo/turbo-build.log +4 -5
- package/CHANGELOG.md +25 -0
- package/LICENSE +21 -0
- package/dist/define.d.ts.map +1 -1
- package/dist/define.js +1 -9
- package/dist/define.js.map +1 -1
- package/package.json +14 -15
- package/src/define.js +259 -0
- package/src/define.ts +1 -13
- 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,500 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Typeform Forms Provider
|
|
3
|
+
*
|
|
4
|
+
* Concrete implementation of FormsProvider using Typeform API.
|
|
5
|
+
*
|
|
6
|
+
* @packageDocumentation
|
|
7
|
+
*/
|
|
8
|
+
import { defineProvider } from '../registry.js';
|
|
9
|
+
const TYPEFORM_API_URL = 'https://api.typeform.com';
|
|
10
|
+
/**
|
|
11
|
+
* Typeform provider info
|
|
12
|
+
*/
|
|
13
|
+
export const typeformInfo = {
|
|
14
|
+
id: 'forms.typeform',
|
|
15
|
+
name: 'Typeform',
|
|
16
|
+
description: 'Beautiful, user-friendly online forms and surveys',
|
|
17
|
+
category: 'forms',
|
|
18
|
+
website: 'https://typeform.com',
|
|
19
|
+
docsUrl: 'https://developer.typeform.com',
|
|
20
|
+
requiredConfig: ['accessToken'],
|
|
21
|
+
optionalConfig: ['workspaceId'],
|
|
22
|
+
};
|
|
23
|
+
/**
|
|
24
|
+
* Create Typeform forms provider
|
|
25
|
+
*/
|
|
26
|
+
export function createTypeformProvider(config) {
|
|
27
|
+
let accessToken;
|
|
28
|
+
let workspaceId;
|
|
29
|
+
return {
|
|
30
|
+
info: typeformInfo,
|
|
31
|
+
async initialize(cfg) {
|
|
32
|
+
accessToken = cfg.accessToken;
|
|
33
|
+
workspaceId = cfg.workspaceId;
|
|
34
|
+
if (!accessToken) {
|
|
35
|
+
throw new Error('Typeform access token is required');
|
|
36
|
+
}
|
|
37
|
+
},
|
|
38
|
+
async healthCheck() {
|
|
39
|
+
const start = Date.now();
|
|
40
|
+
try {
|
|
41
|
+
const response = await fetch(`${TYPEFORM_API_URL}/me`, {
|
|
42
|
+
headers: {
|
|
43
|
+
Authorization: `Bearer ${accessToken}`,
|
|
44
|
+
},
|
|
45
|
+
});
|
|
46
|
+
return {
|
|
47
|
+
healthy: response.ok,
|
|
48
|
+
latencyMs: Date.now() - start,
|
|
49
|
+
message: response.ok ? 'Connected' : `HTTP ${response.status}`,
|
|
50
|
+
checkedAt: new Date(),
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
catch (error) {
|
|
54
|
+
return {
|
|
55
|
+
healthy: false,
|
|
56
|
+
latencyMs: Date.now() - start,
|
|
57
|
+
message: error instanceof Error ? error.message : 'Unknown error',
|
|
58
|
+
checkedAt: new Date(),
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
},
|
|
62
|
+
async dispose() {
|
|
63
|
+
// No cleanup needed
|
|
64
|
+
},
|
|
65
|
+
async createForm(form) {
|
|
66
|
+
const body = {
|
|
67
|
+
title: form.title,
|
|
68
|
+
type: 'form',
|
|
69
|
+
};
|
|
70
|
+
if (workspaceId) {
|
|
71
|
+
body.workspace = { href: `${TYPEFORM_API_URL}/workspaces/${workspaceId}` };
|
|
72
|
+
}
|
|
73
|
+
// Map fields to Typeform format
|
|
74
|
+
if (form.fields && form.fields.length > 0) {
|
|
75
|
+
body.fields = form.fields.map((field, index) => {
|
|
76
|
+
const typeformField = {
|
|
77
|
+
title: field.title,
|
|
78
|
+
ref: `field_${index}`,
|
|
79
|
+
properties: {},
|
|
80
|
+
};
|
|
81
|
+
// Map field types
|
|
82
|
+
switch (field.type) {
|
|
83
|
+
case 'text':
|
|
84
|
+
typeformField.type = 'short_text';
|
|
85
|
+
break;
|
|
86
|
+
case 'textarea':
|
|
87
|
+
typeformField.type = 'long_text';
|
|
88
|
+
break;
|
|
89
|
+
case 'email':
|
|
90
|
+
typeformField.type = 'email';
|
|
91
|
+
break;
|
|
92
|
+
case 'number':
|
|
93
|
+
typeformField.type = 'number';
|
|
94
|
+
break;
|
|
95
|
+
case 'date':
|
|
96
|
+
typeformField.type = 'date';
|
|
97
|
+
break;
|
|
98
|
+
case 'dropdown':
|
|
99
|
+
typeformField.type = 'dropdown';
|
|
100
|
+
if (field.choices) {
|
|
101
|
+
typeformField.properties = {
|
|
102
|
+
choices: field.choices.map((choice) => ({ label: choice })),
|
|
103
|
+
};
|
|
104
|
+
}
|
|
105
|
+
break;
|
|
106
|
+
case 'radio':
|
|
107
|
+
typeformField.type = 'multiple_choice';
|
|
108
|
+
if (field.choices) {
|
|
109
|
+
typeformField.properties = {
|
|
110
|
+
choices: field.choices.map((choice) => ({ label: choice })),
|
|
111
|
+
allow_multiple_selection: false,
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
break;
|
|
115
|
+
case 'checkbox':
|
|
116
|
+
typeformField.type = 'multiple_choice';
|
|
117
|
+
if (field.choices) {
|
|
118
|
+
typeformField.properties = {
|
|
119
|
+
choices: field.choices.map((choice) => ({ label: choice })),
|
|
120
|
+
allow_multiple_selection: true,
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
break;
|
|
124
|
+
case 'file':
|
|
125
|
+
typeformField.type = 'file_upload';
|
|
126
|
+
break;
|
|
127
|
+
default:
|
|
128
|
+
typeformField.type = 'short_text';
|
|
129
|
+
}
|
|
130
|
+
if (field.description) {
|
|
131
|
+
typeformField.properties = {
|
|
132
|
+
...(typeof typeformField.properties === 'object' ? typeformField.properties : {}),
|
|
133
|
+
description: field.description,
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
typeformField.validations = {
|
|
137
|
+
required: field.required || false,
|
|
138
|
+
};
|
|
139
|
+
return typeformField;
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
// Add settings
|
|
143
|
+
if (form.settings) {
|
|
144
|
+
body.settings = {
|
|
145
|
+
is_public: form.settings.isPublic !== undefined ? form.settings.isPublic : true,
|
|
146
|
+
show_progress_bar: form.settings.showProgressBar || false,
|
|
147
|
+
};
|
|
148
|
+
if (form.settings.confirmationMessage) {
|
|
149
|
+
body.thankyou_screens = [
|
|
150
|
+
{
|
|
151
|
+
title: 'Thank you!',
|
|
152
|
+
properties: {
|
|
153
|
+
show_button: false,
|
|
154
|
+
share_icons: false,
|
|
155
|
+
},
|
|
156
|
+
},
|
|
157
|
+
];
|
|
158
|
+
}
|
|
159
|
+
if (form.settings.redirectUrl) {
|
|
160
|
+
body.thankyou_screens = [
|
|
161
|
+
{
|
|
162
|
+
title: 'Thank you!',
|
|
163
|
+
properties: {
|
|
164
|
+
show_button: true,
|
|
165
|
+
button_text: 'Continue',
|
|
166
|
+
button_mode: 'redirect',
|
|
167
|
+
redirect_url: form.settings.redirectUrl,
|
|
168
|
+
},
|
|
169
|
+
},
|
|
170
|
+
];
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
try {
|
|
174
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms`, {
|
|
175
|
+
method: 'POST',
|
|
176
|
+
headers: {
|
|
177
|
+
'Content-Type': 'application/json',
|
|
178
|
+
Authorization: `Bearer ${accessToken}`,
|
|
179
|
+
},
|
|
180
|
+
body: JSON.stringify(body),
|
|
181
|
+
});
|
|
182
|
+
if (!response.ok) {
|
|
183
|
+
const errorData = await response.json().catch(() => ({}));
|
|
184
|
+
throw new Error(`Failed to create form: ${errorData?.message || response.statusText}`);
|
|
185
|
+
}
|
|
186
|
+
const data = await response.json();
|
|
187
|
+
return transformTypeformToFormData(data);
|
|
188
|
+
}
|
|
189
|
+
catch (error) {
|
|
190
|
+
throw new Error(`Failed to create form: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
191
|
+
}
|
|
192
|
+
},
|
|
193
|
+
async getForm(formId) {
|
|
194
|
+
try {
|
|
195
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms/${formId}`, {
|
|
196
|
+
headers: {
|
|
197
|
+
Authorization: `Bearer ${accessToken}`,
|
|
198
|
+
},
|
|
199
|
+
});
|
|
200
|
+
if (response.status === 404) {
|
|
201
|
+
return null;
|
|
202
|
+
}
|
|
203
|
+
if (!response.ok) {
|
|
204
|
+
throw new Error(`HTTP ${response.status}`);
|
|
205
|
+
}
|
|
206
|
+
const data = await response.json();
|
|
207
|
+
return transformTypeformToFormData(data);
|
|
208
|
+
}
|
|
209
|
+
catch (error) {
|
|
210
|
+
throw new Error(`Failed to get form: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
211
|
+
}
|
|
212
|
+
},
|
|
213
|
+
async updateForm(formId, updates) {
|
|
214
|
+
const body = {};
|
|
215
|
+
if (updates.title) {
|
|
216
|
+
body.title = updates.title;
|
|
217
|
+
}
|
|
218
|
+
if (updates.fields) {
|
|
219
|
+
body.fields = updates.fields.map((field, index) => {
|
|
220
|
+
const typeformField = {
|
|
221
|
+
title: field.title,
|
|
222
|
+
ref: `field_${index}`,
|
|
223
|
+
properties: {},
|
|
224
|
+
};
|
|
225
|
+
// Map field types (same as createForm)
|
|
226
|
+
switch (field.type) {
|
|
227
|
+
case 'text':
|
|
228
|
+
typeformField.type = 'short_text';
|
|
229
|
+
break;
|
|
230
|
+
case 'textarea':
|
|
231
|
+
typeformField.type = 'long_text';
|
|
232
|
+
break;
|
|
233
|
+
case 'email':
|
|
234
|
+
typeformField.type = 'email';
|
|
235
|
+
break;
|
|
236
|
+
case 'number':
|
|
237
|
+
typeformField.type = 'number';
|
|
238
|
+
break;
|
|
239
|
+
case 'date':
|
|
240
|
+
typeformField.type = 'date';
|
|
241
|
+
break;
|
|
242
|
+
case 'dropdown':
|
|
243
|
+
typeformField.type = 'dropdown';
|
|
244
|
+
if (field.choices) {
|
|
245
|
+
typeformField.properties = {
|
|
246
|
+
choices: field.choices.map((choice) => ({ label: choice })),
|
|
247
|
+
};
|
|
248
|
+
}
|
|
249
|
+
break;
|
|
250
|
+
case 'radio':
|
|
251
|
+
typeformField.type = 'multiple_choice';
|
|
252
|
+
if (field.choices) {
|
|
253
|
+
typeformField.properties = {
|
|
254
|
+
choices: field.choices.map((choice) => ({ label: choice })),
|
|
255
|
+
allow_multiple_selection: false,
|
|
256
|
+
};
|
|
257
|
+
}
|
|
258
|
+
break;
|
|
259
|
+
case 'checkbox':
|
|
260
|
+
typeformField.type = 'multiple_choice';
|
|
261
|
+
if (field.choices) {
|
|
262
|
+
typeformField.properties = {
|
|
263
|
+
choices: field.choices.map((choice) => ({ label: choice })),
|
|
264
|
+
allow_multiple_selection: true,
|
|
265
|
+
};
|
|
266
|
+
}
|
|
267
|
+
break;
|
|
268
|
+
case 'file':
|
|
269
|
+
typeformField.type = 'file_upload';
|
|
270
|
+
break;
|
|
271
|
+
default:
|
|
272
|
+
typeformField.type = 'short_text';
|
|
273
|
+
}
|
|
274
|
+
if (field.description) {
|
|
275
|
+
typeformField.properties = {
|
|
276
|
+
...(typeof typeformField.properties === 'object' ? typeformField.properties : {}),
|
|
277
|
+
description: field.description,
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
typeformField.validations = {
|
|
281
|
+
required: field.required || false,
|
|
282
|
+
};
|
|
283
|
+
return typeformField;
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
if (updates.settings) {
|
|
287
|
+
body.settings = {
|
|
288
|
+
...(updates.settings.isPublic !== undefined && {
|
|
289
|
+
is_public: updates.settings.isPublic,
|
|
290
|
+
}),
|
|
291
|
+
...(updates.settings.showProgressBar !== undefined && {
|
|
292
|
+
show_progress_bar: updates.settings.showProgressBar,
|
|
293
|
+
}),
|
|
294
|
+
};
|
|
295
|
+
}
|
|
296
|
+
try {
|
|
297
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms/${formId}`, {
|
|
298
|
+
method: 'PATCH',
|
|
299
|
+
headers: {
|
|
300
|
+
'Content-Type': 'application/json',
|
|
301
|
+
Authorization: `Bearer ${accessToken}`,
|
|
302
|
+
},
|
|
303
|
+
body: JSON.stringify(body),
|
|
304
|
+
});
|
|
305
|
+
if (!response.ok) {
|
|
306
|
+
const errorData = await response.json().catch(() => ({}));
|
|
307
|
+
throw new Error(`Failed to update form: ${errorData?.message || response.statusText}`);
|
|
308
|
+
}
|
|
309
|
+
const data = await response.json();
|
|
310
|
+
return transformTypeformToFormData(data);
|
|
311
|
+
}
|
|
312
|
+
catch (error) {
|
|
313
|
+
throw new Error(`Failed to update form: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
314
|
+
}
|
|
315
|
+
},
|
|
316
|
+
async deleteForm(formId) {
|
|
317
|
+
try {
|
|
318
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms/${formId}`, {
|
|
319
|
+
method: 'DELETE',
|
|
320
|
+
headers: {
|
|
321
|
+
Authorization: `Bearer ${accessToken}`,
|
|
322
|
+
},
|
|
323
|
+
});
|
|
324
|
+
return response.ok || response.status === 204;
|
|
325
|
+
}
|
|
326
|
+
catch (error) {
|
|
327
|
+
throw new Error(`Failed to delete form: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
328
|
+
}
|
|
329
|
+
},
|
|
330
|
+
async listForms(options) {
|
|
331
|
+
const params = new URLSearchParams();
|
|
332
|
+
const pageSize = options?.limit || 10;
|
|
333
|
+
const page = options?.offset ? Math.floor(options.offset / pageSize) + 1 : 1;
|
|
334
|
+
params.set('page_size', pageSize.toString());
|
|
335
|
+
params.set('page', page.toString());
|
|
336
|
+
if (workspaceId) {
|
|
337
|
+
params.set('workspace_id', workspaceId);
|
|
338
|
+
}
|
|
339
|
+
try {
|
|
340
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms?${params}`, {
|
|
341
|
+
headers: {
|
|
342
|
+
Authorization: `Bearer ${accessToken}`,
|
|
343
|
+
},
|
|
344
|
+
});
|
|
345
|
+
if (!response.ok) {
|
|
346
|
+
throw new Error(`HTTP ${response.status}`);
|
|
347
|
+
}
|
|
348
|
+
const data = (await response.json());
|
|
349
|
+
const items = (data.items || []).map(transformTypeformToFormData);
|
|
350
|
+
return {
|
|
351
|
+
items,
|
|
352
|
+
total: data.total_items,
|
|
353
|
+
hasMore: data.page_count > page,
|
|
354
|
+
nextCursor: data.page_count > page ? ((page + 1) * pageSize).toString() : undefined,
|
|
355
|
+
};
|
|
356
|
+
}
|
|
357
|
+
catch (error) {
|
|
358
|
+
throw new Error(`Failed to list forms: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
359
|
+
}
|
|
360
|
+
},
|
|
361
|
+
async getResponses(formId, options) {
|
|
362
|
+
const params = new URLSearchParams();
|
|
363
|
+
const pageSize = options?.limit || 25;
|
|
364
|
+
params.set('page_size', pageSize.toString());
|
|
365
|
+
if (options?.since) {
|
|
366
|
+
params.set('since', options.since.toISOString());
|
|
367
|
+
}
|
|
368
|
+
if (options?.until) {
|
|
369
|
+
params.set('until', options.until.toISOString());
|
|
370
|
+
}
|
|
371
|
+
if (options?.completed !== undefined) {
|
|
372
|
+
params.set('completed', options.completed.toString());
|
|
373
|
+
}
|
|
374
|
+
if (options?.cursor) {
|
|
375
|
+
params.set('before', options.cursor);
|
|
376
|
+
}
|
|
377
|
+
try {
|
|
378
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms/${formId}/responses?${params}`, {
|
|
379
|
+
headers: {
|
|
380
|
+
Authorization: `Bearer ${accessToken}`,
|
|
381
|
+
},
|
|
382
|
+
});
|
|
383
|
+
if (!response.ok) {
|
|
384
|
+
throw new Error(`HTTP ${response.status}`);
|
|
385
|
+
}
|
|
386
|
+
const data = (await response.json());
|
|
387
|
+
const items = (data.items || []).map((item) => transformTypeformResponseToFormResponseData(formId, item));
|
|
388
|
+
return {
|
|
389
|
+
items,
|
|
390
|
+
total: data.total_items,
|
|
391
|
+
hasMore: !!data.next_cursor,
|
|
392
|
+
nextCursor: data.next_cursor,
|
|
393
|
+
};
|
|
394
|
+
}
|
|
395
|
+
catch (error) {
|
|
396
|
+
throw new Error(`Failed to get responses: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
397
|
+
}
|
|
398
|
+
},
|
|
399
|
+
async getResponse(formId, responseId) {
|
|
400
|
+
try {
|
|
401
|
+
const response = await fetch(`${TYPEFORM_API_URL}/forms/${formId}/responses`, {
|
|
402
|
+
headers: {
|
|
403
|
+
Authorization: `Bearer ${accessToken}`,
|
|
404
|
+
},
|
|
405
|
+
});
|
|
406
|
+
if (!response.ok) {
|
|
407
|
+
throw new Error(`HTTP ${response.status}`);
|
|
408
|
+
}
|
|
409
|
+
const data = (await response.json());
|
|
410
|
+
const item = data.items?.find((r) => r.response_id === responseId);
|
|
411
|
+
if (!item) {
|
|
412
|
+
return null;
|
|
413
|
+
}
|
|
414
|
+
return transformTypeformResponseToFormResponseData(formId, item);
|
|
415
|
+
}
|
|
416
|
+
catch (error) {
|
|
417
|
+
throw new Error(`Failed to get response: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
418
|
+
}
|
|
419
|
+
},
|
|
420
|
+
};
|
|
421
|
+
}
|
|
422
|
+
/**
|
|
423
|
+
* Transform Typeform API response to FormData
|
|
424
|
+
*/
|
|
425
|
+
function transformTypeformToFormData(data) {
|
|
426
|
+
// Map Typeform fields back to our format
|
|
427
|
+
const fields = (data.fields || []).map((field) => {
|
|
428
|
+
let type = 'text';
|
|
429
|
+
// Map Typeform types back to our types
|
|
430
|
+
switch (field.type) {
|
|
431
|
+
case 'short_text':
|
|
432
|
+
type = 'text';
|
|
433
|
+
break;
|
|
434
|
+
case 'long_text':
|
|
435
|
+
type = 'textarea';
|
|
436
|
+
break;
|
|
437
|
+
case 'email':
|
|
438
|
+
type = 'email';
|
|
439
|
+
break;
|
|
440
|
+
case 'number':
|
|
441
|
+
type = 'number';
|
|
442
|
+
break;
|
|
443
|
+
case 'date':
|
|
444
|
+
type = 'date';
|
|
445
|
+
break;
|
|
446
|
+
case 'dropdown':
|
|
447
|
+
type = 'dropdown';
|
|
448
|
+
break;
|
|
449
|
+
case 'multiple_choice':
|
|
450
|
+
type = field.properties?.allow_multiple_selection ? 'checkbox' : 'radio';
|
|
451
|
+
break;
|
|
452
|
+
case 'file_upload':
|
|
453
|
+
type = 'file';
|
|
454
|
+
break;
|
|
455
|
+
}
|
|
456
|
+
return {
|
|
457
|
+
type,
|
|
458
|
+
title: field.title,
|
|
459
|
+
description: field.properties?.description,
|
|
460
|
+
required: field.validations?.required || false,
|
|
461
|
+
choices: field.properties?.choices?.map((c) => c.label),
|
|
462
|
+
};
|
|
463
|
+
});
|
|
464
|
+
return {
|
|
465
|
+
id: data.id,
|
|
466
|
+
title: data.title,
|
|
467
|
+
description: data.description,
|
|
468
|
+
fields,
|
|
469
|
+
responseCount: data.response_count || 0,
|
|
470
|
+
url: data._links?.display || `https://form.typeform.com/to/${data.id}`,
|
|
471
|
+
createdAt: new Date(data.created_at),
|
|
472
|
+
updatedAt: new Date(data.last_updated_at || data.created_at),
|
|
473
|
+
};
|
|
474
|
+
}
|
|
475
|
+
/**
|
|
476
|
+
* Transform Typeform response to FormResponseData
|
|
477
|
+
*/
|
|
478
|
+
function transformTypeformResponseToFormResponseData(formId, data) {
|
|
479
|
+
const answers = (data.answers || []).map((answer) => ({
|
|
480
|
+
fieldId: answer.field.ref || answer.field.id,
|
|
481
|
+
value: answer[answer.type] || answer.text || answer.choice || answer.choices,
|
|
482
|
+
}));
|
|
483
|
+
return {
|
|
484
|
+
id: data.response_id || data.token,
|
|
485
|
+
formId,
|
|
486
|
+
answers,
|
|
487
|
+
submittedAt: new Date(data.submitted_at),
|
|
488
|
+
metadata: data.metadata
|
|
489
|
+
? {
|
|
490
|
+
ip: data.metadata.user_agent,
|
|
491
|
+
userAgent: data.metadata.user_agent,
|
|
492
|
+
referer: data.metadata.referer,
|
|
493
|
+
}
|
|
494
|
+
: undefined,
|
|
495
|
+
};
|
|
496
|
+
}
|
|
497
|
+
/**
|
|
498
|
+
* Typeform provider definition
|
|
499
|
+
*/
|
|
500
|
+
export const typeformProvider = defineProvider(typeformInfo, async (config) => createTypeformProvider(config));
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Digital Tool Providers
|
|
3
|
+
*
|
|
4
|
+
* Concrete implementations of entity types using third-party APIs.
|
|
5
|
+
*
|
|
6
|
+
* @packageDocumentation
|
|
7
|
+
*/
|
|
8
|
+
// Provider types and registry
|
|
9
|
+
export * from './types.js';
|
|
10
|
+
export { providerRegistry, createProviderRegistry, registerProvider, getProvider, createProvider, listProviders, defineProvider, } from './registry.js';
|
|
11
|
+
// Email providers
|
|
12
|
+
export { sendgridInfo, sendgridProvider, createSendGridProvider, resendInfo, resendProvider, createResendProvider, registerEmailProviders, emailProviders, } from './email/index.js';
|
|
13
|
+
// Messaging providers
|
|
14
|
+
export { slackInfo, slackProvider, createSlackProvider, twilioSmsInfo, twilioSmsProvider, createTwilioSmsProvider, registerMessagingProviders, messagingProviders, } from './messaging/index.js';
|
|
15
|
+
// Spreadsheet providers
|
|
16
|
+
export { xlsxInfo, xlsxProvider, createXlsxProvider, googleSheetsInfo, googleSheetsProvider, createGoogleSheetsProvider, registerSpreadsheetProviders, spreadsheetProviders, } from './spreadsheet/index.js';
|
|
17
|
+
// Media providers
|
|
18
|
+
export { cloudinaryInfo, cloudinaryProvider, createCloudinaryProvider, registerMediaProviders, mediaProviders, } from './media/index.js';
|
|
19
|
+
// Task providers
|
|
20
|
+
export { todoistInfo, todoistProvider, createTodoistProvider, registerTaskProviders, taskProviders, } from './tasks/index.js';
|
|
21
|
+
// Finance providers
|
|
22
|
+
export { stripeInfo, stripeProvider, createStripeProvider, registerFinanceProviders, financeProviders, } from './finance/index.js';
|
|
23
|
+
// Support providers
|
|
24
|
+
export { zendeskInfo, zendeskProvider, createZendeskProvider, registerSupportProviders, supportProviders, } from './support/index.js';
|
|
25
|
+
// Project management providers
|
|
26
|
+
export { linearInfo, linearProvider, createLinearProvider, registerProjectManagementProviders, projectManagementProviders, } from './project-management/index.js';
|
|
27
|
+
// Knowledge providers
|
|
28
|
+
export { notionInfo, notionProvider, createNotionProvider, registerKnowledgeProviders, knowledgeProviders, } from './knowledge/index.js';
|
|
29
|
+
// CRM providers
|
|
30
|
+
export { hubspotInfo, hubspotProvider, createHubSpotProvider, registerCRMProviders, crmProviders, } from './crm/index.js';
|
|
31
|
+
// E-commerce providers
|
|
32
|
+
export { shopifyInfo, shopifyProvider, createShopifyProvider, registerEcommerceProviders, ecommerceProviders, } from './ecommerce/index.js';
|
|
33
|
+
// Development providers
|
|
34
|
+
export { githubInfo, githubProvider, createGitHubProvider, registerDevelopmentProviders, developmentProviders, } from './development/index.js';
|
|
35
|
+
// Forms providers
|
|
36
|
+
export { typeformInfo, typeformProvider, createTypeformProvider, registerFormsProviders, formsProviders, } from './forms/index.js';
|
|
37
|
+
// Analytics providers
|
|
38
|
+
export { mixpanelInfo, mixpanelProvider, createMixpanelProvider, registerAnalyticsProviders, analyticsProviders, } from './analytics/index.js';
|
|
39
|
+
// Video conferencing providers
|
|
40
|
+
export { zoomInfo, zoomProvider, createZoomProvider, googleMeetInfo, googleMeetProvider, createGoogleMeetProvider, teamsInfo, teamsProvider, createTeamsProvider, jitsiInfo, jitsiProvider, createJitsiProvider, registerVideoConferencingProviders, videoConferencingProviders, } from './video-conferencing/index.js';
|
|
41
|
+
// Calendar providers
|
|
42
|
+
export { googleCalendarInfo, googleCalendarProvider, createGoogleCalendarProvider, registerCalendarProviders, calendarProviders, } from './calendar/index.js';
|
|
43
|
+
// Storage providers
|
|
44
|
+
export { s3Info, s3Provider, createS3Provider, registerStorageProviders, storageProviders, } from './storage/index.js';
|
|
45
|
+
// Import for registration
|
|
46
|
+
import { registerEmailProviders } from './email/index.js';
|
|
47
|
+
import { registerMessagingProviders } from './messaging/index.js';
|
|
48
|
+
import { registerSpreadsheetProviders } from './spreadsheet/index.js';
|
|
49
|
+
import { registerMediaProviders } from './media/index.js';
|
|
50
|
+
import { registerTaskProviders } from './tasks/index.js';
|
|
51
|
+
import { registerFinanceProviders } from './finance/index.js';
|
|
52
|
+
import { registerKnowledgeProviders } from './knowledge/index.js';
|
|
53
|
+
import { registerCRMProviders } from './crm/index.js';
|
|
54
|
+
import { registerSupportProviders } from './support/index.js';
|
|
55
|
+
import { registerProjectManagementProviders } from './project-management/index.js';
|
|
56
|
+
import { registerEcommerceProviders } from './ecommerce/index.js';
|
|
57
|
+
import { registerDevelopmentProviders } from './development/index.js';
|
|
58
|
+
import { registerFormsProviders } from './forms/index.js';
|
|
59
|
+
import { registerAnalyticsProviders } from './analytics/index.js';
|
|
60
|
+
import { registerVideoConferencingProviders } from './video-conferencing/index.js';
|
|
61
|
+
import { registerCalendarProviders } from './calendar/index.js';
|
|
62
|
+
import { registerStorageProviders } from './storage/index.js';
|
|
63
|
+
/**
|
|
64
|
+
* Register all built-in providers in the global registry
|
|
65
|
+
*/
|
|
66
|
+
export function registerAllProviders() {
|
|
67
|
+
registerEmailProviders();
|
|
68
|
+
registerMessagingProviders();
|
|
69
|
+
registerSpreadsheetProviders();
|
|
70
|
+
registerMediaProviders();
|
|
71
|
+
registerTaskProviders();
|
|
72
|
+
registerFinanceProviders();
|
|
73
|
+
registerKnowledgeProviders();
|
|
74
|
+
registerSupportProviders();
|
|
75
|
+
registerProjectManagementProviders();
|
|
76
|
+
registerCRMProviders();
|
|
77
|
+
registerEcommerceProviders();
|
|
78
|
+
registerDevelopmentProviders();
|
|
79
|
+
registerFormsProviders();
|
|
80
|
+
registerAnalyticsProviders();
|
|
81
|
+
registerVideoConferencingProviders();
|
|
82
|
+
registerCalendarProviders();
|
|
83
|
+
registerStorageProviders();
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* All providers by category (single-word identifiers)
|
|
87
|
+
*/
|
|
88
|
+
export const allProviders = {
|
|
89
|
+
// Message category (email, messaging)
|
|
90
|
+
email: ['email.sendgrid', 'email.resend'],
|
|
91
|
+
messaging: ['messaging.slack', 'messaging.twilio-sms'],
|
|
92
|
+
// Productivity
|
|
93
|
+
calendar: ['calendar.google-calendar', 'calendar.cal-com'],
|
|
94
|
+
task: ['task.todoist'],
|
|
95
|
+
// Project management & Code
|
|
96
|
+
project: ['project.linear'],
|
|
97
|
+
code: ['code.github'],
|
|
98
|
+
// Business
|
|
99
|
+
sales: ['sales.hubspot'],
|
|
100
|
+
finance: ['finance.stripe'],
|
|
101
|
+
support: ['support.zendesk'],
|
|
102
|
+
commerce: ['commerce.shopify'],
|
|
103
|
+
// Content
|
|
104
|
+
knowledge: ['knowledge.notion'],
|
|
105
|
+
media: ['media.cloudinary'],
|
|
106
|
+
form: ['form.typeform'],
|
|
107
|
+
analytics: ['analytics.mixpanel'],
|
|
108
|
+
// Office suite
|
|
109
|
+
document: [], // TODO: Google Docs, Office365, Markdown providers
|
|
110
|
+
spreadsheet: ['spreadsheet.xlsx', 'spreadsheet.google-sheets'],
|
|
111
|
+
presentation: [], // TODO: Google Slides, PowerPoint providers
|
|
112
|
+
// Communication
|
|
113
|
+
meeting: [
|
|
114
|
+
'meeting.zoom',
|
|
115
|
+
'meeting.google-meet',
|
|
116
|
+
'meeting.teams',
|
|
117
|
+
'meeting.jitsi',
|
|
118
|
+
],
|
|
119
|
+
// Infrastructure
|
|
120
|
+
storage: ['storage.s3'],
|
|
121
|
+
// Signature (e-sign)
|
|
122
|
+
signature: [], // TODO: DocuSign, DocuSeal providers
|
|
123
|
+
};
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Knowledge Providers
|
|
3
|
+
*
|
|
4
|
+
* @packageDocumentation
|
|
5
|
+
*/
|
|
6
|
+
export { notionInfo, notionProvider, createNotionProvider } from './notion.js';
|
|
7
|
+
import { notionProvider } from './notion.js';
|
|
8
|
+
/**
|
|
9
|
+
* Register all knowledge providers
|
|
10
|
+
*/
|
|
11
|
+
export function registerKnowledgeProviders() {
|
|
12
|
+
notionProvider.register();
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* All knowledge providers
|
|
16
|
+
*/
|
|
17
|
+
export const knowledgeProviders = [notionProvider];
|