loading…
Search for a command to run...
loading…
An MCP server that lets AI agents launch, interact with, and inspect Textual TUI applications headlessly.
An MCP server that lets AI agents launch, interact with, and inspect Textual TUI applications headlessly.
An MCP server that lets AI agents launch, interact with, and inspect Textual TUI applications headlessly. Drive any Textual app through its full lifecycle — click buttons, type text, read widget state, take screenshots — all via the Model Context Protocol.
Pilot testing APIRequires Python 3.10+.
pip install textual-mcp-server
For development:
git clone https://github.com/discohead/textual-mcp-server.git
cd textual-mcp-server
pip install -e ".[dev]"
textual-mcp
claude mcp add textual -- textual-mcp
Or add manually to your MCP configuration (e.g., ~/.claude.json or project .mcp.json):
{
"mcpServers": {
"textual": {
"command": "textual-mcp"
}
}
}
1. textual_launch("my_app.py") → session_id
2. textual_snapshot(session_id) → widget tree + focus + bindings
3. textual_click(session_id, "#submit") → interact
4. textual_screenshot(session_id) → visual output
5. textual_stop(session_id) → cleanup
| Tool | Description |
|---|---|
textual_launch |
Launch a Textual app headlessly. Accepts a file path (app.py), path with class (app.py:MyApp), or module path (mypackage.module:MyApp). Returns a session_id. |
textual_stop |
Stop a running session and return any collected errors. |
| Tool | Description |
|---|---|
textual_press |
Simulate key presses (e.g., ["enter"], ["ctrl+s"]). |
textual_click |
Click a widget by CSS selector with optional offset and repeat count. |
textual_type_text |
Type text into the focused input widget, with optional submit (Enter). |
textual_hover |
Hover the mouse over a widget by CSS selector. |
| Tool | Description |
|---|---|
textual_snapshot |
Snapshot the widget tree with ref markers, focus state, active key bindings, and errors. |
textual_screenshot |
Capture the current screen as plain text or SVG. |
textual_query |
Query widgets matching a CSS selector. Returns type, ID, classes, and extracted properties. |
textual_get_screen_stack |
Get the current screen stack with modal indicators. |
| Tool | Description |
|---|---|
textual_wait_for |
Wait for a condition: idle, animation, workers (all complete), or widget (selector appears). |
textual_check_errors |
Check for collected worker errors and app exceptions. |
textual_mcp/
├── server.py # FastMCP server, tool registration, and tool implementations
├── session.py # AppSession — headless app lifecycle via Pilot
├── session_manager.py # Multi-session management
├── app_loader.py # Dynamic app loading from file or module path
├── error_collector.py # Message hook for worker error aggregation
└── serializers/
├── widget_tree.py # DOM → indented text tree with [ref=N] markers
└── widget_state.py # Type-specific property extraction (16 widget types)
Key design decisions:
AppSession wraps Textual's App.run_test() to provide launch/stop semantics with a persistent Pilot handleWidgetTreeSerializer produces LLM-friendly text output — interactive widgets get [ref=N] markers; scrollbars and hidden widgets are excludedWidgetStateExtractor uses an ordered isinstance registry to extract properties from Input, Button, DataTable, TextArea, Tree, and 11 other widget typesErrorCollector hooks into Textual's message system to capture Worker.StateChanged errors without disrupting normal operationThe state extractor provides rich property data for:
Input, Button, Static, Label, Checkbox, Switch, Select, TextArea, DataTable, Tree, ListView, OptionList, TabbedContent, ProgressBar, RadioSet, ContentSwitcher
# Run tests
pytest
# Run a specific test
pytest tests/test_integration_calculator.py -v
MIT
Выполни в терминале:
claude mcp add textual-mcp-server -- npx Безопасность
Низкий рискАвтоматическая эвристика по публичным данным — не гарантия безопасности.