MCP server for Microsoft Outlook via Microsoft Graph API
Project description
outlook-mcp
MCP server for Microsoft Outlook personal accounts via Microsoft Graph API.
Personal Microsoft accounts only —
@outlook.com,@hotmail.com,@live.com. Work/school accounts (Entra ID) are not supported in v1.
Disclaimer: Independent open-source project. Not affiliated with, endorsed by, or supported by Microsoft Corporation. "Outlook" and "Microsoft Graph" are trademarks of Microsoft.
Who this is for
You'll like this if you're:
- An agent builder wiring Outlook into your own infra (OpenClaw, Claude Code, Cursor, custom MCP host) and want a typed tool surface — not stdout you have to parse
- Building on personal Microsoft accounts (Outlook.com / Hotmail / Live) and want full control: BYO Azure app, no enterprise consent flow, no shared client ID
- Looking for real coverage — mail, calendar, contacts, to-do, drafts, folders, batch ops, threading — instead of a mail-only or calendar-only wrapper
- Security-conscious: tokens in the OS keyring (Keychain on macOS), granular
allow_categories, optionalread_onlymode, zero telemetry
This isn't for you if you need work/school M365 accounts (use Microsoft's official tooling — Entra ID auth and admin-consent flows are out of scope here), or if a basic mail-only client would suffice (this has 61 tools — way more than you need for "read my inbox").
How it differs from other Outlook tools you'll find
This is the only first-class MCP server in the personal-Outlook space — most alternatives are bash scripts or skill-shaped CLI wrappers the agent shells out to. That distinction matters: the agent gets typed tool schemas with structured args/returns, not stdout it has to parse. Other things you won't find elsewhere: /$batch-optimized triage (10-20× faster on bulk ops), recursive folder ops with name resolution, granular per-category permissions, multi-account support, and full attachment write paths including >3MB upload sessions for drafts.
What This Enables
Give your AI agent full Outlook access. Example prompts that just work:
- "Summarize my unread email from the past 24 hours and flag anything time-sensitive."
- "What's in my Focused Inbox right now? Anything in Other that looks like it belongs up top?"
- "Any shipping updates in my inbox? Track what I'm waiting on and when it's supposed to arrive."
- "Scan my email for upcoming subscription renewals — what's about to auto-charge in the next two weeks?"
- "I've got a trip to Seattle next week — check my calendar for the itinerary and create a To Do task with a packing checklist."
- "Draft a reply to the last message from my sister saying I'll call her this weekend."
- "Move all newsletter and promotional email from this week to a 'Read Later' folder — batch 20 at a time."
The server exposes 61 discrete tools so the agent can compose its own workflow — read, triage, write, schedule, track tasks — without hardcoded macros.
Works With
- OpenClaw — native MCP support, available via ClawHub
- Claude Code — add to
~/.claude/settings.jsonundermcpServers - Cursor — MCP-compatible
- Any MCP client — it's a standard stdio MCP server
Listed on the official MCP Registry as io.github.mpalermiti/outlook-mcp.
Features
61 tools across 14 categories:
- Auth (1) -- auth status check (login is via CLI)
- Mail Read (4) -- list inbox (with Focused Inbox filter), read message, search (KQL), list folders
- Mail Write (3) -- send, reply/reply-all, forward
- Mail Triage (9) -- move, delete (soft by default), flag, categorize, mark read/unread, reclassify (Focused Inbox), list/set/delete per-sender Focused Inbox overrides
- Calendar Read (2) -- list events (with recurring expansion), get event details
- Calendar Write (4) -- create, update, delete, RSVP (accept/decline/tentative)
- Contacts (6) -- list, search, get, create, update, delete
- To Do (6) -- list task lists, list/create/update/complete/delete tasks
- Drafts (5) -- list, create, update, send, delete
- Attachments (5) -- list, download, send-with-attachments, attach-to-draft, remove-draft-attachment
- Folder Management (3) -- create, rename, delete mail folders
- Threading and Batch (3) -- list thread, copy message, batch triage
- User and Admin (6) -- whoami, list calendars, list categories, mail tips, accounts
- Mailbox Settings (4) -- get/set timezone, get/set auto-reply (out-of-office)
Design principles:
- BYOID -- Bring Your Own ID. You register your own Azure AD app. No shared client ID.
- Zero telemetry -- no analytics, no local caching, no third-party calls.
- Token storage -- OS keyring via
azure-identity(macOS Keychain, Windows Credential Store, Linux Secret Service). - Input validation -- all inputs validated (email, Graph IDs, OData, KQL, datetimes) before any API call.
- Read-only mode -- set
read_only: truein config to block all write operations. - Soft delete -- delete moves to Deleted Items by default. Hard delete requires explicit
permanent: true. - Timezone-aware -- calendar operations respect your configured IANA timezone.
Azure AD App Registration
You need to register a free Azure AD app to get a client ID.
Prerequisites (Personal Microsoft Accounts)
Microsoft has deprecated app registration for personal accounts without an Azure AD tenant. You need to create a free Azure account first:
- Go to azure.microsoft.com/free and sign up with your personal
@outlook.comaccount. Requires a credit card for identity verification but won't charge you. This creates a proper Azure AD tenant.
Register the App
-
Go to App Registrations and sign in with your
@outlook.comaccount. -
Click "+ New registration" and fill in:
- Name: anything except Microsoft-branded terms (e.g.
mp-outlook-mcp— names like "Outlook MCP" will be rejected) - Supported account types: select "Personal Microsoft accounts only"
- Redirect URI: leave blank
- Name: anything except Microsoft-branded terms (e.g.
-
Click Register. Copy the Application (client) ID from the overview page.
-
Go to Authentication (Preview) → Settings tab → toggle "Allow public client flows" to Yes → Save.
-
Go to API permissions → Add a permission → Microsoft Graph → Delegated permissions → add:
Mail.ReadWrite,Mail.SendCalendars.ReadWriteContacts.ReadWrite,Tasks.ReadWriteUser.Read,offline_access
No client secret is needed. The device code flow uses public client auth.
Quick Start
Upgrading from 1.6.x: v1.7.0 introduces four mailbox-settings tools (
outlook_get_timezone,outlook_set_timezone,outlook_get_auto_reply,outlook_set_auto_reply) that require theMailboxSettings.Read/MailboxSettings.ReadWriteGraph scopes. After upgrading, re-runoutlook-mcp authso your cached token picks up the new scopes — otherwise the four mailbox-settings tools will fail with an auth error.
Install
Option A — from PyPI (recommended):
uv tool install outlook-graph-mcp
# or: pipx install outlook-graph-mcp
# or: pip install outlook-graph-mcp
Option B — from source:
git clone https://github.com/mpalermiti/outlook-mcp.git
cd outlook-mcp
uv sync
Configure
Create ~/.outlook-mcp/config.json:
{
"client_id": "YOUR_APPLICATION_CLIENT_ID",
"tenant_id": "consumers",
"timezone": "America/Los_Angeles",
"read_only": true
}
The only required field is client_id. Everything else has sensible defaults. Start with read_only: true — flip to false when you're comfortable.
Register with your MCP client
If installed from PyPI:
{
"mcpServers": {
"outlook": {
"command": "outlook-mcp"
}
}
}
If installed from source:
{
"mcpServers": {
"outlook": {
"command": "uv",
"args": ["--directory", "/path/to/outlook-mcp", "run", "outlook-mcp"]
}
}
}
For OpenClaw, use the openclaw mcp CLI — it writes to mcp.servers in ~/.openclaw/openclaw.json for you:
# If installed from PyPI:
openclaw mcp set outlook '{"command":"outlook-mcp"}'
# If installed from source:
openclaw mcp set outlook '{"command":"uv","args":["--directory","/path/to/outlook-mcp","run","outlook-mcp"]}'
# Verify:
openclaw mcp list
openclaw mcp show outlook --json
Restart the OpenClaw gateway after registering. See the OpenClaw MCP docs for SSE/HTTP transport variants.
Authenticate
Run this once on the machine where the MCP server will run:
uv run outlook-mcp auth
You'll get a URL and a code. Open the URL in any browser, enter the code, and sign in with your Microsoft account. Tokens are cached in the OS keyring — the MCP server picks them up automatically.
Other CLI commands:
uv run outlook-mcp status # Check auth status
uv run outlook-mcp logout # Clear credentials
uv run outlook-mcp serve # Start MCP server (default, used by OpenClaw/Claude)
Tool Reference
Auth
| Tool | Description |
|---|---|
outlook_auth_status |
Check if authenticated and whether read-only mode is active. |
Note: Authentication is handled via the CLI (
outlook-mcp auth), not through MCP tools. See Authenticate above.
Mail Read
| Tool | Description |
|---|---|
outlook_list_inbox |
List messages in a folder. folder accepts display names, well-known names, or Graph IDs. Filter by read status, sender, date range, Focused Inbox classification. Pagination via skip. |
outlook_read_message |
Get full message by ID. Format: text, html, or full (both). Pass include_deferred_send=True to also surface the draft's scheduled delivery time. |
outlook_search_mail |
Search mail using KQL query. Optionally scope to a folder by name or ID. |
outlook_list_folders |
List mail folders with counts, parent_id, and child_count. Pass recursive=true to walk the full folder tree (subfolders included). |
Mail Write
| Tool | Description |
|---|---|
outlook_send_message |
Send email. Supports TO/CC/BCC, HTML body, importance level. |
outlook_reply |
Reply or reply-all to a message. |
outlook_forward |
Forward a message to one or more recipients with optional comment. |
Mail Triage
| Tool | Description |
|---|---|
outlook_move_message |
Move a message to a folder by name or ID. |
outlook_delete_message |
Delete a message. Soft delete (Deleted Items) by default. permanent: true for hard delete. |
outlook_flag_message |
Set follow-up flag: flagged, complete, or notFlagged. |
outlook_categorize_message |
Set categories on a message. |
outlook_mark_read |
Mark a message as read or unread. |
outlook_reclassify_message |
Move a message between Focused Inbox and Other (focused / other). |
outlook_list_inbox_overrides |
List Focused Inbox per-sender override rules. |
outlook_set_inbox_override |
Upsert a per-sender Focused Inbox override (focused / other). Case-insensitive sender matching; PATCH-if-exists, else POST. |
outlook_delete_inbox_override |
Delete a Focused Inbox override by ID. |
Calendar Read
| Tool | Description |
|---|---|
outlook_list_events |
List events in a date range. Expands recurring events. Configurable via days, after, before. |
outlook_get_event |
Get full event details: attendees, body, online meeting URL, recurrence. |
Calendar Write
| Tool | Description |
|---|---|
outlook_create_event |
Create event with location, attendees, recurrence, online meeting support. |
outlook_update_event |
Update event fields (subject, time, location, body). Only patches changed fields. |
outlook_delete_event |
Delete a calendar event. |
outlook_rsvp |
RSVP to an event: accept, decline, or tentative. Optionally include a message. |
Contacts
| Tool | Description |
|---|---|
outlook_list_contacts |
List contacts with cursor pagination. |
outlook_search_contacts |
Search contacts by name or email. |
outlook_get_contact |
Get full contact details by ID. |
outlook_create_contact |
Create a new contact. |
outlook_update_contact |
Update contact fields. |
outlook_delete_contact |
Delete a contact. |
To Do
| Tool | Description |
|---|---|
outlook_list_task_lists |
List To Do lists. |
outlook_list_tasks |
List tasks with status filter and pagination. |
outlook_create_task |
Create task with due date, importance, recurrence. |
outlook_update_task |
Update task fields. |
outlook_complete_task |
Mark task as completed. |
outlook_delete_task |
Delete a task. |
Drafts
| Tool | Description |
|---|---|
outlook_list_drafts |
List draft messages with pagination. |
outlook_create_draft |
Create a draft. Supports scheduled delivery via deferred_send_datetime (server-side, Outlook-desktop-compatible "Delay Delivery"). |
outlook_update_draft |
Update draft fields. Accepts is_html=True for HTML bodies and deferred_send_datetime to set or clear the scheduled delivery time. |
outlook_send_draft |
Send an existing draft. |
outlook_delete_draft |
Delete a draft. |
Attachments
| Tool | Description |
|---|---|
outlook_list_attachments |
List attachments on a message. |
outlook_download_attachment |
Download attachment and save decoded bytes to a file. |
outlook_send_with_attachments |
Send message with file attachments (auto upload session for >3MB). |
outlook_attach_to_draft |
Add attachments to an existing draft (auto upload session for >3MB). |
outlook_remove_draft_attachment |
Remove a single attachment from a draft. |
Folder Management
| Tool | Description |
|---|---|
outlook_create_folder |
Create mail folder (top-level or nested). |
outlook_rename_folder |
Rename a mail folder. |
outlook_delete_folder |
Delete a mail folder (refuses well-known folders). |
Threading and Batch
| Tool | Description |
|---|---|
outlook_list_thread |
Get all messages in a conversation thread. |
outlook_copy_message |
Copy a message to another folder. |
outlook_batch_triage |
Batch move/flag/categorize/mark_read (max 20 per call). Single Graph /$batch round-trip — 10-20× faster than per-message calls for large triage. |
User and Admin
| Tool | Description |
|---|---|
outlook_whoami |
Get current user profile. |
outlook_list_calendars |
List available calendars. |
outlook_list_categories |
List category definitions with colors. |
outlook_get_mail_tips |
Pre-send check (OOF, delivery restrictions). |
outlook_list_accounts |
List configured accounts. |
outlook_switch_account |
Switch active account. |
Mailbox Settings
| Tool | Description |
|---|---|
outlook_get_timezone |
Read the server-side mailbox timezone (/me/mailboxSettings/timeZone). |
outlook_set_timezone |
Write the server-side mailbox timezone. Accepts IANA (America/Los_Angeles) or Windows (Pacific Standard Time) zone names; Graph validates. |
outlook_get_auto_reply |
Read out-of-office / auto-reply configuration. Scheduled datetimes are normalized to UTC ISO 8601 where possible. |
outlook_set_auto_reply |
Configure auto-reply. Supports disabled / always / scheduled status, internal + external messages, none / contacts_only / all external-audience scopes, and scheduled start/end datetimes. |
Configuration
Config lives at ~/.outlook-mcp/config.json (created with 0600 permissions).
| Field | Type | Default | Description |
|---|---|---|---|
client_id |
string |
null |
Azure AD application (client) ID. Required for auth. |
tenant_id |
string |
"consumers" |
Azure AD tenant. Use "consumers" for personal Microsoft accounts. |
timezone |
string |
"UTC" |
IANA timezone (e.g. "America/New_York"). Used for relative date computations in calendar tools. |
read_only |
bool |
false |
When true, all write tools (send, reply, move, delete, create, update, RSVP) return an error. |
allow_categories |
list[string] |
[] |
Optional. Restrict write tools to specific categories (see below). Empty list = all writes allowed when read_only: false. |
Granular Write Permissions (optional)
By default, read_only: false unlocks all write tools. For finer control, set allow_categories to restrict write access to specific categories. Read tools (list, search, get) are always allowed — allow_categories only narrows the write surface.
Available categories:
| Category | Tools | Risk |
|---|---|---|
mail_drafts |
create/update/delete draft | Safe — drafts only, no send |
mail_triage |
move, delete (soft), flag, categorize, mark read, copy, batch | Moderate — reversible except hard delete |
mail_folders |
create/rename/delete folder | Moderate |
mail_send |
send, reply, forward, send_draft, send_with_attachments | Dangerous — sends email on your behalf |
calendar_write |
create/update/delete event, RSVP | Moderate — creates calendar entries |
contacts_write |
create/update/delete contact | Moderate |
todo_write |
create/update/complete/delete task | Safe — your own task list |
mailbox_settings |
get/set timezone, get/set auto-reply (OOO) | Moderate — server-side mailbox state, visible to all senders |
Example policies:
Draft-only assistant (agent can compose drafts, you review and send):
{ "read_only": false, "allow_categories": ["mail_drafts", "mail_triage", "todo_write"] }
Calendar-only (agent can manage your schedule, nothing else):
{ "read_only": false, "allow_categories": ["calendar_write"] }
Full write access (agent can do everything):
{ "read_only": false }
Read-only (safest default, no writes):
{ "read_only": true }
When allow_categories is set, any tool in a non-allowed category returns a permission-denied error (PermissionDeniedError) naming the blocked category. When allow_categories is empty (or unset) and read_only is false, all write tools are permitted. read_only: true always takes precedence — if set, all writes are blocked regardless of allow_categories. Unknown category names are rejected at config load time with a validation error; only the eight names above are accepted.
Privacy and Security
- Zero telemetry. No analytics, no tracking, no usage data collected.
- Zero local caching. Every call goes directly to Microsoft Graph. No local email/calendar storage.
- Zero third-party calls. The server only talks to
graph.microsoft.comandlogin.microsoftonline.com. - Token storage. OAuth tokens are persisted via
azure-identity'sTokenCachePersistenceOptions. On macOS the OS Keychain is used; on Windows, DPAPI; on Linux with PyGObject/libsecret available, gnome-keyring. On Linux without libsecret (e.g. the isolated venv created byuv tool install), tokens fall back to a0600plaintext file at~/.IdentityService/and the MCP logs a one-time warning at startup. For encrypted storage on Linux, installpython3-gi gnome-keyring libsecret-1-0and re-create the venv with--system-site-packages. - No logging of sensitive data. Message bodies, recipient addresses, and tokens are never logged.
- Config permissions. Config directory is
0700, config file is0600. Symlinked configs are rejected. - Input validation. All user inputs (email addresses, Graph IDs, OData filters, KQL queries, datetimes) are validated and sanitized before reaching the Graph API.
Development
# Install dev dependencies
uv sync --extra dev
# Run tests
uv run pytest
# Lint
uv run ruff check src/ tests/
# Format
uv run ruff format src/ tests/
# Run server locally (stdio)
uv run outlook-mcp
Requirements: Python 3.10+
Roadmap
- Inbox Rules -- list, create, delete rules
- Advanced mail -- raw MIME export, internet message headers
- Calendar -- cancel event (with attendee notification)
- Checklists -- checklist items on To Do tasks
- Enterprise (Entra ID) -- work/school account support
License
MIT. See LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file outlook_graph_mcp-1.7.0.tar.gz.
File metadata
- Download URL: outlook_graph_mcp-1.7.0.tar.gz
- Upload date:
- Size: 53.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b645a0ecaa02b0381a5102e34bf346821163d499d6436e88c4c6a62aecbc0334
|
|
| MD5 |
1b9a56fc4d0b3f87bf117c1cbf07e45c
|
|
| BLAKE2b-256 |
34b2df121d457a7eb2b4ec132a679e208fd81c5a8160edcd3502687a46958d7d
|
File details
Details for the file outlook_graph_mcp-1.7.0-py3-none-any.whl.
File metadata
- Download URL: outlook_graph_mcp-1.7.0-py3-none-any.whl
- Upload date:
- Size: 70.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c52776a3326c3143bb57301f836a1f5878194dc000b7146fefe3304e4a9a3afe
|
|
| MD5 |
21c98a488c382e781a41cd0d17c7e782
|
|
| BLAKE2b-256 |
f16381b65ba7917b2b494444e6a819b557866681cca719caa9647e8aa2dbbf34
|