Microsoft Entra MCP Server
A FastMCP server that provides AI assistants with access to Microsoft Entra (Azure AD) directory services. It enables LLMs to search for users, groups, and check memberships using the Microsoft Graph API.
README
Microsoft Entra MCP Server
A FastMCP server that provides AI assistants with access to Microsoft Entra (Azure AD) directory services. This server enables LLMs to search for users, groups, and check memberships using the Microsoft Graph API.
Features
- š 4 Tools: Search users, search groups, get user membership, get group members
- š§ Full-text Search: Uses Microsoft Graph
$searchwith AND tokenization for order-agnostic matches (e.g., "Arun AND Singh" matches "Singh, Arun" and "Arun Kumar Singh") - š Accurate Counts: Uses
ConsistencyLevel: eventualwith$count=trueand pagination across@odata.nextLink - š Pagination: Users, groups, and group members are paginated to return complete results up to limits
- š Robust Identifier Resolution: Membership lookup resolves user ID from email/UPN before querying
- šØ 7 Prompts: Pre-built prompt templates for common Entra queries
- š Secure Authentication: Uses Azure AD app registration with client credentials
- š Health Endpoint: Built-in health check for monitoring
- ā Fully Tested: Comprehensive test suite with pytest
Prerequisites
Azure AD App Registration
- Go to Azure Portal ā Microsoft Entra ID ā App registrations
- Create a new app registration
- Note down:
- Application (client) ID
- Directory (tenant) ID
- Create a client secret under Certificates & secrets
- Grant the following Microsoft Graph API permissions:
User.Read.AllGroup.Read.AllGroupMember.Read.All
Environment Variables
Set these environment variables before running. You can either:
Option 1: Direct environment variables
export ENTRA_TENANT_ID="your-tenant-id-here"
export ENTRA_CLIENT_ID="your-client-id-here"
export ENTRA_CLIENT_SECRET="your-client-secret-here"
Option 2: Use .env file
cp env.template .env
# Edit .env with your actual values
The server will automatically load variables from a .env file if it exists.
Quick Start
Installation
pip install -r requirements.txt
Running Locally
# Create virtual environment
python3 -m venv .venv
# Activate virtual environment
source .venv/bin/activate
# Install dependencies
pip install -r requirements.txt
# Set environment variables
export ENTRA_TENANT_ID="..."
export ENTRA_CLIENT_ID="..."
export ENTRA_CLIENT_SECRET="..."
# Run the server
python main.py
Server will start on http://0.0.0.0:8001
Using Docker
# Build the image
docker build -t entra-mcp .
# Run the container with environment variables
docker run -p 8001:8001 \
-e ENTRA_TENANT_ID="..." \
-e ENTRA_CLIENT_ID="..." \
-e ENTRA_CLIENT_SECRET="..." \
entra-mcp
API Reference
Tools
1. search_entra_users
Search for users by display name, email, or user principal name.
Parameters:
| Parameter | Type | Required | Default | Description |
|---|---|---|---|---|
query |
string | Yes | - | Search term for user name, email, or UPN (full-text $search with AND-tokenized terms) |
max_results |
integer | No | 10 | Maximum number of results to return |
Search Behavior:
- Full-text across
displayName,mail, anduserPrincipalNameusing Graph$search - Tokens are ANDed for better relevance (e.g., "Arun Singh" ā
"Arun" AND "Singh"), matching order-agnostic names and middle names - Uses
ConsistencyLevel: eventualand$count=truefor accurate totals - Paginates across
@odata.nextLinkand returns up tomax_results
Example Usage:
# Order-agnostic and middle-name tolerant
search_entra_users(query="Arun Singh", max_results=25)
# Also matches comma-separated and compound names
search_entra_users(query="Singh, Arun", max_results=25)
# Email or UPN fragments are also matched by `$search`
search_entra_users(query="arun.singh@company.com")
2. search_entra_groups
Search for groups by display name or description.
Parameters:
| Parameter | Type | Required | Default | Description |
|---|---|---|---|---|
query |
string | Yes | - | Search term for group name or description (full-text $search with AND-tokenized terms) |
max_results |
integer | No | 10 | Maximum number of results to return |
Search Behavior:
- Full-text across
displayNameanddescriptionwith Graph$searchand AND-tokenized terms - Uses
ConsistencyLevel: eventualand$count=truefor accurate totals - Paginates across
@odata.nextLinkand returns up tomax_results
3. get_user_group_membership
Get all groups a user belongs to.
Parameters:
| Parameter | Type | Required | Default | Description |
|---|---|---|---|---|
user_identifier |
string | Yes | - | User ID, UPN, or email address |
Behavior:
- Resolves the user ID from email/UPN automatically when needed
- Uses
ConsistencyLevel: eventual,$count=true, and paginates across@odata.nextLink
4. get_group_members
Get all members of a group.
Parameters:
| Parameter | Type | Required | Default | Description |
|---|---|---|---|---|
group_identifier |
string | Yes | - | Group ID or display name |
max_results |
integer | No | 50 | Maximum number of members to return |
Behavior:
- Uses
ConsistencyLevel: eventualand paginates across@odata.nextLink - Returns up to
max_resultsmembers
Available Prompts
1. find_user_by_name
Find a user by their display name.
Default now suggests a broader search and higher limit to take advantage of $search:
# Suggests: Call search_entra_users with: query='{name}', max_results=25
2. find_user_by_email
Find a user by their email address.
3. find_group_by_name
Find a group by name.
4. check_user_groups
Check what groups a user belongs to.
5. list_group_members
List all members of a specific group.
6. user_access_audit
Perform an access audit for a user.
7. group_membership_audit
Audit the membership of a security-sensitive group.
Testing
Health Check
curl http://localhost:8001/health
Run Test Suite
# Run all tests
python -m pytest tests/ -v
Project Structure
streamable-HTTP-Entra-MCP/
āāā main.py # Main server with tools and authentication
āāā promptz.py # Prompt templates for LLMs (7 prompts)
āāā requirements.txt # Python dependencies
āāā README.md # This file
āāā tests/
āāā __init__.py
āāā test_main.py # Tests for main functionality
Security Notes
- The server requires Azure AD application permissions to read user and group data
- All API calls are authenticated using client credentials flow
- No user data is stored locally - all queries go directly to Microsoft Graph API
- Ensure your Azure AD app has minimal required permissions
Graph Query Semantics Used
ConsistencyLevel: eventualis required for$searchand$count$searchvalues are AND-tokenized to improve relevance and handle name order variations$count=trueis requested to return accurate totals- Results are paginated via
@odata.nextLinkuntil limits are reached
Use Cases
- š User Lookup: Find user information by name or email
- š„ Group Discovery: Search for available groups
- š Access Control: Check user group memberships for permissions
- š Audit & Compliance: Review group memberships and user access
- š¤ AI Assistants: Enable LLMs to answer questions about Entra directory
Dependencies
Core:
fastmcp==2.13.0.1- FastMCP framework for MCP serverhttpx==0.28.1- Async HTTP clientazure-identity==1.19.0- Azure authenticationmsal==1.31.0- Microsoft Authentication Library
Development:
pytest==8.3.4- Testing frameworkpytest-asyncio==0.24.0- Async test support
API Data Source
This server uses the Microsoft Graph API:
- Base URL:
https://graph.microsoft.com/v1.0 - Authentication: Client Credentials Flow
- Scopes:
https://graph.microsoft.com/.default
Troubleshooting
Authentication Errors
# Check environment variables are set
echo $ENTRA_TENANT_ID $ENTRA_CLIENT_ID $ENTRA_CLIENT_SECRET
# Verify Azure AD app permissions in Azure Portal
# Ensure client secret is not expired
Import Errors
# Install dependencies
pip install -r requirements.txt
# Verify Azure packages
pip list | grep azure
License
See LICENSE file for details.
Built with ā¤ļø using FastMCP and Microsoft Graph API
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.