Skip to content

Commit e1bdac0

Browse files
fix: Seat Attendee Rescheduling Logic (#14784)
* When rescheduling to new time slot don't trigger reschedule logic * Add unique iCalUID when rescheduling to new slot * Generating iCalUID use v4 uuid * When moving an attendee to a new booking. Only move the attendee --------- Co-authored-by: Udit Takkar <53316345+Udit-takkar@users.noreply.github.com>
1 parent d666f29 commit e1bdac0

File tree

5 files changed

+51
-32
lines changed

5 files changed

+51
-32
lines changed

packages/emails/lib/getICalUID.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import short from "short-uuid";
2-
import { v5 as uuidv5 } from "uuid";
2+
import { v4 as uuidv4 } from "uuid";
33

44
import { APP_NAME } from "@calcom/lib/constants";
55

@@ -15,11 +15,13 @@ const getICalUID = ({
1515
uid,
1616
event,
1717
defaultToEventUid,
18+
attendeeId,
1819
}: {
1920
uid?: string;
2021
// eslint-disable-next-line @typescript-eslint/no-explicit-any
2122
event?: { iCalUID?: string | null; uid?: string | null; [key: string]: any };
2223
defaultToEventUid?: boolean;
24+
attendeeId?: number;
2325
}) => {
2426
if (event?.iCalUID) return event.iCalUID;
2527

@@ -29,8 +31,8 @@ const getICalUID = ({
2931

3032
const translator = short();
3133

32-
uid = translator.fromUUID(uuidv5(APP_NAME, uuidv5.URL));
33-
return `${uid}@${APP_NAME}`;
34+
uid = translator.fromUUID(uuidv4());
35+
return `${uid}${attendeeId ? `${attendeeId}` : ""}@${APP_NAME}`;
3436
};
3537

3638
export default getICalUID;

packages/features/bookings/lib/handleNewBooking.ts

+9-2
Original file line numberDiff line numberDiff line change
@@ -1731,6 +1731,7 @@ async function handler(
17311731
eventTrigger,
17321732
responses,
17331733
});
1734+
17341735
if (newBooking) {
17351736
req.statusCode = 201;
17361737
const bookingResponse = {
@@ -1745,6 +1746,13 @@ async function handler(
17451746
...bookingResponse,
17461747
...luckyUserResponse,
17471748
};
1749+
} else {
1750+
// Rescheduling logic for the original seated event was handled in handleSeats
1751+
// We want to use new booking logic for the new time slot
1752+
originalRescheduledBooking = null;
1753+
evt.iCalUID = getICalUID({
1754+
attendeeId: bookingSeat?.attendeeId,
1755+
});
17481756
}
17491757
}
17501758
if (isTeamEventType) {
@@ -1754,7 +1762,6 @@ async function handler(
17541762
id: eventType.team?.id ?? 0,
17551763
};
17561764
}
1757-
17581765
if (reqBody.recurringEventId && eventType.recurringEvent) {
17591766
// Overriding the recurring event configuration count to be the actual number of events booked for
17601767
// the recurring event (equal or less than recurring event configuration count)
@@ -1876,7 +1883,7 @@ async function handler(
18761883
let videoCallUrl;
18771884

18781885
//this is the actual rescheduling logic
1879-
if (originalRescheduledBooking?.uid) {
1886+
if (!eventType.seatsPerTimeSlot && originalRescheduledBooking?.uid) {
18801887
log.silly("Rescheduling booking", originalRescheduledBooking.uid);
18811888
try {
18821889
// cancel workflow reminders from previous rescheduled booking

packages/features/bookings/lib/handleSeats/reschedule/attendee/attendeeRescheduleSeatedBooking.ts

+30-27
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import { cloneDeep } from "lodash";
33

44
import type EventManager from "@calcom/core/EventManager";
55
import { sendRescheduledSeatEmail } from "@calcom/emails";
6+
import { getTranslation } from "@calcom/lib/server/i18n";
67
import prisma from "@calcom/prisma";
78
import type { Person, CalendarEvent } from "@calcom/types/Calendar";
89

@@ -17,11 +18,28 @@ const attendeeRescheduleSeatedBooking = async (
1718
originalBookingEvt: CalendarEvent,
1819
eventManager: EventManager
1920
) => {
20-
const { tAttendees, bookingSeat, bookerEmail, rescheduleUid, evt } = rescheduleSeatedBookingObject;
21+
const { tAttendees, bookingSeat, bookerEmail, evt } = rescheduleSeatedBookingObject;
2122
let { originalRescheduledBooking } = rescheduleSeatedBookingObject;
2223

2324
seatAttendee["language"] = { translate: tAttendees, locale: bookingSeat?.attendee.locale ?? "en" };
2425

26+
// Update the original calendar event by removing the attendee that is rescheduling
27+
if (originalBookingEvt && originalRescheduledBooking) {
28+
// Event would probably be deleted so we first check than instead of updating references
29+
const filteredAttendees = originalRescheduledBooking?.attendees.filter((attendee) => {
30+
return attendee.email !== bookerEmail;
31+
});
32+
const deletedReference = await lastAttendeeDeleteBooking(
33+
originalRescheduledBooking,
34+
filteredAttendees,
35+
originalBookingEvt
36+
);
37+
38+
if (!deletedReference) {
39+
await eventManager.updateCalendarAttendees(originalBookingEvt, originalRescheduledBooking);
40+
}
41+
}
42+
2543
// If there is no booking then remove the attendee from the old booking and create a new one
2644
if (!newTimeSlotBooking) {
2745
await prisma.attendee.delete({
@@ -30,23 +48,6 @@ const attendeeRescheduleSeatedBooking = async (
3048
},
3149
});
3250

33-
// Update the original calendar event by removing the attendee that is rescheduling
34-
if (originalBookingEvt && originalRescheduledBooking) {
35-
// Event would probably be deleted so we first check than instead of updating references
36-
const filteredAttendees = originalRescheduledBooking?.attendees.filter((attendee) => {
37-
return attendee.email !== bookerEmail;
38-
});
39-
const deletedReference = await lastAttendeeDeleteBooking(
40-
originalRescheduledBooking,
41-
filteredAttendees,
42-
originalBookingEvt
43-
);
44-
45-
if (!deletedReference) {
46-
await eventManager.updateCalendarAttendees(originalBookingEvt, originalRescheduledBooking);
47-
}
48-
}
49-
5051
// We don't want to trigger rescheduling logic of the original booking
5152
originalRescheduledBooking = null;
5253

@@ -76,17 +77,19 @@ const attendeeRescheduleSeatedBooking = async (
7677
]);
7778
}
7879

79-
const copyEvent = cloneDeep(evt);
80-
81-
const updateManager = await eventManager.reschedule(copyEvent, rescheduleUid, newTimeSlotBooking.id);
82-
83-
const results = updateManager.results;
80+
// Add the new attendees to the new time slot booking attendees
81+
for (const attendee of newTimeSlotBooking.attendees) {
82+
const language = await getTranslation(attendee.locale ?? "en", "common");
83+
evt.attendees.push({
84+
email: attendee.email,
85+
name: attendee.name,
86+
language,
87+
});
88+
}
8489

85-
const calendarResult = results.find((result) => result.type.includes("_calendar"));
90+
const copyEvent = cloneDeep({ ...evt, iCalUID: newTimeSlotBooking.iCalUID });
8691

87-
evt.iCalUID = Array.isArray(calendarResult?.updatedEvent)
88-
? calendarResult?.updatedEvent[0]?.iCalUID
89-
: calendarResult?.updatedEvent?.iCalUID || undefined;
92+
await eventManager.updateCalendarAttendees(copyEvent, newTimeSlotBooking);
9093

9194
await sendRescheduledSeatEmail(copyEvent, seatAttendee as Person);
9295
const filteredAttendees = originalRescheduledBooking?.attendees.filter((attendee) => {

packages/features/bookings/lib/handleSeats/reschedule/rescheduleSeatedBooking.ts

+4
Original file line numberDiff line numberDiff line change
@@ -39,11 +39,14 @@ const rescheduleSeatedBooking = async (
3939
select: {
4040
id: true,
4141
uid: true,
42+
iCalUID: true,
43+
userId: true,
4244
attendees: {
4345
include: {
4446
bookingSeat: true,
4547
},
4648
},
49+
references: true,
4750
},
4851
});
4952

@@ -82,6 +85,7 @@ const rescheduleSeatedBooking = async (
8285
startTime: dayjs(originalRescheduledBooking.startTime).utc().format(),
8386
endTime: dayjs(originalRescheduledBooking.endTime).utc().format(),
8487
attendees: updatedBookingAttendees,
88+
iCalUID: originalRescheduledBooking.iCalUID,
8589
// If the location is a video integration then include the videoCallData
8690
...(videoReference && {
8791
videoCallData: {

packages/features/bookings/lib/handleSeats/types.d.ts

+3
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,9 @@ export type NewTimeSlotBooking = Prisma.BookingGetPayload<{
7070
select: {
7171
id: true;
7272
uid: true;
73+
iCalUID: true;
74+
userId: true;
75+
references: true;
7376
attendees: {
7477
include: {
7578
bookingSeat: true;

0 commit comments

Comments
 (0)