name: xradio description: CLI tool for pulling, caching, and reading Twitter/X posts offline
xradio
A command-line tool for managing a local cache of Twitter/X posts. Pull tweets from your feed, read them offline, track read/unread status, and play videos.
Prerequisites
- Browser server must be running for
pullcommand:browser server start - Must be logged into Twitter/X in the browser session
yt-dlpandmpvfor video playback
Commands
Pull Tweets from Feed
xradio pull --limit <n>
Fetches tweets from your Twitter home feed and saves them to local storage.
--limit <n>: Number of tweets to pull (default: 10)- Automatically scrolls to load more if needed
- Never overwrites existing cached tweets - skips duplicates
- Requires browser server to be running and logged into Twitter
Example:
xradio pull --limit 20
Output:
π‘ Pulling up to 20 new tweets from feed...
π Browser connected: https://x.com/home
π Refreshing feed...
scroll 1: +8 new, 2 cached (8/20 total new)
scroll 2: +6 new, 4 cached (14/20 total new)
scroll 3: +6 new, 3 cached (20/20 total new)
β
[a1b2c3d] User Name: Tweet text preview...
β
[e4f5g6h] Another User: More tweet content here...
π¦ Done! Saved 20 new tweets.
List Cached Tweets
xradio list # Show unread tweets only
xradio list -A # Show all tweets (including read)
xradio list --all # Same as -A
Lists tweets in the offline cache.
- By default, only shows unread tweets
- Use
-Aor--allto include read tweets - Shows media indicators: π¬ (video), πΌοΈ (image)
- Shows β for read tweets (when using --all)
Output:
π 4 unread tweets in cache:
a1b2c3dπ¬πΌοΈ Man, Crimson Desert looks like the kind of game that would make me disappear...
e4f5g6hπΌοΈ Breaking: New development in the ongoing situation as officials announce...
Use 'xradio read <id>' to view a tweet, 'xradio mark <id>' to mark as read.
Read a Tweet
xradio read <id>
Display the full content of a cached tweet.
- Supports short IDs (like git):
xradio read a1binstead of full hash - Shows formatted tweet with author, text, media indicators
- If ambiguous (multiple matches), asks for longer prefix
Example:
xradio read a1b
Output:
ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
# User Name
**@handle** Β· 2h
This is the full tweet text content here...
---
π¬ Video | πΌοΈ Image
[View on Twitter](https://x.com/handle/status/123456789)
ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
ID: a1b2c3d (a1b2c3d4e5f6g7h8i9j0...)
Pulled: 2026-01-08T16:00:00.000Z
π¬ This tweet has a video. Use 'xradio play a1b' to play it.
Mark as Read
xradio mark <id>
Mark a tweet as read. It will no longer appear in xradio list (only in xradio list -A).
Example:
xradio mark a1b
# Output: β
Marked a1b2c3d as read
Unmark (Mark as Unread)
xradio unmark <id>
Remove the read status from a tweet, making it appear in the unread list again.
Example:
xradio unmark a1b
# Output: β©οΈ Unmarked a1b2c3d (now unread)
Describe Images
xradio describe <id>
xradio describe <id> "custom prompt here"
Describe images in a tweet using AI vision. Downloads images and generates a text description suitable for radio narration.
- Works only on tweets with images (πΌοΈ indicator)
- Uses AI vision model to describe the image content
- Optional custom prompt for specific questions about the image
Example:
xradio describe 836
Output:
πΌοΈ Describing 1 image(s) from: VisegrΓ‘d 24
This is a line graph from Netblocks.org showing normalized network
connectivity in Iran from January 7 to 8, 2026...
Play Video
xradio play <id>
Play a video from a tweet that has video content.
- Extracts video URL via yt-dlp
- Plays with mpv (video + audio)
- Interactive controls:
qto quit,spaceto pause
Example:
xradio play a1b
Output:
π¬ Extracting video URL for: User Name
https://x.com/handle/status/123456789
πΊ Playing video...
(Press 'q' to quit, 'space' to pause)
Storage Format
Tweets are stored in storage/<id>.md where id is SHA1 hash of the permalink.
Each file has YAML frontmatter + Markdown content:
---
id: a1b2c3d4e5f6...
permalink: https://x.com/handle/status/123456789
displayName: User Name
handle: "@handle"
timestamp: "2h"
hasVideo: true
hasImage: false
pulledAt: 2026-01-08T16:00:00.000Z
offline:
read: true
readAt: 2026-01-08T17:00:00.000Z
---
# User Name
**@handle** Β· 2h
Tweet text content...
---
π¬ Video
[View on Twitter](https://x.com/handle/status/123456789)
Typical Workflow
# 1. Start browser (if not running)
browser server start
# 2. Pull fresh tweets
xradio pull --limit 10
# 3. See what's new
xradio list
# 4. Read a tweet
xradio read a1b
# 5. If it has video, play it
xradio play a1b
# 6. Mark as read when done
xradio mark a1b
# 7. Continue with next unread tweet
xradio list
Notes
- Short IDs work like git refs - use just enough characters to be unique
- The
offline.*namespace in frontmatter is for our metadata (read status, etc.) - Tweet data is never overwritten once cached - pull is additive only
- Video playback requires working yt-dlp Twitter extraction (may have intermittent issues with Twitter API)