fcpxml-mcp-server
Provides tools for analyzing, editing, and generating Final Cut Pro XML files, enabling AI to manage timelines, markers, clips, and perform QC checks on Final Cut Pro projects.
Allows exporting Final Cut Pro timelines to DaVinci Resolve XML format for cross-NLE workflow.
Enables extracting chapter markers from Final Cut Pro timelines and formatting them as YouTube chapter timestamps for video descriptions.
FCPXML MCP
The bridge between Final Cut Pro and AI. 53 tools that turn timeline XML into structured data Claude can read, edit, and generate.
Why This Exists
After directing 350+ music videos (Chief Keef, Migos, Masicka), I noticed the same editing bottlenecks on every project: counting cuts manually, extracting chapter markers one by one, hunting flash frames by scrubbing, building rough cuts clip by clip.
These are batch operations that don't need visual feedback. Export the XML, let Claude handle the tedium, import the result. That's the entire philosophy.
See It In Action
You: "Run a health check on my wedding edit"
Claude: ✓ Analyzed WeddingFinal.fcpxml
├─ 247 clips · 42:18 total · 24fps · 1920×1080
├─ 3 flash frames detected (clips 44, 112, 198)
├─ 2 unintentional gaps at 12:04 and 31:47
├─ 14 duplicate source clips
└─ Health score: 72/100
You: "Fix the flash frames and gaps, then add chapter markers from
this transcript"
Claude: ✓ Extended adjacent clips to cover 3 flash frames
✓ Filled 2 gaps by extending previous clips
✓ Added 18 chapter markers from transcript
→ Saved: WeddingFinal_modified.fcpxmlImport the modified XML back into Final Cut Pro. Every change is non-destructive — your original file is never touched.
What Claude Actually Sees
This is the magic trick. When you export XML from Final Cut Pro, your timeline becomes structured data that Claude can reason about:
<!-- What FCP exports -->
<asset-clip ref="r2" offset="342/24s" name="Interview_A"
start="120s" duration="720/24s" format="r1">
<marker start="48/24s" duration="1/24s" value="Key quote"/>
<keyword start="0s" duration="720/24s" value="Interview"/>
</asset-clip># What Claude works with (after parsing)
Clip(
name="Interview_A",
offset=TimeValue(342, 24), # timeline position: 14.25s
start=TimeValue(120, 1), # source in-point: 2:00
duration=TimeValue(720, 24), # 30 seconds
markers=[Marker(value="Key quote", start=TimeValue(48, 24))],
keywords=["Interview"]
)Every time value stays as a rational fraction — 720/24s, not 30.0 — so trim, split, and speed operations have zero rounding error across any frame rate. Comparisons use cross-multiplication (a/b < c/d → a*d < c*b) to stay in integer-land end to end. Denominators are always normalized to positive values at construction, so sign lives on the numerator and cross-multiplication is always correct. Addition and subtraction share a single _binop() code path that handles same-denominator fast paths and LCM alignment in one place.
How It Works
┌──────────┐ ┌──────────────────────────────┐ ┌──────────┐
│ Final Cut│ │ parser.py → Python objects │ │ Final Cut│
│ Pro │─XML─>│ writer.py → Modify & save │─XML─>│ Pro │
│ │ │ rough_cut.py→ Generate new │ │ │
└──────────┘ │ diff.py → Compare │ └──────────┘
│ export.py → Resolve / FCP7 │
└──────────────────────────────┘
▲
Claude Desktop / MCP clientExport from FCP —
File → Export XML...Ask Claude — analyze, edit, generate, QC, export
Import back —
File → Import → XML
What This Is NOT
Not a plugin — it doesn't run inside Final Cut Pro
Not real-time — you work with the XML between exports
Not for creative calls — color, framing, motion still need your eyes
Quick Start
1. Clone & Install
git clone https://github.com/DareDev256/fcp-mcp-server.git
cd fcp-mcp-server
pip install -e .2. Configure Claude Desktop
Add to ~/Library/Application Support/Claude/claude_desktop_config.json:
Using uv (recommended):
{
"mcpServers": {
"fcpxml": {
"command": "uv",
"args": ["--directory", "/path/to/fcp-mcp-server", "run", "server.py"],
"env": { "FCP_PROJECTS_DIR": "/Users/you/Movies" }
}
}
}Using pip:
{
"mcpServers": {
"fcpxml": {
"command": "python",
"args": ["/path/to/fcp-mcp-server/server.py"],
"env": { "FCP_PROJECTS_DIR": "/Users/you/Movies" }
}
}
}3. Use It
Export XML from Final Cut Pro, open Claude Desktop, and ask it to work with your timeline.
When To Use This
Good For | Not Ideal For |
Batch marker insertion (100 chapters from a transcript) | Creative editing decisions (no visual feedback) |
QC before delivery (flash frames, gaps, duplicates) | Real-time adjustments (export/import cycle) |
Data extraction (EDL, CSV, chapter markers) | Fine-tuning cuts (faster directly in FCP) |
Template generation (rough cuts from tagged clips) | Anything visual (color, framing, motion) |
Automated assembly (montages from keywords + pacing) | |
Timeline health checks (validation, stats, scoring) |
Prompt Cookbook
Copy-paste these into Claude Desktop. Each one maps to a real tool chain under the hood.
Analysis
"Give me a full breakdown of ProjectX.fcpxml — clips, duration, frame rate, markers, everything"
"Show me pacing analysis for my timeline — where are the slow sections?"
"Export an EDL and CSV of all clips with timecodes"QC & Fixes
"Run a health check on my timeline and fix anything under 2 frames"
"Find all gaps and flash frames, then auto-fix them"
"Are there any duplicate source clips I can consolidate?"Markers & Chapters
"Add chapter markers from this transcript: [paste transcript]"
"Import markers from my-subtitles.srt onto the timeline"
"List all markers and export them as YouTube chapter timestamps"Generation
"Build a 60-second rough cut from clips tagged 'Interview' — medium pacing"
"Generate a montage from all B-roll clips with accelerating pacing"
"Create an A/B roll: Interview_A as primary, B-roll cuts every 8 seconds"Cross-NLE & Reformat
"Export this timeline for DaVinci Resolve"
"Convert to FCP7 XML so I can open it in Premiere"
"Reformat my 16:9 timeline to 9:16 for Instagram Reels"Under the Hood
When you say "Run a health check on my wedding edit", Claude chains these tools:
analyze_timeline → stats, frame rate, resolution
detect_flash_frames → clips under threshold duration
detect_gaps → unintentional silence/black
detect_duplicates → repeated source media
validate_timeline → structural health score (0-100)Each tool returns structured text that Claude synthesizes into the summary you see. No magic — just batch XML queries that would take 20 minutes by hand.
Pre-Built Prompts
Select these from Claude's prompt menu (⌘/) — they chain multiple tools automatically.
Prompt | What It Does |
qc-check | Full quality control — flash frames, gaps, duplicates, health score |
youtube-chapters | Extract chapter markers formatted for YouTube descriptions |
rough-cut | Guided rough cut — shows clips, suggests structure, generates |
timeline-summary | Quick overview — stats, pacing, keywords, markers, assessment |
cleanup | Find and auto-fix flash frames and gaps |
All 53 Tools
Category | Tools | What It Does |
Analysis | 11 | Stats, clips, markers, keywords, EDL/CSV, pacing |
Multi-Track | 3 | Connected clips, compound clips, secondary lanes |
Roles | 4 | List, assign, filter, export stems |
QC & Validation | 4 | Flash frames, duplicates, gaps, health score |
Editing | 9 | Markers, trim, reorder, transitions, speed, split |
Batch Fixes | 3 | Auto-fix flash frames, rapid trim, fill gaps |
Comparison | 1 | Diff two timelines — added/removed/moved/trimmed |
Reformat | 1 | Aspect ratio conversion (9:16, 1:1, 4:5, custom) |
Silence | 2 | Detect and remove silence candidates |
NLE Export | 2 | DaVinci Resolve v1.9, FCP7 XMEML v5 |
Generation | 3 | Rough cuts, montages, A/B roll |
Beat Sync | 2 | Import beat markers, snap cuts to beats |
Import | 2 | SRT/VTT subtitles, YouTube chapters → markers |
Audio | 1 | Add audio clips, music beds at any lane |
Compound | 2 | Create/flatten compound clips |
Templates | 2 | Pre-built timeline structures (intro/outro, lower thirds, music video) |
Effects | 1 | List FCP transition effects with UUIDs |
53 |
Analysis — 11 tools
list_projects · analyze_timeline · list_clips · list_library_clips · list_markers · find_short_cuts · find_long_clips · list_keywords · export_edl · export_csv · analyze_pacing
Multi-Track — 3 tools
list_connected_clips · add_connected_clip · list_compound_clips
Roles — 4 tools
list_roles · assign_role · filter_by_role · export_role_stems
QC & Validation — 4 tools
detect_flash_frames · detect_duplicates · detect_gaps · validate_timeline
Editing — 9 tools
add_marker · batch_add_markers · insert_clip · trim_clip · reorder_clips · add_transition · change_speed · delete_clips · split_clip
Batch Fixes — 3 tools
fix_flash_frames · rapid_trim · fill_gaps
Comparison · Reformat · Silence
diff_timelines · reformat_timeline · detect_silence_candidates · remove_silence_candidates
NLE Export — 2 tools
export_resolve_xml (DaVinci Resolve FCPXML v1.9) · export_fcp7_xml (Premiere Pro / Resolve / Avid XMEML v5)
Generation — 3 tools
auto_rough_cut · generate_montage · generate_ab_roll
Beat Sync — 2 tools
import_beat_markers · snap_to_beats
Import — 2 tools
import_srt_markers · import_transcript_markers (supports SMPTE HH:MM:SS:FF with frame-accurate placement)
v0.6.0 — Audio, Compound, Templates, Effects — 6 tools
list_effects · add_audio · create_compound_clip · flatten_compound_clip · list_templates · apply_template
Environment Variables
Variable | Required | Default | Description |
| No |
| Root directory for FCPXML file discovery via |
| No | — | Route LLM calls through any OpenAI-compatible proxy (LiteLLM, OpenRouter, Ollama, vLLM) |
Compatibility
Component | Supported Versions |
FCPXML format | v1.8 – v1.11 |
Final Cut Pro | 10.4+ |
Python | 3.10, 3.11, 3.12 |
MCP protocol | 1.0 |
Export targets | |
→ DaVinci Resolve | FCPXML v1.9 |
→ Premiere Pro / Avid | FCP7 XMEML v5 |
Architecture
fcp-mcp-server/ ~8.9k lines Python
├── server.py MCP entry point — 53 tools, 5 prompts, resource discovery
│ _resolve_io_paths() / _setup_modifier() / _setup_generator()
│ _format_clip_table() / _markdown_table() / _format_batch_result()
│ _raw_markers_to_batch()
│ _detect_flash_frames() / _detect_gaps() / _detect_duplicate_groups()
│ consolidate path validation, QC detection, rendering, handler boilerplate
├── fcpxml/
│ ├── README.md Developer guide — TimeValue, clip hierarchy, modifier patterns
│ ├── models.py TimeValue, Timecode, Clip, ConnectedClip, MarkerType, Timeline
│ ├── parser.py FCPXML → Python (spine, connected clips, roles, markers)
│ ├── writer.py Modify & write (markers, trim, gaps, transitions, silence)
│ │ FCPXMLModifier: index-based editing (clips/resources/formats dicts)
│ │ FCPXMLWriter: generate new FCPXML from Python objects
│ │ Helpers: _resolve_asset, _absorb_into_neighbor, _ripple_from_index
│ ├── rough_cut.py Generate timelines (rough cuts, montages, A/B roll)
│ ├── diff.py Timeline comparison engine (identity matching, threshold docs)
│ ├── export.py DaVinci Resolve v1.9 + FCP7 XMEML v5 export
│ ├── safe_xml.py Centralized defusedxml wrappers (XXE/entity-bomb protection) + serialize_xml()
│ └── templates.py Template system (intro/outro, lower thirds, music video)
├── tests/ 912 tests across 18 suites
│ ├── test_models.py TimeValue math, Timecode formatting, MarkerType contracts
│ ├── test_parser.py FCPXML parsing, connected clips, edge cases
│ ├── test_writer.py Clip editing, marker writing, speed changes
│ ├── test_fcpxml_writer.py FCPXMLWriter generation from Python objects
│ ├── test_server.py MCP tool handlers, dispatch, path validation
│ ├── test_rough_cut.py Rough cut generation, montage, A/B roll
│ ├── test_diff.py Moved clips, transitions, markers, clip identity
│ ├── test_export.py Attribute stripping, compound flattening, audio tracks
│ ├── test_features_v05.py Multi-track, roles, diff, reformat, export
│ ├── test_features_v06.py Audio, compound clips, templates, effects, validation
│ ├── test_marker_pipeline.py Marker builder, batch modes, output format
│ ├── test_speed_cutting.py Speed cutting, montage config, pacing curves
│ ├── test_security.py Input validation, XML sanitization, XXE protection
│ ├── test_edge_cases.py Boundary arithmetic, clip collisions, split/diff edges
│ ├── test_diversity.py Boundary conditions across diff, models, validation
│ ├── test_refactored_helpers.py _index_elements, _iter_spine_clips, serialize_xml edges
│ └── test_targeted_gaps.py Targeted branch coverage for diff, export, models
├── docs/
│ └── WORKFLOWS.md 8 production workflow recipes
└── examples/
└── sample.fcpxml 9 clips, 24fps — test fixtureSecurity
Every tool handler is hardened against adversarial input — critical for MCP servers where prompts may be LLM-generated, not human-typed.
Layer | Protection |
File I/O | Path traversal blocked, null bytes rejected, symlinks resolved, 100 MB size limit |
Output sandbox | All generation, write, export, beat sync, subtitle, and reformat handlers enforce |
Subprocess bounds |
|
Speed validation |
|
Directory listing | Confined to |
XML parsing |
|
JSON depth limit | Iterative BFS depth checker rejects payloads nested beyond 50 levels — immune to RecursionError even at ~1000 nesting |
Batch limits | Marker batch operations capped at 10,000 entries — prevents memory exhaustion from adversarial payloads with millions of markers |
Inline text limits | Inline transcript arguments capped at ~1 MB — file-based inputs go through |
Symlink filtering |
|
Marker strings | Sanitized via |
Role values | Stripped of control characters before XML attribute assignment |
URI parsing | MCP resource URIs parsed via |
Output suffixes | Path separators and special characters stripped — no traversal via suffix injection |
Marker types |
|
132 security-specific tests across test_security.py covering XXE, path traversal, sandbox boundaries, output path anchoring, input validation, subprocess bounds, minidom hardening, JSON depth limits, role sanitization, ffmpeg parameter bounds, symlink filtering, file count caps, and write-handler sandbox enforcement. Ruff S (bandit) rules enforced in CI — S314/S320 block unsafe XML parsing, S105 catches hardcoded passwords, S108 flags insecure temp paths. Security events (null bytes, sandbox escapes, unhandled exceptions) are logged via Python logging for audit trails.
Timestamp Parsing — How Import Tools Place Markers
All subtitle and transcript import tools (import_srt_markers, import_transcript_markers) funnel through a single internal function: _parse_timestamp_parts() in server.py. Understanding it matters when timestamps don't land where you expect.
Supported Formats
Format | Example | Parts | Result |
Minutes:Seconds |
| 2 | 90.0s |
H:MM:SS |
| 3 | 3930.0s |
HH:MM:SS.ms |
| 3 | 135.5s |
SMPTE (HH:MM:SS:FF) |
| 4 | 3610.5s @ 24fps |
The SMPTE 4-part format converts the frame component to fractional seconds: frames / frame_rate. The default rate is 24fps — pass frame_rate= to override for 25fps (PAL) or 30fps (NTSC) projects.
The Import Pipeline
SRT / VTT / YouTube chapters / plain transcript
│
▼
parse_srt() / parse_vtt() / parse_transcript_timestamps()
│ │ │
└────────────────┴──────────────────────┘
│
split on ':'
│
▼
_parse_timestamp_parts(parts, frame_rate=24.0)
│
▼
total seconds (float)
│
▼
marker placed on timelineEdge Cases
Unrecognized part counts (1 part, 5+ parts) return
None— the marker is silently skipped, not placed incorrectlyZero frame rate — falls back to base seconds (frames ignored) rather than dividing by zero
Milliseconds — only carried in 3-part format via
float()on the seconds component ("15.500"→15.5)Frame rounding — SMPTE frames are divided exactly (
12/24 = 0.5), not rounded to the nearest frame boundary. The resulting float is converted to FCPXML's rationalTimeValuedownstream, preserving precision
Why This Matters
Before v0.6.20, the 4-part SMPTE parser silently dropped frames — 01:00:10:12 became 3610.0s instead of 3610.5s. At 24fps, that's up to ~0.96 seconds of drift per marker. If you imported a subtitle file with SMPTE timecodes, every marker was slightly off. This was subtle enough to pass QC but visible when scrubbing.
Design Principles
Principle | Implementation |
Rational time, never floats | All durations are fractions ( |
Non-destructive by default | Modified files get |
Single source of truth |
|
Security-first | 10-layer defense-in-depth across all 53 handlers — see Security for the full matrix |
Dispatch, not conditionals |
|
Documentation
Guide | What's Inside |
8 production recipes — QC pipelines, beat-synced assembly, cross-NLE handoffs, documentary A/B roll | |
How this server composes with GitNexus, filesystem, and memory MCP servers | |
Full version history from v0.1.0 to present |
Testing
uv run --extra dev pytest tests/ -v # or: python3 -m pytest tests/ -v
ruff check . --exclude docs/ # lint — must pass before committing795 tests across 17 suites covering models, parser, writer, FCPXMLWriter generation, server handlers, rough cut generation, speed cutting & pacing curves, marker pipeline, refactored helper functions (_index_elements, _iter_spine_clips, _find_spine_clip_at_seconds, _require_clip, _require_spine_clip, _resolve_asset, serialize_xml), recent fix regressions (rapid_trim directions, min_duration, offset recalculation, interval timing accuracy, gap skipping, duplicate-name clip operations across trim/speed/split/delete/markers), security hardening (XXE, entity expansion, path traversal, sandbox boundaries, minidom defense-in-depth, JSON depth limits, input validation, ffmpeg bounds, write-handler sandboxing), connected clips, roles, diff, export, compound clip flattening, audio track generation, templates, effects, boundary conditions, and backward compatibility.
Requirements
Python 3.10+ · Final Cut Pro 10.4+ (FCPXML 1.8+) · Claude Desktop or any MCP client
Dependencies (auto-installed):
mcp,defusedxmlSee Compatibility for full version matrix
Roadmap
Core FCPXML parsing (v1.8–1.11)
Timeline analysis, markers, EDL/CSV export
Clip editing (trim, reorder, split, speed, transitions)
QC tools (flash frames, gaps, duplicates, health scoring)
Generation (rough cuts, montages, A/B roll, beat sync)
MCP Prompts + Resources (auto-discovery)
Subtitle & transcript import as markers
Multi-track (connected clips, compound clips, roles)
Timeline diff + social media reformat
Silence detection & cleanup
Cross-NLE export (DaVinci Resolve, Premiere Pro, Avid)
Audio sync detection
Premiere Pro native XML support
Known Issues
Issue | Impact | Workaround |
Still images crash FCP | PNG/JPEG assets referenced directly in FCPXML crash Final Cut Pro on import ( | Convert stills to short MOVs before referencing: |
Non-standard timebases | FCP rejects time values with denominators outside its standard set (e.g. | Fixed in v0.5.29 — TimeValue arithmetic now uses LCM, and speed changes snap to frame boundaries in 2400-tick timebase. |
Malformed frameDuration crash | A | Fixed in v0.6.23 — writer now validates both numerator and denominator, falling back to 30.0 fps. |
Duplicate clip names corrupt edits | When multiple spine clips share the same name (e.g. | Fixed in v0.6.37–0.6.39 — all methods now resolve clips via |
Contributing
PRs welcome. If you're a video editor who codes (or a coder who edits), let's build this together.
Credits
Built by @DareDev256 — former music video director (350+ videos), now building AI tools for creators.
License
MIT — see LICENSE.
Maintenance
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/DareDev256/fcpxml-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server