import { FSDB_COLLECTION } from '@andyburke/fsdb'; import { EVENT, get_events_collection_for_zone } from '../../../../../../models/event.ts'; import { ZONE, ZONES } from '../../../../../../models/zone.ts'; import parse_body from '../../../../../../utils/bodyparser.ts'; import * as CANNED_RESPONSES from '../../../../../../utils/canned_responses.ts'; import { get_session, get_user, PRECHECK_TABLE, require_user } from '../../../../../../utils/prechecks.ts'; export const PRECHECKS: PRECHECK_TABLE = {}; // GET /api/zones/:zone_id/events/:id - Get an event PRECHECKS.GET = [get_session, get_user, require_user, async (_req: Request, meta: Record): Promise => { const zone_id: string = meta.params?.zone_id?.toLowerCase().trim() ?? ''; // lurid is 49 chars as we use them, eg: "also-play-flow-want-form-wide-thus-work-burn-same" const zone: ZONE | null = zone_id.length === 49 ? await ZONES.get(zone_id) : null; if (!zone) { return CANNED_RESPONSES.not_found(); } meta.zone = zone; const zone_is_public = zone.permissions.read.length === 0; const user_has_read_for_zone = zone_is_public || zone.permissions.read.includes(meta.user.id); const zone_has_public_events = user_has_read_for_zone && (zone.permissions.read_events.length === 0); const user_has_read_events_for_zone = user_has_read_for_zone && (zone_has_public_events || zone.permissions.read_events.includes(meta.user.id)); if (!user_has_read_events_for_zone) { return CANNED_RESPONSES.permission_denied(); } }]; export async function GET(_req: Request, meta: Record): Promise { const events: FSDB_COLLECTION = get_events_collection_for_zone(meta.zone.id); const event: EVENT | null = await events.get(meta.params.event_id); if (!event) { return CANNED_RESPONSES.not_found(); } return Response.json(event, { status: 200 }); } // PUT /api/zones/:zone_id/events/:event_id - Update event PRECHECKS.PUT = [ get_session, get_user, require_user, (_req: Request, _meta: Record): Response | undefined => { if (Deno.env.get('APPEND_ONLY_EVENTS')) { return CANNED_RESPONSES.append_only_events(); } }, async (_req: Request, meta: Record): Promise => { const zone_id: string = meta.params?.zone_id?.toLowerCase().trim() ?? ''; // lurid is 49 chars as we use them, eg: "also-play-flow-want-form-wide-thus-work-burn-same" const zone: ZONE | null = zone_id.length === 49 ? await ZONES.get(zone_id) : null; if (!zone) { return CANNED_RESPONSES.not_found(); } meta.zone = zone; const zone_is_public: boolean = meta.zone.permissions.read.length === 0; const user_has_read_for_zone = zone_is_public || meta.zone.permissions.read.includes(meta.user.id); const zone_events_are_publicly_writable = meta.zone.permissions.write_events.length === 0; const user_has_write_events_for_zone = user_has_read_for_zone && (zone_events_are_publicly_writable || meta.zone.permissions.write_events.includes(meta.user.id)); if (!user_has_write_events_for_zone) { return CANNED_RESPONSES.permission_denied(); } } ]; export async function PUT(req: Request, meta: Record): Promise { const now = new Date().toISOString(); try { const events: FSDB_COLLECTION = get_events_collection_for_zone(meta.zone.id); const event: EVENT | null = await events.get(meta.params.event_id); if (!event) { return CANNED_RESPONSES.not_found(); } if (event.creator_id !== meta.user.id) { return CANNED_RESPONSES.permission_denied(); } const body = await parse_body(req); const updated = { ...event, ...body, id: event.id, creator_id: event.creator_id, timestamps: { created: event.timestamps.created, updated: now } }; await events.update(updated); return Response.json(updated, { status: 200 }); } catch (err) { return Response.json({ error: { message: (err as Error)?.message ?? 'Unknown error due to invalid data.', cause: (err as Error)?.cause ?? 'invalid_data' } }, { status: 400 }); } } // DELETE /api/zones/:zone_id/events/:event_id - Delete event PRECHECKS.DELETE = [ get_session, get_user, require_user, (_req: Request, _meta: Record): Response | undefined => { if (Deno.env.get('APPEND_ONLY_EVENTS')) { return CANNED_RESPONSES.append_only_events(); } }, async (_req: Request, meta: Record): Promise => { const zone_id: string = meta.params?.zone_id?.toLowerCase().trim() ?? ''; // lurid is 49 chars as we use them, eg: "also-play-flow-want-form-wide-thus-work-burn-same" const zone: ZONE | null = zone_id.length === 49 ? await ZONES.get(zone_id) : null; if (!zone) { return CANNED_RESPONSES.not_found(); } meta.zone = zone; const zone_is_public: boolean = meta.zone.permissions.read.length === 0; const user_has_read_for_zone = zone_is_public || meta.zone.permissions.read.includes(meta.user.id); const zone_events_are_publicly_writable = meta.zone.permissions.write_events.length === 0; const user_has_write_events_for_zone = user_has_read_for_zone && (zone_events_are_publicly_writable || meta.zone.permissions.write_events.includes(meta.user.id)); if (!user_has_write_events_for_zone) { return CANNED_RESPONSES.permission_denied(); } } ]; export async function DELETE(_req: Request, meta: Record): Promise { const events: FSDB_COLLECTION = get_events_collection_for_zone(meta.zone.id); const event: EVENT | null = await events.get(meta.params.event_id); if (!event) { return CANNED_RESPONSES.not_found(); } await events.delete(event); return Response.json({ deleted: true }, { status: 200 }); }