Skip to main content

A timezone-correct CalDAV scheduling MCP server (zoned .ics, validated recurrence, single-occurrence edits).

Project description

If you are an agent installing this MCP, read AGENTS.md first.

cal-scheduler

A thin, timezone-correct MCP server for scheduling against any CalDAV calendar. It gives an LLM agent a small, strict set of calendar tools and does the deterministic, fiddly parts of iCalendar correctly so the model doesn't have to.

It is not an NLP layer: the agent phrases the request, the server validates it, rejects bad input loudly, and persists clean zoned .ics.

Why it exists

Most calendar tooling an LLM reaches for gets three things subtly wrong. cal-scheduler fixes them by construction:

  • Stores zoned, not bare UTC. Events are written with TZID/VTIMEZONE, so a weekly 9am stays 9am across a daylight-saving boundary instead of drifting an hour. Storing bare UTC is the classic cause of that drift.
  • Validates recurrence. It rejects an RRULE whose anchor contradicts it (e.g. a series starting on the 30th but set to repeat on the 1st), and normalises UNTIL to UTC under a zoned DTSTART as RFC 5545 requires.
  • Does real single-occurrence edits. Exclude one instance (EXDATE) or move one instance (RECURRENCE-ID) without disturbing the rest of the series — the operations naive wrappers tend to lack.

How it works

A small uv Python package that composes mature libraries rather than implementing a calendar engine:

Module Role
config.py environment config (CALDAV_*, CAL_DEFAULT_TZ)
timezones.py parse datetimes; naive → assume default-zone wall time, offset → normalise into the zone; report what was assumed
ical.py build/parse VEVENTs (icalendar), expand ranges (recurring-ical-events), recurrence validation, EXDATE/RECURRENCE-ID ops
store.py CalDAV transport (caldav) — list/create/delete calendars, get/put/delete events, read-modify-write with etag
server.py the FastMCP stdio server and the tool surface

It runs as a stdio MCP server that an MCP host (Claude, an agent harness, etc.) spawns as a subprocess.

Install

Requires Python ≥ 3.11. The package is on PyPI as cal-scheduler-mcp:

uv tool install cal-scheduler-mcp

(or pip install cal-scheduler-mcp, uvx --from cal-scheduler-mcp cal-scheduler, etc.)

For editable dev work, install from a clone:

# run straight from the repo with uv (no global install)
uv run --directory /path/to/cal-scheduler-mcp cal-scheduler

# or install the console script into a tool environment
gh repo clone limey/cal-scheduler-mcp
uv tool install --editable /path/to/cal-scheduler-mcp
# or, with SSH GitHub access, in one step:
uv tool install git+ssh://git@github.com/limey/cal-scheduler-mcp

Don't uv add cal-scheduler-mcp for the MCP — uv add writes into whatever project you're sitting in, not into the tool environment. For an MCP server (spawned as a subprocess), uv tool install is the right shape.

Configure

All configuration is via environment variables:

Variable Required Default Meaning
CALDAV_BASE_URL CalDAV server URL, e.g. http://127.0.0.1:5232
CALDAV_USERNAME CalDAV account user
CALDAV_PASSWORD CalDAV account password
CAL_DEFAULT_TZ Pacific/Auckland IANA zone every event is stored in

Many MCP hosts strip inherited environment from stdio servers, so set these in the host's per-server env block rather than relying on the ambient shell.

Example MCP host config

The MCP runs as a stdio subprocess that the host spawns. Many hosts strip inherited PATH from that subprocess, so wire the uv run --directory form rather than relying on the cal-scheduler shim being on the spawn host's PATH:

{
  "mcpServers": {
    "cal-scheduler": {
      "command": "uv",
      "args": ["run", "--directory", "/abs/path/to/cal-scheduler-mcp", "cal-scheduler"],
      "env": {
        "CALDAV_BASE_URL": "http://127.0.0.1:5232",
        "CALDAV_USERNAME": "me",
        "CALDAV_PASSWORD": "secret",
        "CAL_DEFAULT_TZ": "Pacific/Auckland"
      }
    }
  }
}

/abs/path/to/cal-scheduler-mcp is the absolute path to a local clone of this repo (see Install above).

Pair it with any CalDAV server. A simple self-hosted option is Radicale (plain http://, no TLS needed for local use).

Tool surface

Eventslist_events(start, end, [calendar]), create_event(summary, start, [end, calendar, description, location, rrule]), update_event(uid, …), delete_event(uid, [calendar]), exclude_occurrence(uid, occurrence, [calendar]), move_occurrence(uid, occurrence, new_start, [new_end, calendar]).

Calendarslist_calendars, create_calendar(name), delete_calendar(name).

Helperresolve_datetime(value) — preview how a datetime will be interpreted, without writing anything.

The timezone rule (the whole point)

Every event is stored zoned to CAL_DEFAULT_TZ.

  • A naive datetime (2026-06-30T21:00) is assumed to be wall time in that zone, and the tool response says so ("assumed Pacific/Auckland wall time").
  • An offset-qualified datetime (…+12:00) is honoured as an instant and re-expressed in the zone — same wall clock when the offset matches, a correct conversion otherwise.
  • A date-only value (2026-06-30) is an all-day event.

Develop

uv sync                 # install deps + dev tools
uv run ruff check       # lint
uv run pytest           # unit tests (no server required)

The unit tests cover the pure layers (timezone resolution, recurrence validation, EXDATE/RECURRENCE-ID construction) and need no running CalDAV server. To exercise the full stack end to end, point CALDAV_BASE_URL at a throwaway CalDAV account.

License

MIT © 2026 Robert Clark

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

cal_scheduler_mcp-1.0.0.tar.gz (17.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

cal_scheduler_mcp-1.0.0-py3-none-any.whl (21.5 kB view details)

Uploaded Python 3

File details

Details for the file cal_scheduler_mcp-1.0.0.tar.gz.

File metadata

  • Download URL: cal_scheduler_mcp-1.0.0.tar.gz
  • Upload date:
  • Size: 17.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for cal_scheduler_mcp-1.0.0.tar.gz
Algorithm Hash digest
SHA256 fb0d9846b3aa5373e555b6aabe7e360bb1a0640174f0de58215e8418348eb9d0
MD5 122c3ff862d7201d2213524a4a2d8446
BLAKE2b-256 1280d61fbf5bc7043e9e3998b696afad851731de88e54d4265e2620412dbaea6

See more details on using hashes here.

Provenance

The following attestation bundles were made for cal_scheduler_mcp-1.0.0.tar.gz:

Publisher: publish.yml on limey/cal-scheduler-mcp

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file cal_scheduler_mcp-1.0.0-py3-none-any.whl.

File metadata

File hashes

Hashes for cal_scheduler_mcp-1.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 fbadd917fce066bc427fc4eec35c2ec0378f5fd5e505208f877a97acb259345e
MD5 e759c77a2e2b8258b530f879d0a4d389
BLAKE2b-256 48236e8c444a417bdf365fe29546f501fafce7cea20aaf3935da8788bd924c92

See more details on using hashes here.

Provenance

The following attestation bundles were made for cal_scheduler_mcp-1.0.0-py3-none-any.whl:

Publisher: publish.yml on limey/cal-scheduler-mcp

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page