LiveMCP turns Ableton Live into an MCP-accessible control surface.
It gives an MCP client two useful layers:
- tools for doing things inside Live
live://...resources for reading stable state without stitching together a bunch of calls
The point is control and inspection. LiveMCP is built for tasks like selecting tracks, focusing views, firing clips, changing device parameters, checking transport state, handling startup dialogs, and keeping the remote script in sync with the package. It is not trying to be an AI songwriter.
- A Python MCP server that talks to Ableton over a local TCP bridge.
- A bundled Ableton MIDI Remote Script that exposes the Live object model.
- A controller-first interface for transport, views, tracks, clips, devices, mixer state, and session diagnostics.
- Not a generative music framework.
- Not a replacement for Ableton's UI.
- Not a promise that every Live feature is scriptable; some parts of the Ableton API are still read-only or missing entirely.
220tools across session, clips, tracks, devices, mixer, arrangement, grooves, docs, and Max patcher control- controller-oriented MCP resources like
live://status,live://view/current,live://track/{track_index}, andmax://selected-device - local docs sync and offline search across official Ableton and Cycling '74 documentation
- packaged install-status and Ableton restart helpers
- install support for macOS, Windows, and WSL
- macOS-only lifecycle automation for restart / quit / prompt handling
- "Read
live://selection/trackand tell me what's armed." - "Show Session view, select track 3, set monitoring to Auto, and arm it."
- "Read
live://statusbefore touching Ableton so we know whether the installed script is stale." - "Focus the Browser, load Operator onto the selected track, and give me the current parameter values."
- "Restart Live, dismiss the crash-recovery junk if it appears, and wait until the socket comes back."
uvx livemcp --installThat finds Ableton, copies the bundled LiveMCP remote script into the right MIDI Remote Scripts folder, and leaves the package side ready to run.
Install works on macOS, Windows, and WSL. The TCP bridge works across WSL2 → Windows via localhost forwarding, so you can run the MCP server in WSL while Ableton runs on the Windows host. The app lifecycle helpers later in this README are macOS-only.
For local development on macOS, use a symlinked install instead:
uv run livemcp --install --symlink-install
# or: bash scripts/dev_install.sh- Open Ableton Live 12
- Go to Preferences > Link, Tempo & MIDI
- Under Control Surface, select LiveMCP
- Status bar shows:
LiveMCP: Server started on port 9877
Claude Code (~/.mcp.json) or Claude Desktop (claude_desktop_config.json):
{
"mcpServers": {
"LiveMCP": {
"command": "uvx",
"args": ["livemcp"]
}
}
}Once the server is running, the most useful starting points are:
- read
live://status - run
uv run livemcp --validation-readiness - read
live://session/current - read
live://view/current - read
max://statusbefore using Max for Live patcher tools - read
docs://status - call
get_session_info - call
show_view("Session")orfocus_view("Browser") - call
search_docs("warp markers live.object")after syncing docs
If you want a local searchable snapshot of the docs you actually build against, sync them once:
uv run livemcp --sync-docsThat crawls the configured official sources, stores raw HTML under ~/.cache/livemcp/docs/raw/,
and builds a local SQLite full-text index at ~/.cache/livemcp/docs/docs.sqlite3.
Current sources:
ableton-live-manual-12cycling74-max-docs
You can sync just one source:
uv run livemcp --sync-docs --docs-source ableton-live-manual-12Check local docs status:
uv run livemcp --docs-statusAfter syncing, use:
search_docs(query, source_id="all", limit=8)get_docs_chunk(chunk_id)get_docs_page(page_id)docs://statusdocs://chunk/{chunk_id}docs://page/{page_id}
Use resources when you want a clean snapshot of current state:
live://statuslive://selection/tracklive://view/currentlive://track/0
Use tools when you want to change something:
start_playbackset_track_volumeselect_deviceshow_viewpress_current_dialog_button
That split matters. Tools are for actions. Resources are for inspection.
For plugin QA sessions, the quickest local readiness check is:
uv run livemcp --validation-readinessThat reports remote-script reachability, the currently selected track/device, and whether the Max bridge is attached for patcher inspection.
For screenshot QA, use the stricter target check:
uv run livemcp --confirm-validation-target --track-name "PEQ V2" --device-name "Parametric EQ V2"That confirms the intended comparison device is actually selected before you trust a screenshot.
For downstream generated-device QA (for example, validating m4l-builder output), keep the loop minimal: run --validation-readiness, confirm the exact target with --confirm-validation-target, then capture screenshots/measurements only after those checks pass.
Use the packaged helper when Live gets stuck on save, crash-recovery, or restore prompts:
uv run livemcp --restart-ableton
uv run livemcp --launch-ableton
uv run livemcp --quit-ableton--restart-ableton clears remote_script/__pycache__, removes Ableton crash-recovery markers
from ~/Library/Preferences/Ableton/Live */, quits Live, relaunches the detected Ableton app,
accepts common crash/restore dialogs when macOS Accessibility is available, and waits for the
LiveMCP socket on port 9877.
If the installed Ableton-side copy is missing or out of sync, restart also repairs it before relaunching Live.
Config file locations:
| OS | Path |
|---|---|
| macOS | ~/Library/Application Support/Claude/claude_desktop_config.json |
| Windows | %APPDATA%\Claude\claude_desktop_config.json |
| Linux | ~/.config/Claude/claude_desktop_config.json |
You do not need to memorize 220 commands to use LiveMCP well. The useful mental model is:
- session tools for transport, selection, views, dialogs, and global state
- track and mixer tools for routing, arm/solo/mute, levels, and return paths
- device tools for loading instruments/effects and changing parameters
- clip and arrangement tools for manipulating content that already exists in the set
- docs tools for searching local snapshots of official Ableton and Max documentation
- max tools for inspecting and editing supported Max for Live patchers through a local bridge
- resources when you want a stable read surface for the current state
The tables below are the full reference.
| Category | Count |
|---|---|
| Session | 76 |
| Clips | 40 |
| Tracks | 32 |
| Devices | 27 |
| Mixer | 14 |
| Arrangement | 9 |
| Grooves | 5 |
| Docs | 4 |
| Max | 13 |
| Total | 220 |
These are meant for controller-style reads where a client wants stable Ableton state without chaining multiple tools.
| Resource | Description |
|---|---|
live://status |
Combined local install state, remote reachability, and transport warnings |
live://session/current |
Global transport and timing state |
live://song/time |
Current playhead position and undo/redo state |
live://view/current |
Current UI view state, visible views, follow-song, and draw mode |
live://selection/track |
Currently selected track |
live://selection/scene |
Currently selected scene |
live://selection/device |
Currently selected device |
live://application/dialog |
Current Ableton dialog state |
max://status |
Local Max bridge reachability and Max-specific warnings |
max://selected-device |
Selected Live device with Max for Live bridge metadata |
max://patcher/current |
Current attached Max patcher session summary |
docs://status |
Local docs index status and synced source coverage |
| Resource | Description |
|---|---|
live://track/{track_index} |
Detailed track state |
live://scene/{scene_index} |
Detailed scene state |
live://scene/{scene_index}/clips |
Scene clip-slot state across all tracks |
live://device/{track_index}/{device_index} |
Device parameters with display values |
docs://chunk/{chunk_id} |
One indexed docs chunk from the local snapshot |
docs://page/{page_id} |
One full docs page from the local snapshot |
Session Tools (76)
| Tool | Description |
|---|---|
get_session_info |
Tempo, time signature, track count, transport state |
get_song_time / set_song_time |
Current playhead position |
get_song_smpte_time |
SMPTE timecode (hours, minutes, seconds, frames) |
set_tempo |
Set BPM |
start_playback / stop_playback / continue_playing |
Transport controls |
stop_all_clips |
Stop all playing clips |
trigger_record |
Toggle session recording |
get_record_mode / set_record_mode |
Arrangement recording on/off |
get_link_state / set_ableton_link_enabled / set_ableton_link_start_stop_sync_enabled |
Ableton Link and Start/Stop Sync |
set_tempo_follower_enabled |
Tempo Follower on/off |
get_count_in_state |
Count-in duration and active count-in state |
get_session_record_status / set_session_record |
Session Record state and status |
set_time_signature |
Numerator and denominator |
set_loop_region |
Loop on/off, start, length |
undo / redo |
Undo/redo |
tap_tempo |
Tap tempo |
set_metronome |
Metronome on/off |
set_midi_recording_quantization |
Quantization during recording |
capture_midi |
Capture recently played MIDI |
capture_and_insert_scene |
Capture playing clips into a new scene |
get_cue_points |
List all locators/cue points |
create_locator / delete_locator |
Locator CRUD |
jump_to_cue / jump_to_next_cue / jump_to_prev_cue |
Navigate cue points |
get_selected_track / set_selected_track |
Track selection |
get_selected_scene / set_selected_scene |
Scene selection |
get_scene_properties |
Scene tempo and time signature |
get_scene_info |
Scene name, color, is_empty, clip count |
get_scene_clips |
All clips in a scene across all tracks |
set_scene_tempo / set_scene_time_signature |
Per-scene tempo and time sig |
set_scene_name / set_scene_color |
Scene appearance |
fire_scene / duplicate_scene / delete_scene / create_scene |
Scene management |
set_groove_amount / set_scale |
Global groove and scale |
get_application_info |
Live version (major, minor, bugfix) |
get_livemcp_info |
Remote-script transport protocol version and capability flags |
get_livemcp_status |
Combined local install state, remote reachability, and transport warnings |
get_application_dialog / press_current_dialog_button |
Inspect and control Ableton dialog boxes |
get_application_cpu_usage |
Average and peak Live CPU usage |
get_available_main_views / is_view_visible |
Discover valid view names and visibility |
show_view / hide_view / focus_view |
Control Ableton UI views |
toggle_browse |
Toggle browser hot-swap mode |
get_session_metadata |
Song time, length, CPU load |
get_view_state |
Current view, follow song, draw mode, visible views |
get_selected_device / select_device |
Read and change selected device |
get_selected_parameter |
Inspect the selected parameter |
get_selected_chain |
Inspect the selected rack chain |
set_follow_song / set_draw_mode |
Toggle view options |
select_clip_in_detail |
Open clip in Detail View |
get_punch_state / set_punch_in / set_punch_out |
Punch in/out |
re_enable_automation |
Re-enable overridden automation |
get_session_automation_record / set_session_automation_record |
Automation arm |
show_message |
Display text in Ableton's status bar |
Clip Tools (40)
| Tool | Description |
|---|---|
create_clip / create_session_audio_clip / delete_clip / duplicate_clip |
Clip CRUD |
fire_clip / stop_clip |
Launch and stop clips |
get_clip_properties |
Full clip info (type, length, loop, markers) |
set_clip_properties |
Batch set name, color, mute, gain, pitch, loop points |
set_clip_name / set_clip_color / set_clip_muted |
Individual properties |
set_clip_loop |
Loop on/off, start, end |
set_clip_gain / set_clip_pitch |
Audio gain and pitch shift |
set_clip_launch_mode / set_clip_trigger_quantization |
Launch behavior |
set_clip_warp_mode / set_clip_warping |
Warping control |
set_clip_ram_mode / set_clip_velocity_amount |
RAM mode and velocity sensitivity |
get_clip_playing_position |
Current playhead within clip |
get_clip_fades / set_clip_fades |
Audio clip fade in/out lengths |
duplicate_clip_loop / crop_clip |
Loop duplication and cropping |
quantize_clip |
Quantize clip contents |
add_notes_to_clip / get_notes_from_clip / remove_notes_from_clip / clear_clip_notes |
Basic MIDI note editing |
get_notes_extended |
Notes with probability, velocity deviation, release velocity |
add_notes_extended |
Add notes with full MidiNoteSpecification params |
modify_notes |
Modify existing notes in place |
replace_all_notes |
Atomic note replacement (clear + add in one call) |
remove_notes_extended |
Remove notes by pitch/time range |
get_clip_envelope / insert_clip_envelope_step / clear_clip_envelope / clear_all_clip_envelopes |
Automation envelopes |
Track Tools (32)
| Tool | Description |
|---|---|
get_track_info / get_all_tracks_info |
Track details |
create_midi_track / create_audio_track / create_return_track |
Create tracks |
delete_track / delete_return_track / duplicate_track |
Track management |
set_track_name / set_track_color |
Appearance |
set_track_properties |
Batch set name, volume, pan, mute, solo, arm, color |
get_track_routing / set_track_input_routing / set_track_output_routing |
I/O routing |
set_track_monitoring |
Monitor modes (auto, in, off) |
get_group_info |
Group track children |
fold_track |
Fold/unfold group tracks |
get_return_tracks / get_return_track_sends / set_return_track_send |
Return tracks |
get_track_freeze_status |
Check if track is frozen |
get_track_output_meter |
Real-time output level |
get_clip_slot_status |
Playing, recording, triggered state |
set_clip_slot_color |
Clip slot color |
get_take_lanes / create_take_lane |
Take-lane inspection and creation |
create_take_lane_midi_clip / create_take_lane_audio_clip |
Create clips inside take lanes |
Device Tools (27)
| Tool | Description |
|---|---|
get_browser_tree / get_browser_items_at_path |
Browse Ableton's content library |
load_instrument_or_effect |
Load any instrument or effect by URI, path, or name |
load_drum_kit |
Load drum rack presets |
load_device_on_master / load_device_on_return |
Load to master/return tracks |
get_device_parameters / set_device_parameter |
Raw parameter values (0.0–1.0) |
get_device_display_values |
Human-readable values ("2500 Hz", "-12 dB", "On") |
get_master_device_parameters / set_master_device_parameter |
Master track devices |
get_return_device_parameters / set_return_device_parameter |
Return track devices |
get_master_track_devices / get_return_track_devices |
List devices |
get_rack_chains / set_chain_mixer_value |
Rack chain contents, activator, pan, volume, sends |
get_drum_chains / set_drum_chain_property |
Drum-chain note routing and choke settings |
get_drum_pads / set_drum_pad_mute / set_drum_pad_solo |
Drum rack pads |
delete_device / delete_master_device / delete_return_device |
Remove devices |
move_device / enable_device |
Reorder and enable/disable |
Mixer Tools (14)
| Tool | Description |
|---|---|
get_mixer_state |
All tracks: volume, pan, mute, solo, arm, sends |
set_track_volume / set_track_pan |
Level and panning |
set_track_mute / set_track_solo / set_track_arm |
Mute, solo, arm |
set_track_send |
Send levels |
get_master_mixer_state / set_master_volume / set_master_pan |
Master track |
set_crossfade_assign |
Crossfader assignment (A/B/none) |
get_track_output_meter |
Per-track output metering |
get_master_output_meter |
Master output level (L/R) |
get_return_track_output_meter |
Return track metering |
get_all_track_meters |
All meters in one call |
Arrangement Tools (9)
| Tool | Description |
|---|---|
get_arrangement_clips |
All clips on a track's timeline |
get_arrangement_length |
Total arrangement length |
create_arrangement_midi_clip |
Create MIDI clip at position |
create_arrangement_audio_clip |
Place audio file on timeline |
duplicate_to_arrangement |
Copy session clip to arrangement |
delete_arrangement_clip |
Remove arrangement clip |
get_arrangement_overdub / set_arrangement_overdub |
Overdub toggle |
trigger_back_to_arrangement |
Return to arrangement playback |
Groove Tools (5)
| Tool | Description |
|---|---|
get_groove_pool |
List all grooves in the pool |
get_groove_properties |
Timing, random, velocity amounts |
set_groove_property |
Modify groove parameters |
set_clip_groove |
Assign groove to clip |
remove_clip_groove |
Clear groove assignment |
Max Tools (13)
| Tool | Description |
|---|---|
get_selected_max_device |
Describe the selected device and Max bridge attachment state |
open_selected_device_in_max |
Open the selected Max device in the native Max editor |
get_current_patcher |
Summary of the current attached patcher session |
list_patcher_boxes |
List all patcher boxes in the current session |
get_box_attrs / set_box_attrs |
Inspect or change allowlisted object and box attributes |
create_box / delete_box |
Create or remove a patcher box |
create_patchline / delete_patchline |
Connect or disconnect patcher boxes |
set_presentation_rect |
Update a box's presentation rect |
toggle_presentation_mode |
Toggle or force presentation mode |
save_max_device |
Save the attached Max device in place |
V1 Max bridge note: the currently selected Max device must be bridge-enabled and
host the local bridge runtime itself. The packaged LiveMCP Bridge Probe.amxd
in the User Library Max Audio Effect/_Debug folder does this today.
Non-bridge-enabled third-party M4L devices remain explicit unsupported cases
rather than falling back to GUI scripting.
┌─────────────────────────────────────────────────────────┐
│ MCP Client │
│ (Claude, Codex, etc.) │
└────────────────────────┬────────────────────────────────┘
│ MCP Protocol (stdio)
┌────────────────────────▼────────────────────────────────┐
│ Package-Side MCP Server │
│ src/livemcp/ (FastMCP) │
│ │
│ 220 tools + live://, max://, and docs:// resources │
│ structured controller state + action calls │
└────────────────────────┬────────────────────────────────┘
│ TCP Socket (localhost:9877)
┌────────────────────────▼────────────────────────────────┐
│ Ableton Remote Script │
│ remote_script/LiveMCP/ (Python 3.11) │
│ │
│ Live/browser commands: read + main-thread write split │
│ Max bridge commands: socket-thread proxying │
│ browser loading + Live API access │
└────────────────────────┬────────────────────────────────┘
│ TCP Socket (localhost:9881) for Max bridge
┌────────────────────────▼────────────────────────────────┐
│ Selected-Device Max Bridge Runtime │
│ node.script TCP server + js patcher runtime inside │
│ the bridge-enabled Max for Live device │
└────────────────────────┬────────────────────────────────┘
│ Live Object Model + Max patcher API
┌────────────────────────▼────────────────────────────────┐
│ Ableton Live 12 │
└─────────────────────────────────────────────────────────┘
Three design choices matter most:
- Thread safety — reads happen on the socket thread; writes are scheduled onto Ableton's main thread so Live does not get mutated from the wrong place.
- Max bridge dispatch — Max bridge commands stay on the socket thread because the actual patcher mutations happen inside the Max runtime; bouncing those calls through Ableton's main thread deadlocks editor/mutation requests.
- Single-source registration — tool modules and handler registries fail fast on duplicate names instead of silently shadowing commands.
- Controller-first state model — tools handle actions, and resources expose the current state a client usually wants to inspect before acting.
- Native Max for Live support proposal: docs/native-max-for-live-support.md
# Clone and set up
git clone https://github.com/alaarab/livemcp.git
cd livemcp
# Install remote script into Ableton
uv run livemcp --install
# or: bash scripts/install.sh
# See whether Ableton is using the current remote-script copy
uv run livemcp --install-status
# Run the MCP server locally
uv run livemcp
# Restart Ableton and wait for LiveMCP to come back
uv run livemcp --restart-ableton
# Build, test, and publish the current version
bash scripts/publish.sh --dry-run
# Verify tool count
uv run python -c "from livemcp.server import mcp; print(len(mcp._tool_manager._tools), 'tools')"For reproducible local verification from a fresh checkout:
# Install lint/test tooling declared in pyproject.toml
uv sync --group dev
# Unit tests
uv run python -m unittest discover -s tests -v
# Lint
uv run ruff check src tests
# Tool registration sanity check (expected: 220)
uv run python -c "from livemcp.server import mcp; print(len(mcp._tool_manager._tools), 'tools')"For the local dev loop on macOS, the useful rhythm is:
uv run livemcp --install --symlink-install- edit package or remote-script code
uv run livemcp --restart-ableton- test against the running Live instance
bash scripts/publish.sh --dry-runbefore cutting a release
livemcp/
├── src/livemcp/ # MCP server (pip/uvx installable)
│ ├── server.py # FastMCP app, registers all tool modules
│ ├── connection.py # TCP client to remote script
│ ├── resources.py # live:// MCP resources for controller state
│ └── tools/ # 9 tool modules
│ ├── session.py # 76 session tools
│ ├── clips.py # 40 clip tools
│ ├── tracks.py # 32 track tools
│ ├── devices.py # 27 device tools
│ ├── mixer.py # 14 mixer tools
│ ├── arrangement.py # 9 arrangement tools
│ ├── grooves.py # 5 groove tools
│ ├── docs.py # 4 docs tools
│ └── max.py # 13 Max patcher tools
├── remote_script/LiveMCP/ # Ableton MIDI Remote Script
│ ├── __init__.py # create_instance() entry point
│ ├── server.py # TCP server + handler dispatch
│ ├── browser.py # 3-strategy device loading
│ └── handlers/ # 8 handler modules (Live + Max; docs tools are package-local)
└── scripts/
├── install.sh # Wrapper for `uv run livemcp --install`
├── dev_install.sh # Wrapper for `uv run livemcp --install --symlink-install`
├── publish.sh # Test/build/publish helper
├── uninstall.sh # Wrapper for `uv run livemcp --uninstall`
└── restart_ableton.sh # Wrapper for `uv run livemcp --restart-ableton`
Most of these are Ableton Live API limitations, not LiveMCP bugs:
| Feature | Status | Notes |
|---|---|---|
| Follow Actions | Not in API | Confirmed by Cycling '74 |
| Stem Separation | Not in API | UI-only feature (Live 12.3+) |
| Track Freeze/Flatten | Read-only | Can check is_frozen, cannot trigger freeze |
| Warp Markers | Read-only | Cannot add/move/delete warp markers |
| Automation Envelopes | Limited | Can sample/insert values, cannot create from scratch |
| Group Tracks | Cannot create | Can read group info, cannot create groups |
| Groove Removal | API bug | clip.groove = None crashes; raises informative error |
MIT