# protools-mcp
**A natural language interface for Avid Pro Tools, powered by Claude and the Model Context Protocol.**
protools-mcp is a local [MCP (Model Context Protocol)](https://modelcontextprotocol.io) server that connects Claude β or any MCP-compatible AI assistant β to a live Pro Tools session via the PTSL (Pro Tools Scripting Library) API. Instead of navigating menus or writing scripts, you describe what you need in plain language and Claude handles it directly in your session.
Built for podcast post-production workflows, it works equally well across music production, broadcast, and audio post.
---
## What It Looks Like in Practice
Once connected, you can have conversations like:
> *"What's in this session?"*
> *"Search the transcript for everywhere the guest mentions climate change."*
> *"Set a marker at 00:14:32:00 called 'Act Two'."*
> *"Mute all the music tracks and solo the host."*
> *"What clips are on the timeline between 22 and 35 minutes?"*
> *"Save a new version of this session called EP47_mix_v2."*
> *"Export the dialogue tracks as an AAF to the delivery folder."*
Claude reads your live session state, answers questions about your timeline, and executes write operations directly in Pro Tools β no scripting, no keyboard shortcuts, no menu diving.
---
## Capabilities
25+ tools across 7 functional groups:
| Group | Tools | Description |
|-------|-------|-------------|
| **Session** | `get_session_info`, `get_markers`, `get_track_list`, `get_session_snapshot`, `get_show_profile` | Session metadata, tracks, markers, show profiles |
| **Tracks** | `get_track_edl`, `get_track_playlists`, `get_clips_in_range` | Clip-level detail, playlists, time-range queries |
| **Transcript** | `get_transcript`, `search_transcript`, `get_transcript_for_range` | Speech-to-text CSV search with context and speaker labels |
| **Navigation** | `get_playhead_position`, `get_current_selection`, `set_playhead` | Playhead and selection state |
| **Edit** | `select_region`, `create_marker`, `mute_track`, `unmute_track`, `solo_track`, `consolidate_clip` | Session modifications (Claude confirms before executing) |
| **Session Mgmt** | `save_session`, `close_session`, `open_session`, `save_session_as` | Save, close, open, and version sessions |
| **Export** | `export_tracks_as_aaf` | AAF export with configurable format, bit depth, and copy option |
An optional **Show Profile** system lets you define per-show configuration β host names, track layouts, naming conventions β so Claude has the context it needs to work intelligently across multiple shows.
---
## Prerequisites
- **macOS** with Pro Tools running (PTSL listens on `localhost:31416`)
- **Python 3.11+** (tested with 3.11)
- **py-ptsl** installed system-wide or in a virtual environment
- **Claude Desktop** or **Claude Code** (for MCP integration)
- **Accessibility permission** for Claude/terminal (required for AAF export dialog automation)
---
## Setup
1. **Clone this repository:**
```bash
git clone https://github.com/BlueElevatorProductions/protools-mcp.git
cd protools-mcp
```
2. **Create a virtual environment:**
```bash
python3 -m venv venv --system-site-packages
source venv/bin/activate
pip install -r requirements.txt --no-cache-dir
```
The `--system-site-packages` flag reuses your system-wide `py-ptsl` and `grpcio` installs.
3. **Configure `.env`** (optional β defaults shown):
```
PTSL_HOST=localhost
PTSL_PORT=31416
```
4. **Add show profiles** (optional):
Place JSON files in `show_profiles/`. See `show_profiles/holy_uncertain.json` for the format.
5. **Register with Claude Desktop** β add to `~/Library/Application Support/Claude/claude_desktop_config.json`:
```json
{
"mcpServers": {
"protools-mcp": {
"command": "/path/to/protools-mcp/venv/bin/python",
"args": ["/path/to/protools-mcp/server.py"]
}
}
}
```
This makes the server available in Claude Desktop Chat, Cowork, and Code sessions.
For **Claude Code CLI only**, use:
```bash
claude mcp add protools-mcp -s user -- /path/to/protools-mcp/venv/bin/python /path/to/protools-mcp/server.py
```
6. **Grant Accessibility access** (required for AAF export automation):
System Settings > Privacy & Security > Accessibility β enable Claude Desktop and/or your terminal app.
7. **Open Pro Tools** with a session loaded. The MCP server connects lazily on first tool call.
---
## Tool Reference
### Session Context (read-only)
- **`get_session_snapshot()`** β Composite: session info + markers + tracks + auto-matched show profile. **Start here.** Best tool to call at the beginning of any session conversation.
- **`get_session_info()`** β Session name, path, sample rate, bit depth, timecode format, track count, audio file count.
- **`get_markers()`** β All memory location markers with index, name, timecode, and comment.
- **`get_track_list(filter="all")`** β Tracks with active, muted, soloed, and hidden state. Filter: `all`, `active`, `audio`, `inactive`.
- **`get_show_profile(show_id?)`** β Returns show profile config. Auto-infers from session name prefix if no ID given.
### Track Detail (read-only)
- **`get_track_edl(track_name)`** β Full clip list for a track: clip name, start/end timecodes, duration, state.
- **`get_track_playlists(track_name)`** β All playlists on a track, including inactive alternates.
- **`get_clips_in_range(start_timecode, end_timecode, track_filter?)`** β All clips across tracks within a timecode range.
### Transcript (read-only)
- **`get_transcript()`** β Full transcript from a Pro Tools Speech-to-Text CSV export.
- **`search_transcript(query, track_filter?, start_timecode?, end_timecode?)`** β Keyword search with 2-row context window.
- **`get_transcript_for_range(start_timecode, end_timecode)`** β Transcript rows in a time range, formatted as `SPEAKER: text` dialogue.
### Navigation
- **`get_playhead_position()`** β Current playhead timecode.
- **`get_current_selection()`** β Start, end, duration, and selected track names.
- **`set_playhead(timecode)`** β Moves the playhead to a specified timecode.
### Edit Operations (write)
All write tools are labeled `[WRITE]`. Claude will describe the operation and confirm before executing.
- **`select_region(start_timecode, end_timecode, track_names?)`** β Sets timeline selection. Non-destructive.
- **`create_marker(name, timecode, comment?)`** β Adds a memory location marker at the specified timecode.
- **`mute_track(track_name)`** / **`unmute_track(track_name)`** β Toggles track mute state.
- **`solo_track(track_name)`** β Solos a track.
- **`consolidate_clip(track_name, start_timecode, end_timecode)`** β Consolidates a region into a single clip. **Creates a new audio file on disk.**
### Session Management (write)
- **`save_session()`** β Saves the current session to disk.
- **`save_session_as(session_name, session_location)`** β Saves with a new name. `session_name` is the filename without extension; `session_location` is the target directory.
- **`close_session(save_before_close=True)`** β Closes the session, optionally saving first.
- **`open_session(session_path)`** β Opens a `.ptx` or `.ptf` session file.
### Export (write)
- **`export_tracks_as_aaf(...)`** β Exports selected tracks as an AAF. Handles the Pro Tools folder dialog automatically via osascript.
- `audio_format`: `WAV` (default), `AIFF`, `MXF`, `Embedded`
- `bit_depth`: `24` (default), `16`
- `copy_option`: `copy` (default), `consolidate`, `link`
- `quantize_to_frame`: `true` (default)
- `avid_compatible`: `false` (default) β enforce Media Composer compatibility
- `stereo_as_multichannel`: `false` (default)
- `sequence_name`: defaults to `file_name`
---
## Transcript Support
The transcript tools expect a Pro Tools Speech-to-Text CSV export alongside the session. Place the CSV in your session directory (or set `transcript_export_path` in your show profile) and the server will discover it automatically. It reloads on file modification, so the data stays current as you iterate on transcripts.
---
## Show Profile Format
Show profiles let Claude understand the structure of a specific show β which tracks belong to which speakers, naming conventions, and where exports live. Place JSON files in `show_profiles/`. Profiles are auto-matched by session name prefix.
```json
{
"show_id": "HU",
"show_name": "Holy Uncertain",
"session_name_prefix": "HU-",
"hosts": ["Chris", "Lauren"],
"dialogue_tracks": ["Chris", "Lauren Int R", "Chris Int R"],
"guest_tracks": ["Randy Int R"],
"music_tracks": ["Music"],
"transcript_export_path": "/path/to/episodes/",
"naming_conventions": {
"session": "HU-{episode_number}-{guest_last_name}-V{version}",
"export": "HU-{episode_number}-{guest_last_name}-MIX-V{version}"
}
}
```
---
## Architecture
```
Claude Desktop ββstdioβββΆ server.py (FastMCP)
β
βββββββββββββββΌββββββββββββββ
βΌ βΌ βΌ
PTSLBridge Transcript ShowProfile
(gRPC) Watcher Loader
β β β
βΌ βΌ βΌ
Pro Tools CSV files JSON files
:31416
```
- **PTSLBridge** β Lazy gRPC connection with auto-reconnect. The `@ptsl_command` decorator handles errors uniformly. Custom `Operation` subclasses cover PTSL commands not in py-ptsl's ops module.
- **TranscriptWatcher** β Stat-based CSV cache. Reloads only when the file's `mtime` changes. Auto-discovers CSV by searching the session directory.
- **ShowProfileLoader** β Reads `show_profiles/*.json` once at startup, matches sessions by name prefix.
- **osascript integration** β For PTSL commands that trigger Pro Tools dialogs (e.g., AAF export), the bridge runs the command in a background thread and uses System Events to dismiss the dialog automatically. Requires Accessibility permission.
---
## Error Handling
All PTSL errors return structured dicts before being raised as `ToolError`:
| Error Key | Meaning |
|-----------|---------|
| `ptsl_unavailable` | Pro Tools not running or gRPC connection lost |
| `no_session` | No session is open in Pro Tools |
| `ptsl_command_error` | PTSL command failed (details in message) |
| `no_transcript` | No transcript CSV found or configured |
| `dialog_waiting` | AAF export dialog needs manual confirmation (Accessibility not granted) |
---
## Implementation Notes
- **Timecode format**: Pro Tools uses `HH:MM:SS:FF`. Markers return raw sample positions internally; the bridge converts using `samples_to_timecode(samples, sample_rate, fps)`.
- **Track `active` field**: Derived from `is_inactive == TAState_None` on `TrackAttributes`. Distinct from muted/hidden.
- **EDL text**: Parsed from Pro Tools' tab-delimited text export with columns: `CHANNEL`, `EVENT`, `CLIP NAME`, `START TIME`, `END TIME`, `DURATION`, `STATE`.
- **Pro Tools quirks**: `SaveSessionAs` and directory paths require a trailing `/`. Some commands (`GetTrackPlaylists`, `GetPlaylistElements`) need `CId_`-prefixed command IDs. Empty `track_id` fields must be stripped from JSON to avoid "only one of track_id/track_name" errors.
- **Connection management**: gRPC connections can go stale between calls. The `@ptsl_command` decorator catches `grpc.RpcError` and resets the connection automatically.
---
## Troubleshooting
- **"Pro Tools is not running"** β Make sure Pro Tools is open with a session loaded. PTSL listens on port `31416`.
- **Transcript not found** β Set `transcript_export_path` in your show profile, or place the CSV next to the session file.
- **Stale data** β EDL cache expires after 30 seconds. Transcripts reload on file modification. Call tools again for fresh data.
- **AAF export hangs** β Grant Accessibility access in System Settings > Privacy & Security > Accessibility for the app running the MCP server.
- **"only one of track_id and track_name"** β Handled internally by `json_messup()` overrides on custom Operations.
---
## Contributing
Issues and pull requests welcome. If you're using this in a specific workflow and run into edge cases, open an issue β Pro Tools has many quirks and real-world sessions surface them fast.
---
## License
MIT
---
*Built by [Blue Elevator Productions](https://blueelevatorproductions.com)*