Microsoft Planner MCP Server
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@Microsoft Planner MCP ServerCreate a task 'Review Q3 report' in Marketing plan"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
MCP Server for Microsoft Planner
An unofficial MCP server that connects AI assistants to Microsoft Planner. Ask your AI assistant to create tasks, organise plans, manage buckets, and more, all through natural language.
This is an unofficial, community/open-source MCP server for Microsoft Planner.
It is not affiliated with, endorsed by, or sponsored by Microsoft.
Built with FastMCP and authenticated via Microsoft Entra ID (Azure AD) using the On-Behalf-Of (OBO) flow to call Microsoft Graph.
Table of Contents
What Can It Do?
Once connected, you can ask your AI assistant things like:
"Show me all my Planner tasks that are overdue"
"Create a task called 'Prepare Q3 report' in the Marketing plan"
"Move all incomplete tasks in the Sprint bucket to the Backlog bucket"
"Mark the 'Update docs' task as complete"
The AI assistant translates your request into the appropriate tool calls automatically.
Note: This MCP only supports Planner basic tasks and plans
Prerequisites
A Microsoft 365 account with access to Microsoft Planner
An Azure Entra ID (Azure AD) app registration (see Azure Setup below)
An MCP-compatible client — for example:
To run the server locally you also need:
uv (Python package manager)
Or, to run via Docker:
Azure Entra ID Setup
An Azure app registration is required so the server can authenticate users and call Microsoft Graph on their behalf. You need admin access to an Azure Entra ID tenant (or ask your IT administrator).
Step 1 — Register the Application
Go to the Azure Portal → Microsoft Entra ID → App registrations → New registration
Enter a name (e.g.
Microsoft Planner MCP)Under Supported account types, choose the option appropriate for your organisation
Set the Redirect URI to Web →
http://localhost:8000/auth/callbackClick Register
Step 2 — Configure API Permissions
In your new app registration, go to API permissions → Add a permission → Microsoft Graph → Delegated permissions
Add these permissions:
Tasks.ReadWrite— read and write Planner tasksUser.Read— read the signed-in user's profileUser.ReadBasic.All— resolve user display names from the GUIDs in task assignments (required only forlist_userstool)
Click Grant admin consent for your organisation
Step 3 — Expose an API Scope
Go to Expose an API
Set the Application ID URI (accept the default
api://<client-id>or customise it)Click Add a scope:
Scope name:
mcp-accessWho can consent: Admins and users (or Admins only if you prefer)
Fill in the display name and description
Set state to Enabled
Step 4 — Set Token Version
Go to Manifest (or Authentication → Advanced settings in newer portal versions)
Set
"requestedAccessTokenVersion"to2Save
Step 5 — Create a Client Secret
Go to Certificates & secrets → New client secret
Add a description and choose an expiry period
Copy the Value immediately (it is only shown once)
Step 6 — Note Your IDs
You will need these three values for configuration:
Value | Where to Find It |
Application (client) ID | App registration → Overview |
Directory (tenant) ID | App registration → Overview |
Client secret | The value copied in Step 5 |
Installation
Option A — Local (Python + uv)
git clone https://github.com/aixolotl/microsoft-planner-mcp
cd microsoft-planner-mcp
uv syncOption B — Docker
No Python installation needed. See Running with Docker below.
Configuration
Copy the example environment file and fill in your Azure credentials:
cp .env.example .envEdit .env:
# Azure App Registration (required)
CLIENT_ID=your-app-client-id
CLIENT_SECRET=your-app-client-secret
TENANT_ID=your-azure-tenant-id
# Public URL of this server (used for OAuth redirect URI)
BASE_URL=http://localhost:8000
# JSON-encoded list of allowed CORS origins
# Include http://localhost:6274 if using MCP Inspector for testing
ALLOWED_ORIGINS=["http://localhost:8000","http://localhost:6274"]
# Require FastMCP's extra client consent prompt (set false only for local dev)
REQUIRE_AUTHORIZATION_CONSENT=trueRunning the Server
Local
uv run uvicorn src.server:app --host 0.0.0.0 --port 8000The MCP endpoint is available at http://localhost:8000/mcp. A health check endpoint is at http://localhost:8000/health.
Running with Docker
# Pull the latest image
docker pull ghcr.io/aixolotl/microsoft-planner-mcp:latest
# Run with environment variables
docker run --rm -i \
-e BASE_URL=https://localhost:8000 \
-e CLIENT_ID=your_client_id \
-e CLIENT_SECRET=your_api_token \
-e TENANT_ID=your_tenant_id \
-e ALLOWED_ORIGINS=["http://localhost:8000","http://localhost:6274", "http://localhost:3000"] \
-e REQUIRE_AUTHORIZATION_CONSENT=true \
ghcr.io/aixolotl/microsoft-planner-mcp:latestThis starts the MCP server on port 8000.
Running with Docker compose
docker compose upThis starts the MCP server on port 8000. The Docker Compose configuration also includes a Jaeger instance for trace visualisation (see OpenTelemetry Tracing).
Connecting an MCP Client
Once the server is running, configure your MCP client to connect to it.
VS Code
Add the following to your VS Code settings (.vscode/settings.json in your project, or your user settings):
{
"mcp": {
"servers": {
"planner": {
"type": "http",
"url": "http://localhost:8000/mcp"
}
}
}
}Then use Copilot Chat in Agent mode and ask it to interact with your Planner tasks. Copilot will discover the available tools automatically.
Other MCP Clients
Any client that supports the Streamable HTTP transport can connect by pointing to http://localhost:8000/mcp. The server advertises OAuth metadata automatically — the client handles the authentication flow.
Available Tools
All tools are available to your AI assistant automatically once connected. You don't need to call them directly — just describe what you want in natural language. The parameter details below are provided for reference and for client developers.
Read-only tools are annotated with readOnlyHint: true so clients can skip confirmation prompts. Destructive tools (deletes) are annotated with destructiveHint: true.
User
get_me
Return the authenticated user's profile from Microsoft Graph.
Parameters: None
Returns: User profile object (id, displayName, mail, etc.) or
null
list_users
Retrieve Microsoft 365 users by GUID, e-mail address, or free-text search. Useful for resolving the user GUIDs returned in task assignment objects to display names.
Parameter | Type | Required | Description |
| string | No | Comma-separated fields to include (default: |
| string | No | Free text search on display name, or search by field name and value, e.g. |
| list[string] | No | User object GUIDs to look up. Translated to an OData |
| list[string] | No | User principal names (UPNs / e-mail addresses) to look up. Translated to an OData |
| integer | No | Maximum number of users to return (default: |
Returns: List of user objects or
null
Note: When both
guids/emailsandsearchare supplied, the GUID/email filter takes priority. Very large lists of GUIDs or e-mail addresses are silently truncated to stay within the 2 048-character Graph URL limit.
Groups
list_my_groups
List all Microsoft 365 groups the authenticated user is a member of.
Parameter | Type | Required | Description |
| string | No | Comma-separated fields to include (default: |
| string | No | OData filter expression, e.g. |
| string | No | OData search string, e.g. |
Returns: List of group objects or
null
Note: The Groups tool will not return details like name or mail of groups with the standard permissions
Tasks.ReadWritedocumented here, however searching and filtering still works, so you can find a group with a specific name using this tool.
Plans
list_my_plans
List Planner plans shared with the authenticated user.
Parameter | Type | Required | Description |
| string | No | Comma-separated fields to include (default: |
Returns: List of plan objects or
null
list_group_plans
List all Planner plans belonging to a Microsoft 365 group.
Parameter | Type | Required | Description |
| string | Yes | The object ID of the group (from |
| string | No | Comma-separated fields to include (default: |
Returns: List of plan objects or
null
create_plan
Create a new Planner plan for a Microsoft 365 group.
Parameter | Type | Required | Description |
| string | Yes | The object ID of the M365 group that will own the plan |
| string | Yes | Display title for the new plan |
Returns: The created plan object or
null
delete_plan
Delete a Planner plan.
Parameter | Type | Required | Description |
| string | Yes | The ID of the plan to delete |
| string | Yes | The current |
Returns: Confirmation message
list_plan_categories
Get category label definitions for a Planner plan. Returns all 25 category slots with their key (e.g. category1) and display name.
Parameter | Type | Required | Description |
| string | Yes | The ID of the plan |
Returns: List of category objects (
key,display_name) ornull
Buckets
list_buckets
List all buckets in a Planner plan.
Parameter | Type | Required | Description |
| string | Yes | The ID of the plan |
Returns: List of bucket objects or
null
create_bucket
Create a new bucket in a Planner plan.
Parameter | Type | Required | Description |
| string | Yes | The ID of the plan to create the bucket in |
| string | Yes | Display name for the new bucket |
Returns: The created bucket object or
null
delete_bucket
Delete a Planner bucket.
Parameter | Type | Required | Description |
| string | Yes | The ID of the bucket to delete |
| string | Yes | The current |
Returns: Confirmation message
Tasks
list_my_tasks
List all Planner tasks assigned to the authenticated user across all plans.
Parameter | Type | Required | Description |
| string | No | Comma-separated fields to include (default: |
| string | No | OData filter expression, e.g. |
| string | No | Free-text search matched against the task title |
Returns: List of task objects or
null
list_tasks
List all tasks in a Planner plan.
Parameter | Type | Required | Description |
| string | Yes | The ID of the plan |
| string | No | Comma-separated fields to include (default: |
| string | No | OData filter expression, e.g. |
| string | No | Free-text search matched against the task title |
Returns: List of task objects or
null
get_task_details
Get the full details for a task: description, checklist items, and external references.
Parameter | Type | Required | Description |
| string | Yes | The ID of the task |
Returns: Task details object (description, checklist, references) or
null
create_task
Create a new task in a Planner plan.
Parameter | Type | Required | Description |
| string | Yes | The ID of the plan |
| string | Yes | The ID of the bucket to place the task in |
| string | Yes | Title of the task |
| string | No | ISO 8601 start date (e.g. |
| string | No | ISO 8601 due date (e.g. |
| integer | No | Completion percentage, 0–100 |
| list[string] | No | User object IDs to assign to the task |
Returns: The created task object or
null
update_task
Update a task's standard fields and/or detail fields (description, checklist, references). Only provided fields are changed. When detail fields are specified, a separate API call updates the task details resource automatically.
Parameter | Type | Required | Description |
| string | Yes | The ID of the task |
| string | Yes | The current |
| string | No | New title |
| integer | No | Completion percentage, 0–100 |
| string | No | ISO 8601 due date |
| string | No | ID of the bucket to move the task to |
| string | No | Order hint for sorting within the assignee's task list |
| list[string] | No | User IDs to assign |
| list[string] | No | User IDs to remove |
| string | No | Plain-text description (up to 2000 characters) — detail field |
| string | No | Preview style: |
| object | No | Dict keyed by checklist item GUID. Pass |
| object | No | Dict keyed by URL-encoded reference URL. Pass |
| string | No | The |
Returns: The updated task object, the updated task details object, or both (
{ "task": ..., "details": ... }) depending on which fields were provided. Returnsnullif the result is empty.
Note: The
update_task_detailstool has been removed. Useupdate_taskwith detail field parameters (description,checklist_items,references,preview_type) instead. The task and details resources have separate ETags — provideetag_detailswhen available to avoid an extra round-trip, or omit it to let the tool auto-refresh.
list_task_fields
Return metadata for every field on a Planner task, including its data type, description, whether it is writable, and whether it requires a separate get_task_details call.
Parameters: None
Returns: List of field metadata objects (
name,type,description,writable,detailed)
Note: The list of fields is static as Planner Standard doesn't support custom fields. But this endpoint awaits Microsoft's future updates to the API to fully support Planner Premium.
delete_task
Delete a Planner task.
Parameter | Type | Required | Description |
| string | Yes | The ID of the task |
| string | Yes | The current |
Returns: Confirmation message
Development
This section covers building, testing, and contributing to the project.
Project Structure
src/
├── server.py # FastMCP app, middleware, route mounting
├── config.py # Settings via pydantic-settings
├── auth_provider.py # Azure OAuth provider (OBO flow)
├── deps.py # Shared dependency helpers
├── graph_client_manager.py # Singleton GraphClientManager with per-user OBO clients
├── telemetry.py # OpenTelemetry setup
├── types.py # Shared structural types
├── services/
│ └── planner_service.py # Business logic wrapping Graph SDK calls
└── tools/
├── me.py # get_me
├── groups.py # list_my_groups
├── plans.py # plan tools + list_plan_categories
├── tasks.py # task tools + list_task_fields
└── buckets.py # bucket tools
tests/
├── conftest.py
├── test_buckets_tool.py
├── test_groups_tool.py
├── test_planner_service.py
├── test_plans_tool.py
└── test_tasks_tool.pyInstalling Dev Dependencies
uv sync --devRunning Tests
uv run pytestWith verbose output:
uv run pytest -vArchitecture
The server is built with FastMCP and uses these key patterns:
Server composition — Tools are split into five domain routers (me, groups, plans, tasks, buckets), each a standalone FastMCP instance mounted on the main app. This keeps each domain's tools, imports, and tests isolated.
Authentication — The server uses FastMCP's OAuthProxy pattern via a custom AzureProvider. Azure Entra ID does not support Dynamic Client Registration (DCR), so the provider acts as a DCR-compliant proxy facing MCP clients while using the pre-registered app credentials with Azure. When a tool call arrives, the server exchanges the MCP session token for a Microsoft Graph token via the On-Behalf-Of flow, scoped to Tasks.ReadWrite and User.Read.
Middleware — Five built-in middleware layers are stacked on the server (outermost first):
Middleware | Purpose |
| Catches unhandled exceptions, logs full traces server-side, returns clean MCP errors |
| 60 req/min per client to protect the Microsoft Graph quota |
| Records wall-clock duration for every MCP operation |
| Emits one JSON log line per request (method, status, duration, client info) |
| Truncates tool responses above 500 KB to prevent overflowing LLM context windows |
Client logging — Every tool sends real-time progress messages to the MCP client via get_optional_context(), so users see status updates like "Fetching tasks…" and "Found 12 task(s)" in their client.
OpenTelemetry Tracing
The server includes native OpenTelemetry instrumentation with zero overhead when unused. To enable trace export:
# Install the optional OTEL dependency group
uv sync --group otel
# Set the OTLP endpoint and start the server
export OTEL_EXPORTER_OTLP_ENDPOINT=http://localhost:4317
export OTEL_SERVICE_NAME=microsoft-planner-mcp
uv run uvicorn src.server:app --host 0.0.0.0 --port 8000Traces follow the MCP semantic conventions and work with any OTLP-compatible backend (Jaeger, Grafana Tempo, Datadog, New Relic, etc.). The Docker Compose setup includes Jaeger with a UI at http://localhost:16686.
Testing with MCP Inspector
MCP Inspector lets you test tools interactively in a browser:
Start the server:
uv run uvicorn src.server:app --host 0.0.0.0 --port 8000In a separate terminal:
npx @modelcontextprotocol/inspectorOpen the URL printed in the terminal (e.g.
http://localhost:6274/?MCP_PROXY_AUTH_TOKEN=...)Set Transport Type to
Streamable HTTPand URL tohttp://localhost:8000/mcp, then click Connect.
Contributing
Contributions are welcome! Please open an issue or submit a pull request. See the issue templates for bug reports, feature requests, and tasks.
License
This project is licensed under the MIT License.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/aixolotl/microsoft-planner-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server