Compare commits

..

No commits in common. "d6022eeb62e38d9476113a1ec4aa5f21f87d190e" and "5ec42d07bb89b24a6901e86e9c2db636bb386f28" have entirely different histories.

8 changed files with 165 additions and 55 deletions

View File

@ -1,7 +1,7 @@
import ICAL from './lib/ical.timezones.js' import ICAL from './lib/ical.timezones.js';
import fs from 'fs' import fs from 'fs';
import path from 'path' import path from 'path';
import axios from 'axios' import axios from 'axios';
export const MERGED_CALENDARS_DIR = path.join(process.cwd(), 'calendar'); export const MERGED_CALENDARS_DIR = path.join(process.cwd(), 'calendar');
@ -26,6 +26,81 @@ export async function fetchCalendarData(calendar) {
} }
} }
// Helper function to check if TZID exists in the raw property string
function hasTZID(rawProperty) {
return rawProperty.includes('TZID=');
}
// Function to process DTSTART/DTEND
function processDateTimeProperty(event, propertyName, newEvent) {
const rawProperty = event.getFirstProperty(propertyName)?.toICALString();
if (!rawProperty) {
console.log(`No raw property found for ${propertyName}`);
return;
}
console.log(`Raw property: ${rawProperty}`); // Log the raw property
// Check if it's a date-based event (VALUE=DATE)
if (rawProperty.includes('VALUE=DATE')) {
console.log(`Date-based event detected for ${propertyName}: ${rawProperty}`);
// Split to get the date part (should be in the format YYYYMMDD)
const dateOnly = rawProperty.split(':')[1]; // e.g., "20231225"
console.log(`Extracted date string: ${dateOnly}`);
if (!dateOnly) {
console.error(`Error: Could not extract date from ${rawProperty}`);
return;
}
// Ensure the date string is valid (no dashes, just YYYYMMDD)
const year = dateOnly.slice(0, 4);
const month = dateOnly.slice(4, 6);
const day = dateOnly.slice(6, 8);
console.log(`Parsed date: ${year}-${month}-${day}`);
// Check if the date is valid
if (!year || !month || !day || isNaN(new Date(`${year}-${month}-${day}`))) {
console.error(`Invalid date parsed from raw property: ${rawProperty}`);
return;
}
const formattedDate = dateOnly; // Use the date string as is (YYYYMMDD format)
console.log(`Formatted date: ${formattedDate}`);
// Log before adding the property to ensure it's correct
console.log(`Adding date-based property with value: ${propertyName};VALUE=DATE:${formattedDate}`);
// Correct property name usage (DTSTART not dtstart)
const property = new ICAL.Property(propertyName.toUpperCase(), newEvent); // Use uppercase "DTSTART"
property.setValue(`VALUE=DATE:${formattedDate}`);
// Log the property object before adding it to ensure everything is correct
console.log(`Property to add:`, property);
newEvent.addProperty(property);
} else {
console.log(`Time-based event detected for ${propertyName}: ${rawProperty}`);
// Time-based event processing (existing logic)
const dateTime = event.getFirstPropertyValue(propertyName);
const dateTimeString = dateTime.toString();
const property = new ICAL.Property(propertyName, newEvent);
property.setValue(dateTimeString);
if (hasTZID(rawProperty)) {
// If raw property includes TZID, add it
property.setParameter('TZID', dateTime.zone.tzid);
}
newEvent.addProperty(property);
}
}
// Create a top-level VCALENDAR component // Create a top-level VCALENDAR component
export function createCalendarComponent(name) { export function createCalendarComponent(name) {
const calendarComponent = new ICAL.Component(['vcalendar', [], []]); const calendarComponent = new ICAL.Component(['vcalendar', [], []]);
@ -37,71 +112,94 @@ export function createCalendarComponent(name) {
} }
// Add events to the calendar component // Add events to the calendar component
export function addEventsToCalendar(newCalendar, calendars, overrideFlag = false) { export function addEventsToCalendar(calendarComponent, results, overrideFlag = false) {
let defaultTimeZone = null; // To store the first found X-WR-TIMEZONE let defaultTimeZone = null; // To store the first found X-WR-TIMEZONE
calendars.forEach((calendarRaw) => { results.forEach((result) => {
try { try {
const calendar = new ICAL.Component(ICAL.parse(calendarRaw.data)); const parsed = ICAL.parse(result.data);
const component = new ICAL.Component(parsed);
// Extract METHOD from the parsed data (if available) // Extract METHOD from the parsed data (if available)
const method = calendar.getFirstPropertyValue('method'); const method = component.getFirstPropertyValue('method');
if (method) { if (method) {
console.log(`Extracted METHOD: ${method}`); console.log(`Extracted METHOD: ${method}`);
// Only add the METHOD property once // Only add the METHOD property once
if (!newCalendar.getFirstPropertyValue('method')) { if (!calendarComponent.getFirstPropertyValue('method')) {
newCalendar.updatePropertyWithValue('method', method.toUpperCase()); calendarComponent.updatePropertyWithValue('method', method.toUpperCase());
} }
} }
// Extract X-WR-TIMEZONE if available // Extract X-WR-TIMEZONE if available
const wrTimeZone = calendar.getFirstPropertyValue('x-wr-timezone'); const wrTimeZone = component.getFirstPropertyValue('x-wr-timezone');
if (wrTimeZone) { if (wrTimeZone) {
console.log(`Extracted X-WR-TIMEZONE: ${wrTimeZone}`); console.log(`Extracted X-WR-TIMEZONE: ${wrTimeZone}`);
// Set it as the default if not already set // Set it as the default if not already set
if (!defaultTimeZone) { if (!defaultTimeZone) {
defaultTimeZone = wrTimeZone; defaultTimeZone = wrTimeZone;
if (!newCalendar.getFirstPropertyValue('x-wr-timezone')) { if (!calendarComponent.getFirstPropertyValue('x-wr-timezone')) {
newCalendar.updatePropertyWithValue('x-wr-timezone', defaultTimeZone); calendarComponent.updatePropertyWithValue('x-wr-timezone', defaultTimeZone);
} }
} }
} }
// Extract and add VTIMEZONE components // Extract and add VTIMEZONE components
const timezones = calendar.getAllSubcomponents('vtimezone'); const timezones = component.getAllSubcomponents('vtimezone');
timezones.forEach((timezone) => { timezones.forEach((timezone) => {
const tzid = timezone.getFirstPropertyValue('tzid'); const tzid = timezone.getFirstPropertyValue('tzid');
if (!newCalendar.getFirstSubcomponent((comp) => comp.name === 'vtimezone' && comp.getFirstPropertyValue('tzid') === tzid)) { if (!calendarComponent.getFirstSubcomponent((comp) => comp.name === 'vtimezone' && comp.getFirstPropertyValue('tzid') === tzid)) {
newCalendar.addSubcomponent(timezone); calendarComponent.addSubcomponent(timezone);
} }
}); });
// Process VEVENT components // Process VEVENT components
calendar.getAllSubcomponents('vevent').forEach((vevent) => { component.getAllSubcomponents('vevent').forEach((event) => {
const event = new ICAL.Event(vevent); const vevent = new ICAL.Event(event);
const newEvent = new ICAL.Event(); const newEvent = new ICAL.Component('vevent');
newEvent.uid = event.uid; // 1. Add DTSTART
newEvent.startDate = event.startDate processDateTimeProperty(event, 'dtstart', newEvent);
newEvent.endDate = event.endDate
const dtstamp = vevent.getFirstPropertyValue('dtstamp'); // 2. Add DTEND
if (dtstamp) newEvent.component.updatePropertyWithValue('dtstamp', dtstamp); processDateTimeProperty(event, 'dtend', newEvent);
if (overrideFlag) { // 3. Copy DTSTAMP
newEvent.summary = 'Busy' const dtstamp = event.getFirstPropertyValue('dtstamp');
if (dtstamp) newEvent.updatePropertyWithValue('dtstamp', dtstamp);
// 4. Copy UID
newEvent.updatePropertyWithValue('uid', vevent.uid);
// 5. Add LOCATION (conditionally included)
if (!overrideFlag && vevent.location) {
newEvent.updatePropertyWithValue('location', vevent.location);
} else if (overrideFlag && vevent.location) {
// Modify SUMMARY if override is set
const modifiedSummary = `${vevent.summary.trim()} (Location omitted)`;
newEvent.updatePropertyWithValue('summary', modifiedSummary);
} else { } else {
newEvent.summary = event.summary; newEvent.updatePropertyWithValue('summary', vevent.summary.trim());
if (event.location) newEvent.location = event.location;
} }
const rrule = vevent.getFirstPropertyValue('rrule'); // 6. Copy Recurrence Rules (RRULE) and Recurrence ID
if (rrule) newEvent.component.updatePropertyWithValue('rrule', rrule); const rrule = event.getFirstPropertyValue('rrule');
if (rrule) newEvent.updatePropertyWithValue('rrule', rrule);
const recurrenceId = event.getFirstPropertyValue('recurrence-id');
if (recurrenceId) newEvent.updatePropertyWithValue('recurrence-id', recurrenceId);
// 7. Copy SUMMARY
newEvent.updatePropertyWithValue('summary', vevent.summary.trim());
// 8. Add SEQUENCE (if available or default to 0)
const sequence = event.getFirstPropertyValue('sequence') || 0;
newEvent.updatePropertyWithValue('sequence', sequence);
// Add the VEVENT to the calendar // Add the VEVENT to the calendar
newCalendar.addSubcomponent(newEvent.component); calendarComponent.addSubcomponent(newEvent);
}); });
} catch (error) {
console.error(`Error processing calendar:`, calendarRaw, error); console.log(`Processed VEVENT components for calendar: ${result.name}`);
} catch (error) {
console.error('Error processing calendar data:', error.message);
} }
}); });
} }

View File

@ -47,9 +47,9 @@ async function refreshCalendarData(calendarName) {
// Read the JSON file to get the source URL and other details // Read the JSON file to get the source URL and other details
const { calendars } = JSON.parse(fs.readFileSync(jsonFilePath, 'utf-8')); const { calendars } = JSON.parse(fs.readFileSync(jsonFilePath, 'utf-8'));
const calendarResults = await Promise.all(calendars.map(fetchCalendarData)); const results = await Promise.all(calendars.map(fetchCalendarData));
const calendarComponent = createCalendarComponent(calendarName); const calendarComponent = createCalendarComponent(calendarName);
addEventsToCalendar(calendarComponent, calendarResults); addEventsToCalendar(calendarComponent, results);
saveCalendarFile(`${calendarName}.ics`, calendarComponent.toString()); saveCalendarFile(`${calendarName}.ics`, calendarComponent.toString());
console.log('Calendar data refreshed and saved.'); console.log('Calendar data refreshed and saved.');

View File

@ -2,13 +2,13 @@ import express from 'express';
import path from 'path'; import path from 'path';
import routes from './routes.js'; import routes from './routes.js';
console.log(`Starting server in ${process.cwd()}`);
const app = express(); const app = express();
app.use(express.json()); app.use(express.json());
// Serve static files from the 'public' directory // Serve static files from the 'public' directory
app.use(express.static(path.join(process.cwd(), 'public'))); app.use(express.static(path.join(process.cwd(), 'public')));
app.use('/', routes); app.use('/', routes);
export default app; export default app;

View File

@ -1,6 +1,8 @@
import request from 'supertest'; import request from 'supertest';
import express from 'express';
import fs from 'fs'; import fs from 'fs';
import path from 'path'; import path from 'path';
import { jest } from '@jest/globals';
import { fileURLToPath } from 'url'; import { fileURLToPath } from 'url';
// ESM equivalent of __dirname // ESM equivalent of __dirname
@ -13,18 +15,20 @@ const EXPECTED_OUTPUTS_DIR = path.join(__dirname, 'expected_outputs');
let server; let server;
process.chdir(__dirname) process.chdir(__dirname)
const app = await import('../src/server'); console.log(process.cwd());
import app from '../src/server.js';
const normalizeLineEndings = (str) => str.replace(/\r\n/g, '\r\n').trimEnd(); // Normalize to CRLF
describe('Calendar Merging API', () => { describe('Calendar Merging API', () => {
beforeAll(async () => { beforeAll(async () => {
// Start the server
server = app.listen(0);
// Clean up the merged calendars directory before tests // Clean up the merged calendars directory before tests
fs.rmdirSync(CALENDARS_DIR, { recursive: true }); fs.rmdirSync(CALENDARS_DIR, { recursive: true });
if (fs.existsSync(CALENDARS_DIR)) { if (fs.existsSync(CALENDARS_DIR)) {
fs.rmdirSync(CALENDARS_DIR, { recursive: true }); fs.rmdirSync(CALENDARS_DIR, { recursive: true });
} }
// Start the server
server = app.default.listen(0);
}); });
afterAll(async () => { afterAll(async () => {
@ -67,8 +71,12 @@ describe('Calendar Merging API', () => {
const expectedOutput = fs.readFileSync(input, 'utf8'); const expectedOutput = fs.readFileSync(input, 'utf8');
const actualOutput = fs.readFileSync(filePath, 'utf8'); const actualOutput = fs.readFileSync(filePath, 'utf8');
// Normalize line endings
const normalizedActual = normalizeLineEndings(actualOutput);
const normalizedExpected = normalizeLineEndings(expectedOutput);
//compare //compare
expect(actualOutput).toBe(expectedOutput); expect(normalizedActual).toBe(normalizedExpected);
}); });
test('Preserve google calendar', async () => { test('Preserve google calendar', async () => {
@ -95,8 +103,12 @@ describe('Calendar Merging API', () => {
const expectedOutput = fs.readFileSync(input, 'utf8'); const expectedOutput = fs.readFileSync(input, 'utf8');
const actualOutput = fs.readFileSync(filePath, 'utf8'); const actualOutput = fs.readFileSync(filePath, 'utf8');
// Normalize line endings
const normalizedActual = normalizeLineEndings(actualOutput);
const normalizedExpected = normalizeLineEndings(expectedOutput);
//compare //compare
expect(actualOutput).toBe(expectedOutput); expect(normalizedActual).toBe(normalizedExpected);
}); });

View File

@ -1,13 +1,11 @@
BEGIN:VCALENDAR BEGIN:VCALENDAR
NAME:US Holidays NAME:US Holidays
PRODID:-//CalMerge//Calendar Merger 1.0//EN
VERSION:2.0 VERSION:2.0
CALSCALE:GREGORIAN
BEGIN:VEVENT BEGIN:VEVENT
UID:20231225T000000-001@example.com UID:20231225T000000-001@example.com
DTSTAMP:20231225T000000Z
DTSTART;VALUE=DATE:20231225 DTSTART;VALUE=DATE:20231225
DTEND;VALUE=DATE:20231226 DTEND;VALUE=DATE:20231226
DTSTAMP:20231225T000000Z
SUMMARY:Christmas Day SUMMARY:Christmas Day
END:VEVENT END:VEVENT
END:VCALENDAR END:VCALENDAR

View File

@ -6,10 +6,11 @@ CALSCALE:GREGORIAN
METHOD:PUBLISH METHOD:PUBLISH
X-WR-TIMEZONE:Africa/Nairobi X-WR-TIMEZONE:Africa/Nairobi
BEGIN:VEVENT BEGIN:VEVENT
UID:6tbrvsitniuu72li7kk15gou2b@google.com
DTSTART:20241003T190000Z DTSTART:20241003T190000Z
DTEND:20241003T200000Z DTEND:20241003T200000Z
DTSTAMP:20241119T115316Z DTSTAMP:20241119T115316Z
UID:6tbrvsitniuu72li7kk15gou2b@google.com
SUMMARY:progodessey SUMMARY:progodessey
SEQUENCE:0
END:VEVENT END:VEVENT
END:VCALENDAR END:VCALENDAR

View File

@ -30,10 +30,10 @@ SUMMARY:progodessey
TRANSP:OPAQUE TRANSP:OPAQUE
END:VEVENT END:VEVENT
BEGIN:VEVENT BEGIN:VEVENT
UID:7l7n9nltrudluv65gfgll2q930@google.com
DTSTART:20241009T173000Z DTSTART:20241009T173000Z
DTEND:20241009T183000Z DTEND:20241009T183000Z
DTSTAMP:20241119T115316Z DTSTAMP:20241119T115316Z
UID:7l7n9nltrudluv65gfgll2q930@google.com
CREATED:20241010T123337Z CREATED:20241010T123337Z
LAST-MODIFIED:20241010T123337Z LAST-MODIFIED:20241010T123337Z
SEQUENCE:0 SEQUENCE:0

View File

@ -21,12 +21,13 @@ RRULE:FREQ=YEARLY;BYMONTH=10;BYDAY=-1SU
END:STANDARD END:STANDARD
END:VTIMEZONE END:VTIMEZONE
BEGIN:VEVENT BEGIN:VEVENT
UID:5f4ad965-16a8-48eb-8233-78bf93a8b35e
DTSTART;TZID=Europe/Berlin:20241120T211500 DTSTART;TZID=Europe/Berlin:20241120T211500
DTEND;TZID=Europe/Berlin:20241120T215000 DTEND;TZID=Europe/Berlin:20241120T215000
DTSTAMP:20241113T212909Z DTSTAMP:20241113T212909Z
SUMMARY:JR Weekly Check-In UID:5f4ad965-16a8-48eb-8233-78bf93a8b35e
LOCATION:FaceTime LOCATION:FaceTime
RRULE:FREQ=WEEKLY;BYDAY=WE RRULE:FREQ=WEEKLY;BYDAY=WE
SUMMARY:JR Weekly Check-In
SEQUENCE:0
END:VEVENT END:VEVENT
END:VCALENDAR END:VCALENDAR