@mcp-monorepo/ics 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +16 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/event-store.d.ts +9 -0
- package/dist/lib/event-store.d.ts.map +1 -0
- package/dist/lib/event-store.js +84 -0
- package/dist/lib/event-store.js.map +1 -0
- package/dist/lib/filter-events.d.ts +3 -0
- package/dist/lib/filter-events.d.ts.map +1 -0
- package/dist/lib/filter-events.js +10 -0
- package/dist/lib/filter-events.js.map +1 -0
- package/dist/lib/format-date.d.ts +2 -0
- package/dist/lib/format-date.d.ts.map +1 -0
- package/dist/lib/format-date.js +19 -0
- package/dist/lib/format-date.js.map +1 -0
- package/dist/lib/recurrrence.d.ts +3 -0
- package/dist/lib/recurrrence.d.ts.map +1 -0
- package/dist/lib/recurrrence.js +49 -0
- package/dist/lib/recurrrence.js.map +1 -0
- package/dist/lib/types.d.ts +39 -0
- package/dist/lib/types.d.ts.map +1 -0
- package/dist/lib/types.js +2 -0
- package/dist/lib/types.js.map +1 -0
- package/dist/tools/fetch-events.d.ts +3 -0
- package/dist/tools/fetch-events.d.ts.map +1 -0
- package/dist/tools/fetch-events.js +66 -0
- package/dist/tools/fetch-events.js.map +1 -0
- package/dist/tools/get-current-datetime.d.ts +3 -0
- package/dist/tools/get-current-datetime.d.ts.map +1 -0
- package/dist/tools/get-current-datetime.js +24 -0
- package/dist/tools/get-current-datetime.js.map +1 -0
- package/dist/tools/search-events.d.ts +3 -0
- package/dist/tools/search-events.d.ts.map +1 -0
- package/dist/tools/search-events.js +75 -0
- package/dist/tools/search-events.js.map +1 -0
- package/package.json +40 -0
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":""}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { createMcpServer } from '@mcp-monorepo/shared';
|
|
3
|
+
import { logger } from '@mcp-monorepo/shared';
|
|
4
|
+
import { rawEvents } from './lib/event-store.js';
|
|
5
|
+
import { registerFetchEventsTool } from './tools/fetch-events.js';
|
|
6
|
+
import { registerGetCurrentDatetimeTool } from './tools/get-current-datetime.js';
|
|
7
|
+
import { registerSearchEventsTool } from './tools/search-events.js';
|
|
8
|
+
createMcpServer({
|
|
9
|
+
name: 'ics',
|
|
10
|
+
importMetaPath: import.meta.filename,
|
|
11
|
+
title: 'ICS Calendar MCP Server',
|
|
12
|
+
tools: [registerGetCurrentDatetimeTool, registerFetchEventsTool, registerSearchEventsTool],
|
|
13
|
+
}).then(() => logger.error);
|
|
14
|
+
rawEvents.refresh().then(() => logger.error);
|
|
15
|
+
setTimeout(() => rawEvents.refresh().then(() => logger.error), 1000 * 60 * 60);
|
|
16
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAA;AACtD,OAAO,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAA;AAE7C,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAA;AAChD,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAA;AACjE,OAAO,EAAE,8BAA8B,EAAE,MAAM,iCAAiC,CAAA;AAChF,OAAO,EAAE,wBAAwB,EAAE,MAAM,0BAA0B,CAAA;AAEnE,eAAe,CAAC;IACd,IAAI,EAAE,KAAK;IACX,cAAc,EAAE,MAAM,CAAC,IAAI,CAAC,QAAQ;IACpC,KAAK,EAAE,yBAAyB;IAChC,KAAK,EAAE,CAAC,8BAA8B,EAAE,uBAAuB,EAAE,wBAAwB,CAAC;CAC3F,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;AAE3B,SAAS,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;AAC5C,UAAU,CAAC,GAAG,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,IAAI,GAAG,EAAE,GAAG,EAAE,CAAC,CAAA"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { RefreshablePromise } from '@mcp-monorepo/shared';
|
|
2
|
+
import { type CalendarSource, type RawIcalEvent } from './types.js';
|
|
3
|
+
export declare const getIcsUrls: () => CalendarSource[];
|
|
4
|
+
export declare const parseIcsContent: (icsContent: string, source: string) => Promise<RawIcalEvent[]>;
|
|
5
|
+
export declare const rawEvents: RefreshablePromise<{
|
|
6
|
+
events: RawIcalEvent[];
|
|
7
|
+
errors: string[];
|
|
8
|
+
}>;
|
|
9
|
+
//# sourceMappingURL=event-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"event-store.d.ts","sourceRoot":"","sources":["../../src/lib/event-store.ts"],"names":[],"mappings":"AAGA,OAAO,EAAmB,kBAAkB,EAAE,MAAM,sBAAsB,CAAA;AAI1E,OAAO,EAAE,KAAK,cAAc,EAAE,KAAK,YAAY,EAAkB,MAAM,YAAY,CAAA;AAEnF,eAAO,MAAM,UAAU,QAAO,cAAc,EAsB3C,CAAA;AAED,eAAO,MAAM,eAAe,GAAU,YAAY,MAAM,EAAE,QAAQ,MAAM,KAAG,OAAO,CAAC,YAAY,EAAE,CAkBhG,CAAA;AAmDD,eAAO,MAAM,SAAS;YAhDZ,YAAY,EAAE;YACd,MAAM,EAAE;EA+C4C,CAAA"}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { access, mkdir, readFile, writeFile } from 'fs/promises';
|
|
2
|
+
import { dirname, join } from 'path';
|
|
3
|
+
import { findProjectRoot, RefreshablePromise } from '@mcp-monorepo/shared';
|
|
4
|
+
import { logger } from '@mcp-monorepo/shared';
|
|
5
|
+
import ical from 'node-ical';
|
|
6
|
+
export const getIcsUrls = () => {
|
|
7
|
+
const envVars = process.env;
|
|
8
|
+
const calendarSources = [];
|
|
9
|
+
Object.keys(envVars).forEach((key) => {
|
|
10
|
+
const ENV_PREFIX = 'CALENDAR_';
|
|
11
|
+
if (key.startsWith(ENV_PREFIX)) {
|
|
12
|
+
const url = envVars[key];
|
|
13
|
+
if (url && (url.startsWith('http://') || url.startsWith('https://'))) {
|
|
14
|
+
const name = key.substring(ENV_PREFIX.length);
|
|
15
|
+
calendarSources.push({ name, url });
|
|
16
|
+
}
|
|
17
|
+
else {
|
|
18
|
+
throw new Error(`Invalid URL for environment variable ${key}: ${url}`);
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
if (calendarSources.length === 0) {
|
|
23
|
+
throw new Error('No valid ICS URLs provided in environment variables.');
|
|
24
|
+
}
|
|
25
|
+
return calendarSources;
|
|
26
|
+
};
|
|
27
|
+
export const parseIcsContent = async (icsContent, source) => {
|
|
28
|
+
const entries = Object.values(await ical.async.parseICS(icsContent));
|
|
29
|
+
return entries
|
|
30
|
+
.filter((e) => e.type === 'VEVENT')
|
|
31
|
+
.map((event) => {
|
|
32
|
+
return {
|
|
33
|
+
uid: event.uid,
|
|
34
|
+
summary: event.summary,
|
|
35
|
+
description: event.description,
|
|
36
|
+
location: event.location,
|
|
37
|
+
dtstart: event.start,
|
|
38
|
+
dtend: event.end,
|
|
39
|
+
allDay: event.datetype === 'date',
|
|
40
|
+
source,
|
|
41
|
+
rrule: event.rrule,
|
|
42
|
+
rruleOptions: event.rrule ? event.rrule.origOptions : undefined,
|
|
43
|
+
};
|
|
44
|
+
});
|
|
45
|
+
};
|
|
46
|
+
async function refreshEvents() {
|
|
47
|
+
const sources = getIcsUrls();
|
|
48
|
+
logger.info('Refreshing events from: ', sources);
|
|
49
|
+
const events = [];
|
|
50
|
+
const errors = [];
|
|
51
|
+
for (const source of sources) {
|
|
52
|
+
try {
|
|
53
|
+
const dataFile = join(await findProjectRoot(import.meta.dirname), 'data', 'ics', source.url.replace(/[^a-zA-Z0-9]/g, '_') + '.json');
|
|
54
|
+
if (await access(dataFile).then(() => true, () => false)) {
|
|
55
|
+
logger.info('File found. Using cache of: ' + source.name + ' path: ' + dataFile);
|
|
56
|
+
const data = await readFile(dataFile, 'utf-8');
|
|
57
|
+
const parsed = JSON.parse(data);
|
|
58
|
+
events.push(...parsed);
|
|
59
|
+
}
|
|
60
|
+
else {
|
|
61
|
+
logger.info('File not found. Fetching source: ' + source.name + ' url: ' + source.url);
|
|
62
|
+
const icsContent = await fetch(source.url);
|
|
63
|
+
if (!(icsContent && icsContent.ok))
|
|
64
|
+
throw new Error(`Fetch failed for ${source.url}`);
|
|
65
|
+
const text = await icsContent.text();
|
|
66
|
+
const parsed = await parseIcsContent(text, source.name);
|
|
67
|
+
await mkdir(dirname(dataFile), { recursive: true });
|
|
68
|
+
await writeFile(dataFile, JSON.stringify(parsed));
|
|
69
|
+
events.push(...parsed);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
catch (e) {
|
|
73
|
+
logger.error(e);
|
|
74
|
+
errors.push(`${source.name}: ${e.message}`);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
errors.forEach((e) => logger.error(e));
|
|
78
|
+
return {
|
|
79
|
+
events,
|
|
80
|
+
errors,
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
export const rawEvents = new RefreshablePromise(refreshEvents);
|
|
84
|
+
//# sourceMappingURL=event-store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"event-store.js","sourceRoot":"","sources":["../../src/lib/event-store.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"filter-events.d.ts","sourceRoot":"","sources":["../../src/lib/filter-events.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,aAAa,EAAE,MAAM,YAAY,CAAA;AAE/C,wBAAgB,YAAY,CAAC,MAAM,EAAE,aAAa,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,IAAI,mBAQ3E"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export function filterEvents(events, start, end) {
|
|
2
|
+
return events
|
|
3
|
+
.filter((evt) => {
|
|
4
|
+
const st = evt.dtstart;
|
|
5
|
+
const en = evt.dtend || evt.dtstart;
|
|
6
|
+
return st <= end && en >= start;
|
|
7
|
+
})
|
|
8
|
+
.sort((a, b) => a.dtstart.getTime() - b.dtstart.getTime());
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=filter-events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"filter-events.js","sourceRoot":"","sources":["../../src/lib/filter-events.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,YAAY,CAAC,MAAuB,EAAE,KAAW,EAAE,GAAS;IAC1E,OAAO,MAAM;SACV,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;QACd,MAAM,EAAE,GAAG,GAAG,CAAC,OAAO,CAAA;QACtB,MAAM,EAAE,GAAG,GAAG,CAAC,KAAK,IAAI,GAAG,CAAC,OAAO,CAAA;QACnC,OAAO,EAAE,IAAI,GAAG,IAAI,EAAE,IAAI,KAAK,CAAA;IACjC,CAAC,CAAC;SACD,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC,CAAA;AAC9D,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"format-date.d.ts","sourceRoot":"","sources":["../../src/lib/format-date.ts"],"names":[],"mappings":"AAAA,wBAAgB,UAAU,CAAC,IAAI,EAAE,IAAI,EAAE,MAAM,UAAQ,GAAG,MAAM,CAiB7D"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
export function formatDate(date, allDay = false) {
|
|
2
|
+
if (allDay) {
|
|
3
|
+
return new Date(date).toLocaleDateString('en-US', {
|
|
4
|
+
weekday: 'long',
|
|
5
|
+
year: 'numeric',
|
|
6
|
+
month: 'long',
|
|
7
|
+
day: 'numeric',
|
|
8
|
+
});
|
|
9
|
+
}
|
|
10
|
+
return date.toLocaleString('en-US', {
|
|
11
|
+
weekday: 'long',
|
|
12
|
+
year: 'numeric',
|
|
13
|
+
month: 'long',
|
|
14
|
+
day: 'numeric',
|
|
15
|
+
hour: '2-digit',
|
|
16
|
+
minute: '2-digit',
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=format-date.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"format-date.js","sourceRoot":"","sources":["../../src/lib/format-date.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,UAAU,CAAC,IAAU,EAAE,MAAM,GAAG,KAAK;IACnD,IAAI,MAAM,EAAE,CAAC;QACX,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,kBAAkB,CAAC,OAAO,EAAE;YAChD,OAAO,EAAE,MAAM;YACf,IAAI,EAAE,SAAS;YACf,KAAK,EAAE,MAAM;YACb,GAAG,EAAE,SAAS;SACf,CAAC,CAAA;IACJ,CAAC;IACD,OAAO,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE;QAClC,OAAO,EAAE,MAAM;QACf,IAAI,EAAE,SAAS;QACf,KAAK,EAAE,MAAM;QACb,GAAG,EAAE,SAAS;QACd,IAAI,EAAE,SAAS;QACf,MAAM,EAAE,SAAS;KAClB,CAAC,CAAA;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"recurrrence.d.ts","sourceRoot":"","sources":["../../src/lib/recurrrence.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,YAAY,CAAA;AAqB7D,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,YAAY,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,GAAG,aAAa,EAAE,CA0CzG"}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
function getAdjustedRecurrenceDate(rruleOrigOptions, eventStart, recurrence) {
|
|
2
|
+
if (rruleOrigOptions && rruleOrigOptions.tzid) {
|
|
3
|
+
try {
|
|
4
|
+
const utc = recurrence.getTime() + eventStart.getTimezoneOffset() * 60000;
|
|
5
|
+
return new Date(utc);
|
|
6
|
+
}
|
|
7
|
+
catch {
|
|
8
|
+
return recurrence;
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
const origOffset = eventStart.getTimezoneOffset();
|
|
12
|
+
const recOffset = recurrence.getTimezoneOffset();
|
|
13
|
+
const offsetDiff = origOffset - recOffset;
|
|
14
|
+
return new Date(recurrence.getTime() + offsetDiff * 60000);
|
|
15
|
+
}
|
|
16
|
+
export function resolveRecurrence(events, startDate, endDate) {
|
|
17
|
+
const createCalendarEvent = (event, dtstart, dtend, uidSuffix) => ({
|
|
18
|
+
uid: uidSuffix ? `${event.uid}_${uidSuffix}` : event.uid,
|
|
19
|
+
summary: event.summary,
|
|
20
|
+
description: event.description,
|
|
21
|
+
location: event.location,
|
|
22
|
+
dtstart: dtstart ?? event.dtstart,
|
|
23
|
+
dtend: dtend ?? event.dtend,
|
|
24
|
+
allDay: event.allDay,
|
|
25
|
+
source: event.source,
|
|
26
|
+
});
|
|
27
|
+
const expandSingleEvent = (event) => {
|
|
28
|
+
if (!event.rrule) {
|
|
29
|
+
return [createCalendarEvent(event)];
|
|
30
|
+
}
|
|
31
|
+
try {
|
|
32
|
+
const recurrences = event.rrule.between(startDate, endDate);
|
|
33
|
+
if (recurrences.length === 0) {
|
|
34
|
+
return event.dtstart >= startDate && event.dtstart <= endDate ? [createCalendarEvent(event)] : [];
|
|
35
|
+
}
|
|
36
|
+
const duration = event.dtend && event.dtstart ? event.dtend.getTime() - event.dtstart.getTime() : 0;
|
|
37
|
+
return recurrences.map((recurrenceDate) => {
|
|
38
|
+
const adjustedStart = getAdjustedRecurrenceDate(event.rruleOptions, event.dtstart, recurrenceDate);
|
|
39
|
+
const adjustedEnd = duration ? new Date(adjustedStart.getTime() + duration) : undefined;
|
|
40
|
+
return createCalendarEvent(event, adjustedStart, adjustedEnd, adjustedStart.getTime().toString());
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
catch {
|
|
44
|
+
return [createCalendarEvent(event)];
|
|
45
|
+
}
|
|
46
|
+
};
|
|
47
|
+
return events.flatMap(expandSingleEvent);
|
|
48
|
+
}
|
|
49
|
+
//# sourceMappingURL=recurrrence.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"recurrrence.js","sourceRoot":"","sources":["../../src/lib/recurrrence.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
export interface CalendarEvent {
|
|
2
|
+
uid: string;
|
|
3
|
+
summary: string;
|
|
4
|
+
dtstart: Date;
|
|
5
|
+
dtend?: Date;
|
|
6
|
+
allDay: boolean;
|
|
7
|
+
location?: string;
|
|
8
|
+
description?: string;
|
|
9
|
+
source: string;
|
|
10
|
+
isRecurring?: boolean;
|
|
11
|
+
}
|
|
12
|
+
export interface RRuleLike {
|
|
13
|
+
between: (start: Date, end: Date) => Date[];
|
|
14
|
+
origOptions?: {
|
|
15
|
+
tzid?: string;
|
|
16
|
+
dtstart: Date;
|
|
17
|
+
[key: string]: unknown;
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
export interface RawIcalEvent {
|
|
21
|
+
uid: string;
|
|
22
|
+
summary: string;
|
|
23
|
+
description?: string;
|
|
24
|
+
location?: string;
|
|
25
|
+
dtstart: Date;
|
|
26
|
+
dtend?: Date;
|
|
27
|
+
allDay: boolean;
|
|
28
|
+
source: string;
|
|
29
|
+
rrule?: RRuleLike;
|
|
30
|
+
rruleOptions?: {
|
|
31
|
+
tzid?: string;
|
|
32
|
+
dtstart: Date;
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
export interface CalendarSource {
|
|
36
|
+
name: string;
|
|
37
|
+
url: string;
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/lib/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,aAAa;IAC5B,GAAG,EAAE,MAAM,CAAA;IACX,OAAO,EAAE,MAAM,CAAA;IACf,OAAO,EAAE,IAAI,CAAA;IACb,KAAK,CAAC,EAAE,IAAI,CAAA;IACZ,MAAM,EAAE,OAAO,CAAA;IACf,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB,MAAM,EAAE,MAAM,CAAA;IACd,WAAW,CAAC,EAAE,OAAO,CAAA;CACtB;AAED,MAAM,WAAW,SAAS;IACxB,OAAO,EAAE,CAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,IAAI,KAAK,IAAI,EAAE,CAAA;IAC3C,WAAW,CAAC,EAAE;QAAE,IAAI,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,IAAI,CAAC;QAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;KAAE,CAAA;CACvE;AAED,MAAM,WAAW,YAAY;IAC3B,GAAG,EAAE,MAAM,CAAA;IACX,OAAO,EAAE,MAAM,CAAA;IACf,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,OAAO,EAAE,IAAI,CAAA;IACb,KAAK,CAAC,EAAE,IAAI,CAAA;IACZ,MAAM,EAAE,OAAO,CAAA;IACf,MAAM,EAAE,MAAM,CAAA;IACd,KAAK,CAAC,EAAE,SAAS,CAAA;IACjB,YAAY,CAAC,EAAE;QAAE,IAAI,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,IAAI,CAAA;KAAE,CAAA;CAChD;AAED,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAA;IACZ,GAAG,EAAE,MAAM,CAAA;CACZ"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/lib/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fetch-events.d.ts","sourceRoot":"","sources":["../../src/tools/fetch-events.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AAExE,eAAO,MAAM,uBAAuB,GAAI,QAAQ,SAAS,SA8DrD,CAAA"}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { registerTool } from '@mcp-monorepo/shared';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
import { rawEvents } from '../lib/event-store.js';
|
|
4
|
+
import { filterEvents } from '../lib/filter-events.js';
|
|
5
|
+
import { formatDate } from '../lib/format-date.js';
|
|
6
|
+
import { resolveRecurrence } from '../lib/recurrrence.js';
|
|
7
|
+
export const registerFetchEventsTool = (server) => registerTool(server, {
|
|
8
|
+
name: 'fetch-events',
|
|
9
|
+
title: 'Fetch Calendar Events',
|
|
10
|
+
description: 'Fetches events from multiple ICS calendar URLs for a specified time period.',
|
|
11
|
+
inputSchema: {
|
|
12
|
+
offset: z.number().default(0).describe('Offset to start returning events from (default: 0)'),
|
|
13
|
+
limit: z.number().default(50).describe('Maximum number of events to return (default: 50)'),
|
|
14
|
+
startDate: z.string().describe('Start date in YYYY-MM-DD format'),
|
|
15
|
+
endDate: z.string().describe('End date in YYYY-MM-DD format'),
|
|
16
|
+
},
|
|
17
|
+
outputSchema: {
|
|
18
|
+
events: z.array(z.object({
|
|
19
|
+
summary: z.string(),
|
|
20
|
+
start: z.string(),
|
|
21
|
+
end: z.string().optional(),
|
|
22
|
+
description: z.string().optional(),
|
|
23
|
+
location: z.string().optional(),
|
|
24
|
+
uid: z.string(),
|
|
25
|
+
source: z.string(),
|
|
26
|
+
})),
|
|
27
|
+
errors: z.array(z.string()),
|
|
28
|
+
total: z.number(),
|
|
29
|
+
},
|
|
30
|
+
isReadOnly: true,
|
|
31
|
+
async fetcher({ endDate, limit, offset, startDate }) {
|
|
32
|
+
const start = new Date(startDate + 'T00:00:00');
|
|
33
|
+
const end = new Date(endDate + 'T23:59:59');
|
|
34
|
+
if (start > end)
|
|
35
|
+
throw new Error('Error: Start date must be before end date');
|
|
36
|
+
const events = await rawEvents;
|
|
37
|
+
const expandedEvents = resolveRecurrence(events.events, start, end);
|
|
38
|
+
const filteredEvents = filterEvents(expandedEvents, start, end);
|
|
39
|
+
return {
|
|
40
|
+
events: filteredEvents.slice(offset, offset + limit),
|
|
41
|
+
errors: events.errors,
|
|
42
|
+
total: filteredEvents.length,
|
|
43
|
+
};
|
|
44
|
+
},
|
|
45
|
+
formatter({ events, errors, total }) {
|
|
46
|
+
const formattedEvents = events.map((event) => ({
|
|
47
|
+
summary: event.summary,
|
|
48
|
+
start: formatDate(event.dtstart, event.allDay),
|
|
49
|
+
...(event.dtend ? { end: formatDate(event.dtend, event.allDay) } : {}),
|
|
50
|
+
description: event.description
|
|
51
|
+
? event.description.length > 400
|
|
52
|
+
? event.description.substring(0, 400) + '...'
|
|
53
|
+
: event.description
|
|
54
|
+
: undefined,
|
|
55
|
+
location: event.location,
|
|
56
|
+
uid: event.uid,
|
|
57
|
+
source: event.source,
|
|
58
|
+
}));
|
|
59
|
+
return {
|
|
60
|
+
events: formattedEvents,
|
|
61
|
+
errors,
|
|
62
|
+
total,
|
|
63
|
+
};
|
|
64
|
+
},
|
|
65
|
+
});
|
|
66
|
+
//# sourceMappingURL=fetch-events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fetch-events.js","sourceRoot":"","sources":["../../src/tools/fetch-events.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"get-current-datetime.d.ts","sourceRoot":"","sources":["../../src/tools/get-current-datetime.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AAExE,eAAO,MAAM,8BAA8B,GAAI,QAAQ,SAAS,SAoB5D,CAAA"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { registerTool } from '@mcp-monorepo/shared';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
import { formatDate } from '../lib/format-date.js';
|
|
4
|
+
export const registerGetCurrentDatetimeTool = (server) => registerTool(server, {
|
|
5
|
+
name: 'get-current-datetime',
|
|
6
|
+
title: 'Get current date and time',
|
|
7
|
+
description: 'Returns the current date and time in local format.',
|
|
8
|
+
inputSchema: {},
|
|
9
|
+
outputSchema: {
|
|
10
|
+
localDate: z.string(),
|
|
11
|
+
timestamp: z.number(),
|
|
12
|
+
},
|
|
13
|
+
isReadOnly: true,
|
|
14
|
+
async fetcher() {
|
|
15
|
+
return new Date();
|
|
16
|
+
},
|
|
17
|
+
formatter(date) {
|
|
18
|
+
return {
|
|
19
|
+
localDate: formatDate(date, false),
|
|
20
|
+
timestamp: date.valueOf(),
|
|
21
|
+
};
|
|
22
|
+
},
|
|
23
|
+
});
|
|
24
|
+
//# sourceMappingURL=get-current-datetime.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"get-current-datetime.js","sourceRoot":"","sources":["../../src/tools/get-current-datetime.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAA;AACnD,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AAEvB,OAAO,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAA;AAIlD,MAAM,CAAC,MAAM,8BAA8B,GAAG,CAAC,MAAiB,EAAE,EAAE,CAClE,YAAY,CAAC,MAAM,EAAE;IACnB,IAAI,EAAE,sBAAsB;IAC5B,KAAK,EAAE,2BAA2B;IAClC,WAAW,EAAE,oDAAoD;IACjE,WAAW,EAAE,EAAE;IACf,YAAY,EAAE;QACZ,SAAS,EAAE,CAAC,CAAC,MAAM,EAAE;QACrB,SAAS,EAAE,CAAC,CAAC,MAAM,EAAE;KACtB;IACD,UAAU,EAAE,IAAI;IAChB,KAAK,CAAC,OAAO;QACX,OAAO,IAAI,IAAI,EAAE,CAAA;IACnB,CAAC;IACD,SAAS,CAAC,IAAI;QACZ,OAAO;YACL,SAAS,EAAE,UAAU,CAAC,IAAI,EAAE,KAAK,CAAC;YAClC,SAAS,EAAE,IAAI,CAAC,OAAO,EAAE;SAC1B,CAAA;IACH,CAAC;CACF,CAAC,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search-events.d.ts","sourceRoot":"","sources":["../../src/tools/search-events.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AAExE,eAAO,MAAM,wBAAwB,GAAI,QAAQ,SAAS,SAkFtD,CAAA"}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { registerTool } from '@mcp-monorepo/shared';
|
|
2
|
+
import { performKeywordSearch } from '@mcp-monorepo/shared';
|
|
3
|
+
import { z } from 'zod';
|
|
4
|
+
import { rawEvents } from '../lib/event-store.js';
|
|
5
|
+
import { formatDate } from '../lib/format-date.js';
|
|
6
|
+
export const registerSearchEventsTool = (server) => registerTool(server, {
|
|
7
|
+
name: 'search-events',
|
|
8
|
+
title: 'Search Calendar Events',
|
|
9
|
+
description: 'Search through all calendar events using keywords. Searches in title and description, sorts by number of matching words.',
|
|
10
|
+
inputSchema: {
|
|
11
|
+
query: z
|
|
12
|
+
.string()
|
|
13
|
+
.describe('Search query with keywords separated by spaces, at least one keyword must be included'),
|
|
14
|
+
limit: z.number().default(50).describe('Maximum number of events to return (default: 50)'),
|
|
15
|
+
},
|
|
16
|
+
outputSchema: {
|
|
17
|
+
events: z.array(z.object({
|
|
18
|
+
summary: z.string(),
|
|
19
|
+
start: z.string(),
|
|
20
|
+
end: z.string().optional(),
|
|
21
|
+
description: z.string().optional(),
|
|
22
|
+
location: z.string().optional(),
|
|
23
|
+
uid: z.string(),
|
|
24
|
+
source: z.string(),
|
|
25
|
+
matchCount: z.number(),
|
|
26
|
+
matchedWords: z.array(z.string()),
|
|
27
|
+
})),
|
|
28
|
+
errors: z.array(z.string()),
|
|
29
|
+
total: z.number(),
|
|
30
|
+
searchQuery: z.string(),
|
|
31
|
+
},
|
|
32
|
+
isReadOnly: true,
|
|
33
|
+
async fetcher({ query, limit }) {
|
|
34
|
+
if (!query.trim()) {
|
|
35
|
+
throw new Error('Search query cannot be empty');
|
|
36
|
+
}
|
|
37
|
+
const events = await rawEvents;
|
|
38
|
+
const results = performKeywordSearch(query, events.events, (event) => [event.summary, event.description, event.location, event.source], (a, b) => new Date(a.dtstart).getTime() - new Date(b.dtstart).getTime());
|
|
39
|
+
// Limit results based on the specified `limit`
|
|
40
|
+
const limitedResults = results.slice(0, limit);
|
|
41
|
+
return {
|
|
42
|
+
results: limitedResults,
|
|
43
|
+
total: results.length,
|
|
44
|
+
searchQuery: query,
|
|
45
|
+
errors: events.errors,
|
|
46
|
+
};
|
|
47
|
+
},
|
|
48
|
+
formatter({ results, total, searchQuery, errors }) {
|
|
49
|
+
const formattedEvents = results.map((result) => {
|
|
50
|
+
const { description, dtend, dtstart, location, source, summary, uid } = result.match;
|
|
51
|
+
return {
|
|
52
|
+
summary: summary,
|
|
53
|
+
start: formatDate(dtstart, true),
|
|
54
|
+
...(dtend ? { end: formatDate(dtend, true) } : {}),
|
|
55
|
+
description: description
|
|
56
|
+
? description.length > 400
|
|
57
|
+
? description.substring(0, 400) + '...'
|
|
58
|
+
: description
|
|
59
|
+
: undefined,
|
|
60
|
+
location: location,
|
|
61
|
+
uid: uid,
|
|
62
|
+
source: source,
|
|
63
|
+
matchCount: result.matchCount,
|
|
64
|
+
matchedWords: result.matchedWords,
|
|
65
|
+
};
|
|
66
|
+
});
|
|
67
|
+
return {
|
|
68
|
+
events: formattedEvents,
|
|
69
|
+
total,
|
|
70
|
+
searchQuery,
|
|
71
|
+
errors,
|
|
72
|
+
};
|
|
73
|
+
},
|
|
74
|
+
});
|
|
75
|
+
//# sourceMappingURL=search-events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search-events.js","sourceRoot":"","sources":["../../src/tools/search-events.ts"],"names":[],"mappings":"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"}
|
package/package.json
ADDED
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@mcp-monorepo/ics",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"type": "module",
|
|
5
|
+
"description": "MCP server for calendar tools using ICS/ical feeds",
|
|
6
|
+
"main": "dist/index.js",
|
|
7
|
+
"types": "dist/index.d.ts",
|
|
8
|
+
"bin": {
|
|
9
|
+
"mcp-npm-server": "./dist/index.js"
|
|
10
|
+
},
|
|
11
|
+
"files": [
|
|
12
|
+
"dist"
|
|
13
|
+
],
|
|
14
|
+
"publishConfig": {
|
|
15
|
+
"access": "public"
|
|
16
|
+
},
|
|
17
|
+
"scripts": {
|
|
18
|
+
"build": "tsc -b",
|
|
19
|
+
"dev": "tsc --watch",
|
|
20
|
+
"start": "node dist/index.js",
|
|
21
|
+
"test": "vitest run --passWithNoTests",
|
|
22
|
+
"test:watch": "vitest",
|
|
23
|
+
"test:coverage": "vitest run --coverage",
|
|
24
|
+
"typecheck": "tsc --noEmit",
|
|
25
|
+
"clean": "rimraf dist tsconfig.tsbuildinfo"
|
|
26
|
+
},
|
|
27
|
+
"dependencies": {
|
|
28
|
+
"@modelcontextprotocol/sdk": "^1.23.1",
|
|
29
|
+
"zod": "^3.25.76",
|
|
30
|
+
"date-fns-tz": "^3.2.0",
|
|
31
|
+
"node-ical": "^0.20.1",
|
|
32
|
+
"@mcp-monorepo/shared": "*"
|
|
33
|
+
},
|
|
34
|
+
"devDependencies": {
|
|
35
|
+
"@types/node": "^22.14.1",
|
|
36
|
+
"typescript": "^5.8.3",
|
|
37
|
+
"vitest": "^3.2.4",
|
|
38
|
+
"rimraf": "^6.0.1"
|
|
39
|
+
}
|
|
40
|
+
}
|