skills/x-api/SKILL.md
Post tweets, build threads, upload media via the X API.
npx skillsauth add notque/claude-code-toolkit x-apiInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill orchestrates OAuth-authenticated, rate-limit-aware X/Twitter API interactions through a deterministic Python script (scripts/x-api-poster.py). The workflow implements a 4-phase pipeline with an explicit confirmation gate (Phase 2) to prevent accidental public posts.
Core principles:
--confirmed)Goal: Confirm credentials, content, and dependencies before any network call.
Step 1: Check credentials
Test credential presence by running a dry-run credential check:
python3 $HOME/.claude/scripts/x-api-poster.py post --dry-run --text "ping"
This confirms all required environment variables are set: X_API_KEY, X_API_SECRET, X_ACCESS_TOKEN, X_ACCESS_SECRET, X_BEARER_TOKEN.
X_BEARER_TOKEN is requiredStep 2: Validate content length
The script enforces a 280-character limit per tweet. Before posting, validate your content length:
For a single tweet:
python3 $HOME/.claude/scripts/x-api-poster.py post --dry-run --text "your tweet text here"
For a thread:
python3 $HOME/.claude/scripts/x-api-poster.py thread --dry-run --texts "part 1" "part 2" "part 3"
If --dry-run reports a length error, ask the user to shorten the text or approve auto-segmentation into a thread.
Gate: Dry run exits 0, content length validates, credentials confirmed present. Proceed only when gate passes.
Goal: Show the user exactly what will be posted and require explicit approval before writing.
This gate is mandatory because X posts are public and irreversible. Present a content preview in this format:
CONTENT PREVIEW
================
Tweet 1/1:
"Your tweet text here"
Characters: 42/280
Action: POST single tweet
Approve? [yes/no]
For a thread:
CONTENT PREVIEW
================
Tweet 1/3:
"First part text"
Tweet 2/3:
"Second part text"
Tweet 3/3:
"Third part text"
Action: POST thread (3 tweets, chained replies)
Approve? [yes/no]
Wait for explicit user approval. The words "yes", "approve", "go ahead", "post it", or equivalent typed in the current conversation turn constitute approval. Do not infer approval from context or prior conversation turns. Do not pass --confirmed before the user provides explicit typed approval.
Gate: User has typed an explicit approval in this conversation turn. Proceed only when gate passes.
Goal: Execute the write operation and capture tweet IDs.
Only proceed once Phase 2 approval is confirmed. Pass the --confirmed flag when the user approves in this turn.
Single tweet:
python3 $HOME/.claude/scripts/x-api-poster.py post \
--confirmed \
--text "your tweet text here"
Thread:
python3 $HOME/.claude/scripts/x-api-poster.py thread \
--confirmed \
--texts "part 1" "part 2" "part 3"
Tweet with media:
python3 $HOME/.claude/scripts/x-api-poster.py post \
--confirmed \
--text "your tweet text here" \
--media /absolute/path/to/image.jpg
Media constraints: Images must be <= 5 MB (JPG, PNG, GIF); videos must be <= 512 MB (MP4). Media upload is a two-step process; if either step fails, no orphaned media is left behind. Confirm the file exists and is in a supported format before posting.
Watch output for:
[tweet-posted] id=... url=... — success line per tweet; contains canonical URL (https://x.com/i/web/status/{id})[rate-limit-warning] remaining=N reset=EPOCH — surface to user immediately if presentERROR: line — surface verbatim and stopOAuth mode is automatic: Read operations use Bearer token only; write operations require full OAuth 1.0a. The script selects the mode based on operation type — do not override it.
Gate: Script exits 0, at least one [tweet-posted] line in output. Proceed only when gate passes.
Goal: Return tweet URLs, IDs, and engagement baseline to the user.
Step 1: Collect tweet IDs from Phase 3 output
Parse all [tweet-posted] id=... url=... lines from the script output.
Step 2: Read engagement baseline (optional)
For each posted tweet, you may optionally read initial engagement metrics via:
python3 $HOME/.claude/scripts/x-api-poster.py read-timeline --user-id me --max-results 5
Engagement metrics have propagation delay: X API metrics take time to populate. Reading public_metrics immediately after posting with 0 impressions is expected behavior, not failure. Report metrics as baseline at post time and note they will grow asynchronously.
Step 3: Report to user
Provide:
Cause: One or more credential env vars not set in the shell Solution:
Cause: A single tweet segment is too long Solution:
Cause: Script invoked without confirmation (should not happen if Phase 2 gate was followed) Solution: Return to Phase 2, present the confirm gate, and obtain explicit user approval
Cause: Credentials are invalid, expired, or lack the required permissions Solution:
Cause: API rate limit window exhausted Solution:
Cause: Media upload is a two-step process; failure at either step leaves no orphaned media Solution:
$HOME/.claude/scripts/x-api-poster.py: Backing script (exit codes: 0=success, 1=missing credentials, 2=content validation failed, 3=API error, 4=write attempted without --confirmed)documentation
Document translation: quick/normal/refined modes with chunked parallel subagents and glossary support.
development
AI image generation: Gemini and Nano Banana backends; single/series/batch workflows with prompt-to-disk.
testing
Unified voice content generation pipeline with mandatory validation and joy-check. 13-phase pipeline: LOAD, GROUND, STATS-CHECKPOINT, GENERATE, HOOK-GATE, VALIDATE, REFINE, VARIETY-GATE, JOY-CHECK, ANTI-AI, CLOSE-GATE, OUTPUT, CLEANUP. Use when writing articles, blog posts, or any content that uses a voice profile. Use for "write article", "blog post", "write in voice", "generate content", "draft article", "write about".
documentation
Critique-and-rewrite loop for voice fidelity validation.