Re-implemented event confirmation mails object based

This commit is contained in:
nicolas 2021-06-17 00:56:02 +02:00
parent 04e0b55b51
commit f56ced0ff1
7 changed files with 232 additions and 305 deletions

View file

@ -1,4 +1,9 @@
import EventOwnerMail from "./emails/EventOwnerMail"; import EventOwnerMail from "./emails/EventOwnerMail";
import EventAttendeeMail from "./emails/EventAttendeeMail";
import {v5 as uuidv5} from 'uuid';
import short from 'short-uuid';
const translator = short();
const {google} = require('googleapis'); const {google} = require('googleapis');
@ -324,15 +329,22 @@ const getBusyTimes = (withCredentials, dateFrom, dateTo) => Promise.all(
(results) => results.reduce((acc, availability) => acc.concat(availability), []) (results) => results.reduce((acc, availability) => acc.concat(availability), [])
); );
const createEvent = async (credential, calEvent: CalendarEvent, hashUID: string): Promise<any> => { const createEvent = async (credential, calEvent: CalendarEvent): Promise<any> => {
const mail = new EventOwnerMail(calEvent, hashUID); const uid: string = translator.fromUUID(uuidv5(JSON.stringify(calEvent), uuidv5.URL));
const sentMail = await mail.sendEmail();
const creationResult = credential ? await calendars([credential])[0].createEvent(calEvent) : null; const creationResult = credential ? await calendars([credential])[0].createEvent(calEvent) : null;
const ownerMail = new EventOwnerMail(calEvent, uid);
const attendeeMail = new EventAttendeeMail(calEvent, uid);
await ownerMail.sendEmail();
if(!creationResult || !creationResult.disableConfirmationEmail) {
await attendeeMail.sendEmail();
}
return { return {
createdEvent: creationResult, uid,
sentMail: sentMail createdEvent: creationResult
}; };
}; };

View file

