This commit is contained in:
parent
96f04c6de4
commit
b00a37eb17
2
app.ts
2
app.ts
|
@ -29,7 +29,7 @@ app.use(express.json());
|
|||
// Configure CORS
|
||||
let allowedHosts = [
|
||||
'https://www.nachklang.art',
|
||||
'https://calendar.nachklang.art',
|
||||
'https://calendar.nachklang.art'
|
||||
];
|
||||
app.use(cors({
|
||||
origin: function (origin: any, callback: any) {
|
||||
|
|
|
@ -1,3 +1,13 @@
|
|||
export interface Event {
|
||||
event_id: number;
|
||||
calendar_id: number;
|
||||
uuid: string;
|
||||
name: string;
|
||||
description: string;
|
||||
start_datetime: Date;
|
||||
end_datetime: Date;
|
||||
created_date: Date;
|
||||
location: string;
|
||||
created_by: string;
|
||||
url: string;
|
||||
}
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
|
||||
import express, {Request, Response} from 'express';
|
||||
import * as EventService from './events.service';
|
||||
import {Event} from './event.interface';
|
||||
import * as iCalService from './icalgenerator.service';
|
||||
|
||||
|
||||
/**
|
||||
|
@ -13,13 +13,18 @@ import {Event} from './event.interface';
|
|||
|
||||
export const eventsRouter = express.Router();
|
||||
|
||||
/**
|
||||
* Constants
|
||||
*/
|
||||
const fileNames = ['Nachklang_calendar', 'Nachklang_internal_calendar', 'Nachklang_management_calendar'];
|
||||
|
||||
|
||||
/**
|
||||
* Controller Definitions
|
||||
*/
|
||||
|
||||
// POST users/register
|
||||
eventsRouter.get('/all', async (req: Request, res: Response) => {
|
||||
eventsRouter.get('/json', async (req: Request, res: Response) => {
|
||||
try {
|
||||
// Get request params
|
||||
let calendarId: number = parseInt(req.query.calendar as string ?? '', 10);
|
||||
|
@ -27,10 +32,31 @@ eventsRouter.get('/all', async (req: Request, res: Response) => {
|
|||
// Get events
|
||||
let events = await EventService.getAllEvents(calendarId);
|
||||
|
||||
// Send the session details back to the user
|
||||
// Send the events back
|
||||
res.status(200).send(events);
|
||||
} catch (e: any) {
|
||||
console.log('Error handling a request: ' + e.message);
|
||||
res.status(500).send(JSON.stringify({'message': 'Internal Server Error. Try again later.'}));
|
||||
}
|
||||
});
|
||||
|
||||
eventsRouter.get('/ical', async (req: Request, res: Response) => {
|
||||
try {
|
||||
// Get request params
|
||||
let calendarId: number = parseInt(req.query.calendar as string ?? '', 10);
|
||||
|
||||
// Get events
|
||||
let events = await EventService.getAllEvents(calendarId);
|
||||
|
||||
// generate file name
|
||||
let fileName = fileNames[calendarId];
|
||||
let file = await iCalService.convertToIcal(events);
|
||||
|
||||
// Send the ical file back
|
||||
res.set({'Content-Disposition': 'attachment; filename=' + fileName + '.ics'});
|
||||
res.status(200).send(file);
|
||||
} catch (e: any) {
|
||||
console.log('Error handling a request: ' + e.message);
|
||||
res.status(500).send(JSON.stringify({'message': 'Internal Server Error. Try again later.'}));
|
||||
}
|
||||
});
|
||||
|
|
132
src/models/calendar/events/icalgenerator.service.ts
Normal file
132
src/models/calendar/events/icalgenerator.service.ts
Normal file
|
@ -0,0 +1,132 @@
|
|||
import {Event} from './event.interface';
|
||||
|
||||
export const convertToIcal = async (events: Event[]): Promise<string> => {
|
||||
try {
|
||||
let ical: iCalFile = {body: []};
|
||||
generateHeaderInfo(ical);
|
||||
generateFooterInfo(ical);
|
||||
for (let event of events) {
|
||||
addEventToFile(ical, event);
|
||||
}
|
||||
|
||||
return serializeIcalFile(ical);
|
||||
} catch (err) {
|
||||
throw err;
|
||||
}
|
||||
};
|
||||
|
||||
const serializeIcalFile = (ical: iCalFile): string => {
|
||||
let returnString = '';
|
||||
|
||||
returnString += ical.header;
|
||||
for (let event of ical.body) {
|
||||
returnString += serializeIcalEvent(event);
|
||||
}
|
||||
returnString += ical.footer;
|
||||
|
||||
return returnString;
|
||||
};
|
||||
|
||||
const serializeIcalEvent = (icalevent: iCalEvent): string => {
|
||||
let returnString = '';
|
||||
|
||||
returnString += icalevent.header;
|
||||
returnString += 'UID:' + icalevent.uid;
|
||||
returnString += 'DTSTAMP:' + icalevent.created;
|
||||
returnString += 'ORGANIZER:' + icalevent.organizer;
|
||||
returnString += 'DTSTART;TZID=Europe/Berlin:' + icalevent.start;
|
||||
returnString += 'DTEND;TZID=Europe/Berlin:' + icalevent.end;
|
||||
returnString += 'SUMMARY:' + icalevent.summary;
|
||||
returnString += 'DESCRIPTION:' + icalevent.description;
|
||||
returnString += 'LOCATION:' + icalevent.location;
|
||||
returnString += 'URL:' + icalevent.url;
|
||||
returnString += icalevent.footer;
|
||||
|
||||
return returnString;
|
||||
};
|
||||
|
||||
|
||||
const generateHeaderInfo = (ical: iCalFile) => {
|
||||
ical.header = 'BEGIN:VCALENDAR\n' +
|
||||
'VERSION:2.0\n' +
|
||||
'PRODID:-//hacksw/handcal//NONSGML v1.0//EN\n' +
|
||||
'CALSCALE:GREGORIAN\n' +
|
||||
'BEGIN:VTIMEZONE\n' +
|
||||
'TZID:Europe/Berlin\n' +
|
||||
'LAST-MODIFIED:20201011T015911Z\n' +
|
||||
'TZURL:http://tzurl.org/zoneinfo-outlook/Europe/Berlin\n' +
|
||||
'X-LIC-LOCATION:Europe/Berlin\n' +
|
||||
'BEGIN:DAYLIGHT\n' +
|
||||
'TZNAME:CEST\n' +
|
||||
'TZOFFSETFROM:+0100\n' +
|
||||
'TZOFFSETTO:+0200\n' +
|
||||
'DTSTART:19700329T020000\n' +
|
||||
'RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=-1SU\n' +
|
||||
'END:DAYLIGHT\n' +
|
||||
'BEGIN:STANDARD\n' +
|
||||
'TZNAME:CET\n' +
|
||||
'TZOFFSETFROM:+0200\n' +
|
||||
'TZOFFSETTO:+0100\n' +
|
||||
'DTSTART:19701025T030000\n' +
|
||||
'RRULE:FREQ=YEARLY;BYMONTH=10;BYDAY=-1SU\n' +
|
||||
'END:STANDARD\n' +
|
||||
'END:VTIMEZONE\n';
|
||||
};
|
||||
|
||||
const generateFooterInfo = (ical: iCalFile) => {
|
||||
ical.footer = 'END:VCALENDAR';
|
||||
};
|
||||
|
||||
const addEventToFile = (ical: iCalFile, event: Event) => {
|
||||
ical.body.push(createIcalEvent(event));
|
||||
};
|
||||
|
||||
const createIcalEvent = (event: Event): iCalEvent => {
|
||||
return {
|
||||
header: 'BEGIN:VEVENT\n',
|
||||
uid: event.uuid + '\n',
|
||||
created: formatDate(event.created_date) + 'Z\n',
|
||||
organizer: event.created_by + '\n',
|
||||
start: formatDate(event.start_datetime) + '\n',
|
||||
end: formatDate(event.end_datetime) + '\n',
|
||||
summary: event.name + '\n',
|
||||
description: event.description + '\n',
|
||||
location: event.location + '\n',
|
||||
url: event.url + '\n',
|
||||
footer: 'END:VEVENT\n'
|
||||
};
|
||||
};
|
||||
|
||||
const formatDate = (date: Date): string => {
|
||||
let returnString = '';
|
||||
|
||||
returnString += date.getFullYear();
|
||||
returnString += (date.getMonth() + 1).toString().padStart(2, '0'); // +1 Because JS sucks
|
||||
returnString += date.getDate().toString().padStart(2, '0');
|
||||
returnString += 'T';
|
||||
returnString += date.getHours().toString().padStart(2, '0');
|
||||
returnString += date.getMinutes().toString().padStart(2, '0');
|
||||
returnString += date.getSeconds().toString().padStart(2, '0');
|
||||
|
||||
return returnString;
|
||||
};
|
||||
|
||||
export interface iCalFile {
|
||||
header?: string;
|
||||
body: iCalEvent[];
|
||||
footer?: string;
|
||||
}
|
||||
|
||||
export interface iCalEvent {
|
||||
header: string;
|
||||
uid: string;
|
||||
created: string;
|
||||
organizer: string;
|
||||
start: string;
|
||||
end: string;
|
||||
summary: string;
|
||||
description: string;
|
||||
location: string;
|
||||
url: string;
|
||||
footer: string;
|
||||
}
|
Loading…
Reference in New Issue
Block a user