Many an update!

This commit is contained in:
Jennie Robinson Faber 2025-12-01 15:26:42 +00:00
parent 85195d6c7a
commit d588c49946
35 changed files with 3528 additions and 1142 deletions

View file

@ -0,0 +1,103 @@
import Event from "../../../models/event";
export default defineEventHandler(async (event) => {
const id = getRouterParam(event, "id");
if (!id) {
throw createError({
statusCode: 400,
statusMessage: "Event ID is required",
});
}
try {
// Find event by ID or slug
const eventData = await Event.findOne({
$or: [{ _id: id }, { slug: id }],
isVisible: true,
isCancelled: { $ne: true },
}).select("title slug description startDate endDate location");
if (!eventData) {
throw createError({
statusCode: 404,
statusMessage: "Event not found",
});
}
// Generate iCal format for single event
const ical = generateSingleEventCalendar(eventData);
// Set headers for download
const filename = `${eventData.slug || eventData._id}.ics`;
setHeader(event, "Content-Type", "text/calendar; charset=utf-8");
setHeader(
event,
"Content-Disposition",
`attachment; filename="${filename}"`
);
setHeader(event, "Cache-Control", "no-cache");
return ical;
} catch (error) {
console.error("Error generating event calendar:", error);
if (error.statusCode) {
throw error;
}
throw createError({
statusCode: 500,
statusMessage: "Failed to generate calendar file",
});
}
});
function generateSingleEventCalendar(evt) {
const now = new Date();
const timestamp = now
.toISOString()
.replace(/[-:]/g, "")
.replace(/\.\d{3}/, "");
const eventStart = new Date(evt.startDate);
const eventEnd = new Date(evt.endDate);
const dtstart = eventStart
.toISOString()
.replace(/[-:]/g, "")
.replace(/\.\d{3}/, "");
const dtend = eventEnd
.toISOString()
.replace(/[-:]/g, "")
.replace(/\.\d{3}/, "");
// Clean description for iCal format
const description = (evt.description || "")
.replace(/\n/g, "\\n")
.replace(/,/g, "\\,");
const eventUrl = `https://ghostguild.org/events/${evt.slug || evt._id}`;
const ical = [
"BEGIN:VCALENDAR",
"VERSION:2.0",
"PRODID:-//Ghost Guild//Events//EN",
"CALSCALE:GREGORIAN",
"METHOD:PUBLISH",
"BEGIN:VEVENT",
`UID:${evt._id}@ghostguild.org`,
`DTSTAMP:${timestamp}`,
`DTSTART:${dtstart}`,
`DTEND:${dtend}`,
`SUMMARY:${evt.title}`,
`DESCRIPTION:${description}\\n\\nView event: ${eventUrl}`,
`LOCATION:${evt.location || "Online"}`,
`URL:${eventUrl}`,
`STATUS:CONFIRMED`,
"END:VEVENT",
"END:VCALENDAR",
];
return ical.join("\r\n");
}

View file

