Open Paper Trading MCP
Provides AI agents with tools for paper trading stocks, options, ETFs, and bonds, including advanced options strategies and risk analysis, using real market data without financial risk.
README
Open Paper Trading MCP ๐
A comprehensive paper trading simulator with dual interfaces: REST API (FastAPI) and AI agent tools (MCP). Designed for algorithmic trading development, strategy backtesting, options trading simulation, and training AI agents in realistic market environments without financial risk.
๐ฏ Core Capabilities
- Multi-Asset Trading: Stocks, options, ETFs, and bonds with specialized implementations
- Advanced Options Trading: Full options chain support with Greeks calculations and 15+ spread strategies
- Professional Spread Builder: Iron Condors, Butterflies, Straddles, Strangles, and advanced multi-leg strategies
- Real-time Risk Analysis: P&L diagrams, breakeven calculations, and win probability analysis
- AI Agent Training: Native MCP interface for training trading agents and LLMs
- Production-Ready: Type-safe, async architecture with comprehensive testing and monitoring
- Dual Interface Access: Both REST API (web clients) and MCP tools (AI agents) access identical functionality
โ Current Status (2025-01-08)
๐ PRODUCTION READY QUALITY - Successfully implemented and deployed dual-server architecture:
- FastAPI Server (port 2080): Frontend integration + 49 REST API endpoints operational
- MCP Server (port 2081): Independent MCP server with 43 tools + auto-generated list_tools function
- Test Success Rate: 99.8% (576/581 tests passing, comprehensive journey-based testing)
- Code Quality: 100% ruff compliance, 100% mypy clean, all style violations resolved
- Database Integration: PostgreSQL async operations with proper session management
- Service Layer: TradingService fully integrated via dependency injection
- API Documentation: Auto-generated docs available at
/docs - Multi-Account Support: Complete backend implementation with account_id parameter support
- Options Trading: Complete options chain integration with live market data and professional interface
- ADK Evaluation Complete: 42/42 evaluations tested - 100% agent behavior validation with proper multi-step workflows
- MCP Tool Reliability: All 43 tools validated through live agent interaction using real market data
- AsyncIO Infrastructure: Complete resolution of event loop conflicts, 100% test stability
โ Prerequisites
Before you begin, ensure you have the following installed:
- Docker and Docker Compose: For running the application in a containerized environment.
- Python: Version 3.11 or higher.
- uv: The project's package manager.
๐ Quick Start
# 1. Clone the repository
git clone https://github.com/yourusername/open-paper-trading-mcp.git
cd open-paper-trading-mcp
# 2. Start everything with Docker
docker-compose up --build
# 3. Services are now available at:
# - Frontend & API: http://localhost:2080/
# - MCP Server: http://localhost:2081/
# - API Docs: http://localhost:2080/docs
๐๏ธ Architecture Overview
REST Client AI Agent
| |
v v
FastAPI Server MCP Server
(Port 2080) (Port 2081)
| |
+------------------+
|
v
TradingService
|
+---------+---------+
| |
v v
PostgreSQL DB Robinhood API
(Trading State) (Market Data)
Split Architecture Benefits:
- Independent Servers: FastAPI (2080) and MCP (2081) run separately, eliminating mounting conflicts
- Dual Interface Access: Web clients use REST API, AI agents use MCP tools - same underlying functionality
- Service Layer Unity: Both interfaces use identical TradingService for consistency
- Database-First: All trading state persisted in PostgreSQL with async operations
- Real-time Market Data: Direct API calls to Robinhood for current market information
- Type Safety: Full Pydantic validation on all inputs/outputs across both interfaces
๐ Key Achievements & Lessons Learned
Major Technical Achievements
- AsyncIO Infrastructure Mastery: Resolved 164 AsyncIO event loop conflicts that were causing 49% test failure rate
- Split Architecture Success: Overcame FastMCP mounting conflicts by implementing independent server architecture
- Database Session Consistency: Established unified
get_async_session()pattern across entire codebase - Test Infrastructure Stability: Achieved 99.8% test success rate (576/581 tests passing)
- Dual Interface Implementation: Successfully created mirror functionality between REST API and MCP tools
- MCP Tool Validation Complete: 42/42 ADK evaluations completed with 100% agent behavior validation
- Live Market Data Integration: All tools successfully use real Robinhood API with proper error handling
- Production Deployment Ready: Docker containers optimized, real data policy enforced, comprehensive monitoring
Critical Lessons Learned
- Event Loop Management: Create fresh database engines per test in current event loop to prevent AsyncIO conflicts
- Service Architecture: Independent servers solve mounting conflicts better than complex integration
- Database Patterns: Always use
get_async_session()dependency injection, neverAsyncSessionLocal()directly - Testing Patterns: Standardized mocking with
side_effectfor async generators ensures reliable tests - Code Quality: Comprehensive linting (ruff), type checking (mypy), and formatting standards prevent technical debt
Development Workflow Optimizations
- Split Development: FastAPI server (frontend/API) and MCP server (AI tools) can be developed independently
- Service Layer Unity: Changes to TradingService automatically benefit both interfaces
- Test-Driven Stability: Comprehensive test coverage (70%+) with AsyncIO-safe patterns
- Live API Integration: Robinhood API tests with
@pytest.mark.robinhoodfor real-world validation
๐ ๏ธ Technology Stack
- Backend: Python, FastAPI, FastMCP
- Database: PostgreSQL
- ORM: SQLAlchemy
- Market Data: Robinhood API
- Package Management: uv
- Containerization: Docker, Docker Compose
โ๏ธ Development
Local Setup (without Docker)
# Install uv package manager
curl -LsSf https://astral.sh/uv/install.sh | sh
# Create virtual environment and install dependencies
uv venv
source .venv/bin/activate # On Windows: .venv\Scripts\activate
uv pip sync pyproject.toml
# Set up PostgreSQL and update .env
cp .env.example .env
# Edit DATABASE_URL in .env
# Run the application
uv run python app/main.py
Configuration
The application is configured using environment variables. Copy the .env.example file to .env and update the following variables:
DATABASE_URL: The connection string for your PostgreSQL database.ROBINHOOD_USERNAME: Your Robinhood username (for live market data).ROBINHOOD_PASSWORD: Your Robinhood password.QUOTE_ADAPTER_TYPE: The quote adapter to use (testorrobinhood).
Development Commands
# Format code
python scripts/dev.py format
# Run linting
python scripts/dev.py lint
# Type checking
python scripts/dev.py typecheck
# Run all tests
python scripts/dev.py test
# Run all checks
python scripts/dev.py check
Database Development Patterns
Always use consistent database session patterns:
# โ
CORRECT - Use get_async_session()
from app.storage.database import get_async_session
async def database_operation():
async for db in get_async_session():
result = await db.execute(select(Model))
return result.scalars().all()
# โ INCORRECT - Never use AsyncSessionLocal() directly
from app.storage.database import AsyncSessionLocal
async with AsyncSessionLocal() as db: # Breaks testing!
pass
Testing database code:
from unittest.mock import patch
@patch('app.storage.database.get_async_session')
async def test_function(mock_get_session, test_session):
async def mock_generator():
yield test_session
mock_get_session.return_value = mock_generator()
# Your test code here
result = await database_operation()
assert result is not None
This ensures consistent behavior between production and testing environments.
๐งช Testing Best Practices
AsyncIO Event Loop Management
Critical for async test stability:
# tests/conftest.py - Create fresh engines per test
@pytest_asyncio.fixture(scope="function")
async def db_session() -> AsyncGenerator[AsyncSession, None]:
# Create engine in current event loop (critical for AsyncIO compatibility)
test_engine = create_async_engine(
database_url,
echo=False,
future=True,
pool_pre_ping=True, # Verify connections before use
pool_recycle=300 # Recycle connections every 5 minutes
)
test_session_factory = async_sessionmaker(
bind=test_engine,
class_=AsyncSession,
expire_on_commit=False
)
try:
async with test_session_factory() as session:
yield session
finally:
await test_engine.dispose() # Critical for preventing leaks
Common Test Issues & Solutions
1. Missing Await Keywords
# โ WRONG - Async method without await
result = adapter.get_account_ids() # Returns coroutine!
assert len(result) == 3 # TypeError: object of type 'coroutine' has no len()
# โ
CORRECT - Always await async methods
result = await adapter.get_account_ids()
assert len(result) == 3
2. DateTime Timezone Issues
# โ WRONG - Mixed timezone awareness
from datetime import datetime
created_at = datetime.now(timezone.utc) # timezone-aware
updated_at = datetime.now() # timezone-naive
# โ
CORRECT - Consistent timezone handling
from datetime import datetime, timezone
created_at = datetime.now(timezone.utc)
updated_at = datetime.now(timezone.utc)
3. Database Session Mocking Pattern
# โ
CORRECT - Proper async session mocking
async def test_database_operation(self, db_session: AsyncSession):
adapter = DatabaseAccountAdapter()
with patch('app.adapters.accounts.get_async_session') as mock_get_session:
async def mock_session_generator():
yield db_session
mock_get_session.side_effect = lambda: mock_session_generator()
# Test database operations with real session
result = await adapter.get_account("test-id")
assert result is not None
Test Infrastructure Achievements
- AsyncIO Event Loop Issues: โ RESOLVED - 164 AsyncIO errors eliminated
- Success Rate Improvement: 29% โ 99.8% (576/581 tests passing)
- Database Session Consistency: โ Implemented across all core functions
- Test Pattern Standardization: โ Unified mocking patterns established
- Live API Testing: โ
Robinhood tests integrated with
@pytest.mark.robinhoodmarker - Journey-Based Testing: โ User journey organization prevents timeout issues with 581 total tests
Live API Testing with Robinhood
The test suite includes integration tests that make live, read-only calls to the Robinhood API:
# Run all tests including live Robinhood calls
uv run pytest
# Exclude live Robinhood API tests (faster, no external dependencies)
uv run pytest -m "not robinhood"
# Run only Robinhood integration tests
uv run pytest -m "robinhood"
Robinhood Test Features:
- Read-only operations: Stock quotes, market data, company information, search
- Rate limiting protection: All marked as
@pytest.mark.slow - Real data validation: Verifies actual API responses and data formats
- Shared fixtures: Consistent setup via
trading_service_robinhoodfixture
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.