Also implement locks for tasks that can have unintended side-effects if ran simultaneously. Other minor UI tweaks.
214 lines
6.3 KiB
TypeScript
214 lines
6.3 KiB
TypeScript
import { Handler } from 'fresh/server.ts';
|
|
|
|
import { Calendar, FreshContextState } from '/lib/types.ts';
|
|
import { buildRFC822Date, convertObjectToDavXml, DAV_RESPONSE_HEADER, escapeHtml, escapeXml } from '/lib/utils/misc.ts';
|
|
import { formatCalendarEventsToVCalendar, parseVCalendarFromTextContents } from '/lib/utils/calendar.ts';
|
|
import {
|
|
createCalendarEvent,
|
|
getCalendar,
|
|
getCalendarEvent,
|
|
getCalendarEvents,
|
|
updateCalendarEvent,
|
|
} from '/lib/data/calendar.ts';
|
|
import { createSessionCookie } from '/lib/auth.ts';
|
|
|
|
interface Data {}
|
|
|
|
interface DavResponse {
|
|
'd:href': string;
|
|
'd:propstat': {
|
|
'd:prop': {
|
|
'd:resourcetype'?: {
|
|
'd:collection'?: {};
|
|
'cal:calendar'?: {};
|
|
};
|
|
'd:displayname'?: string | {};
|
|
'cal:calendar-data'?: string;
|
|
'd:getlastmodified'?: string;
|
|
'd:getetag'?: string;
|
|
'cs:getctag'?: string;
|
|
'd:getcontenttype'?: string;
|
|
'd:getcontentlength'?: {};
|
|
'd:creationdate'?: string;
|
|
};
|
|
'd:status': string;
|
|
}[];
|
|
}
|
|
|
|
interface DavMultiStatusResponse {
|
|
'd:multistatus': {
|
|
'd:response': DavResponse[];
|
|
};
|
|
'd:multistatus_attributes': {
|
|
'xmlns:d': string;
|
|
'xmlns:s': string;
|
|
'xmlns:cal': string;
|
|
'xmlns:oc': string;
|
|
'xmlns:nc': string;
|
|
'xmlns:cs': string;
|
|
};
|
|
}
|
|
|
|
interface ResponseBody extends DavMultiStatusResponse {}
|
|
|
|
export const handler: Handler<Data, FreshContextState> = async (request, context) => {
|
|
if (!context.state.user) {
|
|
return new Response('Unauthorized', {
|
|
status: 401,
|
|
headers: { 'www-authenticate': 'Basic realm="bewCloud", charset="UTF-8"' },
|
|
});
|
|
}
|
|
|
|
if (
|
|
request.method !== 'PROPFIND' && request.method !== 'REPORT' && request.method !== 'GET' &&
|
|
request.method !== 'PUT' && request.method !== 'MKCALENDAR'
|
|
) {
|
|
return new Response('Bad Request', { status: 400 });
|
|
}
|
|
|
|
const { calendarId } = context.params;
|
|
|
|
let calendar: Calendar | null = null;
|
|
|
|
try {
|
|
calendar = await getCalendar(calendarId, context.state.user.id);
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
|
|
if (!calendar) {
|
|
// TODO: Support MKCALENDAR
|
|
return new Response('Not found', { status: 404 });
|
|
}
|
|
|
|
if (request.method === 'PUT') {
|
|
const requestBody = await request.clone().text();
|
|
|
|
const [partialCalendarEvent] = parseVCalendarFromTextContents(requestBody);
|
|
|
|
if (partialCalendarEvent.title && partialCalendarEvent.start_date && partialCalendarEvent.end_date) {
|
|
const newCalendarEvent = await createCalendarEvent(
|
|
context.state.user.id,
|
|
calendarId,
|
|
partialCalendarEvent.title,
|
|
new Date(partialCalendarEvent.start_date),
|
|
new Date(partialCalendarEvent.end_date),
|
|
partialCalendarEvent.is_all_day,
|
|
);
|
|
|
|
const parsedExtra = JSON.stringify(partialCalendarEvent.extra || {});
|
|
|
|
if (parsedExtra !== '{}') {
|
|
newCalendarEvent.extra = partialCalendarEvent.extra!;
|
|
|
|
if (
|
|
newCalendarEvent.extra.is_recurring && newCalendarEvent.extra.recurring_sequence === 0 &&
|
|
!newCalendarEvent.extra.recurring_id
|
|
) {
|
|
newCalendarEvent.extra.recurring_id = newCalendarEvent.id;
|
|
}
|
|
|
|
await updateCalendarEvent(newCalendarEvent);
|
|
}
|
|
|
|
const calendarEvent = await getCalendarEvent(newCalendarEvent.id, context.state.user.id);
|
|
|
|
return new Response('Created', { status: 201, headers: { 'etag': `"${calendarEvent.revision}"` } });
|
|
}
|
|
|
|
return new Response('Not found', { status: 404 });
|
|
}
|
|
|
|
const calendarEvents = await getCalendarEvents(context.state.user.id, [calendar.id]);
|
|
|
|
if (request.method === 'GET') {
|
|
const response = new Response(formatCalendarEventsToVCalendar(calendarEvents, [calendar]), {
|
|
status: 200,
|
|
});
|
|
|
|
if (context.state.session) {
|
|
return response;
|
|
}
|
|
|
|
return createSessionCookie(request, context.state.user, response, true);
|
|
}
|
|
|
|
const requestBody = (await request.clone().text()).toLowerCase();
|
|
|
|
const includeVCalendar = requestBody.includes('calendar-data');
|
|
|
|
const parsedCalendar: DavResponse = {
|
|
'd:href': `/dav/calendars/${calendar.id}`,
|
|
'd:propstat': [{
|
|
'd:prop': {
|
|
'd:displayname': calendar.name,
|
|
'd:getlastmodified': buildRFC822Date(calendar.updated_at.toISOString()),
|
|
'd:getetag': escapeHtml(`"${calendar.revision}"`),
|
|
'cs:getctag': calendar.revision,
|
|
'd:getcontenttype': 'text/calendar; charset=utf-8',
|
|
'd:resourcetype': {},
|
|
},
|
|
'd:status': 'HTTP/1.1 200 OK',
|
|
}],
|
|
};
|
|
|
|
const parsedCalendarEvents = calendarEvents.map((calendarEvent) => {
|
|
const parsedCalendarEvent: DavResponse = {
|
|
'd:href': `/dav/calendars/${calendar!.id}/${calendarEvent.id}.ics`,
|
|
'd:propstat': [{
|
|
'd:prop': {
|
|
'd:getlastmodified': buildRFC822Date(calendarEvent.updated_at.toISOString()),
|
|
'd:getetag': escapeHtml(`"${calendarEvent.revision}"`),
|
|
'cs:getctag': calendarEvent.revision,
|
|
'd:getcontenttype': 'text/calendar; charset=utf-8',
|
|
'd:resourcetype': {},
|
|
'd:creationdate': buildRFC822Date(calendarEvent.created_at.toISOString()),
|
|
},
|
|
'd:status': 'HTTP/1.1 200 OK',
|
|
}, {
|
|
'd:prop': {
|
|
'd:displayname': {},
|
|
'd:getcontentlength': {},
|
|
},
|
|
'd:status': 'HTTP/1.1 404 Not Found',
|
|
}],
|
|
};
|
|
|
|
if (includeVCalendar) {
|
|
parsedCalendarEvent['d:propstat'][0]['d:prop']['cal:calendar-data'] = escapeXml(
|
|
formatCalendarEventsToVCalendar([calendarEvent], [calendar!]),
|
|
);
|
|
}
|
|
|
|
return parsedCalendarEvent;
|
|
});
|
|
|
|
const responseBody: ResponseBody = {
|
|
'd:multistatus': {
|
|
'd:response': [parsedCalendar, ...parsedCalendarEvents],
|
|
},
|
|
'd:multistatus_attributes': {
|
|
'xmlns:d': 'DAV:',
|
|
'xmlns:s': 'http://sabredav.org/ns',
|
|
'xmlns:cal': 'urn:ietf:params:xml:ns:caldav',
|
|
'xmlns:oc': 'http://owncloud.org/ns',
|
|
'xmlns:nc': 'http://nextcloud.org/ns',
|
|
'xmlns:cs': 'http://calendarserver.org/ns/',
|
|
},
|
|
};
|
|
|
|
const response = new Response(convertObjectToDavXml(responseBody, true), {
|
|
headers: {
|
|
'content-type': 'application/xml; charset=utf-8',
|
|
'dav': DAV_RESPONSE_HEADER,
|
|
},
|
|
status: 207,
|
|
});
|
|
|
|
if (context.state.session) {
|
|
return response;
|
|
}
|
|
|
|
return createSessionCookie(request, context.state.user, response, true);
|
|
};
|