- Documents recent infrastructure cleanup (8 CTs destroyed, 35 DNS records removed, Headscale cleanup) - Adds 24 new runbooks covering Authentik, PeerTube, Meshtastic, RECON, Proxmox, Mailcow, Internet Archive, GPU routing - Adds project documentation for headscale, vaultwarden, peertube, matrix, mmud, advbbs, arr stack - Updates services.md, environment.md, caddy.md, authentik.md to match live infrastructure - Removes 4 deprecated runbook duplicates (canonical versions live in projects/) - Adds .gitignore for binary archives and editor temp files Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
6.3 KiB
Add PeerTube Channel
Overview
Add a YouTube channel to the PeerTube bulk import pipeline. Creates the PeerTube channel, adds to channel-map.json, and the downloader will begin syncing videos automatically.
Prerequisites
- SSH access from CT 130 (RECON) → CT 110 (PeerTube): working
- Sudoers:
/etc/sudoers.d/recon-mgmton CT 110 (allows zvx to run yt-dlp, psql, tee as peertube) - YouTube cookies at
/opt/bulk-import/config/cookies.txton CT 110 (not stale)
Method 1: Web UI (Preferred)
- Open RECON Dashboard → Upload tab:
http://192.168.1.130:8420/upload - Scroll to PeerTube Channels section
- Enter YouTube URL, category, priority
- Click Add Channel
- Wait for "Added: ChannelName" confirmation
Note: If the channel has members-only content, the API will automatically retry with --ignore-errors on the /videos tab.
Method 2: CLI (For Troubleshooting)
Use when the web UI fails or you need manual control.
Variables
YT_URL="https://www.youtube.com/@ChannelName"
CATEGORY="CategoryName"
PRIORITY="M" # H, M, or L
Step 1: Resolve Channel Info
# From CT 130 or cortex:
ssh zvx@192.168.1.170 "sudo -u peertube /usr/local/bin/yt-dlp \
--cookies /opt/bulk-import/config/cookies.txt \
--print channel --print channel_url --print channel_id \
--playlist-items 1 --skip-download '$YT_URL'"
If members-only error: Append /videos to URL and add --ignore-errors --playlist-items 1:5:
ssh zvx@192.168.1.170 "sudo -u peertube /usr/local/bin/yt-dlp \
--cookies /opt/bulk-import/config/cookies.txt \
--print channel --print channel_url --print channel_id \
--ignore-errors --playlist-items 1:5 --skip-download '${YT_URL}/videos' 2>/dev/null" | head -3
Record the output:
CHANNEL_NAME="Civilian Rifleman"
CHANNEL_URL="https://www.youtube.com/channel/UC..."
CHANNEL_ID="UC..."
Step 2: Slugify Actor Name
ACTOR_NAME=$(echo "$CHANNEL_NAME" | tr '[:upper:]' '[:lower:]' | sed 's/[^a-z0-9]/-/g; s/--*/-/g; s/^-//; s/-$//' | cut -c1-50)
echo "$ACTOR_NAME"
Step 3: Check for Duplicates
ssh zvx@192.168.1.170 "cat /opt/bulk-import/config/channel-map.json" \
| python3 -c "import sys,json; d=json.load(sys.stdin); \
matches=[c for c in d if c.get('actor_name')=='$ACTOR_NAME' or c.get('youtube_channel_id')=='$CHANNEL_ID']; \
print('DUPLICATE:', matches[0]['channel_name']) if matches else print('OK - no conflicts')"
Step 4: Create PeerTube Channel
ssh zvx@192.168.1.170 bash << 'REMOTE'
CLIENT=$(curl -s http://localhost:9000/api/v1/oauth-clients/local -H "Host: stream.echo6.co")
CID=$(echo "$CLIENT" | python3 -c "import sys,json; print(json.load(sys.stdin)['client_id'])")
CSEC=$(echo "$CLIENT" | python3 -c "import sys,json; print(json.load(sys.stdin)['client_secret'])")
TOKEN=$(curl -s http://localhost:9000/api/v1/users/token -H "Host: stream.echo6.co" \
--data "client_id=$CID&client_secret=$CSEC&grant_type=password&username=root&password=7redditGold" \
| python3 -c "import sys,json; print(json.load(sys.stdin)['access_token'])")
curl -s -X POST http://localhost:9000/api/v1/video-channels \
-H "Host: stream.echo6.co" \
-H "Authorization: Bearer $TOKEN" \
-H "Content-Type: application/json" \
-d "{\"name\":\"ACTOR_NAME\",\"displayName\":\"(YT)CHANNEL_NAME\"}"
REMOTE
Replace ACTOR_NAME and CHANNEL_NAME in the -d payload. Record the returned videoChannel.id.
Step 5: Update channel-map.json
IMPORTANT: Write to temp file first, then tee into place. Never pipe directly into tee on the same file being read — it causes a race condition that empties the file.
ssh zvx@192.168.1.170 bash << 'REMOTE'
python3 -c "
import json
with open('/opt/bulk-import/config/channel-map.json') as f:
channels = json.load(f)
channels.append({
'category': 'CATEGORY',
'channel_name': '(YT)CHANNEL_NAME',
'actor_name': 'ACTOR_NAME',
'youtube_url': 'CHANNEL_URL',
'youtube_channel_id': 'CHANNEL_ID',
'peertube_channel_id': PT_CHANNEL_ID,
'video_count': 0,
'priority': 'PRIORITY',
'est_videos': 0,
'est_gb': 0
})
print(json.dumps(channels, indent=2))
" > /tmp/channel-map-new.json \
&& sudo -u peertube tee /opt/bulk-import/config/channel-map.json < /tmp/channel-map-new.json > /dev/null \
&& rm -f /tmp/channel-map-new.json \
&& echo "OK"
REMOTE
Replace all placeholder values (CATEGORY, CHANNEL_NAME, ACTOR_NAME, CHANNEL_URL, CHANNEL_ID, PT_CHANNEL_ID, PRIORITY).
Step 6: Verify
# Check channel count
curl -s http://192.168.1.130:8420/api/peertube/channels/stats | python3 -m json.tool
# Verify new channel in list
curl -s http://192.168.1.130:8420/api/peertube/channels \
| python3 -c "import sys,json; d=json.load(sys.stdin); print(d[-1]['actor_name'], d[-1]['category'])"
Recovery: Empty channel-map.json
If tee race condition empties the file:
- Check Contabo backup:
ssh root@100.64.0.1 ls -la /opt/backups/recon/ - Or rebuild from PeerTube DB:
ssh zvx@192.168.1.170 "sudo -u peertube psql peertube_prod -t -A -c \ \"SELECT name, \\\"displayName\\\" FROM \\\"videoChannel\\\" WHERE name != 'root_channel' AND name != 'default' ORDER BY id;\""
API Endpoints (RECON Dashboard)
| Endpoint | Method | Purpose |
|---|---|---|
/api/peertube/channels |
GET | List all channels with video counts |
/api/peertube/channels/stats |
GET | Total channels, videos, downloader status |
/api/peertube/channels/add |
POST | Add channel (JSON: youtube_url, category, priority) |
/api/peertube/channels/<actor_name> |
DELETE | Remove channel from JSON and PeerTube |
Common Issues
| Issue | Cause | Fix |
|---|---|---|
| yt-dlp "Join this channel" error | Members-only first video | API auto-retries with /videos tab. CLI: add --ignore-errors --playlist-items 1:5 and use /videos URL |
| channel-map.json empty (0 bytes) | tee race condition | Always write to temp file first, then tee. Restore from backup or Contabo |
| sudo: password required | Sudoers not set up | Create /etc/sudoers.d/recon-mgmt via pct exec 110 from root@192.168.1.243 |
| PeerTube "actor name already exists" | Channel exists in PeerTube but not in JSON | Add entry to JSON manually with correct peertube_channel_id |
Last updated: 2026-02-18 — Initial creation