Sync your music library to a classic iPod from a modern Mac
Project description
clickwheel
Sync a music library to a classic iPod from a modern Mac — no iTunes required. Scan, clean up metadata, pick what goes on the iPod, and sync, all from the terminal. Optional MCP server lets Claude or other AI clients drive it conversationally.
Install
pipx install clickwheel
Optional extras:
pipx inject clickwheel 'clickwheel[fix]' # metadata cleanup via beets
pipx inject clickwheel 'clickwheel[mcp]' # MCP server for Claude / AI clients
Quick Start
Point clickwheel at your music:
mkdir -p ~/.clickwheel
cat > ~/.clickwheel/config.yaml << 'EOF'
music_dir: /path/to/your/music
EOF
Then index it and pick what goes on the iPod:
clickwheel scan # build the library index
clickwheel select # interactive checkbox picker
clickwheel sync # push to the iPod
Commands
| Command | Description |
|---|---|
clickwheel scan |
Index your music library and report on metadata quality |
clickwheel fix |
Clean up metadata, fetch album art, fill genres via beets |
clickwheel select |
Interactive picker — checkbox artist selection |
clickwheel playlist |
List saved playlists or show details for one |
clickwheel edit |
Add or remove artists via interactive menus or --add/--remove flags |
clickwheel heal |
Drop playlist references to tracks no longer on disk |
clickwheel delete |
Delete a saved playlist (with confirmation) |
clickwheel diff |
Preview what would be added or removed on the iPod |
clickwheel sync |
Push your playlist to the iPod (with live progress table) |
clickwheel ls |
Show what's on the iPod |
clickwheel eject |
Safely unmount the iPod |
clickwheel scrobble |
Submit recent iPod listens to Last.fm |
Configuration
# ~/.clickwheel/config.yaml
music_dir: /Volumes/Music/Library
ipod_capacity_gb: 64
auto_scan: true
auto_scan_staleness_minutes: 1440 # default: 24h
lastfm_api_key: your_key
lastfm_api_secret: your_secret
lastfm_username: your_username
Environment variables (MUSIC_DIR, AUTO_SCAN, etc.) override config values. See docs/configuration.md for the full schema, auto-scan internals, and fix / Last.fm walkthroughs.
MCP server
clickwheel ships an optional MCP server so Claude Code, Claude Desktop, and other MCP-aware clients can drive your library conversationally:
What's on my iPod, and how full is it?
Build me a 45-minute late-night indie folk playlist using only tracks I actually own.
Sync the 'ipod' playlist to my iPod and then eject it.
Quick start with Claude Code:
pipx inject clickwheel 'clickwheel[mcp]'
claude mcp add clickwheel clickwheel-mcp --scope user
The server exposes 21 tools across library, playlist, iPod, and Last.fm domains, plus a build_playlist prompt with anti-hallucination rules. Destructive operations (delete_playlist, sync_playlist_to_ipod) are gated by client confirmation.
For Claude Desktop config, the full tool reference, and other clients (Cursor, Continue, Cline, Zed), see docs/mcp/.
Requirements
- macOS (iPod sync depends on macOS disk utilities)
- Python 3.11+
- iPod Classic with stock firmware, connected via USB
- FLAC files are excluded from sync (stock firmware limitation)
Contributing
See CONTRIBUTING.md for dev setup, testing, and commit conventions.
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 clickwheel-0.5.0.tar.gz.
File metadata
- Download URL: clickwheel-0.5.0.tar.gz
- Upload date:
- Size: 591.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
84c5bc4abeeb1726e013be7451428bd16500f5f0f8ee0f0c9c659aa6517a3076
|
|
| MD5 |
3d74e35a424511f1ab48786d45d029c1
|
|
| BLAKE2b-256 |
b4dff60c84a3e423b433d8ddd774608c1b02db52195360b907dc459d47eddc87
|
Provenance
The following attestation bundles were made for clickwheel-0.5.0.tar.gz:
Publisher:
publish.yml on pdugan20/clickwheel
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
clickwheel-0.5.0.tar.gz -
Subject digest:
84c5bc4abeeb1726e013be7451428bd16500f5f0f8ee0f0c9c659aa6517a3076 - Sigstore transparency entry: 1464250822
- Sigstore integration time:
-
Permalink:
pdugan20/clickwheel@89a6480291965d6c82b6c6fb29895c64a3af14c5 -
Branch / Tag:
refs/tags/v0.5.0 - Owner: https://github.com/pdugan20
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@89a6480291965d6c82b6c6fb29895c64a3af14c5 -
Trigger Event:
push
-
Statement type:
File details
Details for the file clickwheel-0.5.0-py3-none-any.whl.
File metadata
- Download URL: clickwheel-0.5.0-py3-none-any.whl
- Upload date:
- Size: 579.5 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 |
4b58210c3471bbbd6edc6ff032116185950a588db5406fd44130c812447317c5
|
|
| MD5 |
69b83e2c147bba8042ec818dac8986b9
|
|
| BLAKE2b-256 |
e2563050feba3bb417ab4b5dbd4f0c1715f1a814c03d6f9e09b7d861a9b31222
|
Provenance
The following attestation bundles were made for clickwheel-0.5.0-py3-none-any.whl:
Publisher:
publish.yml on pdugan20/clickwheel
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
clickwheel-0.5.0-py3-none-any.whl -
Subject digest:
4b58210c3471bbbd6edc6ff032116185950a588db5406fd44130c812447317c5 - Sigstore transparency entry: 1464250890
- Sigstore integration time:
-
Permalink:
pdugan20/clickwheel@89a6480291965d6c82b6c6fb29895c64a3af14c5 -
Branch / Tag:
refs/tags/v0.5.0 - Owner: https://github.com/pdugan20
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@89a6480291965d6c82b6c6fb29895c64a3af14c5 -
Trigger Event:
push
-
Statement type: