Skip to main content

MCP server for Microsoft Teams, Outlook Calendar, and Mail via Microsoft Graph API

Project description

Graph MCP

An MCP server that connects Claude to Microsoft Teams, Outlook Calendar, and Mail through the Microsoft Graph API.

What it does

Gives Claude access to 41 tools:

Category Tools
Auth Check status, login (browser OAuth), logout
Chats List chats, read/send messages, create chats, list members
Teams & Channels List teams, list channels, read/send messages, list members, read/send replies
Calendar List calendars, list/get events, create/update/delete events
Mail List emails, read email, search emails, send email, reply, list/get attachments
Meetings Find meeting by Join URL, list/get transcripts, list recordings, get recording URL
Files Browse OneDrive folders, search files, get content, upload files, create sharing links
Users Search organization directory
Presence Get/set your presence, get another user's presence
Search Search messages across all chats and channels

Prerequisites

You need an Azure App Registration:

  1. Go to Azure Portal > App registrations > New registration

  2. Set platform to Mobile and desktop applications

  3. Set redirect URI to http://localhost:3000/auth/callback

  4. Mark as Public client (no client secret needed)

  5. Under API permissions, add these delegated permissions:

    • offline_access, openid, profile, User.Read
    • User.ReadBasic.All
    • Chat.Read, Chat.ReadWrite, ChatMessage.Send
    • ChannelMessage.Read.All, ChannelMessage.Send
    • Team.ReadBasic.All, Channel.ReadBasic.All, ChannelMember.Read.All
    • Calendars.ReadWrite
    • Mail.Read, Mail.Send
    • Presence.Read, Presence.Read.All, Presence.ReadWrite
    • OnlineMeetings.Read, OnlineMeetingTranscript.Read.All, OnlineMeetingRecording.Read.All
    • Files.ReadWrite.All

Install

pip install graph-mcp

Or from a cloned repo:

pip install -e .

Setup

graph-mcp setup

This asks for your Azure Client ID and Tenant ID, then gives you the exact command:

claude mcp add graph -e AZURE_CLIENT_ID=your-id -e AZURE_TENANT_ID=your-tenant -- /path/to/graph-mcp

Paste it, start Claude Code, and ask Claude to log in. It opens your browser for OAuth sign-in — that's it.

Message and email formatting

The Teams message tools (graph_send_chat_message, graph_send_channel_message, and graph_reply_to_channel_message) and outbound mail tools (graph_send_mail, graph_reply_mail) now default to HTML mode and convert markdown to HTML automatically using Python-Markdown.

Examples:

  • **bold**<strong>bold</strong>
  • - item lists → <ul><li>...</li></ul>
  • `code`<code>...</code>

If you already have raw HTML, pass it directly and it will be sent as-is.

For email, this means normal LLM-written markdown like **bold**, bullet lists, or inline code renders properly instead of showing raw markdown characters.

These tools also support optional top-level Graph API mentions, either in raw Graph format or a simplified shape such as:

[
  {
    "name": "Jane Smith",
    "user_id": "ef1c916a-3135-4417-ba27-8eb7bd084193"
  }
]

Use the corresponding <at id="0">Jane Smith</at> tag in the HTML body to trigger a real Teams mention.

How it works

Claude Code  ──stdio──>  graph-mcp  ──HTTPS──>  Microsoft Graph API
                              │
                         ~/.graph-mcp/
                           tokens.enc  (encrypted)
                           .key        (auto-generated)
  • Auth: OAuth2 Authorization Code flow with PKCE. Login opens your browser, a local callback server captures the token. No secrets stored in config.
  • Token persistence: Tokens are encrypted with Fernet and stored in ~/.graph-mcp/tokens.enc. The encryption key is auto-generated on first use. Logins survive server restarts.
  • Token refresh: Access tokens are refreshed automatically before they expire. You only need to log in again if the refresh token itself expires.
  • Rate limiting: Sliding window counter with exponential backoff. Respects Retry-After headers on 429 responses.

Configuration

All configuration is passed via environment variables (set in the MCP config's env block):

Variable Required Default Description
AZURE_CLIENT_ID Yes From your Azure App Registration
AZURE_TENANT_ID No common Your Azure tenant ID, or common for multi-tenant
GRAPH_REDIRECT_URI No http://localhost:3000/auth/callback Must match Azure app registration
GRAPH_TOKEN_ENCRYPTION_KEY No auto-generated Fernet key for token encryption
GRAPH_DEBUG No false Enable verbose logging

Troubleshooting

"Approval required" error during login Your Azure app is requesting scopes that aren't registered or need admin consent. Check the API permissions in the Azure portal match the list above.

403 Forbidden on specific tools The endpoint needs a permission that's not in your Azure app registration, or requires admin consent.

Login works but tools say "not authenticated" Restart the MCP server (claude mcp restart graph) — it may be running an old version.

Disclaimer

This project is an independent open-source effort and is not affiliated with, endorsed by, or sponsored by Microsoft Corporation. Microsoft, Microsoft Teams, Outlook, Microsoft 365, Microsoft Graph, and Azure are trademarks of the Microsoft group of companies.

This software is provided "as is", without warranty of any kind. Use it at your own risk. The authors accept no liability for any damages, data loss, or security issues arising from the use of this software. You are responsible for complying with your organization's policies and Microsoft's API Terms of Use when using this tool.

This software accesses Microsoft services on your behalf using your own credentials and Azure app registration. Data retrieved from Microsoft Graph (emails, messages, calendar events, etc.) is passed to the LLM that invoked the tool. Be mindful of your organization's data handling policies when using this with cloud-hosted AI models.

License

MIT — see LICENSE.

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

graph_mcp-0.5.0.tar.gz (20.5 kB view details)

Uploaded Source

Built Distribution

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

graph_mcp-0.5.0-py3-none-any.whl (28.4 kB view details)

Uploaded Python 3

File details

Details for the file graph_mcp-0.5.0.tar.gz.

File metadata

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

File hashes

Hashes for graph_mcp-0.5.0.tar.gz
Algorithm Hash digest
SHA256 aef1a61e188ca59496d82f87fe200b9da7c4fcac343fbd72ec19b35f80e6495b
MD5 4c789e8d175e793bab40e48b7afc0b10
BLAKE2b-256 b28ae8e908cf62b43fe84ee0b743b68270864807243ac35d48470594ed439d19

See more details on using hashes here.

Provenance

The following attestation bundles were made for graph_mcp-0.5.0.tar.gz:

Publisher: publish.yml on JustStas/Graph-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 graph_mcp-0.5.0-py3-none-any.whl.

File metadata

  • Download URL: graph_mcp-0.5.0-py3-none-any.whl
  • Upload date:
  • Size: 28.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for graph_mcp-0.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 4ecdf7cc6ae342ae5d945100c2560655964360683b881b7a0eff61ea68f833d7
MD5 6344e1618f895a5dae513ddbb3350a24
BLAKE2b-256 73c3c0a76ba59ef5cbf50749336b1eaf76eeb68039163e4b61d491769cf9cc47

See more details on using hashes here.

Provenance

The following attestation bundles were made for graph_mcp-0.5.0-py3-none-any.whl:

Publisher: publish.yml on JustStas/Graph-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