Playground
MCP Server example with TypeScript
Installation
npx mcp-playgroundAsk AI about Playground
Powered by Claude Β· Grounded in docs
I know everything about Playground. Ask me about installation, configuration, usage, or troubleshooting.
0/500
Reviews
Documentation
MCP Playground π οΈπ©οΈ
A Streamlit-based playground that lets you chat with large language models and seamlessly plug in external Multi-Server Command Protocol (MCP) tools. Spin up multiple FastMCP servers (Weather & Currency) alongside a Streamlit client, all orchestrated with Docker Compose. The client is provider-agnostic (OpenAI β’ Amazon Bedrock β’ Anthropic β’ Google Gemini β’ Groq) thanks to LangChain + LangGraph. Built with uv for lightning-fast dependency installation.
π Learn More
Want a deep dive into how it all works? Check out the detailed walkthrough in this Medium article: https://medium.com/@elkhan.alizada/your-own-ai-agent-playground-build-it-with-streamlit-langgraph-and-docker-4caeb6fe0ac4
π₯οΈπ Main Interface β Connected View

ποΈ Architecture

β¨ Key Features
| Feature | Description |
|---|---|
| π Multi-Server MCP | Register any number of MCP servers; the agent auto-detects available tools & routes calls. |
| π₯οΈ Streamlit Chat UI | Rich chat experience with history, sidebar controls and live tool execution output. |
| π§© Provider-Agnostic | One LangChain interface for OpenAI, Bedrock, Anthropic, Google, Groq. Switch on the fly. |
| π― Dynamic Model Selection | Override default models with custom model IDs directly from the UI. |
| β Dynamic MCP Server Management | Add/remove MCP servers on-the-fly through the UI without editing config files. |
| π€ React Agent via LangGraph | create_react_agent enables dynamic tool selection and reasoning. |
| π³ Docker-First | Separate Dockerfiles for client & each server + a single docker-compose.yaml. |
| π¦ Extensible | Drop-in new MCP servers or providers without touching UI code. |
π Project Layout
mcp-playground/
ββ docker-compose.yaml # One-command orchestration
ββ client/ # Streamlit UI
β ββ app.py # Main entry-point
β ββ config.py # Typed settings & defaults
β ββ servers_config.json # MCP endpoint catalogue
β ββ ui_components/ # Streamlit widgets
β ββ ...
ββ servers/
ββ server1/ # Weather Service MCP
β ββ main.py
ββ server2/ # Currency Exchange MCP
ββ main.py
π Quick Start
1 Β· Prerequisites
- Docker β₯ 24 & Docker Compose
- At least one LLM provider key (e.g.
OPENAI_API_KEY) or AWS creds for Bedrock.
2 Β· Clone & Run
git clone https://github.com/your-org/mcp-playground.git
cd mcp-playground
docker compose up --build
β‘ Fast Builds with uv: All services use uv for dependency management, providing significantly faster Docker builds compared to traditional pip. Dependencies are cached and only reinstalled when pyproject.toml changes.
| Service | URL | Default Port |
|---|---|---|
| Streamlit Client | http://localhost:8501 | 8501 |
| Weather MCP | http://localhost:8000 | 8000 |
| Currency MCP | http://localhost:8001 | 8001 |
βοΈ Configuration
Default Settings
All runtime settings are concentrated in client/config.py and environment variables.
| Variable | Purpose |
|---|---|
MODEL_ID | Provider selector (OpenAI, Bedrock, Anthropic, Google, Groq). |
TEMPERATURE | Sampling temperature (sidebar slider). |
MAX_TOKENS | Token limit (sidebar). |
Default Model IDs:
MODEL_OPTIONS = {
'OpenAI': 'gpt-4o',
'Antropic': 'claude-3-5-sonnet-20240620',
'Google': 'gemini-2.0-flash-001',
'Bedrock': 'us.anthropic.claude-sonnet-4-5-20250929-v1:0',
'Groq': 'meta-llama/llama-4-scout-17b-16e-instruct'
}
π― Dynamic Model Override
You can override the default model for any provider directly in the UI:
- Select your provider from the dropdown
- See the default model displayed
- Enter a custom model ID in the "π― Custom Model" field
- Leave empty to use the default
Examples:
- OpenAI: Override
gpt-4owitho1for reasoning tasks orgpt-4o-minifor cost efficiency - Anthropic: Use
claude-opus-4.6orclaude-sonnet-4.6for latest models - Google: Try
gemini-2.0-flash-exporgemini-exp-1206for experimental features - Bedrock: Use
us.anthropic.claude-opus-4-6-v1for Opus 4.6 orus.anthropic.claude-sonnet-4-6for Sonnet 4.6 - Groq: Try different Llama models like
llama-3.3-70b-versatile
β Dynamic MCP Server Management
Add custom MCP servers without editing servers_config.json:
- Click "β Add MCP Server" expander in the sidebar
- Fill in the server details:
- Server Name: Unique identifier (e.g.,
MyCustomServer) - Server URL: Full endpoint URL (e.g.,
http://localhost:8002/sse) - Transport Type:
sseorhttp - Timeout: Connection timeout in seconds (default: 600)
- SSE Read Timeout: SSE-specific timeout (default: 900)
- Headers (optional): JSON object with custom headers
{"Authorization": "Bearer token"}
- Server Name: Unique identifier (e.g.,
- Click "Add Server" to add multiple servers
- Servers persist in your session until removed
Server Sources:
- π From Config: Static servers from
servers_config.json(cannot be removed via UI) - π User Added: Dynamic servers added through UI (can be removed with ποΈ button)
MCP endpoints can also be statically defined in servers_config.json for persistent configuration.
π¬ Using the Playground
- Select Provider Β· Pick your LLM in the sidebar and paste the corresponding credentials.
- Customize Model (Optional) Β· Override the default model ID with any model from your provider.
- Add MCP Servers (Optional) Β· Add custom MCP servers through the UI or use the default servers.
- Connect MCP Servers Β· Click connect; available tools appear in the MCP Tools list.
- Chat Β· Type a question.
- If connected, the React agent decides whether to call an MCP tool (e.g. get_current_weather).
- Otherwise it falls back to plain LLM chat.
- Inspect Tool Calls Β· Tool invocations are streamed back as YAML blocks with inputs & outputs.
Try:
"What will the weather be in Baku tomorrow and how much is 100 USD in AZN?"
π οΈ Included MCP Servers
Weather Service :8000
mcp = FastMCP("Weather Service", host="0.0.0.0", port=8000)
@mcp.tool()
async def get_current_weather(location: str) -> dict: ...
@mcp.tool()
async def get_forecast(location: str, days: int = 3) -> dict: ...
Currency Exchange :8001
mcp = FastMCP("Currency Exchange", host="0.0.0.0", port=8001)
@mcp.tool()
async def get_currency_rates(date: str = None) -> dict: ...
@mcp.tool()
async def convert_currency(amount: float, from_currency: str, to_currency: str, date: str = None) -> dict: ...
