A Python library for generating Final Cut Pro XML (FCPXML) timelines programmatically, designed to help video editors collaborate with LLMs on automatic edits.
Project description
steele-fcpxml
A Python library for generating Final Cut Pro XML (FCPXML) timelines programmatically.
steele-fcpxml lets a script - or an LLM acting on behalf of a video editor - assemble an FCPXML timeline from a list of source clips, in/out points, gaps, and markers. The resulting .fcpxml file can be imported directly into DaVinci Resolve or Final Cut Pro and dropped onto the editor's timeline with every clip pre-positioned at the right source timecode.
Use case
Modern video editing increasingly involves an LLM assistant that scans large media archives, identifies relevant clips, and proposes assemblies. The bottleneck has traditionally been the last mile: turning a list of "use clip X from 1:23 to 1:35, then clip Y from 4:02 to 4:18" into something an NLE can actually open.
This library closes that gap. The LLM (or any script) produces a Python program like:
from pathlib import Path
from steele_fcpxml import FCPXML, FrameRate, tc
(
FCPXML("My Episode", "Catchphrase Compilation", timeline_fps=FrameRate.FPS_25)
.add_clip(Path("/media/interview_01.mp4"), in_point=tc("1:23"), out_point=tc("1:35"), name="Setup")
.add_gap(2.0, name="Beat")
.add_clip(Path("/media/interview_02.mp4"), in_point=tc("4:02"), out_point=tc("4:18"), name="Payoff")
.add_marker("Act break")
.write(Path("/edit/compilation.fcpxml"))
)
The editor opens compilation.fcpxml in DaVinci Resolve and gets a working timeline with every clip on the correct source frame, ready to trim and refine. No manual timecode entry, no scrubbing through hours of footage to find a moment the LLM already identified by transcript search.
Typical workflows it supports:
- Catchphrase / supercut compilations. Grep transcripts for a phrase, hand the matches to this library, ship an FCPXML.
- Multi-source montages. Mix clips at different frame rates (25 fps, 29.97 fps, 23.976 fps) - the library auto-probes each file with
ffprobeand emits the correct rational-time arithmetic so timeline offsets never drift. - Iterative LLM-assisted edits. Generate a draft FCPXML, the editor refines it in Resolve, the LLM regenerates with new clips and the editor merges. Round-trip via plain text Python source.
- Validation in CI. A separate
steele-fcpxml validate <file>CLI checks structure, missing media references, and frame-rate consistency before the file ever reaches the NLE.
Public API
The top-level surface is deliberately tiny - three names cover almost every use:
from steele_fcpxml import FCPXML, FrameRate, tc
FCPXML- the fluent timeline builder (add_clip,add_gap,add_marker,fork,write).FrameRate- standard frame rates (FPS_25,FPS_29_97, ...) for the timeline.tc- parse"MM:SS"/"HH:MM:SS"timecodes to seconds.
A lower-level API (the spec dataclasses, the FCPXMLWriter, the probe result, and the FCPXMLValidator) is available from the submodules steele_fcpxml.specs, steele_fcpxml.writer, steele_fcpxml.probe, and steele_fcpxml.validator for callers who need it.
Requirements
- Python 3.12+
ffmpeginstalled on PATH (for theffprobebinary used to detect frame rates and durations)
Installation
pip install steele-fcpxml
The only runtime dependency is click (for the CLI). ffprobe must be available on PATH for clip probing.
CLI
steele-fcpxml validate path/to/timeline.fcpxml
steele-fcpxml validate --json path/to/timeline.fcpxml
Development
This project uses uv for dependency management and ruff + pyright for quality gates.
git clone https://github.com/salimfadhley/steele_fcpxml.git
cd steele_fcpxml
uv sync
uv run pytest # tests (mocked ffprobe; fast)
uv run ruff check src tests
uv run ruff format --check src tests
uv run pyright
The default test suite mocks ffprobe and needs no media. A small set of tiny, GPL-licensed fixture clips in tests/fixtures/ drives the real-ffprobe integration tests (tests/test_probe_integration.py), which skip automatically when ffprobe is not installed.
Making a release
Versioning is automatic and tag-driven via hatch-vcs: there is no version string to edit in the repository - the version is computed from the latest git tag at build time. A tag v0.3.0 produces the distribution steele_fcpxml-0.3.0.
To cut a release:
-
Make sure
mainis green in CI (ruff, pyright, tests, and the wheel build all pass). -
Tag the commit and push the tag:
git tag v0.1.0 git push origin v0.1.0
-
Pushing a
v*tag triggers.github/workflows/release.yaml, which builds the wheel and sdist and publishes them to PyPI using Trusted Publishing (OIDC) - no API token or password is stored anywhere.
Tags follow semantic versioning: vMAJOR.MINOR.PATCH. Between tags, local and CI builds produce dev versions like 0.1.0.dev4+g<hash>, which is expected.
One-time PyPI setup (already configured for this project)
Trusted Publishing must be registered once on the PyPI side, under the project's Publishing settings (or as a "pending publisher" before the project's first release):
| Field | Value |
|---|---|
| Owner | salimfadhley |
| Repository | steele_fcpxml |
| Workflow | release.yaml |
| Environment | pypi |
License
GPL-3.0-or-later. A copyleft license. If you redistribute this code or build something on top of it that you distribute, downstream users must receive the same freedoms. The intent is to keep the library and any improvements available to the wider community of video editors using LLM tools.
See also
- DaVinci Resolve's FCPXML import documentation: https://documentation.blackmagicdesign.com/
- The FCPXML schema reference: https://developer.apple.com/documentation/professional_video_applications/fcpxml_reference
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file steele_fcpxml-0.1.0.tar.gz.
File metadata
- Download URL: steele_fcpxml-0.1.0.tar.gz
- Upload date:
- Size: 81.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
478dac6f354ccfa50a75de100e85c037462f986f922fbd6ef019890a9c82f61d
|
|
| MD5 |
983c0434cb1a4b09192c7fc50cea722e
|
|
| BLAKE2b-256 |
df82bc5dcca77b68b698c610d7f2f63a9e14869e8a60f3f7b203f4806b70f8ba
|
Provenance
The following attestation bundles were made for steele_fcpxml-0.1.0.tar.gz:
Publisher:
release.yaml on salimfadhley/steele_fcpxml
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
steele_fcpxml-0.1.0.tar.gz -
Subject digest:
478dac6f354ccfa50a75de100e85c037462f986f922fbd6ef019890a9c82f61d - Sigstore transparency entry: 1796046354
- Sigstore integration time:
-
Permalink:
salimfadhley/steele_fcpxml@a2dd4f05281087479c789ca844cd67a22becfc67 -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/salimfadhley
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yaml@a2dd4f05281087479c789ca844cd67a22becfc67 -
Trigger Event:
push
-
Statement type:
File details
Details for the file steele_fcpxml-0.1.0-py3-none-any.whl.
File metadata
- Download URL: steele_fcpxml-0.1.0-py3-none-any.whl
- Upload date:
- Size: 33.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
428adea659e1ddfdfa8412a82a6ef5709513348c911f46fe0db7ef781bdc7384
|
|
| MD5 |
df02eb4aff58ac88e2cd63ec4dea8502
|
|
| BLAKE2b-256 |
02fb7c2be2f7f6584e54320fda17282d2f98242df82fe45de6460d70b1c12398
|
Provenance
The following attestation bundles were made for steele_fcpxml-0.1.0-py3-none-any.whl:
Publisher:
release.yaml on salimfadhley/steele_fcpxml
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
steele_fcpxml-0.1.0-py3-none-any.whl -
Subject digest:
428adea659e1ddfdfa8412a82a6ef5709513348c911f46fe0db7ef781bdc7384 - Sigstore transparency entry: 1796046530
- Sigstore integration time:
-
Permalink:
salimfadhley/steele_fcpxml@a2dd4f05281087479c789ca844cd67a22becfc67 -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/salimfadhley
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yaml@a2dd4f05281087479c789ca844cd67a22becfc67 -
Trigger Event:
push
-
Statement type: