PyEye - MCP server for semantic analysis and code navigation
Project description
PyEye ๐๏ธ
PyEye Server
An extensible MCP (Model Context Protocol) server that provides intelligent Python code analysis, navigation, and understanding capabilities for AI assistants like Claude.
Features
- ๐ Semantic Code Navigation: Find symbols, go to definitions, find references using Jedi
- ๐ Module & Package Analysis: List packages/modules, analyze dependencies, detect circular imports
- ๐๏ธ Multi-Project Support: Analyze multiple projects and dependencies simultaneously
- ๐ฆ Namespace Packages: Handle packages distributed across multiple repositories
- ๐ Standalone Scripts: Analyze notebooks, scripts, and examples alongside formal packages
- ๐ Auto-Update: Automatically detects and reflects file changes with smart cache invalidation
- โ๏ธ Configuration System: Flexible configuration via files, env vars, or auto-discovery
- ๐ Extensible Plugin System: Framework-specific analyzers (Pydantic, Django, Flask)
- ๐ Fast & Cached: Intelligent caching with LRU eviction and performance optimization
- ๐ฏ Type-Aware: Full understanding of Python type hints and annotations
- ๐ Performance Monitoring: Built-in metrics tracking with p50/p95/p99 latencies
- ๐ก๏ธ Input Validation: Secure parameter validation and path checking
- ๐ค Development Automation: Release automation, dogfooding metrics, and worktree safety
Installation
The PyEye can be installed in two ways:
Option 1: Project-Specific Installation (Recommended)
Install directly into your Python project's virtual environment:
# Activate your project's virtual environment
source /path/to/your/project/venv/bin/activate
# Install from PyPI
pip install pyeye-mcp
# Or install from source
git clone https://github.com/okeefeco/pyeye-mcp.git
pip install -e ./pyeye-mcp
Then create a .mcp.json file in your project root:
{
"mcpServers": {
"pyeye": {
"type": "stdio",
"command": "python",
"args": ["-m", "pyeye.mcp"],
"env": {}
}
}
}
This way, the MCP server uses your project's environment and has access to all your project's dependencies.
Option 2: Global Installation
For analyzing multiple projects or using with global Python:
# Install globally with pipx (recommended for isolation)
pipx install pyeye-mcp
# Or with pip
pip install --user pyeye-mcp
# Or from source
git clone https://github.com/okeefeco/pyeye-mcp.git
cd pyeye-mcp
pip install --user .
Configure with Claude Code (Global)
# Add the MCP server globally (available in all projects)
claude mcp add pyeye -s user -- python -m pyeye.mcp
# Verify it's connected
claude mcp list
Configure with Claude Desktop
Add to your Claude Desktop configuration (~/Library/Application Support/Claude/claude_desktop_config.json on macOS):
{
"mcpServers": {
"pyeye": {
"command": "python",
"args": ["-m", "pyeye.mcp"],
"env": {}
}
}
}
Note: Use the full path to Python if needed (e.g., /usr/local/bin/python3 or C:\\Python311\\python.exe).
Configure with GitHub Copilot (VS Code)
As of 2025, GitHub Copilot has full MCP support in VS Code, JetBrains, Eclipse, and Xcode. Follow these steps to use this PyEye server with GitHub Copilot:
Prerequisites
- GitHub Copilot Business or Enterprise subscription (required for MCP support)
- VS Code version 1.102 or later (MCP support is GA)
- Organization MCP policy enabled by your admin
Step 1: Enable MCP in Your Organization
Your GitHub Copilot administrator needs to enable the MCP servers policy:
- Go to your organization settings on GitHub
- Navigate to Copilot โ Policies
- Enable "MCP servers in Copilot" policy
- Save changes
Step 2: Install the MCP Server
Install the PyEye server in your project or globally:
# Option A: Install in your project's virtual environment (recommended)
pip install pyeye-mcp
# Option B: Install globally with pipx
pipx install pyeye-mcp
# Option C: Install from source
git clone https://github.com/okeefeco/pyeye-mcp.git
pip install -e ./pyeye-mcp
Step 3: Configure VS Code
Add the MCP server configuration to your VS Code settings:
User Settings (applies to all projects):
// File: ~/.config/Code/User/settings.json (Linux/Mac)
// or %APPDATA%\Code\User\settings.json (Windows)
{
"github.copilot.chat.mcpServers": {
"pyeye": {
"command": "python",
"args": ["-m", "pyeye.mcp"],
"env": {}
}
}
}
Workspace Settings (project-specific):
// File: .vscode/settings.json in your project root
{
"github.copilot.chat.mcpServers": {
"pyeye": {
"command": "${workspaceFolder}/.venv/bin/python",
"args": ["-m", "pyeye.mcp"],
"env": {
"PYTHONPATH": "${workspaceFolder}"
}
}
}
}
Step 4: Verify Connection
- Open VS Code in your Python project
- Open the GitHub Copilot Chat panel
- Type:
@mcp listto see available MCP servers - You should see
pyeyein the list - Test with:
@mcp pyeye find_symbol MyClass
Troubleshooting
MCP not available:
- Ensure you have Copilot Business/Enterprise (not Free/Pro)
- Check that your organization admin enabled the MCP policy
- Update VS Code to version 1.102 or later
Server not connecting:
- Verify Python path in the configuration
- Check that
pyeyeis installed:python -m pyeye.mcp --help - Look for errors in VS Code Output panel โ GitHub Copilot Logs
Import errors:
- If using a virtual environment, ensure the path points to the venv Python
- Add
PYTHONPATHto the env configuration if needed
Other IDEs
JetBrains IDEs (IntelliJ, PyCharm, etc.):
- MCP support is GA - configure in Settings โ Tools โ GitHub Copilot โ MCP Servers
Visual Studio:
- MCP support is in preview - configure in Tools โ Options โ GitHub Copilot โ MCP Servers
Eclipse & Xcode:
- MCP support is GA - see IDE-specific documentation for configuration
Configuration
The server can be configured to analyze packages in other locations. Create a .pyeye.json file in your project:
{
"packages": [
"../my-shared-library",
"~/repos/company-utils",
"/absolute/path/to/package"
],
"namespaces": {
"mycompany": [
"~/repos/mycompany-auth",
"~/repos/mycompany-api"
]
}
}
Configuration Methods
Configuration is loaded in the following order (later sources override earlier ones):
- Global Config:
~/.config/pyeye/config.jsonor~/.pyeye.json- User defaults - Project Config:
.pyeye.jsonin project root or[tool.pyeye]inpyproject.toml - Override File:
.pyeye.override.json- Local development overrides (git-ignored) - Auto-Discovery: Automatically detects source layouts and sibling packages if no packages configured
Using Override Files
Override files are perfect for local development configurations that shouldn't be committed:
// .pyeye.override.json (git-ignored)
{
"packages": [
"../my-local-dev-package",
"~/dev/experimental"
],
"namespaces": {
"company.feature": ["/home/user/feature-branch"]
}
}
Performance Settings
All performance-critical settings can be configured via environment variables to tune for your specific workload:
| Environment Variable | Default | Description | Valid Range |
|---|---|---|---|
PYEYE_MAX_PROJECTS |
10 | Maximum number of projects in memory | 1-1000 |
PYEYE_CACHE_TTL |
300 | Cache time-to-live in seconds | 0-86400 (24h) |
PYEYE_WATCHER_DEBOUNCE |
0.5 | File watcher debounce delay in seconds | 0.0-10.0 |
PYEYE_MAX_FILE_SIZE |
1048576 | Maximum file size to analyze (bytes) | 1KB-100MB |
PYEYE_MAX_WORKERS |
4 | Maximum concurrent analysis workers | 1-32 |
PYEYE_ANALYSIS_TIMEOUT |
30.0 | Analysis timeout in seconds | 1.0-300.0 |
PYEYE_ENABLE_MEMORY_PROFILING |
false | Enable memory profiling | true/false |
PYEYE_ENABLE_PERFORMANCE_METRICS |
false | Enable performance metrics | true/false |
| Connection Pooling | Optimize multi-project workflows | ||
PYEYE_ENABLE_CONNECTION_POOLING |
true | Enable connection pooling for multiple projects | true/false |
PYEYE_POOL_MAX_CONNECTIONS |
10 | Maximum pooled project connections | 1-100 |
PYEYE_POOL_TTL |
3600 | Connection time-to-live in seconds | 60-86400 |
Performance Tuning Examples
Large codebase with stable files:
export PYEYE_MAX_PROJECTS=50 # Handle more projects
export PYEYE_CACHE_TTL=1800 # 30 minute cache
export PYEYE_WATCHER_DEBOUNCE=2.0 # Less frequent updates
Active development with frequent changes:
export PYEYE_MAX_PROJECTS=5 # Fewer projects, faster switching
export PYEYE_CACHE_TTL=60 # 1 minute cache
export PYEYE_WATCHER_DEBOUNCE=0.1 # Near real-time updates
Memory-constrained environment:
export PYEYE_MAX_PROJECTS=3 # Minimal project cache
export PYEYE_MAX_FILE_SIZE=524288 # 512KB file limit
export PYEYE_MAX_WORKERS=2 # Fewer workers
This file is automatically ignored by git and takes precedence over all other configuration sources.
Auto-Detection of Source Layouts
PyEye automatically detects source layouts from pyproject.toml build backend metadata, supporting projects that use the src/ directory pattern. This works with multiple build backends:
Setuptools:
[tool.setuptools.packages.find]
where = ["src"]
Poetry:
[[tool.poetry.packages]]
include = "mypackage"
from = "src"
Hatch:
[tool.hatch.build.targets.wheel]
sources = ["src"]
PDM:
[tool.pdm.build]
package-dir = "src"
If no configuration is found in pyproject.toml, PyEye will also check for the presence of a src/ directory containing Python packages and automatically add it to the package paths.
Note: Explicit [tool.pyeye] configuration always takes precedence over auto-detected layouts.
Workflow Resources
pyeye provides workflow guidance as MCP Resources to help AI agents and users discover how to use tools effectively for common multi-step tasks.
Discovering Workflows
List available workflows:
"List pyeye workflow resources"
Or with Claude Code's MCP tools:
ListMcpResourcesTool(server="pyeye")
Using Workflows
On-demand (trial):
"Use pyeye find-references workflow"
"Use pyeye refactoring workflow"
"Use pyeye code-understanding workflow"
"Use pyeye dependency-analysis workflow"
"Use pyeye code-review-standards workflow"
"Use pyeye code-review-security workflow"
"Use pyeye code-review-pr workflow"
Permanent (adoption):
"Add pyeye find-references workflow to my CLAUDE.md"
The AI will fetch the workflow, add it to your context file, and automatically follow it in future sessions.
Available Workflows
-
find-references - Find ALL class/function references including packages AND notebooks/scripts
- Addresses limitation that
find_referencesonly works with packages (issue #236) - Combines
get_type_info,find_references, andGrepfor complete coverage
- Addresses limitation that
-
refactoring - Safe refactoring with impact analysis
- Analyze subclasses, references, and dependencies before changing code
- Includes change planning and validation steps
- Prevents breaking changes through comprehensive analysis
-
code-understanding - Understand unfamiliar code structure
- Systematic exploration from "What is this?" to "How does it work?"
- Covers symbol location, inspection, hierarchy, execution flow, and usage patterns
- Progressive understanding from basic to deep integration knowledge
-
dependency-analysis - Analyze module dependencies and architecture
- Map import relationships and identify circular dependencies
- Calculate coupling metrics and assess change impact
- Understand architectural patterns and module relationships
-
code-review-standards - Python code review best practices (2025)
- Industry standards: PEP 8, PEP 257, PEP 484, modern Python features
- MCP-enhanced analysis: Type safety, anti-patterns, architecture review
- Automated checks combined with semantic understanding
-
code-review-security - OWASP security code review
- Security checklist: Input validation, injection prevention, auth patterns
- Data flow analysis using MCP tools (trace user input through code)
- Framework-specific security (Flask/Django plugin integration)
-
code-review-pr - Complete pull request review workflow
- Combines automated checks, semantic analysis, and manual review
- Step-by-step process: CI validation โ impact analysis โ standards โ security
- Constructive feedback guidelines and time budgets
Example Usage Flow
Discovery (README): User learns workflows exist
User: "How do I find all references to a class?"
AI: "There's a find-references workflow for that. Let me show you..."
Trial (On-Demand): User tries workflow
User: "Use pyeye find-references workflow for this class"
AI: [fetches workflow from MCP Resources]
AI: [executes: get_type_info โ find_references โ Grep]
AI: "Found 15 references: 12 in packages, 3 in notebooks"
Adoption (Self-Service): User adds to context
User: "That's useful - add it to my CLAUDE.md"
AI: [reads workflow resource, appends to CLAUDE.md]
AI: "โ
Workflow added to your context"
Automatic Usage: Future sessions use workflow automatically
User: "Find all uses of this class"
AI: [sees workflow in context, follows steps automatically]
AI: [returns complete results without prompting]
Benefits
- Discover best practices - Learn optimal tool combinations
- Avoid trial and error - Workflows encode discovered patterns
- Handle tool limitations - Includes workarounds (e.g., issue #236)
- Self-service adoption - Add workflows to your context as needed
- Always up-to-date - Workflows maintained with MCP server
Technical Details
Workflows are exposed via MCP Resources protocol:
- URI scheme:
workflows://[workflow-name] - Format: Markdown (human and AI readable)
- Access: Via
ListMcpResourcesToolandReadMcpResourceTool - Integration: Can be programmatically added to user context files
Core Tools
Basic Navigation
find_symbol- Find class, function, or variable definitions with re-export tracking- Now includes
import_pathsfield showing all available import paths for a symbol - Automatically detects re-exports through
__init__.pyfiles - Lists shorter/preferred import paths first (e.g.,
from models import Userbeforefrom models.user import User)
- Now includes
goto_definition- Jump to where a symbol is definedfind_references- Find all places where a symbol is usedget_type_info- Get type hints and docstringsfind_imports- Track module imports across the projectget_call_hierarchy- Analyze function call relationshipsfind_subclasses- Find all classes inheriting from a given base class- Supports direct and indirect inheritance
- Works with built-in classes (Exception, str, etc.)
- Can show full inheritance hierarchy chains
- Handles multiple inheritance correctly
Multi-Project Tools
configure_packages- Set up additional package locationsfind_symbol_multi- Search across multiple projectsconfigure_namespace_package- Set up distributed namespace packagesfind_in_namespace- Search within namespace packages
Project Structure & Analysis
list_project_structure- View Python project file organizationlist_packages- List all Python packages with structurelist_modules- List modules with exports, classes, functions, and metricsanalyze_dependencies- Analyze module imports and detect circular dependenciesget_module_info- Get detailed module information including metrics and dependencies
Framework-Specific Tools (Auto-Activated)
Django (when Django is detected)
find_django_models- Find all Django modelsfind_django_views- Find all viewsfind_django_urls- Find URL patternsfind_django_templates- Find templatesfind_django_migrations- Find migrations
Pydantic (when Pydantic is detected)
find_pydantic_models- Discover all BaseModel classesget_model_schema- Extract complete model schemafind_validators- Locate all validation methodsfind_field_validators- Find field-specific validatorsfind_model_config- Extract model configurationstrace_model_inheritance- Map model inheritance hierarchiesfind_computed_fields- Find computed_field and @property fields
Flask (when Flask is detected)
find_flask_routes- Discover all route decorators with methods and endpointsfind_flask_blueprints- Locate Blueprint definitions and registrationsfind_flask_views- Find view functions and MethodView classesfind_flask_templates- Locate Jinja2 templates and render_template callsfind_flask_extensions- Identify Flask extensions (SQLAlchemy, Login, CORS, etc.)find_flask_config- Find configuration files and app.config usagefind_error_handlers- Locate @app.errorhandler decoratorsfind_cli_commands- Find Flask CLI commands (@app.cli.command)
Advanced Features
Multi-Project Support
Analyze your main project along with local dependencies:
# Configure to analyze multiple packages
configure_packages(
packages=["../my-lib", "~/repos/shared-utils"],
namespaces={"company": ["~/repos/company-*"]}
)
Namespace Packages
Handle packages distributed across multiple repositories:
# company.auth in repo A, company.api in repo B
configure_namespace_package(
namespace="company",
repo_paths=["~/repos/company-auth", "~/repos/company-api"]
)
Auto-Update on File Changes
The server uses file watching to automatically update when code changes:
- Detects modifications in real-time
- Invalidates cache for changed files
- Maintains separate watchers per project
Architecture
PyEye
โโโ Core Server (FastMCP)
โ โโโ 17 MCP tools registered
โโโ Project Manager
โ โโโ Multi-project support (LRU cache, max 10)
โ โโโ Connection pooling (optional optimization)
โ โโโ Namespace resolver
โ โโโ Configuration loader (multiple sources)
โโโ Analysis Engine
โ โโโ Jedi (semantic analysis & type inference)
โโโ Caching & Performance
โ โโโ File watchers (watchdog with debouncing)
โ โโโ Granular cache (5min TTL with smart invalidation)
โ โโโ Metrics collection (p50/p95/p99 latencies)
โ โโโ Performance monitoring & reporting
โโโ Validation & Security
โ โโโ Input validation (MCP tool parameters)
โ โโโ Path security checks
โ โโโ Safe file operations
โโโ Plugin System
โ โโโ Base plugin class (AnalyzerPlugin)
โ โโโ Pydantic plugin (7 specialized tools)
โ โโโ Django plugin (5 specialized tools)
โ โโโ Flask plugin (8 specialized tools)
โโโ Utility Systems
โ โโโ Dependency tracking & circular detection
โ โโโ Import analysis & re-export resolution
โ โโโ Scope management (main/all/namespace scoping)
โ โโโ Async utilities (concurrent operations)
โ โโโ Cross-platform path handling
โโโ Development & Automation
โโโ Release automation agent
โโโ Dogfooding metrics tracking
โโโ Worktree safety management
Plugin Development
Create custom plugins for your project patterns:
from pyeye.plugins.base import AnalyzerPlugin
class MyProjectPlugin(AnalyzerPlugin):
def name(self) -> str:
return "MyProject"
def detect(self) -> bool:
# Return True if this plugin should activate
return (self.project_path / "my_framework.conf").exists()
def find_patterns(self, pattern_name: str):
# Find your custom patterns
pass
Performance Monitoring Details
The server includes comprehensive performance monitoring to help identify bottlenecks and optimize performance for large-scale deployments.
Connection Pooling for Multi-Project Workflows
Connection pooling is enabled by default to optimize performance when working with multiple projects. You can customize the pooling behavior:
# Customize connection pooling (already enabled by default)
export PYEYE_POOL_MAX_CONNECTIONS=20 # Increase pool size for many projects
export PYEYE_POOL_TTL=7200 # Increase TTL to 2 hours
# Or disable pooling if needed
export PYEYE_ENABLE_CONNECTION_POOLING=false
# Start the server
uv run mcp dev src/pyeye/server.py
Connection pooling provides significant performance improvements:
- Reduced initialization time for frequently accessed projects
- Shared connections across multiple analysis operations
- Automatic eviction of idle connections based on TTL
- Memory-efficient pooling with configurable limits
Metrics Tracked
- Operation Latencies: p50, p95, p99 percentiles for all MCP tools
- Cache Performance: Hit rate, miss rate, evictions
- Connection Pool Stats: Pool size, hits, misses, evictions, reuse rate
- Memory Usage: RSS, VMS, percentage used
- Error Rates: Track failures per operation
- Throughput: Operations per second
Performance Baselines
The following performance baselines are enforced in CI:
| Operation | p50 (ms) | p95 (ms) | p99 (ms) |
|---|---|---|---|
| symbol_search | 50 | 100 | 200 |
| goto_definition | 30 | 75 | 150 |
| find_references | 100 | 250 | 500 |
| cache_lookup | 0.1 | 0.5 | 1.0 |
Development
# Install development dependencies
uv add --dev pytest black ruff mypy
# Run tests
uv run pytest
# Format code
uv run black src/
uv run ruff check src/
# Test the server
uv run mcp dev src/pyeye/server.py
Documentation
- Installation Guide
- Configuration Guide
- Troubleshooting Guide - Common issues and solutions
- API Reference
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
License
MIT License - see LICENSE file for details
Acknowledgments
Built on top of:
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 pyeye_mcp-1.1.1.tar.gz.
File metadata
- Download URL: pyeye_mcp-1.1.1.tar.gz
- Upload date:
- Size: 490.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b6397e60e70184bf8a63aa10244edca685f7401fc7610c773c0130cc3da98394
|
|
| MD5 |
cdd517f208d85ac0bda6381bc71d11eb
|
|
| BLAKE2b-256 |
43c3ec60720a1a16c35d77dde7215c8bca83897e45f3de40ce8686738d8833c0
|
Provenance
The following attestation bundles were made for pyeye_mcp-1.1.1.tar.gz:
Publisher:
release.yml on okeefeco/pyeye-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pyeye_mcp-1.1.1.tar.gz -
Subject digest:
b6397e60e70184bf8a63aa10244edca685f7401fc7610c773c0130cc3da98394 - Sigstore transparency entry: 1077394831
- Sigstore integration time:
-
Permalink:
okeefeco/pyeye-mcp@9ddd19cce5c18dced05098117a7bb4691df1ead3 -
Branch / Tag:
refs/tags/v1.1.1 - Owner: https://github.com/okeefeco
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@9ddd19cce5c18dced05098117a7bb4691df1ead3 -
Trigger Event:
push
-
Statement type:
File details
Details for the file pyeye_mcp-1.1.1-py3-none-any.whl.
File metadata
- Download URL: pyeye_mcp-1.1.1-py3-none-any.whl
- Upload date:
- Size: 148.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b101eda8d922759702262795e2af726c214cf36353622e9855c5e4b41a4e913f
|
|
| MD5 |
77d4cc9f9cfa3c638822d8262de68462
|
|
| BLAKE2b-256 |
b9de2bd4d212096455a9ce10763ea9be3c0ee0a2864720c00b08f09b0d972571
|
Provenance
The following attestation bundles were made for pyeye_mcp-1.1.1-py3-none-any.whl:
Publisher:
release.yml on okeefeco/pyeye-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pyeye_mcp-1.1.1-py3-none-any.whl -
Subject digest:
b101eda8d922759702262795e2af726c214cf36353622e9855c5e4b41a4e913f - Sigstore transparency entry: 1077394842
- Sigstore integration time:
-
Permalink:
okeefeco/pyeye-mcp@9ddd19cce5c18dced05098117a7bb4691df1ead3 -
Branch / Tag:
refs/tags/v1.1.1 - Owner: https://github.com/okeefeco
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@9ddd19cce5c18dced05098117a7bb4691df1ead3 -
Trigger Event:
push
-
Statement type: