diff --git a/CHANGELOG.md b/CHANGELOG.md
index 1e20d4d..2993e9f 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,18 @@
+## v0.6.0 (2025-07-29)
+
+### Feat
+
+- **calendar**: add comprehensive Calendar app support via CalDAV protocol
+
+### Fix
+
+- apply ruff formatting to pass CI checks
+- **calendar**: address PR feedback from maintainer
+
+### Refactor
+
+- **calendar**: optimize logging for production readiness
+
## v0.5.0 (2025-07-26)
### Feat
diff --git a/README.md b/README.md
index 5b4a8bf..0d7f730 100644
--- a/README.md
+++ b/README.md
@@ -13,6 +13,7 @@ The server provides integration with multiple Nextcloud apps, enabling LLMs to i
| App | Support Status | Description |
|-----|----------------|-------------|
| **Notes** | ✅ Full Support | Create, read, update, delete, and search notes. Handle attachments via WebDAV. |
+| **Calendar** | ✅ Full Support | Complete calendar integration - create, update, delete events. Support for recurring events, reminders, attendees, and all-day events via CalDAV. |
| **Tables** | ⚠️ Row Operations | Read table schemas and perform CRUD operations on table rows. Table management not yet supported. |
| **Files (WebDAV)** | ✅ Full Support | Complete file system access - browse directories, read/write files, create/delete resources. |
@@ -29,6 +30,22 @@ The server provides integration with multiple Nextcloud apps, enabling LLMs to i
| `nc_notes_delete_note` | Delete a note by ID |
| `nc_notes_search_notes` | Search notes by title or content |
+### Calendar Tools
+
+| Tool | Description |
+|------|-------------|
+| `nc_calendar_list_calendars` | List all available calendars for the user |
+| `nc_calendar_create_event` | Create a comprehensive calendar event with full feature support (recurring, reminders, attendees, etc.) |
+| `nc_calendar_list_events` | **Enhanced:** List events with advanced filtering (min attendees, duration, categories, status, search across all calendars) |
+| `nc_calendar_get_event` | Get detailed information about a specific event |
+| `nc_calendar_update_event` | Update any aspect of an existing event |
+| `nc_calendar_delete_event` | Delete a calendar event |
+| `nc_calendar_create_meeting` | Quick meeting creation with smart defaults |
+| `nc_calendar_get_upcoming_events` | Get upcoming events in the next N days |
+| `nc_calendar_find_availability` | **New:** Intelligent availability finder - find free time slots for meetings with attendee conflict detection |
+| `nc_calendar_bulk_operations` | **New:** Bulk update, delete, or move events matching filter criteria |
+| `nc_calendar_manage_calendar` | **New:** Create, delete, and manage calendar properties |
+
### Tables Tools
| Tool | Description |
@@ -89,6 +106,98 @@ await nc_webdav_write_file("NewProject/docs/notes.md", "# My Notes\n\nContent he
await nc_webdav_delete_resource("old_file.txt")
```
+### Calendar Integration
+
+The server provides comprehensive calendar integration through CalDAV, enabling you to:
+
+- List all available calendars
+- Create, read, update, and delete calendar events
+- Handle recurring events with RRULE support
+- Manage event reminders and notifications
+- Support all-day and timed events
+- Handle attendees and meeting invitations
+- Organize events with categories and priorities
+
+**Usage Examples:**
+
+```python
+# List available calendars
+calendars = await nc_calendar_list_calendars()
+
+# Create a simple event
+await nc_calendar_create_event(
+ calendar_name="personal",
+ title="Team Meeting",
+ start_datetime="2025-07-28T14:00:00",
+ end_datetime="2025-07-28T15:00:00",
+ description="Weekly team sync",
+ location="Conference Room A"
+)
+
+# Create a recurring weekly meeting
+await nc_calendar_create_event(
+ calendar_name="work",
+ title="Weekly Standup",
+ start_datetime="2025-07-28T09:00:00",
+ end_datetime="2025-07-28T09:30:00",
+ recurring=True,
+ recurrence_rule="FREQ=WEEKLY;BYDAY=MO"
+)
+
+# Quick meeting creation
+await nc_calendar_create_meeting(
+ title="Client Call",
+ date="2025-07-28",
+ time="15:00",
+ duration_minutes=60,
+ attendees="client@example.com,colleague@company.com"
+)
+
+# Get upcoming events
+events = await nc_calendar_get_upcoming_events(days_ahead=7)
+
+# Advanced search - find all meetings with 5+ attendees lasting 2+ hours
+long_meetings = await nc_calendar_list_events(
+ calendar_name="", # Search all calendars
+ search_all_calendars=True,
+ start_date="2025-07-01",
+ end_date="2025-07-31",
+ min_attendees=5,
+ min_duration_minutes=120,
+ title_contains="meeting"
+)
+
+# Find availability for a 1-hour meeting with specific attendees
+availability = await nc_calendar_find_availability(
+ duration_minutes=60,
+ attendees="sarah@company.com,mike@company.com",
+ date_range_start="2025-07-28",
+ date_range_end="2025-08-04",
+ business_hours_only=True,
+ exclude_weekends=True,
+ preferred_times="09:00-12:00,14:00-17:00"
+)
+
+# Bulk update all team meetings to new location
+bulk_result = await nc_calendar_bulk_operations(
+ operation="update",
+ title_contains="team meeting",
+ start_date="2025-08-01",
+ end_date="2025-08-31",
+ new_location="Conference Room B",
+ new_reminder_minutes=15
+)
+
+# Create a new project calendar
+new_calendar = await nc_calendar_manage_calendar(
+ action="create",
+ calendar_name="project-alpha",
+ display_name="Project Alpha Calendar",
+ description="Calendar for Project Alpha team",
+ color="#FF5722"
+)
+```
+
### Note Attachments
This server supports adding and retrieving note attachments via WebDAV. Please note the following behavior regarding attachments:
diff --git a/docker-compose.yml b/docker-compose.yml
index 94ff076..5a858fb 100644
--- a/docker-compose.yml
+++ b/docker-compose.yml
@@ -21,7 +21,7 @@ services:
restart: always
app:
- image: nextcloud:31.0.7@sha256:31d564f5f9f43f2aed0633854a2abd39155f85aa156997f7252f5af908efa99b
+ image: nextcloud:31.0.7@sha256:81dc361f8f216d8acff20bd3dea2226fb6cea883c277505cbb2ddd6327c867fa
#user: www-data:www-data
restart: always
#post_start:
diff --git a/nextcloud_mcp_server/client/__init__.py b/nextcloud_mcp_server/client/__init__.py
index 22db2e5..e6a27d8 100644
--- a/nextcloud_mcp_server/client/__init__.py
+++ b/nextcloud_mcp_server/client/__init__.py
@@ -11,6 +11,7 @@ import logging
from .notes import NotesClient
from .webdav import WebDAVClient
from .tables import TablesClient
+from .calendar import CalendarClient
from ..controllers.notes_search import NotesSearchController
logger = logging.getLogger(__name__)
@@ -46,6 +47,7 @@ class NextcloudClient:
self.notes = NotesClient(self._client, username)
self.webdav = WebDAVClient(self._client, username)
self.tables = TablesClient(self._client, username)
+ self.calendar = CalendarClient(self._client, username)
# Initialize controllers
self._notes_search = NotesSearchController()
diff --git a/nextcloud_mcp_server/client/calendar.py b/nextcloud_mcp_server/client/calendar.py
new file mode 100644
index 0000000..c268a67
--- /dev/null
+++ b/nextcloud_mcp_server/client/calendar.py
@@ -0,0 +1,977 @@
+"""CalDAV client for NextCloud calendar operations."""
+
+import xml.etree.ElementTree as ET
+from datetime import datetime, date
+from typing import Dict, Any, List, Optional, Tuple
+import logging
+from httpx import HTTPStatusError
+from icalendar import Calendar, Event as ICalEvent, vRecur, Alarm
+from datetime import timedelta
+import uuid
+
+from .base import BaseNextcloudClient
+
+logger = logging.getLogger(__name__)
+
+
+class CalendarClient(BaseNextcloudClient):
+ """Client for NextCloud CalDAV calendar operations."""
+
+ def _get_caldav_base_path(self) -> str:
+ """Helper to get the base CalDAV path for calendars."""
+ return f"/remote.php/dav/calendars/{self.username}"
+
+ def _get_principals_path(self) -> str:
+ """Helper to get the principals path for the user."""
+ return f"/remote.php/dav/principals/users/{self.username}"
+
+ async def list_calendars(self) -> List[Dict[str, Any]]:
+ """List all available calendars for the user."""
+ caldav_path = self._get_caldav_base_path()
+
+ propfind_body = """
+
+
+
+
+
+
+
+
+ """
+
+ headers = {
+ "Depth": "1",
+ "Content-Type": "application/xml",
+ "Accept": "application/xml",
+ }
+
+ try:
+ response = await self._client.request(
+ "PROPFIND", caldav_path, content=propfind_body, headers=headers
+ )
+ response.raise_for_status()
+
+ # Parse XML response
+ root = ET.fromstring(response.content)
+ calendars = []
+
+ for response_elem in root.findall(".//{DAV:}response"):
+ href = response_elem.find(".//{DAV:}href")
+ if href is None:
+ continue
+
+ href_text = href.text or ""
+ if not href_text.endswith("/"):
+ continue # Skip non-calendar resources
+
+ # Extract calendar name from href
+ calendar_name = href_text.rstrip("/").split("/")[-1]
+ if not calendar_name or calendar_name == self.username:
+ continue
+
+ # Get properties
+ propstat = response_elem.find(".//{DAV:}propstat")
+ if propstat is None:
+ continue
+
+ prop = propstat.find(".//{DAV:}prop")
+ if prop is None:
+ continue
+
+ # Check if it's a calendar resource
+ resourcetype = prop.find(".//{DAV:}resourcetype")
+ is_calendar = (
+ resourcetype is not None
+ and resourcetype.find(".//{urn:ietf:params:xml:ns:caldav}calendar")
+ is not None
+ )
+
+ if not is_calendar:
+ continue
+
+ # Extract calendar properties
+ displayname_elem = prop.find(".//{DAV:}displayname")
+ displayname = (
+ displayname_elem.text
+ if displayname_elem is not None
+ else calendar_name
+ )
+
+ description_elem = prop.find(
+ ".//{urn:ietf:params:xml:ns:caldav}calendar-description"
+ )
+ description = (
+ description_elem.text if description_elem is not None else ""
+ )
+
+ color_elem = prop.find(
+ ".//{http://calendarserver.org/ns/}calendar-color"
+ )
+ color = color_elem.text if color_elem is not None else "#1976D2"
+
+ calendars.append(
+ {
+ "name": calendar_name,
+ "display_name": displayname,
+ "description": description,
+ "color": color,
+ "href": href_text,
+ }
+ )
+
+ logger.debug(f"Found {len(calendars)} calendars")
+ return calendars
+
+ except HTTPStatusError as e:
+ if e.response.status_code == 401:
+ logger.warning(
+ "Authentication failed for CalDAV - Calendar app may not be enabled for this user"
+ )
+ return []
+ elif e.response.status_code == 404:
+ logger.warning(
+ "CalDAV endpoint not found - Calendar app may not be installed"
+ )
+ return []
+ logger.error(f"HTTP error listing calendars: {e}")
+ raise e
+ except Exception as e:
+ logger.error(f"Unexpected error listing calendars: {e}")
+ raise e
+
+ async def get_calendar_events(
+ self,
+ calendar_name: str,
+ start_date: str = "",
+ end_date: str = "",
+ limit: int = 50,
+ ) -> List[Dict[str, Any]]:
+ """List events in a calendar within date range."""
+ calendar_path = f"{self._get_caldav_base_path()}/{calendar_name}/"
+
+ # Build time range filter if dates provided
+ time_range_filter = ""
+ if start_date or end_date:
+ start_dt = start_date or "19700101T000000Z"
+ end_dt = end_date or "20301231T235959Z"
+ time_range_filter = f"""
+
+ """
+
+ report_body = f"""
+
+
+
+
+
+
+
+
+ {time_range_filter}
+
+
+
+ """
+
+ headers = {
+ "Depth": "1",
+ "Content-Type": "application/xml",
+ "Accept": "application/xml",
+ }
+
+ try:
+ response = await self._client.request(
+ "REPORT", calendar_path, content=report_body, headers=headers
+ )
+ response.raise_for_status()
+
+ # Parse XML response and extract events
+ root = ET.fromstring(response.content)
+ events = []
+
+ for response_elem in root.findall(".//{DAV:}response"):
+ href = response_elem.find(".//{DAV:}href")
+ if href is None:
+ continue
+
+ propstat = response_elem.find(".//{DAV:}propstat")
+ if propstat is None:
+ continue
+
+ prop = propstat.find(".//{DAV:}prop")
+ if prop is None:
+ continue
+
+ calendar_data = prop.find(
+ ".//{urn:ietf:params:xml:ns:caldav}calendar-data"
+ )
+ etag_elem = prop.find(".//{DAV:}getetag")
+
+ if calendar_data is not None and calendar_data.text:
+ event_data = self._parse_ical_event(calendar_data.text)
+ if event_data:
+ event_data["href"] = href.text
+ event_data["etag"] = (
+ etag_elem.text if etag_elem is not None else ""
+ )
+ events.append(event_data)
+
+ if len(events) >= limit:
+ break
+
+ logger.debug(f"Found {len(events)} events")
+ return events
+
+ except HTTPStatusError as e:
+ logger.error(f"HTTP error getting calendar events: {e}")
+ raise e
+ except Exception as e:
+ logger.error(f"Unexpected error getting calendar events: {e}")
+ raise e
+
+ async def create_event(
+ self, calendar_name: str, event_data: Dict[str, Any]
+ ) -> Dict[str, Any]:
+ """Create a new calendar event with comprehensive features."""
+ event_uid = str(uuid.uuid4())
+ event_filename = f"{event_uid}.ics"
+ event_path = f"{self._get_caldav_base_path()}/{calendar_name}/{event_filename}"
+
+ # Create iCalendar event
+ ical_content = self._create_ical_event(event_data, event_uid)
+
+ headers = {
+ "Content-Type": "text/calendar; charset=utf-8",
+ "If-None-Match": "*", # Ensure we're creating, not updating
+ }
+
+ try:
+ response = await self._client.put(
+ event_path, content=ical_content, headers=headers
+ )
+ response.raise_for_status()
+
+ logger.debug(f"Created event {event_uid}")
+ return {
+ "uid": event_uid,
+ "href": event_path,
+ "etag": response.headers.get("etag", ""),
+ "status_code": response.status_code,
+ }
+
+ except HTTPStatusError as e:
+ logger.error(f"HTTP error creating event: {e}")
+ raise e
+ except Exception as e:
+ logger.error(f"Unexpected error creating event: {e}")
+ raise e
+
+ async def update_event(
+ self,
+ calendar_name: str,
+ event_uid: str,
+ event_data: Dict[str, Any],
+ etag: str = "",
+ ) -> Dict[str, Any]:
+ """Update an existing calendar event."""
+ event_filename = f"{event_uid}.ics"
+ event_path = f"{self._get_caldav_base_path()}/{calendar_name}/{event_filename}"
+
+ # Get existing event data to merge with updates
+ existing_event_data = {}
+ if not etag:
+ try:
+ existing_event_data, current_etag = await self.get_event(
+ calendar_name, event_uid
+ )
+ etag = current_etag
+ except Exception:
+ # Continue without etag if we can't get it
+ pass
+
+ # Merge existing data with new data (new data takes precedence)
+ merged_data = {**existing_event_data, **event_data}
+
+ # Create updated iCalendar event
+ ical_content = self._create_ical_event(merged_data, event_uid)
+
+ headers = {
+ "Content-Type": "text/calendar; charset=utf-8",
+ }
+ if etag:
+ headers["If-Match"] = etag
+
+ try:
+ response = await self._client.put(
+ event_path, content=ical_content, headers=headers
+ )
+ response.raise_for_status()
+
+ logger.debug(f"Updated event {event_uid}")
+ return {
+ "uid": event_uid,
+ "href": event_path,
+ "etag": response.headers.get("etag", ""),
+ "status_code": response.status_code,
+ }
+
+ except HTTPStatusError as e:
+ logger.error(f"HTTP error updating event: {e}")
+ raise e
+ except Exception as e:
+ logger.error(f"Unexpected error updating event: {e}")
+ raise e
+
+ async def delete_event(self, calendar_name: str, event_uid: str) -> Dict[str, Any]:
+ """Delete a calendar event."""
+ event_filename = f"{event_uid}.ics"
+ event_path = f"{self._get_caldav_base_path()}/{calendar_name}/{event_filename}"
+
+ try:
+ response = await self._client.delete(event_path)
+ response.raise_for_status()
+
+ logger.debug(f"Deleted event {event_uid}")
+ return {"status_code": response.status_code}
+
+ except HTTPStatusError as e:
+ if e.response.status_code == 404:
+ logger.debug(f"Event {event_uid} not found")
+ return {"status_code": 404}
+ logger.error(f"HTTP error deleting event: {e}")
+ raise e
+ except Exception as e:
+ logger.error(f"Unexpected error deleting event: {e}")
+ raise e
+
+ async def get_event(
+ self, calendar_name: str, event_uid: str
+ ) -> Tuple[Dict[str, Any], str]:
+ """Get detailed information about a specific event."""
+ event_filename = f"{event_uid}.ics"
+ event_path = f"{self._get_caldav_base_path()}/{calendar_name}/{event_filename}"
+
+ headers = {"Accept": "text/calendar"}
+
+ try:
+ response = await self._client.get(event_path, headers=headers)
+ response.raise_for_status()
+
+ etag = response.headers.get("etag", "")
+ event_data = self._parse_ical_event(response.text)
+
+ if not event_data:
+ raise ValueError(f"Failed to parse event data for {event_uid}")
+
+ event_data["href"] = event_path
+ event_data["etag"] = etag
+
+ logger.debug(f"Retrieved event {event_uid}")
+ return event_data, etag
+
+ except HTTPStatusError as e:
+ logger.error(f"HTTP error getting event: {e}")
+ raise e
+ except Exception as e:
+ logger.error(f"Unexpected error getting event: {e}")
+ raise e
+
+ def _create_ical_event(self, event_data: Dict[str, Any], event_uid: str) -> str:
+ """Create iCalendar content from event data."""
+ cal = Calendar()
+ cal.add("prodid", "-//NextCloud MCP Server//EN")
+ cal.add("version", "2.0")
+
+ event = ICalEvent()
+ event.add("uid", event_uid)
+ event.add("summary", event_data.get("title", ""))
+ event.add("description", event_data.get("description", ""))
+ event.add("location", event_data.get("location", ""))
+
+ # Handle dates/times
+ start_str = event_data.get("start_datetime", "")
+ end_str = event_data.get("end_datetime", "")
+ all_day = event_data.get("all_day", False)
+
+ if start_str: # Only parse if start_datetime is provided
+ if all_day:
+ start_date = datetime.fromisoformat(start_str.split("T")[0]).date()
+ event.add("dtstart", start_date)
+ if end_str:
+ end_date = datetime.fromisoformat(end_str.split("T")[0]).date()
+ event.add("dtend", end_date)
+ else:
+ start_dt = datetime.fromisoformat(start_str.replace("Z", "+00:00"))
+ event.add("dtstart", start_dt)
+ if end_str:
+ end_dt = datetime.fromisoformat(end_str.replace("Z", "+00:00"))
+ event.add("dtend", end_dt)
+
+ # Add categories
+ categories = event_data.get("categories", "")
+ if categories:
+ event.add("categories", categories.split(","))
+
+ # Add priority and status
+ priority = event_data.get("priority", 5)
+ event.add("priority", priority)
+
+ status = event_data.get("status", "CONFIRMED")
+ event.add("status", status)
+
+ # Add privacy classification
+ privacy = event_data.get("privacy", "PUBLIC")
+ event.add("class", privacy)
+
+ # Add URL
+ url = event_data.get("url", "")
+ if url:
+ event.add("url", url)
+
+ # Handle recurrence
+ recurring = event_data.get("recurring", False)
+ if recurring:
+ recurrence_rule = event_data.get("recurrence_rule", "")
+ if recurrence_rule:
+ event.add("rrule", vRecur.from_ical(recurrence_rule))
+
+ # Add alarms/reminders
+ reminder_minutes = event_data.get("reminder_minutes", 0)
+ if reminder_minutes > 0:
+ alarm = Alarm()
+ alarm.add("action", "DISPLAY")
+ alarm.add("description", "Event reminder")
+ alarm.add("trigger", timedelta(minutes=-reminder_minutes))
+ event.add_component(alarm)
+
+ # Add attendees
+ attendees = event_data.get("attendees", "")
+ if attendees:
+ for email in attendees.split(","):
+ if email.strip():
+ event.add("attendee", f"mailto:{email.strip()}")
+
+ # Add timestamps
+ now = datetime.utcnow()
+ event.add("created", now)
+ event.add("dtstamp", now)
+ event.add("last-modified", now)
+
+ cal.add_component(event)
+ return cal.to_ical().decode("utf-8")
+
+ def _parse_ical_event(self, ical_text: str) -> Optional[Dict[str, Any]]:
+ """Parse iCalendar text and extract event data."""
+ try:
+ cal = Calendar.from_ical(ical_text)
+ for component in cal.walk():
+ if component.name == "VEVENT":
+ event_data = {
+ "uid": str(component.get("uid", "")),
+ "title": str(component.get("summary", "")),
+ "description": str(component.get("description", "")),
+ "location": str(component.get("location", "")),
+ "status": str(component.get("status", "CONFIRMED")),
+ "priority": int(component.get("priority", 5)),
+ "privacy": str(component.get("class", "PUBLIC")),
+ "url": str(component.get("url", "")),
+ }
+
+ # Handle dates
+ dtstart = component.get("dtstart")
+ if dtstart:
+ if isinstance(dtstart.dt, date) and not isinstance(
+ dtstart.dt, datetime
+ ):
+ event_data["start_datetime"] = dtstart.dt.isoformat()
+ event_data["all_day"] = True
+ else:
+ event_data["start_datetime"] = dtstart.dt.isoformat()
+ event_data["all_day"] = False
+
+ dtend = component.get("dtend")
+ if dtend:
+ if isinstance(dtend.dt, date) and not isinstance(
+ dtend.dt, datetime
+ ):
+ event_data["end_datetime"] = dtend.dt.isoformat()
+ else:
+ event_data["end_datetime"] = dtend.dt.isoformat()
+
+ # Handle categories
+ categories = component.get("categories")
+ if categories:
+ event_data["categories"] = self._extract_categories(categories)
+
+ # Handle recurrence
+ rrule = component.get("rrule")
+ if rrule:
+ event_data["recurring"] = True
+ event_data["recurrence_rule"] = str(rrule)
+
+ # Handle attendees
+ attendees = []
+ for attendee in component.get("attendee", []):
+ if isinstance(attendee, list):
+ attendees.extend(
+ str(a).replace("mailto:", "") for a in attendee
+ )
+ else:
+ attendees.append(str(attendee).replace("mailto:", ""))
+ if attendees:
+ event_data["attendees"] = ",".join(attendees)
+
+ return event_data
+
+ return None
+
+ except Exception as e:
+ logger.error(f"Error parsing iCalendar: {e}")
+ return None
+
+ def _extract_categories(self, categories_obj) -> str:
+ """Extract categories from icalendar object to string."""
+ if not categories_obj:
+ return ""
+
+ try:
+ # Handle icalendar vCategory objects
+ if hasattr(categories_obj, "cats"):
+ # vCategory object has a 'cats' attribute that's a list
+ return ", ".join(str(cat) for cat in categories_obj.cats)
+ elif hasattr(categories_obj, "__iter__") and not isinstance(
+ categories_obj, str
+ ):
+ # Handle lists or other iterables
+ return ", ".join(str(cat) for cat in categories_obj)
+ else:
+ # Handle strings or other objects
+ return str(categories_obj)
+ except Exception:
+ # Fallback to string conversion
+ return str(categories_obj)
+
+ async def search_events_across_calendars(
+ self,
+ start_date: str = "",
+ end_date: str = "",
+ filters: Optional[Dict[str, Any]] = None,
+ ) -> List[Dict[str, Any]]:
+ """Search events across all calendars with advanced filtering."""
+ try:
+ calendars = await self.list_calendars()
+ all_events = []
+
+ for calendar in calendars:
+ try:
+ events = await self.get_calendar_events(
+ calendar["name"], start_date, end_date
+ )
+
+ # Apply filters if provided
+ if filters:
+ events = self._apply_event_filters(events, filters)
+
+ # Add calendar info to each event
+ for event in events:
+ event["calendar_name"] = calendar["name"]
+ event["calendar_display_name"] = calendar.get(
+ "display_name", calendar["name"]
+ )
+
+ all_events.extend(events)
+ except Exception as e:
+ logger.warning(
+ f"Error getting events from calendar {calendar['name']}: {e}"
+ )
+ continue
+
+ return all_events
+
+ except Exception as e:
+ logger.error(f"Error searching events across calendars: {e}")
+ raise
+
+ def _apply_event_filters(
+ self, events: List[Dict[str, Any]], filters: Dict[str, Any]
+ ) -> List[Dict[str, Any]]:
+ """Apply advanced filters to event list."""
+ filtered_events = []
+
+ for event in events:
+ # Skip if event doesn't match filters
+ if not self._event_matches_filters(event, filters):
+ continue
+ filtered_events.append(event)
+
+ return filtered_events
+
+ def _event_matches_filters(
+ self, event: Dict[str, Any], filters: Dict[str, Any]
+ ) -> bool:
+ """Check if an event matches the provided filters."""
+ try:
+ # Filter by minimum attendees
+ if "min_attendees" in filters:
+ attendees = event.get("attendees", "")
+ attendee_count = len(attendees.split(",")) if attendees else 0
+ if attendee_count < filters["min_attendees"]:
+ return False
+
+ # Filter by minimum duration
+ if "min_duration_minutes" in filters:
+ start_str = event.get("start_datetime", "")
+ end_str = event.get("end_datetime", "")
+ if start_str and end_str:
+ try:
+ start_dt = datetime.fromisoformat(
+ start_str.replace("Z", "+00:00")
+ )
+ end_dt = datetime.fromisoformat(end_str.replace("Z", "+00:00"))
+ duration_minutes = (end_dt - start_dt).total_seconds() / 60
+ if duration_minutes < filters["min_duration_minutes"]:
+ return False
+ except Exception:
+ pass
+
+ # Filter by categories
+ if "categories" in filters:
+ event_categories = event.get("categories", "").lower()
+ required_categories = [cat.lower() for cat in filters["categories"]]
+ if not any(cat in event_categories for cat in required_categories):
+ return False
+
+ # Filter by status
+ if "status" in filters:
+ if event.get("status", "").upper() != filters["status"].upper():
+ return False
+
+ # Filter by title contains
+ if "title_contains" in filters:
+ title = event.get("title", "").lower()
+ search_term = filters["title_contains"].lower()
+ if search_term not in title:
+ return False
+
+ # Filter by location contains
+ if "location_contains" in filters:
+ location = event.get("location", "").lower()
+ search_term = filters["location_contains"].lower()
+ if search_term not in location:
+ return False
+
+ return True
+
+ except Exception:
+ # If filtering fails, include the event
+ return True
+
+ async def find_availability(
+ self,
+ duration_minutes: int,
+ attendees: Optional[List[str]] = None,
+ date_range_start: str = "",
+ date_range_end: str = "",
+ constraints: Optional[Dict[str, Any]] = None,
+ ) -> List[Dict[str, Any]]:
+ """Find available time slots for scheduling."""
+ try:
+ # Set default date range if not provided
+ if not date_range_start:
+ date_range_start = datetime.now().strftime("%Y-%m-%d")
+ if not date_range_end:
+ end_date = datetime.now() + timedelta(days=7)
+ date_range_end = end_date.strftime("%Y-%m-%d")
+
+ # Get all events in the date range
+ busy_events = await self.search_events_across_calendars(
+ start_date=date_range_start, end_date=date_range_end
+ )
+
+ # Filter events for relevant attendees if specified
+ if attendees:
+ relevant_events = []
+ for event in busy_events:
+ event_attendees = event.get("attendees", "").lower()
+ if any(
+ attendee.lower() in event_attendees for attendee in attendees
+ ):
+ relevant_events.append(event)
+ busy_events = relevant_events
+
+ # Apply constraints
+ constraints = constraints or {}
+ business_hours_only = constraints.get("business_hours_only", False)
+ exclude_weekends = constraints.get("exclude_weekends", False)
+ preferred_times = constraints.get("preferred_times", [])
+
+ # Generate time slots
+ available_slots = self._generate_available_slots(
+ busy_events,
+ duration_minutes,
+ date_range_start,
+ date_range_end,
+ business_hours_only,
+ exclude_weekends,
+ preferred_times,
+ )
+
+ return available_slots
+
+ except Exception as e:
+ logger.error(f"Error finding availability: {e}")
+ raise
+
+ def _generate_available_slots(
+ self,
+ busy_events: List[Dict[str, Any]],
+ duration_minutes: int,
+ start_date: str,
+ end_date: str,
+ business_hours_only: bool,
+ exclude_weekends: bool,
+ preferred_times: List[str],
+ ) -> List[Dict[str, Any]]:
+ """Generate available time slots."""
+ available_slots = []
+
+ try:
+ current_date = datetime.fromisoformat(start_date)
+ end_date_dt = datetime.fromisoformat(end_date)
+
+ while current_date <= end_date_dt:
+ # Skip weekends if requested
+ if exclude_weekends and current_date.weekday() >= 5:
+ current_date += timedelta(days=1)
+ continue
+
+ # Generate slots for this day
+ day_slots = self._generate_day_slots(
+ current_date,
+ busy_events,
+ duration_minutes,
+ business_hours_only,
+ preferred_times,
+ )
+ available_slots.extend(day_slots)
+
+ current_date += timedelta(days=1)
+
+ return available_slots[:10] # Limit to 10 slots
+
+ except Exception as e:
+ logger.error(f"Error generating available slots: {e}")
+ return []
+
+ def _generate_day_slots(
+ self,
+ date: datetime,
+ busy_events: List[Dict[str, Any]],
+ duration_minutes: int,
+ business_hours_only: bool,
+ preferred_times: List[str],
+ ) -> List[Dict[str, Any]]:
+ """Generate available slots for a specific day."""
+ slots = []
+
+ try:
+ # Define working hours
+ if business_hours_only:
+ start_hour, end_hour = 9, 17
+ else:
+ start_hour, end_hour = 8, 20
+
+ # Get busy periods for this day
+ day_busy_periods = []
+ for event in busy_events:
+ try:
+ event_start = datetime.fromisoformat(
+ event["start_datetime"].replace("Z", "+00:00")
+ )
+ event_end = datetime.fromisoformat(
+ event["end_datetime"].replace("Z", "+00:00")
+ )
+
+ # Check if event is on this day
+ if event_start.date() == date.date():
+ day_busy_periods.append((event_start.time(), event_end.time()))
+ except Exception:
+ continue
+
+ # Sort busy periods
+ day_busy_periods.sort()
+
+ # Generate potential slots
+ current_time = date.replace(
+ hour=start_hour, minute=0, second=0, microsecond=0
+ )
+ end_time = date.replace(hour=end_hour, minute=0, second=0, microsecond=0)
+ slot_duration = timedelta(minutes=duration_minutes)
+
+ while current_time + slot_duration <= end_time:
+ slot_end = current_time + slot_duration
+
+ # Check if slot conflicts with any busy period
+ if not self._slot_conflicts(
+ current_time.time(), slot_end.time(), day_busy_periods
+ ):
+ # Check preferred times if specified
+ if not preferred_times or self._slot_in_preferred_times(
+ current_time.time(), preferred_times
+ ):
+ slots.append(
+ {
+ "start_datetime": current_time.isoformat(),
+ "end_datetime": slot_end.isoformat(),
+ "duration_minutes": duration_minutes,
+ "date": date.date().isoformat(),
+ }
+ )
+
+ current_time += timedelta(minutes=30) # 30-minute increments
+
+ return slots
+
+ except Exception as e:
+ logger.error(f"Error generating day slots: {e}")
+ return []
+
+ def _slot_conflicts(self, slot_start, slot_end, busy_periods):
+ """Check if a time slot conflicts with busy periods."""
+ for busy_start, busy_end in busy_periods:
+ if slot_start < busy_end and slot_end > busy_start:
+ return True
+ return False
+
+ def _slot_in_preferred_times(self, slot_start, preferred_times):
+ """Check if slot falls within preferred time ranges."""
+ if not preferred_times:
+ return True
+
+ for time_range in preferred_times:
+ try:
+ start_str, end_str = time_range.split("-")
+ pref_start = datetime.strptime(start_str, "%H:%M").time()
+ pref_end = datetime.strptime(end_str, "%H:%M").time()
+
+ if pref_start <= slot_start <= pref_end:
+ return True
+ except Exception:
+ continue
+
+ return False
+
+ async def bulk_update_events(
+ self, filter_criteria: Dict[str, Any], update_data: Dict[str, Any]
+ ) -> Dict[str, Any]:
+ """Bulk update events matching filter criteria."""
+ try:
+ # Find events matching criteria
+ events = await self.search_events_across_calendars(
+ start_date=filter_criteria.get("start_date", ""),
+ end_date=filter_criteria.get("end_date", ""),
+ filters=filter_criteria,
+ )
+
+ updated_count = 0
+ failed_count = 0
+ results = []
+
+ for event in events:
+ try:
+ # Update the event
+ await self.update_event(
+ event["calendar_name"], event["uid"], update_data
+ )
+ updated_count += 1
+ results.append(
+ {
+ "uid": event["uid"],
+ "status": "updated",
+ "title": event.get("title", ""),
+ }
+ )
+ except Exception as e:
+ failed_count += 1
+ results.append(
+ {
+ "uid": event["uid"],
+ "status": "failed",
+ "error": str(e),
+ "title": event.get("title", ""),
+ }
+ )
+
+ return {
+ "total_found": len(events),
+ "updated_count": updated_count,
+ "failed_count": failed_count,
+ "results": results,
+ }
+
+ except Exception as e:
+ logger.error(f"Error in bulk update: {e}")
+ raise
+
+ async def create_calendar(
+ self,
+ calendar_name: str,
+ display_name: str = "",
+ description: str = "",
+ color: str = "#1976D2",
+ ) -> Dict[str, Any]:
+ """Create a new calendar."""
+ try:
+ # Calendar creation via CalDAV MKCALENDAR
+ calendar_path = f"{self._get_caldav_base_path()}/{calendar_name}/"
+
+ # Create MKCALENDAR body
+ mkcol_body = f"""
+
+
+
+ {display_name or calendar_name}
+ {color}
+ {description}
+
+
+
+
+
+ """
+
+ headers = {"Content-Type": "application/xml", "Depth": "0"}
+
+ response = await self._client.request(
+ "MKCALENDAR", calendar_path, content=mkcol_body, headers=headers
+ )
+ response.raise_for_status()
+
+ logger.debug(f"Created calendar: {calendar_name}")
+ return {
+ "name": calendar_name,
+ "display_name": display_name or calendar_name,
+ "description": description,
+ "color": color,
+ "status_code": response.status_code,
+ }
+
+ except Exception as e:
+ logger.error(f"Error creating calendar {calendar_name}: {e}")
+ raise
+
+ async def delete_calendar(self, calendar_name: str) -> Dict[str, Any]:
+ """Delete a calendar."""
+ try:
+ calendar_path = f"{self._get_caldav_base_path()}/{calendar_name}/"
+
+ response = await self._client.delete(calendar_path)
+ response.raise_for_status()
+
+ logger.debug(f"Deleted calendar: {calendar_name}")
+ return {"status_code": response.status_code}
+
+ except Exception as e:
+ logger.error(f"Error deleting calendar {calendar_name}: {e}")
+ raise
diff --git a/nextcloud_mcp_server/server/__init__.py b/nextcloud_mcp_server/server/__init__.py
index f938dc2..389a5ec 100644
--- a/nextcloud_mcp_server/server/__init__.py
+++ b/nextcloud_mcp_server/server/__init__.py
@@ -9,6 +9,7 @@ from collections.abc import AsyncIterator
from nextcloud_mcp_server.server.notes import configure_notes_tools
from nextcloud_mcp_server.server.tables import configure_tables_tools
from nextcloud_mcp_server.server.webdav import configure_webdav_tools
+from nextcloud_mcp_server.server.calendar import configure_calendar_tools
setup_logging()
@@ -51,6 +52,7 @@ async def nc_get_capabilities():
configure_notes_tools(mcp)
configure_tables_tools(mcp)
configure_webdav_tools(mcp)
+configure_calendar_tools(mcp)
def run():
diff --git a/nextcloud_mcp_server/server/calendar.py b/nextcloud_mcp_server/server/calendar.py
new file mode 100644
index 0000000..01b21fe
--- /dev/null
+++ b/nextcloud_mcp_server/server/calendar.py
@@ -0,0 +1,722 @@
+import logging
+from mcp.server.fastmcp import FastMCP, Context
+from nextcloud_mcp_server.client import NextcloudClient
+
+logger = logging.getLogger(__name__)
+def configure_calendar_tools(mcp: FastMCP):
+ # Calendar tools
+ @mcp.tool()
+ async def nc_calendar_list_calendars(ctx: Context):
+ """List all available calendars for the user"""
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+ return await client.calendar.list_calendars()
+
+
+ @mcp.tool()
+ async def nc_calendar_create_event(
+ calendar_name: str,
+ title: str,
+ start_datetime: str,
+ ctx: Context,
+ end_datetime: str = "",
+ all_day: bool = False,
+ description: str = "",
+ location: str = "",
+ categories: str = "",
+ recurring: bool = False,
+ recurrence_rule: str = "",
+ recurrence_end_date: str = "",
+ reminder_minutes: int = 15,
+ reminder_email: bool = False,
+ status: str = "CONFIRMED",
+ priority: int = 5,
+ privacy: str = "PUBLIC",
+ attendees: str = "",
+ url: str = "",
+ color: str = "",
+ ):
+ """Create a comprehensive calendar event with full feature support
+
+ Args:
+ calendar_name: Name of the calendar to create the event in
+ title: Event title
+ start_datetime: ISO format: "2025-01-15T14:00:00" or "2025-01-15" for all-day
+ ctx: MCP context
+ end_datetime: ISO format end time, empty for all-day events
+ all_day: Whether this is an all-day event
+ description: Event description/details
+ location: Event location
+ categories: Comma-separated categories (e.g., "work,meeting")
+ recurring: Whether this is a recurring event
+ recurrence_rule: RFC5545 RRULE (e.g., "FREQ=WEEKLY;BYDAY=MO,WE,FR")
+ recurrence_end_date: When to stop recurring
+ reminder_minutes: Minutes before event to send reminder
+ reminder_email: Whether to send email notification
+ status: Event status: CONFIRMED, TENTATIVE, or CANCELLED
+ priority: Priority level 1-9 (1=highest, 9=lowest, 5=normal)
+ privacy: Privacy level: PUBLIC, PRIVATE, or CONFIDENTIAL
+ attendees: Comma-separated email addresses
+ url: Related URL for the event
+ color: Event color (hex or name)
+
+ Returns:
+ Dict with event creation result
+ """
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ event_data = {
+ "title": title,
+ "start_datetime": start_datetime,
+ "end_datetime": end_datetime,
+ "all_day": all_day,
+ "description": description,
+ "location": location,
+ "categories": categories,
+ "recurring": recurring,
+ "recurrence_rule": recurrence_rule,
+ "recurrence_end_date": recurrence_end_date,
+ "reminder_minutes": reminder_minutes,
+ "reminder_email": reminder_email,
+ "status": status,
+ "priority": priority,
+ "privacy": privacy,
+ "attendees": attendees,
+ "url": url,
+ "color": color,
+ }
+
+ return await client.calendar.create_event(calendar_name, event_data)
+
+
+ @mcp.tool()
+ async def nc_calendar_list_events(
+ calendar_name: str,
+ ctx: Context,
+ start_date: str = "",
+ end_date: str = "",
+ limit: int = 50,
+ min_attendees: Optional[int] = None,
+ min_duration_minutes: Optional[int] = None,
+ categories: Optional[str] = None,
+ status: Optional[str] = None,
+ title_contains: Optional[str] = None,
+ location_contains: Optional[str] = None,
+ search_all_calendars: bool = False,
+ ):
+ """List events in a calendar (or all calendars) within date range with advanced filtering.
+
+ Args:
+ calendar_name: Name of the calendar to search. Ignored if search_all_calendars=True.
+ ctx: MCP context
+ start_date: Start date for search (YYYY-MM-DD format, e.g., "2025-01-01")
+ end_date: End date for search (YYYY-MM-DD format, e.g., "2025-01-31")
+ limit: Maximum number of events to return
+ min_attendees: Filter events with at least this many attendees
+ min_duration_minutes: Filter events with at least this duration
+ categories: Filter events containing any of these categories (comma-separated, e.g., "work,meeting")
+ status: Filter events by status (CONFIRMED, TENTATIVE, or CANCELLED)
+ title_contains: Filter events where title contains this text
+ location_contains: Filter events where location contains this text
+ search_all_calendars: If True, search across all calendars instead of just one
+
+ Returns:
+ List of events matching the filters
+ """
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ # Build filters dictionary
+ filters = {}
+ if min_attendees is not None:
+ filters["min_attendees"] = min_attendees
+ if min_duration_minutes is not None:
+ filters["min_duration_minutes"] = min_duration_minutes
+ if categories is not None:
+ filters["categories"] = [cat.strip() for cat in categories.split(",")]
+ if status is not None:
+ filters["status"] = status
+ if title_contains is not None:
+ filters["title_contains"] = title_contains
+ if location_contains is not None:
+ filters["location_contains"] = location_contains
+
+ if search_all_calendars:
+ # Search across all calendars with filters
+ events = await client.calendar.search_events_across_calendars(
+ start_date=start_date,
+ end_date=end_date,
+ filters=filters if filters else None,
+ )
+ return events[:limit]
+ else:
+ # Search in specific calendar
+ events = await client.calendar.get_calendar_events(
+ calendar_name=calendar_name,
+ start_date=start_date,
+ end_date=end_date,
+ limit=limit,
+ )
+
+ # Apply filters if provided
+ if filters:
+ events = client.calendar._apply_event_filters(events, filters)
+
+ return events
+
+
+ @mcp.tool()
+ async def nc_calendar_get_event(
+ calendar_name: str,
+ event_uid: str,
+ ctx: Context,
+ ):
+ """Get detailed information about a specific event"""
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+ event_data, etag = await client.calendar.get_event(calendar_name, event_uid)
+ return event_data
+
+
+ @mcp.tool()
+ async def nc_calendar_update_event(
+ calendar_name: str,
+ event_uid: str,
+ ctx: Context,
+ # All the same parameters as create_event but optional
+ title: str | None = None,
+ start_datetime: str | None = None,
+ end_datetime: str | None = None,
+ all_day: bool | None = None,
+ description: str | None = None,
+ location: str | None = None,
+ categories: str | None = None,
+ # Recurrence updates
+ recurring: bool | None = None,
+ recurrence_rule: str | None = None,
+ # Notification updates
+ reminder_minutes: int | None = None,
+ reminder_email: bool | None = None,
+ # Event property updates
+ status: str | None = None,
+ priority: int | None = None,
+ privacy: str | None = None,
+ attendees: str | None = None,
+ url: str | None = None,
+ color: str | None = None,
+ etag: str = "",
+ ):
+ """Update any aspect of an existing event"""
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ # Build update data with only non-None values
+ event_data = {}
+ if title is not None:
+ event_data["title"] = title
+ if start_datetime is not None:
+ event_data["start_datetime"] = start_datetime
+ if end_datetime is not None:
+ event_data["end_datetime"] = end_datetime
+ if all_day is not None:
+ event_data["all_day"] = all_day
+ if description is not None:
+ event_data["description"] = description
+ if location is not None:
+ event_data["location"] = location
+ if categories is not None:
+ event_data["categories"] = categories
+ if recurring is not None:
+ event_data["recurring"] = recurring
+ if recurrence_rule is not None:
+ event_data["recurrence_rule"] = recurrence_rule
+ if reminder_minutes is not None:
+ event_data["reminder_minutes"] = reminder_minutes
+ if reminder_email is not None:
+ event_data["reminder_email"] = reminder_email
+ if status is not None:
+ event_data["status"] = status
+ if priority is not None:
+ event_data["priority"] = priority
+ if privacy is not None:
+ event_data["privacy"] = privacy
+ if attendees is not None:
+ event_data["attendees"] = attendees
+ if url is not None:
+ event_data["url"] = url
+ if color is not None:
+ event_data["color"] = color
+
+ return await client.calendar.update_event(
+ calendar_name, event_uid, event_data, etag
+ )
+
+
+ @mcp.tool()
+ async def nc_calendar_delete_event(
+ calendar_name: str,
+ event_uid: str,
+ ctx: Context,
+ ):
+ """Delete a calendar event"""
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+ return await client.calendar.delete_event(calendar_name, event_uid)
+
+
+ @mcp.tool()
+ async def nc_calendar_create_meeting(
+ title: str,
+ date: str,
+ time: str,
+ ctx: Context,
+ duration_minutes: int = 60,
+ calendar_name: str = "personal",
+ attendees: str = "",
+ location: str = "",
+ description: str = "",
+ reminder_minutes: int = 15,
+ ):
+ """Quick meeting creation with smart defaults
+
+ This is a convenience function for creating events with common meeting defaults.
+ It automatically:
+ - Calculates end time based on duration
+ - Sets status to CONFIRMED
+ - Adds a reminder
+ - Uses simpler date/time inputs instead of full ISO format
+
+ For full control over all event properties, use nc_calendar_create_event instead.
+
+ Args:
+ title: Meeting title
+ date: Meeting date (YYYY-MM-DD format, e.g., "2025-01-15")
+ time: Meeting start time (HH:MM format, e.g., "14:00")
+ ctx: MCP context
+ duration_minutes: Meeting duration in minutes (default: 60)
+ calendar_name: Calendar to create the meeting in (default: "personal")
+ attendees: Comma-separated email addresses of attendees
+ location: Meeting location
+ description: Meeting description/agenda
+ reminder_minutes: Minutes before meeting to send reminder (default: 15)
+
+ Returns:
+ Dict with meeting creation result
+ """
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ # Combine date and time for start_datetime
+ start_datetime = f"{date}T{time}:00"
+
+ # Calculate end_datetime
+
+ start_dt = datetime.fromisoformat(start_datetime)
+ end_dt = start_dt + timedelta(minutes=duration_minutes)
+ end_datetime = end_dt.isoformat()
+
+ event_data = {
+ "title": title,
+ "start_datetime": start_datetime,
+ "end_datetime": end_datetime,
+ "all_day": False,
+ "description": description,
+ "location": location,
+ "attendees": attendees,
+ "reminder_minutes": reminder_minutes,
+ "status": "CONFIRMED",
+ "priority": 5,
+ "privacy": "PUBLIC",
+ }
+
+ return await client.calendar.create_event(calendar_name, event_data)
+
+
+ @mcp.tool()
+ async def nc_calendar_get_upcoming_events(
+ ctx: Context,
+ calendar_name: str = "", # Empty = all calendars
+ days_ahead: int = 7,
+ limit: int = 10,
+ ):
+ """Get upcoming events in next N days"""
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ now = datetime.now()
+ end_date = now + timedelta(days=days_ahead)
+
+ start_date_str = now.strftime("%Y%m%dT%H%M%SZ")
+ end_date_str = end_date.strftime("%Y%m%dT%H%M%SZ")
+
+ if calendar_name:
+ # Get events from specific calendar
+ return await client.calendar.get_calendar_events(
+ calendar_name=calendar_name,
+ start_date=start_date_str,
+ end_date=end_date_str,
+ limit=limit,
+ )
+ else:
+ # Get events from all calendars
+ all_calendars = await client.calendar.list_calendars()
+ all_events = []
+
+ for calendar in all_calendars:
+ try:
+ events = await client.calendar.get_calendar_events(
+ calendar_name=calendar["name"],
+ start_date=start_date_str,
+ end_date=end_date_str,
+ limit=limit,
+ )
+ # Add calendar info to each event
+ for event in events:
+ event["calendar_name"] = calendar["name"]
+ event["calendar_display_name"] = calendar["display_name"]
+ all_events.extend(events)
+ except Exception as e:
+ logger.warning(
+ f"Error getting events from calendar {calendar['name']}: {e}"
+ )
+ continue
+
+ # Sort by start time and limit
+ all_events.sort(key=lambda x: x.get("start_datetime", ""))
+ return all_events[:limit]
+
+
+ @mcp.tool()
+ async def nc_calendar_find_availability(
+ duration_minutes: int,
+ ctx: Context,
+ attendees: str = "", # Comma-separated email list
+ date_range_start: str = "", # "2025-07-28"
+ date_range_end: str = "", # "2025-08-04"
+ business_hours_only: bool = True,
+ exclude_weekends: bool = True,
+ preferred_times: str = "", # Comma-separated time ranges like "09:00-12:00,14:00-17:00"
+ ):
+ """Find available time slots for scheduling meetings.
+
+ This tool intelligently analyzes existing calendar events to find free time slots
+ that work for all specified attendees within the given constraints.
+
+ Args:
+ duration_minutes: Required duration for the meeting in minutes
+ attendees: Comma-separated list of attendee email addresses to check availability for
+ date_range_start: Start date for availability search (YYYY-MM-DD)
+ date_range_end: End date for availability search (YYYY-MM-DD)
+ business_hours_only: Only suggest slots during business hours (9 AM - 5 PM)
+ exclude_weekends: Skip weekends when finding availability
+ preferred_times: Preferred time ranges as "HH:MM-HH:MM" (comma-separated)
+
+ Returns:
+ List of available time slots with start/end times and duration
+ """
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ # Parse attendees
+ attendee_list = []
+ if attendees:
+ attendee_list = [
+ email.strip() for email in attendees.split(",") if email.strip()
+ ]
+
+ # Parse preferred times
+ preferred_time_list = []
+ if preferred_times:
+ preferred_time_list = [
+ time_range.strip()
+ for time_range in preferred_times.split(",")
+ if time_range.strip()
+ ]
+
+ # Build constraints
+ constraints = {
+ "business_hours_only": business_hours_only,
+ "exclude_weekends": exclude_weekends,
+ "preferred_times": preferred_time_list,
+ }
+
+ return await client.calendar.find_availability(
+ duration_minutes=duration_minutes,
+ attendees=attendee_list,
+ date_range_start=date_range_start,
+ date_range_end=date_range_end,
+ constraints=constraints,
+ )
+
+
+ @mcp.tool()
+ async def nc_calendar_bulk_operations(
+ operation: str, # "update", "delete", "move"
+ ctx: Context,
+ title_contains: Optional[str] = None,
+ categories: Optional[str] = None, # Comma-separated
+ calendar_name: Optional[str] = None,
+ start_date: str = "", # "2025-07-01"
+ end_date: str = "", # "2025-07-31"
+ status: Optional[str] = None,
+ location_contains: Optional[str] = None,
+ # Update operation parameters
+ new_title: Optional[str] = None,
+ new_description: Optional[str] = None,
+ new_location: Optional[str] = None,
+ new_categories: Optional[str] = None,
+ new_priority: Optional[int] = None,
+ new_reminder_minutes: Optional[int] = None,
+ # Move operation parameters
+ target_calendar: Optional[str] = None,
+ ):
+ """Perform bulk operations (update/delete) on events matching filter criteria.
+
+ This tool allows you to efficiently modify or delete multiple events at once
+ by applying filters to find matching events and then performing the specified operation.
+
+ Args:
+ operation: Type of operation - "update" or "delete"
+ title_contains: Filter events where title contains this text
+ categories: Filter events containing any of these categories (comma-separated)
+ calendar_name: Filter events from this specific calendar
+ start_date: Filter events starting from this date (YYYY-MM-DD)
+ end_date: Filter events ending before this date (YYYY-MM-DD)
+ status: Filter events by status (CONFIRMED, TENTATIVE, CANCELLED)
+ location_contains: Filter events where location contains this text
+
+ # For update operations:
+ new_title: New title for matching events
+ new_description: New description for matching events
+ new_location: New location for matching events
+ new_categories: New categories for matching events (comma-separated)
+ new_priority: New priority for matching events (1-9, 5=normal)
+ new_reminder_minutes: New reminder time in minutes before event
+
+ # For move operations:
+ target_calendar: Calendar to move events to (requires operation="move")
+
+ Returns:
+ Summary of operation results including counts and details
+ """
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ if operation not in ["update", "delete", "move"]:
+ raise ValueError("Operation must be 'update', 'delete', or 'move'")
+
+ # Build filter criteria
+ filter_criteria = {}
+ if title_contains is not None:
+ filter_criteria["title_contains"] = title_contains
+ if categories is not None:
+ filter_criteria["categories"] = [cat.strip() for cat in categories.split(",")]
+ if status is not None:
+ filter_criteria["status"] = status
+ if location_contains is not None:
+ filter_criteria["location_contains"] = location_contains
+ if start_date:
+ filter_criteria["start_date"] = start_date
+ if end_date:
+ filter_criteria["end_date"] = end_date
+
+ if operation == "delete":
+ # Find matching events and delete them
+ if calendar_name:
+ events = await client.calendar.get_calendar_events(
+ calendar_name=calendar_name, start_date=start_date, end_date=end_date
+ )
+ if filter_criteria:
+ events = client.calendar._apply_event_filters(events, filter_criteria)
+ else:
+ events = await client.calendar.search_events_across_calendars(
+ start_date=start_date, end_date=end_date, filters=filter_criteria
+ )
+
+ deleted_count = 0
+ failed_count = 0
+ results = []
+
+ for event in events:
+ try:
+ await client.calendar.delete_event(
+ event.get("calendar_name", calendar_name), event["uid"]
+ )
+ deleted_count += 1
+ results.append(
+ {
+ "uid": event["uid"],
+ "status": "deleted",
+ "title": event.get("title", ""),
+ }
+ )
+ except Exception as e:
+ failed_count += 1
+ results.append(
+ {
+ "uid": event["uid"],
+ "status": "failed",
+ "error": str(e),
+ "title": event.get("title", ""),
+ }
+ )
+
+ return {
+ "operation": "delete",
+ "total_found": len(events),
+ "deleted_count": deleted_count,
+ "failed_count": failed_count,
+ "results": results,
+ }
+
+ elif operation == "update":
+ # Build update data
+ update_data = {}
+ if new_title is not None:
+ update_data["title"] = new_title
+ if new_description is not None:
+ update_data["description"] = new_description
+ if new_location is not None:
+ update_data["location"] = new_location
+ if new_categories is not None:
+ update_data["categories"] = new_categories
+ if new_priority is not None:
+ update_data["priority"] = new_priority
+ if new_reminder_minutes is not None:
+ update_data["reminder_minutes"] = new_reminder_minutes
+
+ if not update_data:
+ raise ValueError("No update data provided for update operation")
+
+ return await client.calendar.bulk_update_events(filter_criteria, update_data)
+
+ elif operation == "move":
+ if not target_calendar:
+ raise ValueError("target_calendar is required for move operation")
+
+ # Find matching events
+ if calendar_name:
+ events = await client.calendar.get_calendar_events(
+ calendar_name=calendar_name, start_date=start_date, end_date=end_date
+ )
+ if filter_criteria:
+ events = client.calendar._apply_event_filters(events, filter_criteria)
+ else:
+ events = await client.calendar.search_events_across_calendars(
+ start_date=start_date, end_date=end_date, filters=filter_criteria
+ )
+
+ moved_count = 0
+ failed_count = 0
+ results = []
+
+ for event in events:
+ try:
+ # Create event in target calendar
+ event_data = {
+ k: v
+ for k, v in event.items()
+ if k
+ not in [
+ "uid",
+ "href",
+ "etag",
+ "calendar_name",
+ "calendar_display_name",
+ ]
+ }
+
+ await client.calendar.create_event(target_calendar, event_data)
+
+ # Delete from source calendar
+ await client.calendar.delete_event(
+ event.get("calendar_name", calendar_name), event["uid"]
+ )
+
+ moved_count += 1
+ results.append(
+ {
+ "uid": event["uid"],
+ "status": "moved",
+ "title": event.get("title", ""),
+ "from_calendar": event.get("calendar_name", calendar_name),
+ "to_calendar": target_calendar,
+ }
+ )
+ except Exception as e:
+ failed_count += 1
+ results.append(
+ {
+ "uid": event["uid"],
+ "status": "failed",
+ "error": str(e),
+ "title": event.get("title", ""),
+ }
+ )
+
+ return {
+ "operation": "move",
+ "total_found": len(events),
+ "moved_count": moved_count,
+ "failed_count": failed_count,
+ "target_calendar": target_calendar,
+ "results": results,
+ }
+
+
+ @mcp.tool()
+ async def nc_calendar_manage_calendar(
+ action: str, # "create", "delete", "update", "list"
+ ctx: Context,
+ calendar_name: str = "",
+ display_name: str = "",
+ description: str = "",
+ color: str = "#1976D2", # Default blue color
+ ):
+ """Manage calendar creation, deletion, and properties.
+
+ This tool provides comprehensive calendar management functionality including
+ creating new calendars, deleting existing ones, and updating calendar properties.
+
+ Args:
+ action: Action to perform - "create", "delete", "update", or "list"
+ calendar_name: Internal name for the calendar (required for create/delete/update)
+ display_name: Human-readable name for the calendar (used for create/update)
+ description: Description for the calendar (used for create/update)
+ color: Hex color code for the calendar (e.g., "#1976D2" for blue)
+
+ Returns:
+ Result of the calendar management operation
+ """
+ client: NextcloudClient = ctx.request_context.lifespan_context.client
+
+ if action == "list":
+ return await client.calendar.list_calendars()
+
+ elif action == "create":
+ if not calendar_name:
+ raise ValueError("calendar_name is required for create action")
+
+ return await client.calendar.create_calendar(
+ calendar_name=calendar_name,
+ display_name=display_name or calendar_name,
+ description=description,
+ color=color,
+ )
+
+ elif action == "delete":
+ if not calendar_name:
+ raise ValueError("calendar_name is required for delete action")
+
+ return await client.calendar.delete_calendar(calendar_name)
+
+ elif action == "update":
+ if not calendar_name:
+ raise ValueError("calendar_name is required for update action")
+
+ # Note: Calendar property updates require additional CalDAV PROPPATCH implementation
+ # For now, return an informative message
+ return {
+ "status": "not_implemented",
+ "message": "Calendar property updates require PROPPATCH implementation",
+ "calendar_name": calendar_name,
+ "requested_changes": {
+ "display_name": display_name,
+ "description": description,
+ "color": color,
+ },
+ }
+
+ else:
+ raise ValueError("Action must be 'create', 'delete', 'update', or 'list'")
diff --git a/pyproject.toml b/pyproject.toml
index fc7ca8a..d1f7406 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -1,6 +1,6 @@
[project]
name = "nextcloud-mcp-server"
-version = "0.5.0"
+version = "0.6.0"
description = ""
authors = [
{name = "Chris Coutinho",email = "chris@coutinho.io"}
@@ -10,7 +10,8 @@ requires-python = ">=3.11"
dependencies = [
"mcp[cli] (>=1.10,<1.11)",
"httpx (>=0.28.1,<0.29.0)",
- "pillow (>=11.2.1,<12.0.0)"
+ "pillow (>=11.2.1,<12.0.0)",
+ "icalendar (>=6.0.0,<7.0.0)"
]
[tool.pytest.ini_options]
diff --git a/tests/integration/test_calendar_operations.py b/tests/integration/test_calendar_operations.py
new file mode 100644
index 0000000..2ac0239
--- /dev/null
+++ b/tests/integration/test_calendar_operations.py
@@ -0,0 +1,409 @@
+"""Integration tests for Calendar CalDAV operations."""
+
+import pytest
+import logging
+import uuid
+from datetime import datetime, timedelta
+from httpx import HTTPStatusError
+
+from nextcloud_mcp_server.client import NextcloudClient
+
+logger = logging.getLogger(__name__)
+
+# Mark all tests in this module as integration tests
+pytestmark = pytest.mark.integration
+
+
+@pytest.fixture
+def test_calendar_name():
+ """Unique calendar name for testing."""
+ return f"test_calendar_{uuid.uuid4().hex[:8]}"
+
+
+@pytest.fixture
+async def temporary_calendar(nc_client: NextcloudClient, test_calendar_name: str):
+ """Create a temporary calendar for testing and clean up afterward."""
+ calendar_name = test_calendar_name
+
+ try:
+ # Create a test calendar if possible
+ # Note: Calendar creation might require admin permissions
+ # For now, we'll use an existing calendar or create events in default calendar
+
+ # Try to find an existing calendar to use
+ calendars = await nc_client.calendar.list_calendars()
+ if calendars:
+ calendar_name = calendars[0]["name"]
+ logger.info(f"Using existing calendar: {calendar_name}")
+ yield calendar_name
+ else:
+ pytest.skip("No calendars available for testing")
+
+ except Exception as e:
+ logger.error(f"Error setting up temporary calendar: {e}")
+ pytest.skip(f"Calendar setup failed: {e}")
+
+
+@pytest.fixture
+async def temporary_event(nc_client: NextcloudClient, temporary_calendar: str):
+ """Create a temporary event for testing and clean up afterward."""
+ event_uid = None
+ calendar_name = temporary_calendar
+
+ # Create a test event
+ tomorrow = datetime.now() + timedelta(days=1)
+ event_data = {
+ "title": f"Test Event {uuid.uuid4().hex[:8]}",
+ "start_datetime": tomorrow.strftime("%Y-%m-%dT14:00:00"),
+ "end_datetime": tomorrow.strftime("%Y-%m-%dT15:00:00"),
+ "description": "Test event created by integration tests",
+ "location": "Test Location",
+ "categories": "testing",
+ "status": "CONFIRMED",
+ "priority": 5,
+ }
+
+ try:
+ logger.info(f"Creating temporary event in calendar: {calendar_name}")
+ result = await nc_client.calendar.create_event(calendar_name, event_data)
+ event_uid = result.get("uid")
+
+ if not event_uid:
+ pytest.fail("Failed to create temporary event")
+
+ logger.info(f"Created temporary event with UID: {event_uid}")
+ yield {"uid": event_uid, "calendar_name": calendar_name, "data": event_data}
+
+ finally:
+ # Cleanup
+ if event_uid:
+ try:
+ logger.info(f"Cleaning up temporary event: {event_uid}")
+ await nc_client.calendar.delete_event(calendar_name, event_uid)
+ logger.info(f"Successfully deleted temporary event: {event_uid}")
+ except HTTPStatusError as e:
+ if e.response.status_code != 404:
+ logger.error(f"Error deleting temporary event {event_uid}: {e}")
+ except Exception as e:
+ logger.error(
+ f"Unexpected error deleting temporary event {event_uid}: {e}"
+ )
+
+
+async def test_list_calendars(nc_client: NextcloudClient):
+ """Test listing available calendars."""
+ calendars = await nc_client.calendar.list_calendars()
+
+ assert isinstance(calendars, list)
+
+ if not calendars:
+ pytest.skip("No calendars available - Calendar app may not be enabled")
+
+ logger.info(f"Found {len(calendars)} calendars")
+
+ # Check structure of calendars
+ for calendar in calendars:
+ assert "name" in calendar
+ assert "display_name" in calendar
+ assert "href" in calendar
+ # Optional fields
+ assert "description" in calendar
+ assert "color" in calendar
+
+ logger.info(f"Calendar: {calendar['name']} - {calendar['display_name']}")
+
+
+async def test_create_and_delete_event(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test creating and deleting a basic event."""
+ calendar_name = temporary_calendar
+
+ # Create event
+ tomorrow = datetime.now() + timedelta(days=1)
+ event_data = {
+ "title": "Integration Test Event",
+ "start_datetime": tomorrow.strftime("%Y-%m-%dT10:00:00"),
+ "end_datetime": tomorrow.strftime("%Y-%m-%dT11:00:00"),
+ "description": "Test event for integration testing",
+ "location": "Test Room",
+ "categories": "testing,integration",
+ "status": "CONFIRMED",
+ "priority": 3,
+ }
+
+ try:
+ result = await nc_client.calendar.create_event(calendar_name, event_data)
+ assert "uid" in result
+ assert result["status_code"] in [200, 201, 204]
+
+ event_uid = result["uid"]
+ logger.info(f"Created event with UID: {event_uid}")
+
+ # Verify event was created by retrieving it
+ retrieved_event, etag = await nc_client.calendar.get_event(
+ calendar_name, event_uid
+ )
+ assert retrieved_event["uid"] == event_uid
+ assert retrieved_event["title"] == "Integration Test Event"
+ assert retrieved_event["location"] == "Test Room"
+
+ # Delete event
+ delete_result = await nc_client.calendar.delete_event(calendar_name, event_uid)
+ assert delete_result["status_code"] in [200, 204, 404]
+
+ logger.info(f"Successfully deleted event: {event_uid}")
+
+ except Exception as e:
+ logger.error(f"Test failed: {e}")
+ raise
+
+
+async def test_create_all_day_event(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test creating an all-day event."""
+ calendar_name = temporary_calendar
+
+ tomorrow = datetime.now() + timedelta(days=1)
+ event_data = {
+ "title": "All Day Test Event",
+ "start_datetime": tomorrow.strftime("%Y-%m-%d"),
+ "all_day": True,
+ "description": "Test all-day event",
+ "categories": "testing",
+ }
+
+ try:
+ result = await nc_client.calendar.create_event(calendar_name, event_data)
+ event_uid = result["uid"]
+ logger.info(f"Created all-day event with UID: {event_uid}")
+
+ # Verify event
+ retrieved_event, _ = await nc_client.calendar.get_event(
+ calendar_name, event_uid
+ )
+ assert retrieved_event["title"] == "All Day Test Event"
+ assert retrieved_event.get("all_day") is True
+
+ # Cleanup
+ await nc_client.calendar.delete_event(calendar_name, event_uid)
+
+ except Exception as e:
+ logger.error(f"All-day event test failed: {e}")
+ raise
+
+
+async def test_create_recurring_event(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test creating a recurring event."""
+ calendar_name = temporary_calendar
+
+ tomorrow = datetime.now() + timedelta(days=1)
+ event_data = {
+ "title": "Weekly Recurring Test",
+ "start_datetime": tomorrow.strftime("%Y-%m-%dT14:00:00"),
+ "end_datetime": tomorrow.strftime("%Y-%m-%dT15:00:00"),
+ "description": "Test recurring event",
+ "recurring": True,
+ "recurrence_rule": "FREQ=WEEKLY;BYDAY=MO,WE,FR",
+ "reminder_minutes": 30,
+ }
+
+ try:
+ result = await nc_client.calendar.create_event(calendar_name, event_data)
+ event_uid = result["uid"]
+ logger.info(f"Created recurring event with UID: {event_uid}")
+
+ # Verify event
+ retrieved_event, _ = await nc_client.calendar.get_event(
+ calendar_name, event_uid
+ )
+ assert retrieved_event["title"] == "Weekly Recurring Test"
+ assert retrieved_event.get("recurring") is True
+
+ # Cleanup
+ await nc_client.calendar.delete_event(calendar_name, event_uid)
+
+ except Exception as e:
+ logger.error(f"Recurring event test failed: {e}")
+ raise
+
+
+async def test_list_events_in_range(nc_client: NextcloudClient, temporary_event: dict):
+ """Test listing events within a date range."""
+ calendar_name = temporary_event["calendar_name"]
+
+ # Get events for the next week
+ start_date = datetime.now().strftime("%Y%m%dT000000Z")
+ end_date = (datetime.now() + timedelta(days=7)).strftime("%Y%m%dT235959Z")
+
+ events = await nc_client.calendar.get_calendar_events(
+ calendar_name=calendar_name, start_date=start_date, end_date=end_date, limit=50
+ )
+
+ assert isinstance(events, list)
+ logger.info(f"Found {len(events)} events in date range")
+
+ # Our temporary event should be in the list
+ event_uids = [event.get("uid") for event in events]
+ assert temporary_event["uid"] in event_uids
+
+ # Check event structure
+ for event in events:
+ assert "uid" in event
+ assert "title" in event
+ assert "start_datetime" in event
+
+
+async def test_update_event(nc_client: NextcloudClient, temporary_event: dict):
+ """Test updating an existing event."""
+ calendar_name = temporary_event["calendar_name"]
+ event_uid = temporary_event["uid"]
+
+ # Update event data
+ updated_data = {
+ "title": "Updated Test Event Title",
+ "description": "Updated description for test event",
+ "location": "Updated Location",
+ "priority": 1, # High priority
+ }
+
+ try:
+ result = await nc_client.calendar.update_event(
+ calendar_name, event_uid, updated_data
+ )
+ assert result["uid"] == event_uid
+
+ # Verify updates
+ updated_event, _ = await nc_client.calendar.get_event(calendar_name, event_uid)
+ assert updated_event["title"] == "Updated Test Event Title"
+ assert updated_event["description"] == "Updated description for test event"
+ assert updated_event["location"] == "Updated Location"
+ assert updated_event["priority"] == 1
+
+ logger.info(f"Successfully updated event: {event_uid}")
+
+ except Exception as e:
+ logger.error(f"Event update test failed: {e}")
+ raise
+
+
+async def test_create_event_with_attendees(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test creating an event with attendees."""
+ calendar_name = temporary_calendar
+
+ tomorrow = datetime.now() + timedelta(days=1)
+ event_data = {
+ "title": "Meeting with Attendees",
+ "start_datetime": tomorrow.strftime("%Y-%m-%dT16:00:00"),
+ "end_datetime": tomorrow.strftime("%Y-%m-%dT17:00:00"),
+ "description": "Test meeting with multiple attendees",
+ "location": "Conference Room A",
+ "attendees": "test1@example.com,test2@example.com",
+ "reminder_minutes": 15,
+ "status": "TENTATIVE",
+ }
+
+ try:
+ result = await nc_client.calendar.create_event(calendar_name, event_data)
+ event_uid = result["uid"]
+ logger.info(f"Created event with attendees, UID: {event_uid}")
+
+ # Verify event
+ retrieved_event, _ = await nc_client.calendar.get_event(
+ calendar_name, event_uid
+ )
+ assert retrieved_event["title"] == "Meeting with Attendees"
+ assert "test1@example.com" in retrieved_event.get("attendees", "")
+ assert retrieved_event["status"] == "TENTATIVE"
+
+ # Cleanup
+ await nc_client.calendar.delete_event(calendar_name, event_uid)
+
+ except Exception as e:
+ logger.error(f"Event with attendees test failed: {e}")
+ raise
+
+
+async def test_get_nonexistent_event(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test retrieving a non-existent event."""
+ calendar_name = temporary_calendar
+ fake_uid = f"nonexistent-{uuid.uuid4()}"
+
+ with pytest.raises(HTTPStatusError) as exc_info:
+ await nc_client.calendar.get_event(calendar_name, fake_uid)
+
+ assert exc_info.value.response.status_code == 404
+ logger.info(f"Correctly got 404 for nonexistent event: {fake_uid}")
+
+
+async def test_delete_nonexistent_event(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test deleting a non-existent event."""
+ calendar_name = temporary_calendar
+ fake_uid = f"nonexistent-{uuid.uuid4()}"
+
+ result = await nc_client.calendar.delete_event(calendar_name, fake_uid)
+ assert result["status_code"] == 404
+ logger.info(f"Correctly got 404 for deleting nonexistent event: {fake_uid}")
+
+
+async def test_event_with_url_and_categories(
+ nc_client: NextcloudClient, temporary_calendar: str
+):
+ """Test creating an event with URL and multiple categories."""
+ calendar_name = temporary_calendar
+
+ tomorrow = datetime.now() + timedelta(days=1)
+ event_data = {
+ "title": "Event with URL and Categories",
+ "start_datetime": tomorrow.strftime("%Y-%m-%dT09:00:00"),
+ "end_datetime": tomorrow.strftime("%Y-%m-%dT10:30:00"),
+ "description": "Test event with additional metadata",
+ "categories": "work,meeting,important,quarterly",
+ "url": "https://zoom.us/j/123456789",
+ "privacy": "PRIVATE",
+ "priority": 2,
+ }
+
+ try:
+ result = await nc_client.calendar.create_event(calendar_name, event_data)
+ event_uid = result["uid"]
+ logger.info(f"Created event with metadata, UID: {event_uid}")
+
+ # Verify event
+ retrieved_event, _ = await nc_client.calendar.get_event(
+ calendar_name, event_uid
+ )
+ assert retrieved_event["title"] == "Event with URL and Categories"
+ assert "work" in retrieved_event.get("categories", "")
+ assert "important" in retrieved_event.get("categories", "")
+ assert retrieved_event.get("url") == "https://zoom.us/j/123456789"
+ assert retrieved_event.get("privacy") == "PRIVATE"
+ assert retrieved_event.get("priority") == 2
+
+ # Cleanup
+ await nc_client.calendar.delete_event(calendar_name, event_uid)
+
+ except Exception as e:
+ logger.error(f"Event with metadata test failed: {e}")
+ raise
+
+
+async def test_calendar_operations_error_handling(nc_client: NextcloudClient):
+ """Test error handling for calendar operations."""
+
+ # Test with non-existent calendar
+ fake_calendar = f"nonexistent_calendar_{uuid.uuid4().hex}"
+
+ with pytest.raises(HTTPStatusError):
+ await nc_client.calendar.get_calendar_events(fake_calendar)
+
+ logger.info("Error handling tests completed successfully")
diff --git a/uv.lock b/uv.lock
index b88423d..f72f869 100644
--- a/uv.lock
+++ b/uv.lock
@@ -279,6 +279,19 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/e1/9b/a181f281f65d776426002f330c31849b86b31fc9d848db62e16f03ff739f/httpx_sse-0.4.0-py3-none-any.whl", hash = "sha256:f329af6eae57eaa2bdfd962b42524764af68075ea87370a2de920af5341e318f", size = 7819, upload-time = "2023-12-22T08:01:19.89Z" },
]
+[[package]]
+name = "icalendar"
+version = "6.3.1"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "python-dateutil" },
+ { name = "tzdata" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/08/13/e5899c916dcf1343ea65823eb7278d3e1a1d679f383f6409380594b5f322/icalendar-6.3.1.tar.gz", hash = "sha256:a697ce7b678072941e519f2745704fc29d78ef92a2dc53d9108ba6a04aeba466", size = 177169, upload-time = "2025-05-20T07:42:50.683Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/6c/25/b5fc00e85d2dfaf5c806ac8b5f1de072fa11630c5b15b4ae5bbc228abd51/icalendar-6.3.1-py3-none-any.whl", hash = "sha256:7ea1d1b212df685353f74cdc6ec9646bf42fa557d1746ea645ce8779fdfbecdd", size = 242349, upload-time = "2025-05-20T07:42:48.589Z" },
+]
+
[[package]]
name = "idna"
version = "3.10"
@@ -492,10 +505,11 @@ wheels = [
[[package]]
name = "nextcloud-mcp-server"
-version = "0.5.0"
+version = "0.6.0"
source = { editable = "." }
dependencies = [
{ name = "httpx" },
+ { name = "icalendar" },
{ name = "mcp", extra = ["cli"] },
{ name = "pillow" },
]
@@ -513,6 +527,7 @@ dev = [
[package.metadata]
requires-dist = [
{ name = "httpx", specifier = ">=0.28.1,<0.29.0" },
+ { name = "icalendar", specifier = ">=6.0.0,<7.0.0" },
{ name = "mcp", extras = ["cli"], specifier = ">=1.10,<1.11" },
{ name = "pillow", specifier = ">=11.2.1,<12.0.0" },
]
@@ -798,6 +813,18 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/28/d0/def53b4a790cfb21483016430ed828f64830dd981ebe1089971cd10cab25/pytest_cov-6.1.1-py3-none-any.whl", hash = "sha256:bddf29ed2d0ab6f4df17b4c55b0a657287db8684af9c42ea546b21b1041b3dde", size = 23841, upload-time = "2025-04-05T14:07:49.641Z" },
]
+[[package]]
+name = "python-dateutil"
+version = "2.9.0.post0"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "six" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/66/c0/0c8b6ad9f17a802ee498c46e004a0eb49bc148f2fd230864601a86dcf6db/python-dateutil-2.9.0.post0.tar.gz", hash = "sha256:37dd54208da7e1cd875388217d5e00ebd4179249f90fb72437e91a35459a0ad3", size = 342432, upload-time = "2024-03-01T18:36:20.211Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/ec/57/56b9bcc3c9c6a792fcbaf139543cee77261f3651ca9da0c93f5c1221264b/python_dateutil-2.9.0.post0-py2.py3-none-any.whl", hash = "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427", size = 229892, upload-time = "2024-03-01T18:36:18.57Z" },
+]
+
[[package]]
name = "python-dotenv"
version = "1.1.0"
@@ -998,6 +1025,15 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/e0/f9/0595336914c5619e5f28a1fb793285925a8cd4b432c9da0a987836c7f822/shellingham-1.5.4-py2.py3-none-any.whl", hash = "sha256:7ecfff8f2fd72616f7481040475a65b2bf8af90a56c89140852d1120324e8686", size = 9755, upload-time = "2023-10-24T04:13:38.866Z" },
]
+[[package]]
+name = "six"
+version = "1.17.0"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/94/e7/b2c673351809dca68a0e064b6af791aa332cf192da575fd474ed7d6f16a2/six-1.17.0.tar.gz", hash = "sha256:ff70335d468e7eb6ec65b95b99d3a2836546063f63acc5171de367e834932a81", size = 34031, upload-time = "2024-12-04T17:35:28.174Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/b7/ce/149a00dd41f10bc29e5921b496af8b574d8413afcd5e30dfa0ed46c2cc5e/six-1.17.0-py2.py3-none-any.whl", hash = "sha256:4721f391ed90541fddacab5acf947aa0d3dc7d27b2e1e8eda2be8970586c3274", size = 11050, upload-time = "2024-12-04T17:35:26.475Z" },
+]
+
[[package]]
name = "sniffio"
version = "1.3.1"
@@ -1148,6 +1184,15 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/31/08/aa4fdfb71f7de5176385bd9e90852eaf6b5d622735020ad600f2bab54385/typing_inspection-0.4.0-py3-none-any.whl", hash = "sha256:50e72559fcd2a6367a19f7a7e610e6afcb9fac940c650290eed893d61386832f", size = 14125, upload-time = "2025-02-25T17:27:57.754Z" },
]
+[[package]]
+name = "tzdata"
+version = "2025.2"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/95/32/1a225d6164441be760d75c2c42e2780dc0873fe382da3e98a2e1e48361e5/tzdata-2025.2.tar.gz", hash = "sha256:b60a638fcc0daffadf82fe0f57e53d06bdec2f36c4df66280ae79bce6bd6f2b9", size = 196380, upload-time = "2025-03-23T13:54:43.652Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/5c/23/c7abc0ca0a1526a0774eca151daeb8de62ec457e77262b66b359c3c7679e/tzdata-2025.2-py2.py3-none-any.whl", hash = "sha256:1a403fada01ff9221ca8044d701868fa132215d84beb92242d9acd2147f667a8", size = 347839, upload-time = "2025-03-23T13:54:41.845Z" },
+]
+
[[package]]
name = "uvicorn"
version = "0.34.2"