diff --git a/CHANGELOG.md b/CHANGELOG.md
index 1e20d4d..f395bda 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,32 @@
+## [Unreleased]
+
+### Feat
+
+- **calendar**: Add comprehensive Calendar app support via CalDAV protocol (Issue #74)
+- **calendar**: Add `nc_calendar_list_calendars` tool for listing available calendars
+- **calendar**: Add `nc_calendar_create_event` tool with full feature support (recurrence, reminders, attendees, categories)
+- **calendar**: Add `nc_calendar_list_events` tool with advanced filtering (date range, attendees, categories, status)
+- **calendar**: Add `nc_calendar_get_event` tool for retrieving detailed event information
+- **calendar**: Add `nc_calendar_update_event` tool for modifying existing events
+- **calendar**: Add `nc_calendar_delete_event` tool for removing events
+- **calendar**: Add `nc_calendar_create_meeting` tool for quick meeting creation with smart defaults
+- **calendar**: Add `nc_calendar_get_upcoming_events` tool for viewing upcoming events
+- **calendar**: Add `nc_calendar_find_availability` tool for intelligent scheduling assistance
+- **calendar**: Add `nc_calendar_bulk_operations` tool for efficient batch event management
+- **calendar**: Add `nc_calendar_manage_calendar` tool for calendar creation and management
+
+### Fix
+
+- **calendar**: Fix type annotations in calendar client for better Pylance compatibility
+- **calendar**: Fix alarm trigger formatting using proper timedelta objects
+- **calendar**: Fix event update handling to merge existing data with new changes
+- **calendar**: Fix categories extraction from icalendar objects
+
+### Refactor
+
+- **calendar**: Implement CalDAV client following existing NextCloud client patterns
+- **calendar**: Add comprehensive calendar integration tests covering all scenarios
+
## 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/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..eefe2cc
--- /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
+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.info(f"Found {len(calendars)} calendars for user {self.username}")
+ return calendars
+
+ except HTTPStatusError as e:
+ 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.info(f"Found {len(events)} events in calendar {calendar_name}")
+ 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.info(
+ f"Successfully created event {event_uid} in calendar {calendar_name}"
+ )
+ 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.info(
+ f"Successfully updated event {event_uid} in calendar {calendar_name}"
+ )
+ 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.info(
+ f"Successfully deleted event {event_uid} from calendar {calendar_name}"
+ )
+ return {"status_code": response.status_code}
+
+ except HTTPStatusError as e:
+ if e.response.status_code == 404:
+ logger.info(f"Event {event_uid} not found in calendar {calendar_name}")
+ 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.info(
+ f"Successfully retrieved event {event_uid} from calendar {calendar_name}"
+ )
+ 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:
+ from icalendar import Alarm
+
+ 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.info(f"Successfully 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.info(f"Successfully 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.py b/nextcloud_mcp_server/server.py
index eef5f12..bba66c1 100644
--- a/nextcloud_mcp_server/server.py
+++ b/nextcloud_mcp_server/server.py
@@ -1,5 +1,6 @@
# server.py
import logging
+from typing import Optional
from nextcloud_mcp_server.config import setup_logging
from contextlib import asynccontextmanager
from dataclasses import dataclass
@@ -338,6 +339,677 @@ async def nc_webdav_delete_resource(path: str, ctx: Context):
return await client.webdav.delete_resource(path)
+# 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, # ISO format: "2025-01-15T14:00:00" or "2025-01-15" for all-day
+ ctx: Context,
+ end_datetime: str = "", # Empty for all-day events
+ all_day: bool = False,
+ description: str = "",
+ location: str = "",
+ categories: str = "", # "work,meeting" - comma separated
+ # Recurrence
+ recurring: bool = False,
+ recurrence_rule: str = "", # "FREQ=WEEKLY;BYDAY=MO,WE,FR" (RFC5545 RRULE)
+ recurrence_end_date: str = "", # When to stop recurring
+ # Notifications/Alarms
+ reminder_minutes: int = 15, # Minutes before event to remind
+ reminder_email: bool = False, # Email notification
+ # Event properties
+ status: str = "CONFIRMED", # CONFIRMED, TENTATIVE, CANCELLED
+ priority: int = 5, # 1-9 (1=highest, 9=lowest, 5=normal)
+ privacy: str = "PUBLIC", # PUBLIC, PRIVATE, CONFIDENTIAL
+ # Attendees
+ attendees: str = "", # "email1@domain.com,email2@domain.com"
+ # Additional
+ url: str = "", # Related URL
+ color: str = "", # Event color (hex or name)
+):
+ """Create a comprehensive calendar event with full feature support"""
+ 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 = "", # "2025-01-01"
+ end_date: str = "", # "2025-01-31"
+ limit: int = 50,
+ min_attendees: Optional[int] = None,
+ min_duration_minutes: Optional[int] = None,
+ categories: Optional[str] = None, # Comma-separated: "work,meeting"
+ status: Optional[str] = None, # "CONFIRMED", "TENTATIVE", "CANCELLED"
+ 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.
+ start_date: Start date for search (YYYY-MM-DD format)
+ end_date: End date for search (YYYY-MM-DD format)
+ 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)
+ status: Filter events by status (CONFIRMED, TENTATIVE, 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
+ """
+ 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, # "2025-01-15"
+ time: str, # "14:00"
+ 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"""
+ 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
+ from datetime import datetime, timedelta
+
+ 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
+
+ from datetime import datetime, timedelta
+
+ 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'")
+
+
def run():
mcp.run()
diff --git a/pyproject.toml b/pyproject.toml
index 78b549d..6ae0c13 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -10,7 +10,9 @@ 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)",
+ "caldav (>=1.3.6,<2.0.0)",
+ "icalendar (>=6.0.0,<7.0.0)"
]
[project.scripts]
diff --git a/tests/conftest.py b/tests/conftest.py
index 3664720..13389fc 100644
--- a/tests/conftest.py
+++ b/tests/conftest.py
@@ -5,6 +5,10 @@ import uuid
from nextcloud_mcp_server.client import NextcloudClient
from httpx import HTTPStatusError
import asyncio
+from dotenv import load_dotenv
+
+# Load environment variables from .env file
+load_dotenv()
logger = logging.getLogger(__name__)
diff --git a/tests/integration/test_calendar_operations.py b/tests/integration/test_calendar_operations.py
new file mode 100644
index 0000000..7657bc7
--- /dev/null
+++ b/tests/integration/test_calendar_operations.py
@@ -0,0 +1,405 @@
+"""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)
+ 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 86ebd24..5857730 100644
--- a/uv.lock
+++ b/uv.lock
@@ -52,6 +52,22 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/77/06/bb80f5f86020c4551da315d78b3ab75e8228f89f0162f2c3a819e407941a/attrs-25.3.0-py3-none-any.whl", hash = "sha256:427318ce031701fea540783410126f03899a97ffc6f61596ad581ac2e40e3bc3", size = 63815, upload-time = "2025-03-13T11:10:21.14Z" },
]
+[[package]]
+name = "caldav"
+version = "1.6.0"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "icalendar" },
+ { name = "lxml" },
+ { name = "recurring-ical-events" },
+ { name = "requests" },
+ { name = "vobject" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/9c/97/aeb04abddd3009146c3cfb092343cb4e05090d08e4cfcb859db888ddf9f5/caldav-1.6.0.tar.gz", hash = "sha256:6e742601ec9ca1a0bc6e871fffe0392145bcc67de730f398ba5cefa5c49773f8", size = 156248, upload-time = "2025-05-30T14:55:36.371Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/8d/8a/b30d53c02ab508287a46cc7c3c1ce88624db003376a435d30806439f8f5d/caldav-1.6.0-py3-none-any.whl", hash = "sha256:077ab30726036e80d75ba6da4bcd0134f475189ee0e161aab08062adbf59f099", size = 87737, upload-time = "2025-05-30T14:55:34.423Z" },
+]
+
[[package]]
name = "certifi"
version = "2025.4.26"
@@ -279,6 +295,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"
@@ -382,6 +411,60 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/01/0e/b27cdbaccf30b890c40ed1da9fd4a3593a5cf94dae54fb34f8a4b74fcd3f/jsonschema_specifications-2025.4.1-py3-none-any.whl", hash = "sha256:4653bffbd6584f7de83a67e0d620ef16900b390ddc7939d56684d6c81e33f1af", size = 18437, upload-time = "2025-04-23T12:34:05.422Z" },
]
+[[package]]
+name = "lxml"
+version = "6.0.0"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/c5/ed/60eb6fa2923602fba988d9ca7c5cdbd7cf25faa795162ed538b527a35411/lxml-6.0.0.tar.gz", hash = "sha256:032e65120339d44cdc3efc326c9f660f5f7205f3a535c1fdbf898b29ea01fb72", size = 4096938, upload-time = "2025-06-26T16:28:19.373Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/7c/23/828d4cc7da96c611ec0ce6147bbcea2fdbde023dc995a165afa512399bbf/lxml-6.0.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:4ee56288d0df919e4aac43b539dd0e34bb55d6a12a6562038e8d6f3ed07f9e36", size = 8438217, upload-time = "2025-06-26T16:25:34.349Z" },
+ { url = "https://files.pythonhosted.org/packages/f1/33/5ac521212c5bcb097d573145d54b2b4a3c9766cda88af5a0e91f66037c6e/lxml-6.0.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b8dd6dd0e9c1992613ccda2bcb74fc9d49159dbe0f0ca4753f37527749885c25", size = 4590317, upload-time = "2025-06-26T16:25:38.103Z" },
+ { url = "https://files.pythonhosted.org/packages/2b/2e/45b7ca8bee304c07f54933c37afe7dd4d39ff61ba2757f519dcc71bc5d44/lxml-6.0.0-cp311-cp311-manylinux2010_i686.manylinux2014_i686.manylinux_2_12_i686.manylinux_2_17_i686.whl", hash = "sha256:d7ae472f74afcc47320238b5dbfd363aba111a525943c8a34a1b657c6be934c3", size = 5221628, upload-time = "2025-06-26T16:25:40.878Z" },
+ { url = "https://files.pythonhosted.org/packages/32/23/526d19f7eb2b85da1f62cffb2556f647b049ebe2a5aa8d4d41b1fb2c7d36/lxml-6.0.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:5592401cdf3dc682194727c1ddaa8aa0f3ddc57ca64fd03226a430b955eab6f6", size = 4949429, upload-time = "2025-06-28T18:47:20.046Z" },
+ { url = "https://files.pythonhosted.org/packages/ac/cc/f6be27a5c656a43a5344e064d9ae004d4dcb1d3c9d4f323c8189ddfe4d13/lxml-6.0.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:58ffd35bd5425c3c3b9692d078bf7ab851441434531a7e517c4984d5634cd65b", size = 5087909, upload-time = "2025-06-28T18:47:22.834Z" },
+ { url = "https://files.pythonhosted.org/packages/3b/e6/8ec91b5bfbe6972458bc105aeb42088e50e4b23777170404aab5dfb0c62d/lxml-6.0.0-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f720a14aa102a38907c6d5030e3d66b3b680c3e6f6bc95473931ea3c00c59967", size = 5031713, upload-time = "2025-06-26T16:25:43.226Z" },
+ { url = "https://files.pythonhosted.org/packages/33/cf/05e78e613840a40e5be3e40d892c48ad3e475804db23d4bad751b8cadb9b/lxml-6.0.0-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c2a5e8d207311a0170aca0eb6b160af91adc29ec121832e4ac151a57743a1e1e", size = 5232417, upload-time = "2025-06-26T16:25:46.111Z" },
+ { url = "https://files.pythonhosted.org/packages/ac/8c/6b306b3e35c59d5f0b32e3b9b6b3b0739b32c0dc42a295415ba111e76495/lxml-6.0.0-cp311-cp311-manylinux_2_31_armv7l.whl", hash = "sha256:2dd1cc3ea7e60bfb31ff32cafe07e24839df573a5e7c2d33304082a5019bcd58", size = 4681443, upload-time = "2025-06-26T16:25:48.837Z" },
+ { url = "https://files.pythonhosted.org/packages/59/43/0bd96bece5f7eea14b7220476835a60d2b27f8e9ca99c175f37c085cb154/lxml-6.0.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:2cfcf84f1defed7e5798ef4f88aa25fcc52d279be731ce904789aa7ccfb7e8d2", size = 5074542, upload-time = "2025-06-26T16:25:51.65Z" },
+ { url = "https://files.pythonhosted.org/packages/e2/3d/32103036287a8ca012d8518071f8852c68f2b3bfe048cef2a0202eb05910/lxml-6.0.0-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:a52a4704811e2623b0324a18d41ad4b9fabf43ce5ff99b14e40a520e2190c851", size = 4729471, upload-time = "2025-06-26T16:25:54.571Z" },
+ { url = "https://files.pythonhosted.org/packages/ca/a8/7be5d17df12d637d81854bd8648cd329f29640a61e9a72a3f77add4a311b/lxml-6.0.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:c16304bba98f48a28ae10e32a8e75c349dd742c45156f297e16eeb1ba9287a1f", size = 5256285, upload-time = "2025-06-26T16:25:56.997Z" },
+ { url = "https://files.pythonhosted.org/packages/cd/d0/6cb96174c25e0d749932557c8d51d60c6e292c877b46fae616afa23ed31a/lxml-6.0.0-cp311-cp311-win32.whl", hash = "sha256:f8d19565ae3eb956d84da3ef367aa7def14a2735d05bd275cd54c0301f0d0d6c", size = 3612004, upload-time = "2025-06-26T16:25:59.11Z" },
+ { url = "https://files.pythonhosted.org/packages/ca/77/6ad43b165dfc6dead001410adeb45e88597b25185f4479b7ca3b16a5808f/lxml-6.0.0-cp311-cp311-win_amd64.whl", hash = "sha256:b2d71cdefda9424adff9a3607ba5bbfc60ee972d73c21c7e3c19e71037574816", size = 4003470, upload-time = "2025-06-26T16:26:01.655Z" },
+ { url = "https://files.pythonhosted.org/packages/a0/bc/4c50ec0eb14f932a18efc34fc86ee936a66c0eb5f2fe065744a2da8a68b2/lxml-6.0.0-cp311-cp311-win_arm64.whl", hash = "sha256:8a2e76efbf8772add72d002d67a4c3d0958638696f541734304c7f28217a9cab", size = 3682477, upload-time = "2025-06-26T16:26:03.808Z" },
+ { url = "https://files.pythonhosted.org/packages/89/c3/d01d735c298d7e0ddcedf6f028bf556577e5ab4f4da45175ecd909c79378/lxml-6.0.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:78718d8454a6e928470d511bf8ac93f469283a45c354995f7d19e77292f26108", size = 8429515, upload-time = "2025-06-26T16:26:06.776Z" },
+ { url = "https://files.pythonhosted.org/packages/06/37/0e3eae3043d366b73da55a86274a590bae76dc45aa004b7042e6f97803b1/lxml-6.0.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:84ef591495ffd3f9dcabffd6391db7bb70d7230b5c35ef5148354a134f56f2be", size = 4601387, upload-time = "2025-06-26T16:26:09.511Z" },
+ { url = "https://files.pythonhosted.org/packages/a3/28/e1a9a881e6d6e29dda13d633885d13acb0058f65e95da67841c8dd02b4a8/lxml-6.0.0-cp312-cp312-manylinux2010_i686.manylinux2014_i686.manylinux_2_12_i686.manylinux_2_17_i686.whl", hash = "sha256:2930aa001a3776c3e2601cb8e0a15d21b8270528d89cc308be4843ade546b9ab", size = 5228928, upload-time = "2025-06-26T16:26:12.337Z" },
+ { url = "https://files.pythonhosted.org/packages/9a/55/2cb24ea48aa30c99f805921c1c7860c1f45c0e811e44ee4e6a155668de06/lxml-6.0.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:219e0431ea8006e15005767f0351e3f7f9143e793e58519dc97fe9e07fae5563", size = 4952289, upload-time = "2025-06-28T18:47:25.602Z" },
+ { url = "https://files.pythonhosted.org/packages/31/c0/b25d9528df296b9a3306ba21ff982fc5b698c45ab78b94d18c2d6ae71fd9/lxml-6.0.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:bd5913b4972681ffc9718bc2d4c53cde39ef81415e1671ff93e9aa30b46595e7", size = 5111310, upload-time = "2025-06-28T18:47:28.136Z" },
+ { url = "https://files.pythonhosted.org/packages/e9/af/681a8b3e4f668bea6e6514cbcb297beb6de2b641e70f09d3d78655f4f44c/lxml-6.0.0-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:390240baeb9f415a82eefc2e13285016f9c8b5ad71ec80574ae8fa9605093cd7", size = 5025457, upload-time = "2025-06-26T16:26:15.068Z" },
+ { url = "https://files.pythonhosted.org/packages/99/b6/3a7971aa05b7be7dfebc7ab57262ec527775c2c3c5b2f43675cac0458cad/lxml-6.0.0-cp312-cp312-manylinux_2_27_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:d6e200909a119626744dd81bae409fc44134389e03fbf1d68ed2a55a2fb10991", size = 5657016, upload-time = "2025-07-03T19:19:06.008Z" },
+ { url = "https://files.pythonhosted.org/packages/69/f8/693b1a10a891197143c0673fcce5b75fc69132afa81a36e4568c12c8faba/lxml-6.0.0-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ca50bd612438258a91b5b3788c6621c1f05c8c478e7951899f492be42defc0da", size = 5257565, upload-time = "2025-06-26T16:26:17.906Z" },
+ { url = "https://files.pythonhosted.org/packages/a8/96/e08ff98f2c6426c98c8964513c5dab8d6eb81dadcd0af6f0c538ada78d33/lxml-6.0.0-cp312-cp312-manylinux_2_31_armv7l.whl", hash = "sha256:c24b8efd9c0f62bad0439283c2c795ef916c5a6b75f03c17799775c7ae3c0c9e", size = 4713390, upload-time = "2025-06-26T16:26:20.292Z" },
+ { url = "https://files.pythonhosted.org/packages/a8/83/6184aba6cc94d7413959f6f8f54807dc318fdcd4985c347fe3ea6937f772/lxml-6.0.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:afd27d8629ae94c5d863e32ab0e1d5590371d296b87dae0a751fb22bf3685741", size = 5066103, upload-time = "2025-06-26T16:26:22.765Z" },
+ { url = "https://files.pythonhosted.org/packages/ee/01/8bf1f4035852d0ff2e36a4d9aacdbcc57e93a6cd35a54e05fa984cdf73ab/lxml-6.0.0-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:54c4855eabd9fc29707d30141be99e5cd1102e7d2258d2892314cf4c110726c3", size = 4791428, upload-time = "2025-06-26T16:26:26.461Z" },
+ { url = "https://files.pythonhosted.org/packages/29/31/c0267d03b16954a85ed6b065116b621d37f559553d9339c7dcc4943a76f1/lxml-6.0.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:c907516d49f77f6cd8ead1322198bdfd902003c3c330c77a1c5f3cc32a0e4d16", size = 5678523, upload-time = "2025-07-03T19:19:09.837Z" },
+ { url = "https://files.pythonhosted.org/packages/5c/f7/5495829a864bc5f8b0798d2b52a807c89966523140f3d6fa3a58ab6720ea/lxml-6.0.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:36531f81c8214e293097cd2b7873f178997dae33d3667caaae8bdfb9666b76c0", size = 5281290, upload-time = "2025-06-26T16:26:29.406Z" },
+ { url = "https://files.pythonhosted.org/packages/79/56/6b8edb79d9ed294ccc4e881f4db1023af56ba451909b9ce79f2a2cd7c532/lxml-6.0.0-cp312-cp312-win32.whl", hash = "sha256:690b20e3388a7ec98e899fd54c924e50ba6693874aa65ef9cb53de7f7de9d64a", size = 3613495, upload-time = "2025-06-26T16:26:31.588Z" },
+ { url = "https://files.pythonhosted.org/packages/0b/1e/cc32034b40ad6af80b6fd9b66301fc0f180f300002e5c3eb5a6110a93317/lxml-6.0.0-cp312-cp312-win_amd64.whl", hash = "sha256:310b719b695b3dd442cdfbbe64936b2f2e231bb91d998e99e6f0daf991a3eba3", size = 4014711, upload-time = "2025-06-26T16:26:33.723Z" },
+ { url = "https://files.pythonhosted.org/packages/55/10/dc8e5290ae4c94bdc1a4c55865be7e1f31dfd857a88b21cbba68b5fea61b/lxml-6.0.0-cp312-cp312-win_arm64.whl", hash = "sha256:8cb26f51c82d77483cdcd2b4a53cda55bbee29b3c2f3ddeb47182a2a9064e4eb", size = 3674431, upload-time = "2025-06-26T16:26:35.959Z" },
+ { url = "https://files.pythonhosted.org/packages/79/21/6e7c060822a3c954ff085e5e1b94b4a25757c06529eac91e550f3f5cd8b8/lxml-6.0.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:6da7cd4f405fd7db56e51e96bff0865b9853ae70df0e6720624049da76bde2da", size = 8414372, upload-time = "2025-06-26T16:26:39.079Z" },
+ { url = "https://files.pythonhosted.org/packages/a4/f6/051b1607a459db670fc3a244fa4f06f101a8adf86cda263d1a56b3a4f9d5/lxml-6.0.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:b34339898bb556a2351a1830f88f751679f343eabf9cf05841c95b165152c9e7", size = 4593940, upload-time = "2025-06-26T16:26:41.891Z" },
+ { url = "https://files.pythonhosted.org/packages/8e/74/dd595d92a40bda3c687d70d4487b2c7eff93fd63b568acd64fedd2ba00fe/lxml-6.0.0-cp313-cp313-manylinux2010_i686.manylinux2014_i686.manylinux_2_12_i686.manylinux_2_17_i686.whl", hash = "sha256:51a5e4c61a4541bd1cd3ba74766d0c9b6c12d6a1a4964ef60026832aac8e79b3", size = 5214329, upload-time = "2025-06-26T16:26:44.669Z" },
+ { url = "https://files.pythonhosted.org/packages/52/46/3572761efc1bd45fcafb44a63b3b0feeb5b3f0066886821e94b0254f9253/lxml-6.0.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:d18a25b19ca7307045581b18b3ec9ead2b1db5ccd8719c291f0cd0a5cec6cb81", size = 4947559, upload-time = "2025-06-28T18:47:31.091Z" },
+ { url = "https://files.pythonhosted.org/packages/94/8a/5e40de920e67c4f2eef9151097deb9b52d86c95762d8ee238134aff2125d/lxml-6.0.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:d4f0c66df4386b75d2ab1e20a489f30dc7fd9a06a896d64980541506086be1f1", size = 5102143, upload-time = "2025-06-28T18:47:33.612Z" },
+ { url = "https://files.pythonhosted.org/packages/7c/4b/20555bdd75d57945bdabfbc45fdb1a36a1a0ff9eae4653e951b2b79c9209/lxml-6.0.0-cp313-cp313-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9f4b481b6cc3a897adb4279216695150bbe7a44c03daba3c894f49d2037e0a24", size = 5021931, upload-time = "2025-06-26T16:26:47.503Z" },
+ { url = "https://files.pythonhosted.org/packages/b6/6e/cf03b412f3763d4ca23b25e70c96a74cfece64cec3addf1c4ec639586b13/lxml-6.0.0-cp313-cp313-manylinux_2_27_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:8a78d6c9168f5bcb20971bf3329c2b83078611fbe1f807baadc64afc70523b3a", size = 5645469, upload-time = "2025-07-03T19:19:13.32Z" },
+ { url = "https://files.pythonhosted.org/packages/d4/dd/39c8507c16db6031f8c1ddf70ed95dbb0a6d466a40002a3522c128aba472/lxml-6.0.0-cp313-cp313-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2ae06fbab4f1bb7db4f7c8ca9897dc8db4447d1a2b9bee78474ad403437bcc29", size = 5247467, upload-time = "2025-06-26T16:26:49.998Z" },
+ { url = "https://files.pythonhosted.org/packages/4d/56/732d49def0631ad633844cfb2664563c830173a98d5efd9b172e89a4800d/lxml-6.0.0-cp313-cp313-manylinux_2_31_armv7l.whl", hash = "sha256:1fa377b827ca2023244a06554c6e7dc6828a10aaf74ca41965c5d8a4925aebb4", size = 4720601, upload-time = "2025-06-26T16:26:52.564Z" },
+ { url = "https://files.pythonhosted.org/packages/8f/7f/6b956fab95fa73462bca25d1ea7fc8274ddf68fb8e60b78d56c03b65278e/lxml-6.0.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:1676b56d48048a62ef77a250428d1f31f610763636e0784ba67a9740823988ca", size = 5060227, upload-time = "2025-06-26T16:26:55.054Z" },
+ { url = "https://files.pythonhosted.org/packages/97/06/e851ac2924447e8b15a294855caf3d543424364a143c001014d22c8ca94c/lxml-6.0.0-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:0e32698462aacc5c1cf6bdfebc9c781821b7e74c79f13e5ffc8bfe27c42b1abf", size = 4790637, upload-time = "2025-06-26T16:26:57.384Z" },
+ { url = "https://files.pythonhosted.org/packages/06/d4/fd216f3cd6625022c25b336c7570d11f4a43adbaf0a56106d3d496f727a7/lxml-6.0.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:4d6036c3a296707357efb375cfc24bb64cd955b9ec731abf11ebb1e40063949f", size = 5662049, upload-time = "2025-07-03T19:19:16.409Z" },
+ { url = "https://files.pythonhosted.org/packages/52/03/0e764ce00b95e008d76b99d432f1807f3574fb2945b496a17807a1645dbd/lxml-6.0.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:7488a43033c958637b1a08cddc9188eb06d3ad36582cebc7d4815980b47e27ef", size = 5272430, upload-time = "2025-06-26T16:27:00.031Z" },
+ { url = "https://files.pythonhosted.org/packages/5f/01/d48cc141bc47bc1644d20fe97bbd5e8afb30415ec94f146f2f76d0d9d098/lxml-6.0.0-cp313-cp313-win32.whl", hash = "sha256:5fcd7d3b1d8ecb91445bd71b9c88bdbeae528fefee4f379895becfc72298d181", size = 3612896, upload-time = "2025-06-26T16:27:04.251Z" },
+ { url = "https://files.pythonhosted.org/packages/f4/87/6456b9541d186ee7d4cb53bf1b9a0d7f3b1068532676940fdd594ac90865/lxml-6.0.0-cp313-cp313-win_amd64.whl", hash = "sha256:2f34687222b78fff795feeb799a7d44eca2477c3d9d3a46ce17d51a4f383e32e", size = 4013132, upload-time = "2025-06-26T16:27:06.415Z" },
+ { url = "https://files.pythonhosted.org/packages/b7/42/85b3aa8f06ca0d24962f8100f001828e1f1f1a38c954c16e71154ed7d53a/lxml-6.0.0-cp313-cp313-win_arm64.whl", hash = "sha256:21db1ec5525780fd07251636eb5f7acb84003e9382c72c18c542a87c416ade03", size = 3672642, upload-time = "2025-06-26T16:27:09.888Z" },
+]
+
[[package]]
name = "markdown-it-py"
version = "3.0.0"
@@ -495,7 +578,9 @@ name = "nextcloud-mcp-server"
version = "0.5.0"
source = { editable = "." }
dependencies = [
+ { name = "caldav" },
{ name = "httpx" },
+ { name = "icalendar" },
{ name = "mcp", extra = ["cli"] },
{ name = "pillow" },
]
@@ -512,7 +597,9 @@ dev = [
[package.metadata]
requires-dist = [
+ { name = "caldav", specifier = ">=1.3.6,<2.0.0" },
{ 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 +885,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"
@@ -816,6 +915,15 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/45/58/38b5afbc1a800eeea951b9285d3912613f2603bdf897a4ab0f4bd7f405fc/python_multipart-0.0.20-py3-none-any.whl", hash = "sha256:8a62d3a8335e06589fe01f2a3e178cdcc632f3fbe0d492ad9ee0ec35aab1f104", size = 24546, upload-time = "2024-12-16T19:45:44.423Z" },
]
+[[package]]
+name = "pytz"
+version = "2025.2"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/f8/bf/abbd3cdfb8fbc7fb3d4d38d320f2441b1e7cbe29be4f23797b4a2b5d8aac/pytz-2025.2.tar.gz", hash = "sha256:360b9e3dbb49a209c21ad61809c7fb453643e048b38924c765813546746e81c3", size = 320884, upload-time = "2025-03-25T02:25:00.538Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/81/c4/34e93fe5f5429d7570ec1fa436f1986fb1f00c3e0f43a589fe2bbcd22c3f/pytz-2025.2-py2.py3-none-any.whl", hash = "sha256:5ddf76296dd8c44c26eb8f4b6f35488f3ccbf6fbbd7adee0b7262d43f0ec2f00", size = 509225, upload-time = "2025-03-25T02:24:58.468Z" },
+]
+
[[package]]
name = "pyyaml"
version = "6.0.2"
@@ -863,6 +971,21 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/ad/3f/11dd4cd4f39e05128bfd20138faea57bec56f9ffba6185d276e3107ba5b2/questionary-2.1.0-py3-none-any.whl", hash = "sha256:44174d237b68bc828e4878c763a9ad6790ee61990e0ae72927694ead57bab8ec", size = 36747, upload-time = "2024-12-29T11:49:16.734Z" },
]
+[[package]]
+name = "recurring-ical-events"
+version = "3.8.0"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "icalendar" },
+ { name = "python-dateutil" },
+ { name = "tzdata" },
+ { name = "x-wr-timezone" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/83/90/05dfcc02ecf58bd170305c88db9e3e3933aa73a1f8abac2b326c7fdc1a98/recurring_ical_events-3.8.0.tar.gz", hash = "sha256:3e8c7c35d9bd8956a7ab91afad51477c60d972e1236d3fd1b55087a66bce7d04", size = 602665, upload-time = "2025-06-10T13:23:50.662Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/36/25/88a4218cccae06ce6b15e41d2f263dd4a73e8e8cbe41537cd7784a17479b/recurring_ical_events-3.8.0-py3-none-any.whl", hash = "sha256:cf958eb17c92d4dca5c621e44c2b3fffd4ba700dca0db66287c5dc11438f63ba", size = 238228, upload-time = "2025-06-10T13:23:49.048Z" },
+]
+
[[package]]
name = "referencing"
version = "0.36.2"
@@ -877,6 +1000,21 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/c1/b1/3baf80dc6d2b7bc27a95a67752d0208e410351e3feb4eb78de5f77454d8d/referencing-0.36.2-py3-none-any.whl", hash = "sha256:e8699adbbf8b5c7de96d8ffa0eb5c158b3beafce084968e2ea8bb08c6794dcd0", size = 26775, upload-time = "2025-01-25T08:48:14.241Z" },
]
+[[package]]
+name = "requests"
+version = "2.32.4"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "certifi" },
+ { name = "charset-normalizer" },
+ { name = "idna" },
+ { name = "urllib3" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/e1/0a/929373653770d8a0d7ea76c37de6e41f11eb07559b103b1c02cafb3f7cf8/requests-2.32.4.tar.gz", hash = "sha256:27d0316682c8a29834d3264820024b62a36942083d52caf2f14c0591336d3422", size = 135258, upload-time = "2025-06-09T16:43:07.34Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/7c/e4/56027c4a6b4ae70ca9de302488c5ca95ad4a39e190093d6c1a8ace08341b/requests-2.32.4-py3-none-any.whl", hash = "sha256:27babd3cda2a6d50b30443204ee89830707d396671944c998b5975b031ac2b2c", size = 64847, upload-time = "2025-06-09T16:43:05.728Z" },
+]
+
[[package]]
name = "rich"
version = "14.0.0"
@@ -998,6 +1136,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 +1295,24 @@ 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 = "urllib3"
+version = "2.5.0"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/15/22/9ee70a2574a4f4599c47dd506532914ce044817c7752a79b6a51286319bc/urllib3-2.5.0.tar.gz", hash = "sha256:3fc47733c7e419d4bc3f6b3dc2b4f890bb743906a30d56ba4a5bfa4bbff92760", size = 393185, upload-time = "2025-06-18T14:07:41.644Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/a7/c2/fe1e52489ae3122415c51f387e221dd0773709bad6c6cdaa599e8a2c5185/urllib3-2.5.0-py3-none-any.whl", hash = "sha256:e6b01673c0fa6a13e374b50871808eb3bf7046c4b125b216f6bf1cc604cff0dc", size = 129795, upload-time = "2025-06-18T14:07:40.39Z" },
+]
+
[[package]]
name = "uvicorn"
version = "0.34.2"
@@ -1161,6 +1326,20 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/b1/4b/4cef6ce21a2aaca9d852a6e84ef4f135d99fcd74fa75105e2fc0c8308acd/uvicorn-0.34.2-py3-none-any.whl", hash = "sha256:deb49af569084536d269fe0a6d67e3754f104cf03aba7c11c40f01aadf33c403", size = 62483, upload-time = "2025-04-19T06:02:48.42Z" },
]
+[[package]]
+name = "vobject"
+version = "0.9.9"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "python-dateutil" },
+ { name = "pytz" },
+ { name = "six" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/d0/8a/15c34b3d27c43fc81a467d0f66577afc5542326804c42f30557e31c3259e/vobject-0.9.9.tar.gz", hash = "sha256:ac44e5d7e2079d84c1d52c50a615b9bec4b1ba958608c4c7fe40cbf33247b38e", size = 1208905, upload-time = "2024-12-16T07:29:39.178Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/68/20/6bba813bbd498c28edbbcf8253a6398cf4266ecf7bfa6129835c0a2bfbb1/vobject-0.9.9-py2.py3-none-any.whl", hash = "sha256:0fbdb982065cf4d1843a5d5950c88510041c6de026bda49c3502721de1c6ac3d", size = 47526, upload-time = "2024-12-16T07:31:08.493Z" },
+]
+
[[package]]
name = "wcwidth"
version = "0.2.13"
@@ -1169,3 +1348,17 @@ sdist = { url = "https://files.pythonhosted.org/packages/6c/63/53559446a878410fc
wheels = [
{ url = "https://files.pythonhosted.org/packages/fd/84/fd2ba7aafacbad3c4201d395674fc6348826569da3c0937e75505ead3528/wcwidth-0.2.13-py2.py3-none-any.whl", hash = "sha256:3da69048e4540d84af32131829ff948f1e022c1c6bdb8d6102117aac784f6859", size = 34166, upload-time = "2024-01-06T02:10:55.763Z" },
]
+
+[[package]]
+name = "x-wr-timezone"
+version = "2.0.1"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "click" },
+ { name = "icalendar" },
+ { name = "tzdata" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/79/2b/8ae5f59ab852c8fe32dd37c1aa058eb98aca118fec2d3af5c3cd56fffb7b/x_wr_timezone-2.0.1.tar.gz", hash = "sha256:9166c40e6ffd4c0edebabc354e1a1e2cffc1bb473f88007694793757685cc8c3", size = 18212, upload-time = "2025-02-06T17:10:40.913Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/0f/b7/4bac35b4079b76c07d8faddf89467e9891b1610cfe8d03b0ebb5610e4423/x_wr_timezone-2.0.1-py3-none-any.whl", hash = "sha256:e74a53b9f4f7def8138455c240e65e47c224778bce3c024fcd6da2cbe91ca038", size = 11102, upload-time = "2025-02-06T17:10:39.192Z" },
+]