import type { NextApiRequest, NextApiResponse } from "next"; import prisma from "../../../lib/prisma"; import { CalendarEvent, createEvent, updateEvent } from "../../../lib/calendarClient"; import async from "async"; import { v5 as uuidv5 } from "uuid"; import short from "short-uuid"; import { createMeeting, updateMeeting } from "../../../lib/videoClient"; import EventAttendeeMail from "../../../lib/emails/EventAttendeeMail"; import { getEventName } from "../../../lib/event"; import { LocationType } from "../../../lib/location"; import merge from "lodash.merge"; import dayjs from "dayjs"; const translator = short(); // Commented out because unused and thus throwing an error in linter. // const isAvailable = (busyTimes, time, length) => { // // Check for conflicts // let t = true; // busyTimes.forEach((busyTime) => { // const startTime = dayjs(busyTime.start); // const endTime = dayjs(busyTime.end); // // // Check if start times are the same // if (dayjs(time).format("HH:mm") == startTime.format("HH:mm")) { // t = false; // } // // // Check if time is between start and end times // if (dayjs(time).isBetween(startTime, endTime)) { // t = false; // } // // // Check if slot end time is between start and end time // if (dayjs(time).add(length, "minutes").isBetween(startTime, endTime)) { // t = false; // } // // // Check if startTime is between slot // if (startTime.isBetween(dayjs(time), dayjs(time).add(length, "minutes"))) { // t = false; // } // }); // // return t; // }; interface GetLocationRequestFromIntegrationRequest { location: string; } const getLocationRequestFromIntegration = ({ location }: GetLocationRequestFromIntegrationRequest) => { if (location === LocationType.GoogleMeet.valueOf()) { const requestId = uuidv5(location, uuidv5.URL); return { conferenceData: { createRequest: { requestId: requestId, }, }, }; } return null; }; export default async function handler(req: NextApiRequest, res: NextApiResponse): Promise { const { user } = req.query; const isTimeInPast = (time) => { return dayjs(time).isBefore(new Date(), "day"); }; if (isTimeInPast(req.body.start)) { return res .status(400) .json({ errorCode: "BookingDateInPast", message: "Attempting to create a meeting in the past." }); } let currentUser = await prisma.user.findFirst({ where: { username: user, }, select: { id: true, credentials: true, timeZone: true, email: true, name: true, }, }); // Commented out because unused and thus throwing an error in linter. // const selectedCalendars = await prisma.selectedCalendar.findMany({ // where: { // userId: currentUser.id, // }, // }); // Split credentials up into calendar credentials and video credentials let calendarCredentials = currentUser.credentials.filter((cred) => cred.type.endsWith("_calendar")); let videoCredentials = currentUser.credentials.filter((cred) => cred.type.endsWith("_video")); // Commented out because unused and thus throwing an error in linter. // const hasCalendarIntegrations = // currentUser.credentials.filter((cred) => cred.type.endsWith("_calendar")).length > 0; // const hasVideoIntegrations = // currentUser.credentials.filter((cred) => cred.type.endsWith("_video")).length > 0; // Commented out because unused and thus throwing an error in linter. // const calendarAvailability = await getBusyCalendarTimes( // currentUser.credentials, // dayjs(req.body.start).startOf("day").utc().format(), // dayjs(req.body.end).endOf("day").utc().format(), // selectedCalendars // ); // const videoAvailability = await getBusyVideoTimes( // currentUser.credentials, // dayjs(req.body.start).startOf("day").utc().format(), // dayjs(req.body.end).endOf("day").utc().format() // ); // let commonAvailability = []; // Commented out because unused and thus throwing an error in linter. // if (hasCalendarIntegrations && hasVideoIntegrations) { // commonAvailability = calendarAvailability.filter((availability) => // videoAvailability.includes(availability) // ); // } else if (hasVideoIntegrations) { // commonAvailability = videoAvailability; // } else if (hasCalendarIntegrations) { // commonAvailability = calendarAvailability; // } // Now, get the newly stored credentials (new refresh token for example). currentUser = await prisma.user.findFirst({ where: { username: user, }, select: { id: true, credentials: true, timeZone: true, email: true, name: true, }, }); calendarCredentials = currentUser.credentials.filter((cred) => cred.type.endsWith("_calendar")); videoCredentials = currentUser.credentials.filter((cred) => cred.type.endsWith("_video")); const rescheduleUid = req.body.rescheduleUid; const selectedEventType = await prisma.eventType.findFirst({ where: { userId: currentUser.id, id: req.body.eventTypeId, }, select: { eventName: true, title: true, length: true, }, }); const rawLocation = req.body.location; let evt: CalendarEvent = { type: selectedEventType.title, title: getEventName(req.body.name, selectedEventType.title, selectedEventType.eventName), description: req.body.notes, startTime: req.body.start, endTime: req.body.end, organizer: { email: currentUser.email, name: currentUser.name, timeZone: currentUser.timeZone }, attendees: [{ email: req.body.email, name: req.body.name, timeZone: req.body.timeZone }], }; // If phone or inPerson use raw location // set evt.location to req.body.location if (!rawLocation?.includes("integration")) { evt.location = rawLocation; } // If location is set to an integration location // Build proper transforms for evt object // Extend evt object with those transformations if (rawLocation?.includes("integration")) { const maybeLocationRequestObject = getLocationRequestFromIntegration({ location: rawLocation, }); evt = merge(evt, maybeLocationRequestObject); } const eventType = await prisma.eventType.findFirst({ where: { userId: currentUser.id, title: evt.type, }, select: { id: true, }, }); // TODO isAvailable was throwing an error const isAvailableToBeBooked = true; //isAvailable(commonAvailability, req.body.start, selectedEventType.length); if (!isAvailableToBeBooked) { return res.status(400).json({ message: `${currentUser.name} is unavailable at this time.` }); } let results = []; let referencesToCreate = []; if (rescheduleUid) { // Reschedule event const booking = await prisma.booking.findFirst({ where: { uid: rescheduleUid, }, select: { id: true, references: { select: { id: true, type: true, uid: true, }, }, }, }); // Use all integrations results = results.concat( await async.mapLimit(calendarCredentials, 5, async (credential) => { const bookingRefUid = booking.references.filter((ref) => ref.type === credential.type)[0].uid; return updateEvent(credential, bookingRefUid, evt) .then((response) => ({ type: credential.type, success: true, response })) .catch((e) => { console.error("updateEvent failed", e); return { type: credential.type, success: false }; }); }) ); results = results.concat( await async.mapLimit(videoCredentials, 5, async (credential) => { const bookingRefUid = booking.references.filter((ref) => ref.type === credential.type)[0].uid; return updateMeeting(credential, bookingRefUid, evt) .then((response) => ({ type: credential.type, success: true, response })) .catch((e) => { console.error("updateMeeting failed", e); return { type: credential.type, success: false }; }); }) ); if (results.length > 0 && results.every((res) => !res.success)) { res.status(500).json({ message: "Rescheduling failed" }); return; } // Clone elements referencesToCreate = [...booking.references]; // Now we can delete the old booking and its references. const bookingReferenceDeletes = prisma.bookingReference.deleteMany({ where: { bookingId: booking.id, }, }); const attendeeDeletes = prisma.attendee.deleteMany({ where: { bookingId: booking.id, }, }); const bookingDeletes = prisma.booking.delete({ where: { uid: rescheduleUid, }, }); await Promise.all([bookingReferenceDeletes, attendeeDeletes, bookingDeletes]); } else { // Schedule event results = results.concat( await async.mapLimit(calendarCredentials, 5, async (credential) => { return createEvent(credential, evt) .then((response) => ({ type: credential.type, success: true, response })) .catch((e) => { console.error("createEvent failed", e); return { type: credential.type, success: false }; }); }) ); results = results.concat( await async.mapLimit(videoCredentials, 5, async (credential) => { return createMeeting(credential, evt) .then((response) => ({ type: credential.type, success: true, response })) .catch((e) => { console.error("createMeeting failed", e); return { type: credential.type, success: false }; }); }) ); if (results.length > 0 && results.every((res) => !res.success)) { res.status(500).json({ message: "Booking failed" }); return; } referencesToCreate = results.map((result) => { return { type: result.type, uid: result.response.createdEvent.id.toString(), }; }); } const hashUID = results.length > 0 ? results[0].response.uid : translator.fromUUID(uuidv5(JSON.stringify(evt), uuidv5.URL)); // 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. if (results.length === 0) { // Legacy as well, as soon as we have a separate email integration class. Just used // to send an email even if there is no integration at all. try { const mail = new EventAttendeeMail(evt, hashUID); await mail.sendEmail(); } catch (e) { console.error("Sending legacy event mail failed", e); res.status(500).json({ message: "Booking failed" }); return; } } try { await prisma.booking.create({ data: { uid: hashUID, userId: currentUser.id, references: { create: referencesToCreate, }, eventTypeId: eventType.id, title: evt.title, description: evt.description, startTime: evt.startTime, endTime: evt.endTime, attendees: { create: evt.attendees, }, }, }); } catch (e) { console.error("Error when saving booking to db", e); res.status(500).json({ message: "Booking already exists" }); return; } res.status(204).json({}); }