@ -1,6 +1,7 @@
import {VideoCallData} from "./confirm-booked";
import {CalendarEvent} from "../calendarClient"; import {CalendarEvent} from "../calendarClient";
import EventAttendeeMail from "./EventAttendeeMail"; import EventAttendeeMail from "./EventAttendeeMail";
import {getFormattedMeetingId, getIntegrationName} from "./helpers";
import {VideoCallData} from "../videoClient";
export default class VideoEventAttendeeMail extends EventAttendeeMail { export default class VideoEventAttendeeMail extends EventAttendeeMail {
videoCallData: VideoCallData; videoCallData: VideoCallData;
@ -10,25 +11,6 @@ export default class VideoEventAttendeeMail extends EventAttendeeMail {
this.videoCallData = videoCallData; this.videoCallData = videoCallData;
} }
private getIntegrationName(): string {
//TODO: When there are more complex integration type strings, we should consider using an extra field in the DB for that.
const nameProto = this.videoCallData.type.split("_")[0];
return nameProto.charAt(0).toUpperCase() + nameProto.slice(1);
}
private getFormattedMeetingId(): string {
switch(this.videoCallData.type) {
case 'zoom_video':
const strId = this.videoCallData.id.toString();
const part1 = strId.slice(0, 3);
const part2 = strId.slice(3, 7);
const part3 = strId.slice(7, 11);
return part1 + " " + part2 + " " + part3;
default:
return this.videoCallData.id.toString();
}
}
/** /**
* Adds the video call information to the mail body. * Adds the video call information to the mail body.
* *
@ -36,8 +18,8 @@ export default class VideoEventAttendeeMail extends EventAttendeeMail {
*/ */
protected getAdditionalBody(): string { protected getAdditionalBody(): string {
return ` return `
<strong>Video call provider:</strong> ${this.getIntegrationName()}<br /> <strong>Video call provider:</strong> ${getIntegrationName(this.videoCallData)}<br />
<strong>Meeting ID:</strong> ${this.getFormattedMeetingId()}<br /> <strong>Meeting ID:</strong> ${getFormattedMeetingId(this.videoCallData)}<br />
<strong>Meeting Password:</strong> ${this.videoCallData.password}<br /> <strong>Meeting Password:</strong> ${this.videoCallData.password}<br />
<strong>Meeting URL:</strong> <a href="${this.videoCallData.url}">${this.videoCallData.url}</a><br /> <strong>Meeting URL:</strong> <a href="${this.videoCallData.url}">${this.videoCallData.url}</a><br />
`; `;

View file

@ -1,6 +1,7 @@
import {CalendarEvent} from "../calendarClient"; import {CalendarEvent} from "../calendarClient";
import EventOwnerMail from "./EventOwnerMail"; import EventOwnerMail from "./EventOwnerMail";
import {formattedId, integrationTypeToName, VideoCallData} from "./confirm-booked"; import {VideoCallData} from "../videoClient";
import {getFormattedMeetingId, getIntegrationName} from "./helpers";
export default class VideoEventOwnerMail extends EventOwnerMail { export default class VideoEventOwnerMail extends EventOwnerMail {
videoCallData: VideoCallData; videoCallData: VideoCallData;
@ -18,8 +19,8 @@ export default class VideoEventOwnerMail extends EventOwnerMail {
*/ */
protected getAdditionalBody(): string { protected getAdditionalBody(): string {
return ` return `
<strong>Video call provider:</strong> ${integrationTypeToName(this.videoCallData.type)}<br /> <strong>Video call provider:</strong> ${getIntegrationName(this.videoCallData)}<br />
<strong>Meeting ID:</strong> ${formattedId(this.videoCallData)}<br /> <strong>Meeting ID:</strong> ${getFormattedMeetingId(this.videoCallData)}<br />
<strong>Meeting Password:</strong> ${this.videoCallData.password}<br /> <strong>Meeting Password:</strong> ${this.videoCallData.password}<br />
<strong>Meeting URL:</strong> <a href="${this.videoCallData.url}">${this.videoCallData.url}</a><br /> <strong>Meeting URL:</strong> <a href="${this.videoCallData.url}">${this.videoCallData.url}</a><br />
`; `;

View file

@ -1,101 +0,0 @@
import nodemailer from 'nodemailer';
import {serverConfig} from "../serverConfig";
import {CalendarEvent} from "../calendarClient";
import dayjs, {Dayjs} from "dayjs";
import localizedFormat from "dayjs/plugin/localizedFormat";
import utc from "dayjs/plugin/utc";
import timezone from "dayjs/plugin/timezone";
dayjs.extend(localizedFormat);
dayjs.extend(utc);
dayjs.extend(timezone);
export interface VideoCallData {
type: string;
id: string;
password: string;
url: string;
};
export function integrationTypeToName(type: string): string {
//TODO: When there are more complex integration type strings, we should consider using an extra field in the DB for that.
const nameProto = type.split("_")[0];
return nameProto.charAt(0).toUpperCase() + nameProto.slice(1);
}
export function formattedId(videoCallData: VideoCallData): string {
switch(videoCallData.type) {
case 'zoom_video':
const strId = videoCallData.id.toString();
const part1 = strId.slice(0, 3);
const part2 = strId.slice(3, 7);
const part3 = strId.slice(7, 11);
return part1 + " " + part2 + " " + part3;
default:
return videoCallData.id.toString();
}
}
export default function createConfirmBookedEmail(calEvent: CalendarEvent, cancelLink: string, rescheduleLink: string, options: any = {}, videoCallData?: VideoCallData) {
return sendEmail(calEvent, cancelLink, rescheduleLink, {
provider: {
transport: serverConfig.transport,
from: serverConfig.from,
},
...options
}, videoCallData);
}
const sendEmail = (calEvent: CalendarEvent, cancelLink: string, rescheduleLink: string, {
provider,
}, videoCallData?: VideoCallData) => new Promise((resolve, reject) => {
const {from, transport} = provider;
const inviteeStart: Dayjs = <Dayjs>dayjs(calEvent.startTime).tz(calEvent.attendees[0].timeZone);
nodemailer.createTransport(transport).sendMail(
{
to: `${calEvent.attendees[0].name} <${calEvent.attendees[0].email}>`,
from: `${calEvent.organizer.name} <${from}>`,
replyTo: calEvent.organizer.email,
subject: `Confirmed: ${calEvent.type} with ${calEvent.organizer.name} on ${inviteeStart.format('dddd, LL')}`,
html: html(calEvent, cancelLink, rescheduleLink, videoCallData),
text: text(calEvent, cancelLink, rescheduleLink, videoCallData),
},
(error, info) => {
if (error) {
console.error("SEND_BOOKING_CONFIRMATION_ERROR", calEvent.attendees[0].email, error);
return reject(new Error(error));
}
return resolve();
}
)
});
const html = (calEvent: CalendarEvent, cancelLink, rescheduleLink: string, videoCallData?: VideoCallData) => {
const inviteeStart: Dayjs = <Dayjs>dayjs(calEvent.startTime).tz(calEvent.attendees[0].timeZone);
return `
<div>
Hi ${calEvent.attendees[0].name},<br />
<br />
Your ${calEvent.type} with ${calEvent.organizer.name} at ${inviteeStart.format('h:mma')}
(${calEvent.attendees[0].timeZone}) on ${inviteeStart.format('dddd, LL')} is scheduled.<br />
<br />` + (
videoCallData ? `<strong>Video call provider:</strong> ${integrationTypeToName(videoCallData.type)}<br />
<strong>Meeting ID:</strong> ${formattedId(videoCallData)}<br />
<strong>Meeting Password:</strong> ${videoCallData.password}<br />
<strong>Meeting URL:</strong> <a href="${videoCallData.url}">${videoCallData.url}</a><br /><br />` : ''
) + (
calEvent.location ? `<strong>Location:</strong> ${calEvent.location}<br /><br />` : ''
) +
`<strong>Additional notes:</strong><br />
${calEvent.description}<br />
<br />
Need to change this event?<br />
Cancel: <a href="${cancelLink}">${cancelLink}</a><br />
Reschedule: <a href="${rescheduleLink}">${rescheduleLink}</a>
</div>
`;
};
const text = (evt: CalendarEvent, cancelLink: string, rescheduleLink: string, videoCallData?: VideoCallData) => html(evt, cancelLink, rescheduleLink, videoCallData).replace('<br />', "\n").replace(/<[^>]+>/g, '');

20
lib/emails/helpers.ts Normal file
View file

@ -0,0 +1,20 @@
import {VideoCallData} from "../videoClient";
export function getIntegrationName(videoCallData: VideoCallData): string {
//TODO: When there are more complex integration type strings, we should consider using an extra field in the DB for that.
const nameProto = videoCallData.type.split("_")[0];
return nameProto.charAt(0).toUpperCase() + nameProto.slice(1);
}
export function getFormattedMeetingId(videoCallData: VideoCallData): string {
switch(videoCallData.type) {
case 'zoom_video':
const strId = videoCallData.id.toString();
const part1 = strId.slice(0, 3);
const part2 = strId.slice(3, 7);
const part3 = strId.slice(7, 11);
return part1 + " " + part2 + " " + part3;
default:
return videoCallData.id.toString();
}
}

View file

@ -1,218 +1,236 @@
import prisma from "./prisma"; import prisma from "./prisma";
import {VideoCallData} from "./emails/confirm-booked";
import {CalendarEvent} from "./calendarClient"; import {CalendarEvent} from "./calendarClient";
import VideoEventOwnerMail from "./emails/VideoEventOwnerMail"; import VideoEventOwnerMail from "./emails/VideoEventOwnerMail";
import VideoEventAttendeeMail from "./emails/VideoEventAttendeeMail";
import {v5 as uuidv5} from 'uuid';
import short from 'short-uuid';
const translator = short();
export interface VideoCallData {
type: string;
id: string;
password: string;
url: string;
}
function handleErrorsJson(response) { function handleErrorsJson(response) {
if (!response.ok) { if (!response.ok) {
response.json().then(console.log); response.json().then(console.log);
throw Error(response.statusText); throw Error(response.statusText);
} }
return response.json(); return response.json();
} }
function handleErrorsRaw(response) { function handleErrorsRaw(response) {
if (!response.ok) { if (!response.ok) {
response.text().then(console.log); response.text().then(console.log);
throw Error(response.statusText); throw Error(response.statusText);
} }
return response.text(); return response.text();
} }
const zoomAuth = (credential) => { const zoomAuth = (credential) => {
const isExpired = (expiryDate) => expiryDate < +(new Date()); const isExpired = (expiryDate) => expiryDate < +(new Date());
const authHeader = 'Basic ' + Buffer.from(process.env.ZOOM_CLIENT_ID + ':' + process.env.ZOOM_CLIENT_SECRET).toString('base64'); const authHeader = 'Basic ' + Buffer.from(process.env.ZOOM_CLIENT_ID + ':' + process.env.ZOOM_CLIENT_SECRET).toString('base64');
const refreshAccessToken = (refreshToken) => fetch('https://zoom.us/oauth/token', { const refreshAccessToken = (refreshToken) => fetch('https://zoom.us/oauth/token', {
method: 'POST', method: 'POST',
headers: { headers: {
'Authorization': authHeader, 'Authorization': authHeader,
'Content-Type': 'application/x-www-form-urlencoded' 'Content-Type': 'application/x-www-form-urlencoded'
}, },
body: new URLSearchParams({ body: new URLSearchParams({
'refresh_token': refreshToken, 'refresh_token': refreshToken,
'grant_type': 'refresh_token', 'grant_type': 'refresh_token',
}) })
})
.then(handleErrorsJson)
.then(async (responseBody) => {
// Store new tokens in database.
await prisma.credential.update({
where: {
id: credential.id
},
data: {
key: responseBody
}
});
credential.key.access_token = responseBody.access_token;
credential.key.expires_in = Math.round((+(new Date()) / 1000) + responseBody.expires_in);
return credential.key.access_token;
}) })
.then(handleErrorsJson)
.then(async (responseBody) => {
// Store new tokens in database.
await prisma.credential.update({
where: {
id: credential.id
},
data: {
key: responseBody
}
});
credential.key.access_token = responseBody.access_token;
credential.key.expires_in = Math.round((+(new Date()) / 1000) + responseBody.expires_in);
return credential.key.access_token;
})
return { return {
getToken: () => !isExpired(credential.key.expires_in) ? Promise.resolve(credential.key.access_token) : refreshAccessToken(credential.key.refresh_token) getToken: () => !isExpired(credential.key.expires_in) ? Promise.resolve(credential.key.access_token) : refreshAccessToken(credential.key.refresh_token)
}; };
}; };
interface VideoApiAdapter { interface VideoApiAdapter {
createMeeting(event: CalendarEvent): Promise<any>; createMeeting(event: CalendarEvent): Promise<any>;
updateMeeting(uid: String, event: CalendarEvent); updateMeeting(uid: String, event: CalendarEvent);
deleteMeeting(uid: String); deleteMeeting(uid: String);
getAvailability(dateFrom, dateTo): Promise<any>; getAvailability(dateFrom, dateTo): Promise<any>;
} }
const ZoomVideo = (credential): VideoApiAdapter => { const ZoomVideo = (credential): VideoApiAdapter => {
const auth = zoomAuth(credential); const auth = zoomAuth(credential);
const translateEvent = (event: CalendarEvent) => {
// Documentation at: https://marketplace.zoom.us/docs/api-reference/zoom-api/meetings/meetingcreate
return {
topic: event.title,
type: 2, // Means that this is a scheduled meeting
start_time: event.startTime,
duration: ((new Date(event.endTime)).getTime() - (new Date(event.startTime)).getTime()) / 60000,
//schedule_for: "string", TODO: Used when scheduling the meeting for someone else (needed?)
timezone: event.attendees[0].timeZone,
//password: "string", TODO: Should we use a password? Maybe generate a random one?
agenda: event.description,
settings: {
host_video: true,
participant_video: true,
cn_meeting: false, // TODO: true if host meeting in China
in_meeting: false, // TODO: true if host meeting in India
join_before_host: true,
mute_upon_entry: false,
watermark: false,
use_pmi: false,
approval_type: 2,
audio: "both",
auto_recording: "none",
enforce_login: false,
registrants_email_notification: true
}
};
};
const translateEvent = (event: CalendarEvent) => {
// Documentation at: https://marketplace.zoom.us/docs/api-reference/zoom-api/meetings/meetingcreate
return { return {
getAvailability: (dateFrom, dateTo) => { topic: event.title,
/*const payload = { type: 2, // Means that this is a scheduled meeting
schedules: [credential.key.email], start_time: event.startTime,
startTime: { duration: ((new Date(event.endTime)).getTime() - (new Date(event.startTime)).getTime()) / 60000,
dateTime: dateFrom, //schedule_for: "string", TODO: Used when scheduling the meeting for someone else (needed?)
timeZone: 'UTC', timezone: event.attendees[0].timeZone,
}, //password: "string", TODO: Should we use a password? Maybe generate a random one?
endTime: { agenda: event.description,
dateTime: dateTo, settings: {
timeZone: 'UTC', host_video: true,
}, participant_video: true,
availabilityViewInterval: 60 cn_meeting: false, // TODO: true if host meeting in China
}; in_meeting: false, // TODO: true if host meeting in India
join_before_host: true,
mute_upon_entry: false,
watermark: false,
use_pmi: false,
approval_type: 2,
audio: "both",
auto_recording: "none",
enforce_login: false,
registrants_email_notification: true
}
};
};
return auth.getToken().then( return {
(accessToken) => fetch('https://graph.microsoft.com/v1.0/me/calendar/getSchedule', { getAvailability: (dateFrom, dateTo) => {
method: 'post', /*const payload = {
headers: { schedules: [credential.key.email],
'Authorization': 'Bearer ' + accessToken, startTime: {
'Content-Type': 'application/json' dateTime: dateFrom,
}, timeZone: 'UTC',
body: JSON.stringify(payload) },
}) endTime: {
.then(handleErrorsJson) dateTime: dateTo,
.then(responseBody => { timeZone: 'UTC',
return responseBody.value[0].scheduleItems.map((evt) => ({ },
start: evt.start.dateTime + 'Z', availabilityViewInterval: 60
end: evt.end.dateTime + 'Z' };
}))
}) return auth.getToken().then(
).catch((err) => { (accessToken) => fetch('https://graph.microsoft.com/v1.0/me/calendar/getSchedule', {
console.log(err); method: 'post',
});*/ headers: {
}, 'Authorization': 'Bearer ' + accessToken,
createMeeting: (event: CalendarEvent) => auth.getToken().then(accessToken => fetch('https://api.zoom.us/v2/users/me/meetings', { 'Content-Type': 'application/json'
method: 'POST', },
headers: { body: JSON.stringify(payload)
'Authorization': 'Bearer ' + accessToken, })
'Content-Type': 'application/json', .then(handleErrorsJson)
}, .then(responseBody => {
body: JSON.stringify(translateEvent(event)) return responseBody.value[0].scheduleItems.map((evt) => ({
}).then(handleErrorsJson)), start: evt.start.dateTime + 'Z',
deleteMeeting: (uid: String) => auth.getToken().then(accessToken => fetch('https://api.zoom.us/v2/meetings/' + uid, { end: evt.end.dateTime + 'Z'
method: 'DELETE', }))
headers: { })
'Authorization': 'Bearer ' + accessToken ).catch((err) => {
} console.log(err);
}).then(handleErrorsRaw)), });*/
updateMeeting: (uid: String, event: CalendarEvent) => auth.getToken().then(accessToken => fetch('https://api.zoom.us/v2/meetings/' + uid, { },
method: 'PATCH', createMeeting: (event: CalendarEvent) => auth.getToken().then(accessToken => fetch('https://api.zoom.us/v2/users/me/meetings', {
headers: { method: 'POST',
'Authorization': 'Bearer ' + accessToken, headers: {
'Content-Type': 'application/json' 'Authorization': 'Bearer ' + accessToken,
}, 'Content-Type': 'application/json',
body: JSON.stringify(translateEvent(event)) },
}).then(handleErrorsRaw)), body: JSON.stringify(translateEvent(event))
} }).then(handleErrorsJson)),
deleteMeeting: (uid: String) => auth.getToken().then(accessToken => fetch('https://api.zoom.us/v2/meetings/' + uid, {
method: 'DELETE',
headers: {
'Authorization': 'Bearer ' + accessToken
}
}).then(handleErrorsRaw)),
updateMeeting: (uid: String, event: CalendarEvent) => auth.getToken().then(accessToken => fetch('https://api.zoom.us/v2/meetings/' + uid, {
method: 'PATCH',
headers: {
'Authorization': 'Bearer ' + accessToken,
'Content-Type': 'application/json'
},
body: JSON.stringify(translateEvent(event))
}).then(handleErrorsRaw)),
}
}; };
// factory // factory
const videoIntegrations = (withCredentials): VideoApiAdapter[] => withCredentials.map((cred) => { const videoIntegrations = (withCredentials): VideoApiAdapter[] => withCredentials.map((cred) => {
switch (cred.type) { switch (cred.type) {
case 'zoom_video': case 'zoom_video':
return ZoomVideo(cred); return ZoomVideo(cred);
default: default:
return; // unknown credential, could be legacy? In any case, ignore return; // unknown credential, could be legacy? In any case, ignore
} }
}).filter(Boolean); }).filter(Boolean);
const getBusyTimes = (withCredentials, dateFrom, dateTo) => Promise.all( const getBusyTimes = (withCredentials, dateFrom, dateTo) => Promise.all(
videoIntegrations(withCredentials).map(c => c.getAvailability(dateFrom, dateTo)) videoIntegrations(withCredentials).map(c => c.getAvailability(dateFrom, dateTo))
).then( ).then(
(results) => results.reduce((acc, availability) => acc.concat(availability), []) (results) => results.reduce((acc, availability) => acc.concat(availability), [])
); );
const createMeeting = async (credential, calEvent: CalendarEvent, hashUID: string): Promise<any> => { const createMeeting = async (credential, calEvent: CalendarEvent): Promise<any> => {
if(!credential) { const uid: string = translator.fromUUID(uuidv5(JSON.stringify(calEvent), uuidv5.URL));
throw new Error("Credentials must be set! Video platforms are optional, so this method shouldn't even be called.");
}
const creationResult = await videoIntegrations([credential])[0].createMeeting(calEvent); if (!credential) {
throw new Error("Credentials must be set! Video platforms are optional, so this method shouldn't even be called when no video credentials are set.");
}
const videoCallData: VideoCallData = { const creationResult = await videoIntegrations([credential])[0].createMeeting(calEvent);
type: credential.type,
id: creationResult.id,
password: creationResult.password,
url: creationResult.join_url,
};
const mail = new VideoEventOwnerMail(calEvent, hashUID, videoCallData); const videoCallData: VideoCallData = {
const sentMail = await mail.sendEmail(); type: credential.type,
id: creationResult.id,
password: creationResult.password,
url: creationResult.join_url,
};
return { const ownerMail = new VideoEventOwnerMail(calEvent, uid, videoCallData);
createdEvent: creationResult, const attendeeMail = new VideoEventAttendeeMail(calEvent, uid, videoCallData);
sentMail: sentMail await ownerMail.sendEmail();
};
if(!creationResult || !creationResult.disableConfirmationEmail) {
await attendeeMail.sendEmail();
}
return {
uid,
createdEvent: creationResult
};
}; };
const updateMeeting = (credential, uid: String, event: CalendarEvent): Promise<any> => { const updateMeeting = (credential, uid: String, event: CalendarEvent): Promise<any> => {
if (credential) { if (credential) {
return videoIntegrations([credential])[0].updateMeeting(uid, event); return videoIntegrations([credential])[0].updateMeeting(uid, event);
} }
return Promise.resolve({}); return Promise.resolve({});
}; };
const deleteMeeting = (credential, uid: String): Promise<any> => { const deleteMeeting = (credential, uid: String): Promise<any> => {
if (credential) { if (credential) {
return videoIntegrations([credential])[0].deleteMeeting(uid); return videoIntegrations([credential])[0].deleteMeeting(uid);
} }
return Promise.resolve({}); return Promise.resolve({});
}; };
export {getBusyTimes, createMeeting, updateMeeting, deleteMeeting}; export {getBusyTimes, createMeeting, updateMeeting, deleteMeeting};

View file

@ -43,8 +43,6 @@ export default async function handler(req: NextApiRequest, res: NextApiResponse)
] ]
}; };
const hashUID: string = translator.fromUUID(uuidv5(JSON.stringify(evt), uuidv5.URL));
const eventType = await prisma.eventType.findFirst({ const eventType = await prisma.eventType.findFirst({
where: { where: {
userId: currentUser.id, userId: currentUser.id,
@ -115,7 +113,7 @@ export default async function handler(req: NextApiRequest, res: NextApiResponse)
} else { } else {
// Schedule event // Schedule event
results = results.concat(await async.mapLimit(calendarCredentials, 5, async (credential) => { results = results.concat(await async.mapLimit(calendarCredentials, 5, async (credential) => {
const response = await createEvent(credential, evt, hashUID); const response = await createEvent(credential, evt);
return { return {
type: credential.type, type: credential.type,
response response
@ -123,7 +121,7 @@ export default async function handler(req: NextApiRequest, res: NextApiResponse)
})); }));
results = results.concat(await async.mapLimit(videoCredentials, 5, async (credential) => { results = results.concat(await async.mapLimit(videoCredentials, 5, async (credential) => {
const response = await createMeeting(credential, evt, hashUID); const response = await createMeeting(credential, evt);
return { return {
type: credential.type, type: credential.type,
response response
@ -138,6 +136,10 @@ export default async function handler(req: NextApiRequest, res: NextApiResponse)
})); }));
} }
// TODO Should just be set to the true case as soon as we have a "bare email" integration class.
// UID generation should happen in the integration itself, not here.
const hashUID = results.length > 0 ? results[0].response.uid : translator.fromUUID(uuidv5(JSON.stringify(evt), uuidv5.URL));
await prisma.booking.create({ await prisma.booking.create({
data: { data: {
uid: hashUID, uid: hashUID,
@ -158,12 +160,5 @@ export default async function handler(req: NextApiRequest, res: NextApiResponse)
} }
}); });
// If one of the integrations allows email confirmations or no integrations are added, send it.
/*if (currentUser.credentials.length === 0 || !results.every((result) => result.disableConfirmationEmail)) {
await createConfirmBookedEmail(
evt, cancelLink, rescheduleLink, {}, videoCallData
);
}*/
res.status(200).json(results); res.status(200).json(results);
} }