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.0b1.tar.gz (17.7 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.0b1-py3-none-any.whl (21.4 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: cal_scheduler_mcp-1.0.0b1.tar.gz
  • Upload date:
  • Size: 17.7 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.0b1.tar.gz
Algorithm Hash digest
SHA256 06f5660e4ed7379b63dd3d0797596bc4101e9a81bfabd557fa4453669f24ca5d
MD5 3541afae9bf11295c0a536c9898a08e1
BLAKE2b-256 cf9bb7500627aea4f2ffa21aba14ae44ca8d0d2b605799c5d00b7f5e9ef06acd

See more details on using hashes here.

Provenance

The following attestation bundles were made for cal_scheduler_mcp-1.0.0b1.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.0b1-py3-none-any.whl.

File metadata

File hashes

Hashes for cal_scheduler_mcp-1.0.0b1-py3-none-any.whl
Algorithm Hash digest
SHA256 f9ec6f0fed6a2a43f481f8c57c8ee1fd9bdc2285fd2938c07b16d50ac008ff1c
MD5 96c6660095a461adf3341743a561bcc8
BLAKE2b-256 5dbde33affac1784d82fc1deb2972b02d18a397351302353c82565ae6dcd2fe9

See more details on using hashes here.

Provenance

The following attestation bundles were made for cal_scheduler_mcp-1.0.0b1-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