@ -1,5 +1,8 @@
import Event from "../../../models/event";
import { sendEventCancellationEmail } from "../../../utils/resend.js";
import {
sendEventCancellationEmail,
sendWaitlistNotificationEmail,
} from "../../../utils/resend.js";
export default defineEventHandler(async (event) => {
const id = getRouterParam(event, "id");
@ -71,6 +74,39 @@ export default defineEventHandler(async (event) => {
console.error("Failed to send cancellation email:", emailError);
}
// Notify waitlisted users if waitlist is enabled and there are entries
if (
eventDoc.tickets?.waitlist?.enabled &&
eventDoc.tickets.waitlist.entries?.length > 0
) {
try {
const eventData = {
title: eventDoc.title,
slug: eventDoc.slug,
_id: eventDoc._id,
startDate: eventDoc.startDate,
endDate: eventDoc.endDate,
location: eventDoc.location,
};
// Notify the first person on the waitlist who hasn't been notified yet
const waitlistEntry = eventDoc.tickets.waitlist.entries.find(
(entry) => !entry.notified
);
if (waitlistEntry) {
await sendWaitlistNotificationEmail(waitlistEntry, eventData);
// Mark as notified
waitlistEntry.notified = true;
await eventDoc.save();
}
} catch (waitlistError) {
// Log error but don't fail the cancellation
console.error("Failed to notify waitlist:", waitlistError);
}
}
return {
success: true,
message: "Registration cancelled successfully",

View file

@ -0,0 +1,59 @@
import Event from "../../../models/event";
export default defineEventHandler(async (event) => {
const id = getRouterParam(event, "id");
const body = await readBody(event);
const { email } = body;
if (!email) {
throw createError({
statusCode: 400,
statusMessage: "Email is required",
});
}
try {
// Find event by ID or slug
const eventData = await Event.findOne({
$or: [{ _id: id }, { slug: id }],
});
if (!eventData) {
throw createError({
statusCode: 404,
statusMessage: "Event not found",
});
}
// Find and remove from waitlist
const waitlistIndex = eventData.tickets?.waitlist?.entries?.findIndex(
(entry) => entry.email.toLowerCase() === email.toLowerCase()
);
if (waitlistIndex === -1 || waitlistIndex === undefined) {
throw createError({
statusCode: 404,
statusMessage: "You are not on the waitlist for this event",
});
}
eventData.tickets.waitlist.entries.splice(waitlistIndex, 1);
await eventData.save();
return {
success: true,
message: "You have been removed from the waitlist",
};
} catch (error) {
if (error.statusCode) {
throw error;
}
console.error("Error leaving waitlist:", error);
throw createError({
statusCode: 500,
statusMessage: "Failed to leave waitlist",
});
}
});

View file

@ -0,0 +1,119 @@
import Event from "../../../models/event";
import Member from "../../../models/member";
export default defineEventHandler(async (event) => {
const id = getRouterParam(event, "id");
const body = await readBody(event);
const { name, email, membershipLevel } = body;
if (!email) {
throw createError({
statusCode: 400,
statusMessage: "Email is required",
});
}
try {
// Find event by ID or slug
const eventData = await Event.findOne({
$or: [{ _id: id }, { slug: id }],
});
if (!eventData) {
throw createError({
statusCode: 404,
statusMessage: "Event not found",
});
}
// Check if waitlist is enabled
if (!eventData.tickets?.waitlist?.enabled) {
throw createError({
statusCode: 400,
statusMessage: "Waitlist is not enabled for this event",
});
}
// Check if already on waitlist
const existingEntry = eventData.tickets.waitlist.entries?.find(
(entry) => entry.email.toLowerCase() === email.toLowerCase()
);
if (existingEntry) {
throw createError({
statusCode: 400,
statusMessage: "You are already on the waitlist for this event",
});
}
// Check if already registered
const existingRegistration = eventData.registrations?.find(
(reg) => reg.email?.toLowerCase() === email.toLowerCase() && !reg.cancelledAt
);
if (existingRegistration) {
throw createError({
statusCode: 400,
statusMessage: "You are already registered for this event",
});
}
// Check waitlist capacity
const currentWaitlistSize = eventData.tickets.waitlist.entries?.length || 0;
const maxSize = eventData.tickets.waitlist.maxSize;
if (maxSize && currentWaitlistSize >= maxSize) {
throw createError({
statusCode: 400,
statusMessage: "The waitlist is full",
});
}
// Get member info if authenticated
let memberName = name;
let memberLevel = membershipLevel || "non-member";
// Try to find member by email
const member = await Member.findOne({ email: email.toLowerCase() });
if (member) {
memberName = memberName || member.name;
memberLevel = `${member.circle}-${member.contributionTier}`;
}
// Add to waitlist
if (!eventData.tickets.waitlist.entries) {
eventData.tickets.waitlist.entries = [];
}
eventData.tickets.waitlist.entries.push({
name: memberName || "Guest",
email: email.toLowerCase(),
membershipLevel: memberLevel,
addedAt: new Date(),
notified: false,
});
await eventData.save();
// Get position in waitlist
const position = eventData.tickets.waitlist.entries.length;
return {
success: true,
message: "You have been added to the waitlist",
position,
totalWaitlisted: position,
};
} catch (error) {
if (error.statusCode) {
throw error;
}
console.error("Error joining waitlist:", error);
throw createError({
statusCode: 500,
statusMessage: "Failed to join waitlist",
});
}
});