Claude-code-minoan disc-forge
Burn Red Book audio CDs or MP3 data CDs on macOS via cdrdao, sourcing from Hoodrat HDD or local files. Audio CDs carry CD-Text from ID3 tags; MP3 discs use ISO9660+Joliet (~700 MB). Handles full-album burns and cherry-picked playlists. Triggers on 'burn CD', 'burn album', 'make audio CD', 'mixtape disc', 'soundtrack for the car'.
git clone https://github.com/tdimino/claude-code-minoan
T=$(mktemp -d) && git clone --depth=1 https://github.com/tdimino/claude-code-minoan "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/integration-automation/disc-forge" ~/.claude/skills/tdimino-claude-code-minoan-disc-forge && rm -rf "$T"
skills/integration-automation/disc-forge/SKILL.mddisc-forge
Automated CD burning for macOS via cdrdao. Two modes:
- Audio CD (default) — Red Book, plays in any CD player, 80-min capacity, CD-Text from ID3 tags
- MP3 data CD (
) — plays in MP3-capable stereos (most 2005+ car head units), 700 MB capacity, ISO9660+Joliet filesystem--mp3-disc
Turns "burn the BSG soundtrack to a CD" or "make an MP3 disc of these mantras" into a single command.
When to Use
- User asks to burn an album, soundtrack, or playlist to an audio CD for a car stereo / home stereo
- Source music lives on Hoodrat HDD (Mac Mini) or in a local folder
- A USB CD burner is plugged into the current Mac
- User wants either a Red Book audio CD or an MP3 data CD
Out of scope: DVD burning, CD ripping (use XLD — see
~/.claude/agent_docs/music.md), multi-session discs, DDP mastering, mixed-mode discs (audio + data on one disc).
Quick Start
Burn a full album from Hoodrat HDD with one command:
python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py \ --source "mac-mini-ts:/Volumes/Hoodrat HDD/Musica/BSG.S1 - OST/" \ --name BSG-S1
That runs the full pipeline: preflight → tar-over-SSH stage → duration check → parallel WAV conversion → TOC build with CD-Text → cdrdao write at 16x → cleanup → eject. Takes ~5–7 minutes for an 80-minute disc. cdrdao streams its own progress to stdout; a 10-second countdown precedes the actual burn so
Ctrl-C still aborts safely.
Before the first burn, make sure a blank writable CD-R or CD-RW is in the drive —
preflight.py will abort early if not.
The Pipeline
burn_audio_cd.py is the master orchestrator. It chains the stage scripts in sequence and halts on any failure, preserving staging for debugging. Each script is also independently usable.
Shared stages (both modes):
| Stage | Script | What it does |
|---|---|---|
| 1 | | Verifies + , drive reachable, blank writable disc inserted. Retries for ~3 s after (which briefly locks the drive). Surfaces the drutil label as an informational note — cdrdao bypasses DiscRecording. |
| 2 | | Pulls source files. Remote sources () use tar-over-SSH, not rsync. Local sources use . Filters to one format (default ) to dedupe MP3+WMA and MP3+M4A siblings. |
| final | | Actual burn. by default. Streams progress. |
Audio CD stages (default):
| Stage | Script | What it does |
|---|---|---|
| 3 | | Totals runtime via ffprobe. Aborts if the album overruns 80-min Red Book capacity. Prints a format histogram so leftover duplicates show up. |
| 4 | | Parallel ffmpeg MP3→44.1 kHz/16-bit/stereo PCM (the exact Red Book format). Idempotent — skips WAVs already fresh. |
| 5 | | cdrdao TOC with blocks. Pulls title/artist/album from the original source file's ID3 tags (WAVs don't carry ID3). Gapless by default; inserts 2-sec spacers. |
| cleanup | inline | Deletes on success. Keeps MP3 staging for re-burns. |
MP3 data CD stages (
--mp3-disc):
| Stage | Script | What it does |
|---|---|---|
| 3 | inline (orchestrator) | Sum of file sizes vs. 700 MB ceiling. Aborts if over. |
| 4 | | builds an ISO9660+Joliet image containing the files. Emits a cdrdao data TOC ( / / ). Volume label derived from (sanitized A-Z/0-9/_). |
| cleanup | inline | Deletes on success. Keeps source staging for re-burns. |
Scripts
All scripts support
--help. All live under ~/.claude/skills/disc-forge/scripts/. Default staging root is ~/burn-staging/<name>/.
preflight.py # preflight.py [--json] stage_tracks.py # stage_tracks.py --source <path|host:/path> --dest <name> [--format mp3] check_duration.py # check_duration.py <staging-dir> [--ceiling 80] [--json] convert_to_wav.py # convert_to_wav.py <staging-dir> [--output <wav-dir>] [--jobs N] build_toc.py # build_toc.py --wav-dir <wavs> --source-dir <mp3s> --output <toc> [--gaps] build_data_toc.py # build_data_toc.py --source-dir <dir> --volume-label <label> --iso-output <iso> --toc-output <toc> burn_audio_cd.py # burn_audio_cd.py --source <path|host:/path> --name <name> [--mp3-disc] [--speed 16] [--gaps] [--dry-run] [--keep-wavs] [--ceiling 80]
Key
burn_audio_cd.py flags:
— burn an MP3 data CD (ISO9660+Joliet) instead of a Red Book audio CD. 700 MB capacity, plays in MP3-capable car stereos (most 2005+ head units).--mp3-disc
— MP3 data disc capacity ceiling in MB (default 700, only with--data-ceiling N
).--mp3-disc
— runs everything through TOC generation, skips the cdrdao write. Use to validate the whole pipeline on a disposable run.--dry-run
— 2-sec spacers between tracks (pop albums). Default is gapless (right for OSTs, film scores, live albums). Audio mode only.--gaps
— cdrdao write speed multiplier. Default 16. Drop to 10 for maximum reliability on cheap media.--speed N
— preserve the intermediate WAV staging dir (audio mode) or ISO (data mode) after the burn.--keep-wavs
— audio CD capacity in minutes for the fit check. Default 80. Set to 74 for older 74-min media.--ceiling N
Recipes
Burn a full soundtrack from Hoodrat HDD (the canonical path):
python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py \ --source "mac-mini-ts:/Volumes/Hoodrat HDD/Musica/Blade Runner - OST/" \ --name BladeRunner-OST
Burn from a pre-staged local directory (when tracks are already on the laptop):
python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py \ --source ~/burn-staging/BSG-S1 \ --name BSG-S1
The orchestrator detects that source == staging dir and skips the stage step.
Dry-run the pipeline (validate without burning a coaster):
python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py \ --source ~/burn-staging/BSG-S1 \ --name BSG-S1 \ --dry-run
Runs every stage through TOC generation. Prints the cdrdao command you'd run next and the
cdrdao show-toc command to validate the generated TOC.
MP3 data CD for a modern car stereo (2005+ head units, ~700 MB capacity):
python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py \ --source ~/burn-staging/mantras \ --name mantras \ --mp3-disc
Volume label derives from
--name (sanitized to uppercase A-Z/0-9/_). Stage MP3s with numeric prefixes (01 Foo.mp3, 02 Bar.mp3) to force track order — most stereos play alphabetically. For MP4/Opus/M4A sources, transcode to MP3 first (stage_tracks.py filters to .mp3 by default).
Pop album with 2-sec gaps (Led Zeppelin, Pink Floyd, etc.):
python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py \ --source "mac-mini-ts:/Volumes/Hoodrat HDD/Musica/Dark Side Of The Moon/" \ --name DSOTM \ --gaps
Cherry-picked mixtape (tracks from multiple albums):
Stage tracks manually into a single folder first, then point the orchestrator at it:
mkdir -p ~/burn-staging/my-mixtape scp "mac-mini-ts:/Volumes/Hoodrat HDD/Musica/Blade Runner - OST/01 Main Titles.mp3" ~/burn-staging/my-mixtape/ scp "mac-mini-ts:/Volumes/Hoodrat HDD/Musica/Inception OST [2010]/01 Half Remembered Dream.mp3" ~/burn-staging/my-mixtape/ # ... rename files with numeric prefixes to force desired order ... python3 ~/.claude/skills/disc-forge/scripts/burn_audio_cd.py --source ~/burn-staging/my-mixtape --name my-mixtape
Track order is determined by
sorted() on filenames — use numeric prefixes (01_, 02_, …) to force a specific sequence.
Critical Reminders
Full expansions (with why) live in
references/gotchas.md.
- SMB is not reachable over Tailscale. Port 445 is refused on
; only port 22 works. Use SSH transport.mac-mini-ts
already does this.stage_tracks.py - Apple's bundled
(openrsync 2.6.9) splits remote paths on spaces.rsync
uses tar-over-SSH instead. Never usestage_tracks.py
for Hoodrat HDD.rsync - cdrdao 1.2.6 rejects MP3 input directly. The binary contains the literal string
. WAV conversion is mandatory —"AIFF and MP3 not supported by cdrdao"
handles it. (Burn.app is the alternative if you want to skip WAV conversion — seeconvert_to_wav.py
.)references/burn-app-fallback.md
"Unsupported" label is a red herring for cdrdao. It only affects Apple's DiscRecording framework (Finder burn / Music.app burn / Burn.app), not cdrdao.drutil
surfaces the label as a warning, not an error.preflight.py- CD-Text must come from source ID3 tags, not WAVs. PCM WAV has no tagging container.
reads tags from the original file, references the matching WAV inbuild_toc.py
.AUDIOFILE - macOS Sequoia killed Music.app and Finder's burn-to-disc. Don't fall back to them.
References
| File | Contents |
|---|---|
| Full expanded failure-mode inventory from the author's first burn. Every trap explained with why, so the rule generalizes to novel drives and macOS versions. |
| cdrdao TOC syntax cheatsheet: minimal skeleton, blocks, PREGAP semantics, special-character escaping, how to run to validate. |
| Pointer to (the canonical Hoodrat HDD layout) plus a one-line cheatsheet. |
| If cdrdao refuses the drive: Burn.app GUI fallback with exact click sequence. |
Dependencies
—cdrdaobrew install cdrdao
+ffmpeg
—ffprobebrew install ffmpeg- SSH access to
(only when sourcing from Hoodrat HDD)mac-mini-ts - A USB CD burner that speaks MMC. The ASUS SDRW-08U9M-U works despite drutil labeling it Unsupported.