import { SchedulingType, Prisma, Credential } from "@prisma/client"; import async from "async"; import dayjs from "dayjs"; import dayjsBusinessDays from "dayjs-business-days"; import isBetween from "dayjs/plugin/isBetween"; import timezone from "dayjs/plugin/timezone"; import utc from "dayjs/plugin/utc"; import type { NextApiRequest, NextApiResponse } from "next"; import short from "short-uuid"; import { v5 as uuidv5 } from "uuid"; import { handlePayment } from "@ee/lib/stripe/server"; import { CalendarEvent, getBusyCalendarTimes } from "@lib/calendarClient"; import EventOrganizerRequestMail from "@lib/emails/EventOrganizerRequestMail"; import { getEventName } from "@lib/event"; import EventManager, { CreateUpdateResult, EventResult, PartialReference } from "@lib/events/EventManager"; import logger from "@lib/logger"; import prisma from "@lib/prisma"; import { BookingCreateBody } from "@lib/types/booking"; import { getBusyVideoTimes } from "@lib/videoClient"; dayjs.extend(dayjsBusinessDays); dayjs.extend(utc); dayjs.extend(isBetween); dayjs.extend(timezone); const translator = short(); const log = logger.getChildLogger({ prefix: ["[api] book:user"] }); type BufferedBusyTimes = { start: string; end: string }[]; /** * Refreshes a Credential with fresh data from the database. * * @param credential */ async function refreshCredential(credential: Credential): Promise { const newCredential = await prisma.credential.findUnique({ where: { id: credential.id, }, }); if (!newCredential) { return credential; } else { return newCredential; } } /** * Refreshes the given set of credentials. * * @param credentials */ async function refreshCredentials(credentials: Array): Promise> { return await async.mapLimit(credentials, 5, refreshCredential); } function isAvailable(busyTimes: BufferedBusyTimes, time: string, length: number): boolean { // Check for conflicts let t = true; if (Array.isArray(busyTimes) && busyTimes.length > 0) { busyTimes.forEach((busyTime) => { const startTime = dayjs(busyTime.start); const endTime = dayjs(busyTime.end); // Check if time is between start and end times if (dayjs(time).isBetween(startTime, endTime, null, "[)")) { 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; } function isOutOfBounds( time: dayjs.ConfigType, { periodType, periodDays, periodCountCalendarDays, periodStartDate, periodEndDate, timeZone } ): boolean { const date = dayjs(time); switch (periodType) { case "rolling": { const periodRollingEndDay = periodCountCalendarDays ? dayjs().tz(timeZone).add(periodDays, "days").endOf("day") : dayjs().tz(timeZone).businessDaysAdd(periodDays, "days").endOf("day"); return date.endOf("day").isAfter(periodRollingEndDay); } case "range": { const periodRangeStartDay = dayjs(periodStartDate).tz(timeZone).endOf("day"); const periodRangeEndDay = dayjs(periodEndDate).tz(timeZone).endOf("day"); return date.endOf("day").isBefore(periodRangeStartDay) || date.endOf("day").isAfter(periodRangeEndDay); } case "unlimited": default: return false; } } export default async function handler(req: NextApiRequest, res: NextApiResponse) { const reqBody = req.body as BookingCreateBody; const eventTypeId = reqBody.eventTypeId; log.debug(`Booking eventType ${eventTypeId} started`); const isTimeInPast = (time: string): boolean => { return dayjs(time).isBefore(new Date(), "day"); }; if (isTimeInPast(reqBody.start)) { const error = { errorCode: "BookingDateInPast", message: "Attempting to create a meeting in the past.", }; log.error(`Booking ${eventTypeId} failed`, error); return res.status(400).json(error); } const userSelect = Prisma.validator()({ id: true, email: true, name: true, username: true, timeZone: true, credentials: true, bufferTime: true, }); const userData = Prisma.validator()({ select: userSelect, }); const eventType = await prisma.eventType.findUnique({ where: { id: eventTypeId, }, select: { users: { select: userSelect, }, team: { select: { id: true, name: true, }, }, title: true, length: true, eventName: true, schedulingType: true, periodType: true, periodStartDate: true, periodEndDate: true, periodDays: true, periodCountCalendarDays: true, requiresConfirmation: true, userId: true, price: true, currency: true, }, }); if (!eventType) return res.status(404).json({ message: "eventType.notFound" }); let users = eventType.users; /* If this event was pre-relationship migration */ if (!users.length && eventType.userId) { const eventTypeUser = await prisma.user.findUnique({ where: { id: eventType.userId, }, select: userSelect, }); if (!eventTypeUser) return res.status(404).json({ message: "eventTypeUser.notFound" }); users.push(eventTypeUser); } if (eventType.schedulingType === SchedulingType.ROUND_ROBIN) { const selectedUsers = reqBody.users || []; const selectedUsersDataWithBookingsCount = await prisma.user.findMany({ where: { username: { in: selectedUsers }, bookings: { every: { startTime: { gt: new Date(), }, }, }, }, select: { username: true, _count: { select: { bookings: true }, }, }, }); const bookingCounts = selectedUsersDataWithBookingsCount.map((userData) => ({ username: userData.username, bookingCount: userData._count?.bookings || 0, })); if (!bookingCounts.length) users.slice(0, 1); const [firstMostAvailableUser] = bookingCounts.sort((a, b) => (a.bookingCount > b.bookingCount ? 1 : -1)); const luckyUser = users.find((user) => user.username === firstMostAvailableUser?.username); users = luckyUser ? [luckyUser] : users; } const invitee = [{ email: reqBody.email, name: reqBody.name, timeZone: reqBody.timeZone }]; const guests = reqBody.guests.map((guest) => { const g = { email: guest, name: "", timeZone: reqBody.timeZone, }; return g; }); const teamMembers = eventType.schedulingType === SchedulingType.COLLECTIVE ? users.slice(1).map((user) => ({ email: user.email, name: user.name, timeZone: user.timeZone, })) : []; const attendeesList = [...invitee, ...guests, ...teamMembers]; const seed = `${users[0].username}:${dayjs(reqBody.start).utc().format()}`; const uid = translator.fromUUID(uuidv5(seed, uuidv5.URL)); const evt: CalendarEvent = { type: eventType.title, title: getEventName(reqBody.name, eventType.title, eventType.eventName), description: reqBody.notes, startTime: reqBody.start, endTime: reqBody.end, organizer: { name: users[0].name, email: users[0].email, timeZone: users[0].timeZone, }, attendees: attendeesList, location: reqBody.location, // Will be processed by the EventManager later. }; if (eventType.schedulingType === SchedulingType.COLLECTIVE) { evt.team = { members: users.map((user) => user.name || user.username || "Nameless"), name: eventType.team?.name || "Nameless", }; // used for invitee emails } // Initialize EventManager with credentials const rescheduleUid = reqBody.rescheduleUid; function createBooking() { return prisma.booking.create({ include: { user: { select: { email: true, name: true, timeZone: true }, }, attendees: true, }, data: { uid, title: evt.title, startTime: dayjs(evt.startTime).toDate(), endTime: dayjs(evt.endTime).toDate(), description: evt.description, confirmed: !eventType.requiresConfirmation || !!rescheduleUid, location: evt.location, eventType: { connect: { id: eventTypeId, }, }, attendees: { createMany: { data: evt.attendees, }, }, user: { connect: { id: users[0].id, }, }, }, }); } type Booking = Prisma.PromiseReturnType; let booking: Booking | null = null; try { booking = await createBooking(); } catch (e) { log.error(`Booking ${eventTypeId} failed`, "Error when saving booking to db", e.message); if (e.code === "P2002") { res.status(409).json({ message: "booking.conflict" }); return; } res.status(500).end(); return; } let results: EventResult[] = []; let referencesToCreate: PartialReference[] = []; type User = Prisma.UserGetPayload; let user: User | null = null; for (const currentUser of users) { if (!currentUser) { console.error(`currentUser not found`); return; } if (!user) user = currentUser; const selectedCalendars = await prisma.selectedCalendar.findMany({ where: { userId: currentUser.id, }, }); const credentials = currentUser.credentials; if (credentials) { const calendarBusyTimes = await getBusyCalendarTimes( credentials, reqBody.start, reqBody.end, selectedCalendars ); const videoBusyTimes = await getBusyVideoTimes(credentials); calendarBusyTimes.push(...videoBusyTimes); const bufferedBusyTimes: BufferedBusyTimes = calendarBusyTimes.map((a) => ({ start: dayjs(a.start).subtract(currentUser.bufferTime, "minute").toString(), end: dayjs(a.end).add(currentUser.bufferTime, "minute").toString(), })); let isAvailableToBeBooked = true; try { isAvailableToBeBooked = isAvailable(bufferedBusyTimes, reqBody.start, eventType.length); } catch { log.debug({ message: "Unable set isAvailableToBeBooked. Using true. ", }); } if (!isAvailableToBeBooked) { const error = { errorCode: "BookingUserUnAvailable", message: `${currentUser.name} is unavailable at this time.`, }; log.debug(`Booking ${currentUser.name} failed`, error); } let timeOutOfBounds = false; try { timeOutOfBounds = isOutOfBounds(reqBody.start, { periodType: eventType.periodType, periodDays: eventType.periodDays, periodEndDate: eventType.periodEndDate, periodStartDate: eventType.periodStartDate, periodCountCalendarDays: eventType.periodCountCalendarDays, timeZone: currentUser.timeZone, }); } catch { log.debug({ message: "Unable set timeOutOfBounds. Using false. ", }); } if (timeOutOfBounds) { const error = { errorCode: "BookingUserUnAvailable", message: `${currentUser.name} is unavailable at this time.`, }; log.debug(`Booking ${currentUser.name} failed`, error); res.status(400).json(error); } } } // After polling videoBusyTimes, credentials might have been changed due to refreshment, so query them again. const eventManager = new EventManager(await refreshCredentials(user.credentials)); if (rescheduleUid) { // Use EventManager to conditionally use all needed integrations. const updateResults: CreateUpdateResult = await eventManager.update(evt, rescheduleUid); results = updateResults.results; referencesToCreate = updateResults.referencesToCreate; if (results.length > 0 && results.every((res) => !res.success)) { const error = { errorCode: "BookingReschedulingMeetingFailed", message: "Booking Rescheduling failed", }; log.error(`Booking ${user.name} failed`, error, results); } } else if (!eventType.requiresConfirmation && !eventType.price) { // Use EventManager to conditionally use all needed integrations. const createResults: CreateUpdateResult = await eventManager.create(evt, uid); results = createResults.results; referencesToCreate = createResults.referencesToCreate; if (results.length > 0 && results.every((res) => !res.success)) { const error = { errorCode: "BookingCreatingMeetingFailed", message: "Booking failed", }; log.error(`Booking ${user.username} failed`, error, results); } } if (eventType.requiresConfirmation && !rescheduleUid) { await new EventOrganizerRequestMail(evt, uid).sendEmail(); } if (typeof eventType.price === "number" && eventType.price > 0) { try { const [firstStripeCredential] = user.credentials.filter((cred) => cred.type == "stripe_payment"); // eslint-disable-next-line @typescript-eslint/ban-ts-comment /* @ts-ignore https://github.com/prisma/prisma/issues/9389 */ if (!booking.user) booking.user = user; // eslint-disable-next-line @typescript-eslint/ban-ts-comment /* @ts-ignore https://github.com/prisma/prisma/issues/9389 */ const payment = await handlePayment(evt, eventType, firstStripeCredential, booking); res.status(201).json({ ...booking, message: "Payment required", paymentUid: payment.uid }); return; } catch (e) { log.error(`Creating payment failed`, e); res.status(500).json({ message: "Payment Failed" }); return; } } log.debug(`Booking ${user.username} completed`); await prisma.booking.update({ where: { uid: booking.uid, }, data: { references: { createMany: { data: referencesToCreate, }, }, }, }); // booking successful return res.status(201).json(booking); }