-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Sync pregame calendar events to database
- Loading branch information
Showing
3 changed files
with
217 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,154 @@ | ||
import { NextApiRequest, NextApiResponse } from "next"; | ||
import { Booking, MediaServices } from "@/components/src/types"; | ||
import admin from "@/firebaseAdmin"; | ||
import { getCalendarClient } from "@/lib/googleClient"; | ||
import { toFirebaseTimestampFromString } from "@/components/src/client/utils/serverDate"; | ||
import { Timestamp } from "@firebase/firestore"; | ||
import { NextResponse } from "next/server"; | ||
import { TableNames } from "@/components/src/policy"; | ||
|
||
const db = admin.firestore(); | ||
const createBookingWithDefaults = ( | ||
partialBooking: Partial<Booking>, | ||
): Booking => { | ||
return { | ||
title: "", | ||
description: "", | ||
email: "", | ||
firstName: "", | ||
lastName: "", | ||
secondaryName: "", | ||
nNumber: "", | ||
netId: "", | ||
phoneNumber: "", | ||
department: "", | ||
role: "", | ||
sponsorFirstName: "", | ||
sponsorLastName: "", | ||
sponsorEmail: "", | ||
bookingType: "", | ||
attendeeAffiliation: "", | ||
roomSetup: "", | ||
setupDetails: "", | ||
mediaServices: "", | ||
mediaServicesDetails: "", | ||
catering: "", | ||
hireSecurity: "", | ||
expectedAttendance: "", | ||
cateringService: "", | ||
chartFieldForCatering: "", | ||
chartFieldForSecurity: "", | ||
chartFieldForRoomSetup: "", | ||
calendarEventId: "", | ||
roomId: "", | ||
requestNumber: 0, | ||
equipmentCheckedOut: false, | ||
startDate: null, | ||
endDate: null, | ||
...partialBooking, | ||
}; | ||
}; | ||
const findNyuEmail = (event: any): string => { | ||
const attendees = event.attendees || []; | ||
const nyuEmail = attendees.find( | ||
(attendee: any) => attendee.email && attendee.email.endsWith("@nyu.edu"), | ||
); | ||
return nyuEmail ? nyuEmail.email : ""; | ||
}; | ||
export async function POST(request: Request) { | ||
try { | ||
const calendar = await getCalendarClient(); | ||
// Fetch all calendar IDs from the Resource table | ||
const resourcesSnapshot = await db.collection("resources").get(); | ||
const resources = resourcesSnapshot.docs.map(doc => ({ | ||
id: doc.id, | ||
calendarId: doc.data().calendarId, | ||
roomId: doc.data().roomId, | ||
})); | ||
|
||
let totalNewBookings = 0; | ||
for (const resource of resources) { | ||
try { | ||
// Fetch events for each calendar | ||
const now = new Date(); | ||
const events = await calendar.events.list({ | ||
calendarId: resource.calendarId, | ||
timeMin: now.toISOString(), | ||
maxResults: 100, // Adjust as needed | ||
singleEvents: true, | ||
orderBy: "startTime", | ||
}); | ||
|
||
for (const event of events.data.items || []) { | ||
const bookingRef = db | ||
.collection("bookings") | ||
.where("calendarEventId", "==", event.id); | ||
const bookingSnapshot = await bookingRef.get(); | ||
|
||
if (bookingSnapshot.empty) { | ||
// Create a new booking | ||
const calendarEventId = event.id; | ||
const nyuEmail = findNyuEmail(event); | ||
const newBooking = createBookingWithDefaults({ | ||
title: event.summary || "", | ||
description: event.description || "", | ||
email: nyuEmail || "", | ||
startDate: toFirebaseTimestampFromString( | ||
event.start?.dateTime, | ||
) as Timestamp, | ||
endDate: toFirebaseTimestampFromString( | ||
event.end?.dateTime, | ||
) as Timestamp, | ||
calendarEventId: calendarEventId || "", | ||
equipmentCheckedOut: true, | ||
roomId: resource.roomId, | ||
mediaServices: MediaServices.CHECKOUT_EQUIPMENT, | ||
}); | ||
const bookingDocRef = await db | ||
.collection(TableNames.BOOKING) | ||
.add(newBooking); | ||
|
||
console.log(`New Booking created with ID: ${bookingDocRef.id}`); | ||
|
||
const newBookingStatus = { | ||
calendarEventId: calendarEventId, | ||
email: nyuEmail, | ||
requestedAt: admin.firestore.FieldValue.serverTimestamp(), | ||
firstApprovedAt: admin.firestore.FieldValue.serverTimestamp(), | ||
finalApprovedAt: admin.firestore.FieldValue.serverTimestamp(), | ||
}; | ||
const statusDocRef = await db | ||
.collection(TableNames.BOOKING_STATUS) | ||
.add(newBookingStatus); | ||
console.log( | ||
`New BookingStatus created with ID: ${statusDocRef.id}`, | ||
); | ||
|
||
totalNewBookings++; | ||
} | ||
} | ||
} catch (error) { | ||
console.error( | ||
`Error processing calendar ${resource.calendarId}:`, | ||
error, | ||
); | ||
// Continue with the next calendar | ||
} | ||
} | ||
|
||
return NextResponse.json( | ||
{ | ||
message: `${totalNewBookings} new bookings have been synchronized.`, | ||
}, | ||
{ status: 200 }, | ||
); | ||
} catch (error) { | ||
console.error("Error syncing calendars:", error); | ||
return NextResponse.json( | ||
{ | ||
error: "An error occurred while syncing calendars.", | ||
}, | ||
{ status: 500 }, | ||
); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
60 changes: 60 additions & 0 deletions
60
booking-app/components/src/client/routes/admin/components/SyncCalendars.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
import { Box, Button, Typography } from "@mui/material"; | ||
import React, { useState } from "react"; | ||
import AlertToast from "../../components/AlertToast"; | ||
|
||
const SyncCalendars = () => { | ||
const [loading, setLoading] = useState(false); | ||
const [showAlert, setShowAlert] = useState(false); | ||
const [alertSeverity, setAlertSeverity] = useState<"error" | "success">( | ||
"success" | ||
); | ||
const [message, setMessage] = useState(""); | ||
|
||
const handleSync = async () => { | ||
setLoading(true); | ||
setShowAlert(false); | ||
try { | ||
const response = await fetch("/api/syncCalendars", { method: "POST" }); | ||
const data = await response.json(); | ||
if (response.ok) { | ||
setMessage(`Sync successful: ${data.message}`); | ||
setAlertSeverity("success"); | ||
} else { | ||
setMessage(`Error: ${data.error}`); | ||
setAlertSeverity("error"); | ||
} | ||
} catch (error) { | ||
setMessage("An error occurred while syncing calendars."); | ||
setAlertSeverity("error"); | ||
} finally { | ||
setLoading(false); | ||
setShowAlert(true); | ||
} | ||
}; | ||
|
||
return ( | ||
<Box> | ||
<Typography variant="h6"> | ||
{" "} | ||
Sync Current Semester Calendar Events | ||
</Typography> | ||
<p> | ||
This function saves existing events from the current semester's calendar | ||
to the database. | ||
</p> | ||
<Box sx={{ marginTop: 2 }}> | ||
<Button onClick={handleSync} variant="contained" disabled={loading}> | ||
Sync Calendar Events | ||
</Button> | ||
</Box> | ||
<AlertToast | ||
message={message} | ||
severity={alertSeverity} | ||
open={showAlert} | ||
handleClose={() => setShowAlert(false)} | ||
/> | ||
</Box> | ||
); | ||
}; | ||
|
||
export default SyncCalendars; |