A reeln-cli plugin for Google platform integration — YouTube livestreams, uploads, playlists, and comments.
pip install reeln-plugin-googleOr for development:
git clone https://github.com/StreamnDad/reeln-plugin-google
cd reeln-plugin-google
make dev-installBefore using this plugin, you need a Google Cloud project with OAuth credentials. Follow these steps exactly.
- Go to Google Cloud Console
- Click the project dropdown (top-left) → New Project
- Name it (e.g.,
reeln-livestream) and click Create - Select the new project from the dropdown
- Go to APIs & Services → Library
- Search for YouTube Data API v3
- Click it and press Enable
- Go to APIs & Services → OAuth consent screen
- Select External user type → Create
- Fill in:
- App name — anything (e.g.,
reeln) - User support email — your Gmail
- Developer contact email — your Gmail
- App name — anything (e.g.,
- Click Save and Continue
- On the Scopes page, click Add or Remove Scopes and add:
https://www.googleapis.com/auth/youtubehttps://www.googleapis.com/auth/youtube.uploadhttps://www.googleapis.com/auth/youtube.force-ssl
- Click Update → Save and Continue
- On the Test users page, click Add Users and enter your Google email address
- Click Save and Continue → Back to Dashboard
Testing vs Production mode: Your app starts in "Testing" mode with these limitations:
- Only the test users you added can authorize
- OAuth tokens expire every 7 days (you'll need to re-authorize)
- Maximum 100 test users
Once your setup is stable, you can publish to "Production" mode to remove token expiry and the user cap. Production apps requesting sensitive scopes require Google verification, but apps for personal use with limited users typically pass quickly.
- Go to APIs & Services → Credentials
- Click Create Credentials → OAuth client ID
- Application type: Desktop app
- Name it (e.g.,
reeln-desktop) and click Create - Click Download JSON (the download icon)
The downloaded client_secrets.json contains your OAuth client ID and client secret. Treat it like a password.
# Move it somewhere safe outside your repo
mv ~/Downloads/client_secret_*.json ~/.config/reeln/client_secrets.json
# Restrict permissions (macOS/Linux)
chmod 600 ~/.config/reeln/client_secrets.jsonNever commit this file to git. It allows anyone to impersonate your application and request access to YouTube accounts.
Add the Google plugin to your reeln config. All capabilities are off by default — enable only what you need:
{
"plugins": {
"enabled": ["google"],
"settings": {
"google": {
"client_secrets_file": "~/.config/reeln/client_secrets.json",
"create_livestream": true,
"manage_playlists": true,
"upload_highlights": true,
"upload_shorts": false,
"privacy_status": "unlisted"
}
}
}
}Every capability must be explicitly enabled. Nothing runs unless you set the flag to true.
| Flag | Default | What it does |
|---|---|---|
create_livestream |
false |
Create a YouTube broadcast on reeln game init and write the URL to shared context |
manage_playlists |
false |
Create a per-game playlist on reeln game init; auto-add uploads when combined with upload_highlights |
upload_highlights |
false |
Upload the merged highlights video after reeln highlights merge |
upload_shorts |
false |
Upload Shorts after reeln render short (detected via filter_complex) |
| Key | Required | Default | Description |
|---|---|---|---|
client_secrets_file |
Yes | — | Path to GCP OAuth client secrets JSON |
credentials_cache |
No | <data_dir>/google/oauth.json |
OAuth credentials cache path |
privacy_status |
No | unlisted |
Video/livestream privacy: public, unlisted, or private |
category_id |
No | 20 |
YouTube category ID (20 = Gaming) |
tags |
No | [] |
Default video tags |
scopes |
No | [youtube, youtube.upload, youtube.force-ssl] |
OAuth scopes |
On the first reeln game init, the plugin opens a browser for OAuth consent:
- Sign in with the Google account you added as a test user
- Click Continue (past the "unverified app" warning)
- Grant the requested YouTube permissions
- The browser shows "The authentication flow has completed" — you can close it
Credentials are cached at ~/Library/Application Support/reeln/data/google/oauth.json (macOS) or ~/.local/share/reeln/data/google/oauth.json (Linux). The file is automatically chmod 600'd. Tokens auto-refresh on subsequent runs — you won't see the browser again unless tokens expire or you re-authenticate.
"Access blocked" or "Error 403: access_denied" You didn't add your Google email as a test user. Go to OAuth consent screen → Test users and add it.
"Quota exceeded" The YouTube Data API has a daily quota (default 10,000 units). Creating a livestream costs ~1,600 units. If you're hitting limits, check your quota dashboard.
Token expired after 7 days This happens in Testing mode. Either re-run to trigger the browser flow again, or publish your OAuth consent screen to Production mode (see Testing vs Production above).
"redirect_uri_mismatch" Your OAuth client type isn't set to Desktop app. Delete the credential and create a new one with the correct type.
The plugin hooks into reeln-cli lifecycle events. Here's the typical game-day flow:
reeln game init
└─ ON_GAME_INIT
├─ create_livestream → YouTube broadcast bound to your OBS stream key
└─ manage_playlists → per-game playlist (livestream auto-added if both enabled)
reeln highlights merge
└─ ON_HIGHLIGHTS_MERGED
└─ upload_highlights → upload merged video; auto-add to playlist if enabled
reeln render short
└─ POST_RENDER (for each short)
└─ upload_shorts → upload as YouTube Short (#Shorts appended to title)
reeln game finish
└─ ON_GAME_FINISH → reset cached state for next game
The plugin writes results to context.shared so sibling plugins (e.g., a Discord notifier) can consume them:
| Key | Value |
|---|---|
shared["livestreams"]["google"] |
Livestream URL (https://youtube.com/live/...) |
shared["playlists"]["google"] |
Playlist ID |
shared["uploads"]["google"]["video_id"] |
Uploaded highlights video ID |
shared["uploads"]["google"]["url"] |
Uploaded highlights URL |
shared["uploads"]["google"]["shorts"] |
List of {"video_id", "url"} for each Short |
If an LLM plugin writes metadata to shared["uploads"]["google"] before the upload hooks fire, the plugin uses it instead of auto-generating from game info:
| Key | Used by |
|---|---|
title |
Highlights upload |
description |
Highlights upload |
tags |
Highlights and Shorts upload |
short_title |
Shorts upload |
short_description |
Shorts upload |
make dev-install # uv venv + editable install with dev deps
make test # pytest with 100% coverage
make lint # ruff check
make format # ruff format
make check # lint + mypy + testAGPL-3.0-only