* wip * copy polugin files * wip type changes * refactor: improve Twitch plugin code quality and fix all tests - Extract client manager registry for centralized lifecycle management - Refactor to use early returns and reduce mutations - Fix status check logic for clientId detection - Add comprehensive test coverage for new modules - Remove tests for unimplemented features (index.test.ts, resolver.test.ts) - Fix mock setup issues in test suite (149 tests now passing) - Improve error handling with errorResponse helper in actions.ts - Normalize token handling to eliminate duplication Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * use accountId * delete md file * delte tsconfig * adjust log level * fix probe logic * format * fix monitor * code review fixes * format * no mutation * less mutation * chain debug log * await authProvider setup * use uuid * use spread * fix tests * update docs and remove bot channel fallback * more readme fixes * remove comments + fromat * fix tests * adjust access control logic * format * install * simplify config object * remove duplicate log tags + log received messages * update docs * update tests * format * strip markdown in monitor * remove strip markdown config, enabled by default * default requireMention to true * fix store path arg * fix multi account id + add unit test * fix multi account id + add unit test * make channel required and update docs * remove whisper functionality * remove duplicate connect log * update docs with convert twitch link * make twitch message processing non blocking * schema consistent casing * remove noisy ignore log * use coreLogger --------- Co-authored-by: Claude Sonnet 4.5 <noreply@anthropic.com>
367 lines
9.2 KiB
Markdown
367 lines
9.2 KiB
Markdown
---
|
|
summary: "Twitch chat bot configuration and setup"
|
|
read_when:
|
|
- Setting up Twitch chat integration for Clawdbot
|
|
---
|
|
# Twitch (plugin)
|
|
|
|
Twitch chat support via IRC connection. Clawdbot connects as a Twitch user (bot account) to receive and send messages in channels.
|
|
|
|
## Plugin required
|
|
|
|
Twitch ships as a plugin and is not bundled with the core install.
|
|
|
|
Install via CLI (npm registry):
|
|
|
|
```bash
|
|
clawdbot plugins install @clawdbot/twitch
|
|
```
|
|
|
|
Local checkout (when running from a git repo):
|
|
|
|
```bash
|
|
clawdbot plugins install ./extensions/twitch
|
|
```
|
|
|
|
Details: [Plugins](/plugin)
|
|
|
|
## Quick setup (beginner)
|
|
|
|
1) Create a dedicated Twitch account for the bot (or use an existing account).
|
|
2) Generate credentials: [Twitch Token Generator](https://twitchtokengenerator.com/)
|
|
- Select **Bot Token**
|
|
- Verify scopes `chat:read` and `chat:write` are selected
|
|
- Copy the **Client ID** and **Access Token**
|
|
3) Find your Twitch user ID: https://www.streamweasels.com/tools/convert-twitch-username-to-user-id/
|
|
4) Configure the token:
|
|
- Env: `CLAWDBOT_TWITCH_ACCESS_TOKEN=...` (default account only)
|
|
- Or config: `channels.twitch.accessToken`
|
|
- If both are set, config takes precedence (env fallback is default-account only).
|
|
5) Start the gateway.
|
|
|
|
**⚠️ Important:** Add access control (`allowFrom` or `allowedRoles`) to prevent unauthorized users from triggering the bot. `requireMention` defaults to `true`.
|
|
|
|
Minimal config:
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
enabled: true,
|
|
username: "clawdbot", // Bot's Twitch account
|
|
accessToken: "oauth:abc123...", // OAuth Access Token (or use CLAWDBOT_TWITCH_ACCESS_TOKEN env var)
|
|
clientId: "xyz789...", // Client ID from Token Generator
|
|
channel: "vevisk", // Which Twitch channel's chat to join (required)
|
|
allowFrom: ["123456789"] // (recommended) Your Twitch user ID only - get it from https://www.streamweasels.com/tools/convert-twitch-username-to-user-id/
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
## What it is
|
|
|
|
- A Twitch channel owned by the Gateway.
|
|
- Deterministic routing: replies always go back to Twitch.
|
|
- Each account maps to an isolated session key `agent:<agentId>:twitch:<accountName>`.
|
|
- `username` is the bot's account (who authenticates), `channel` is which chat room to join.
|
|
|
|
## Setup (detailed)
|
|
|
|
### Generate credentials
|
|
|
|
Use [Twitch Token Generator](https://twitchtokengenerator.com/):
|
|
- Select **Bot Token**
|
|
- Verify scopes `chat:read` and `chat:write` are selected
|
|
- Copy the **Client ID** and **Access Token**
|
|
|
|
No manual app registration needed. Tokens expire after several hours.
|
|
|
|
### Configure the bot
|
|
|
|
**Env var (default account only):**
|
|
```bash
|
|
CLAWDBOT_TWITCH_ACCESS_TOKEN=oauth:abc123...
|
|
```
|
|
|
|
**Or config:**
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
enabled: true,
|
|
username: "clawdbot",
|
|
accessToken: "oauth:abc123...",
|
|
clientId: "xyz789...",
|
|
channel: "vevisk"
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
If both env and config are set, config takes precedence.
|
|
|
|
### Access control (recommended)
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
allowFrom: ["123456789"], // (recommended) Your Twitch user ID only
|
|
allowedRoles: ["moderator"] // Or restrict to roles
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
**Available roles:** `"moderator"`, `"owner"`, `"vip"`, `"subscriber"`, `"all"`.
|
|
|
|
**Why user IDs?** Usernames can change, allowing impersonation. User IDs are permanent.
|
|
|
|
Find your Twitch user ID: https://www.streamweasels.com/tools/convert-twitch-username-%20to-user-id/ (Convert your Twitch username to ID)
|
|
|
|
## Token refresh (optional)
|
|
|
|
Tokens from [Twitch Token Generator](https://twitchtokengenerator.com/) cannot be automatically refreshed - regenerate when expired.
|
|
|
|
For automatic token refresh, create your own Twitch application at [Twitch Developer Console](https://dev.twitch.tv/console) and add to config:
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
clientSecret: "your_client_secret",
|
|
refreshToken: "your_refresh_token"
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
The bot automatically refreshes tokens before expiration and logs refresh events.
|
|
|
|
## Multi-account support
|
|
|
|
Use `channels.twitch.accounts` with per-account tokens. See [`gateway/configuration`](/gateway/configuration) for the shared pattern.
|
|
|
|
Example (one bot account in two channels):
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
accounts: {
|
|
channel1: {
|
|
username: "clawdbot",
|
|
accessToken: "oauth:abc123...",
|
|
clientId: "xyz789...",
|
|
channel: "vevisk"
|
|
},
|
|
channel2: {
|
|
username: "clawdbot",
|
|
accessToken: "oauth:def456...",
|
|
clientId: "uvw012...",
|
|
channel: "secondchannel"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
**Note:** Each account needs its own token (one token per channel).
|
|
|
|
## Access control
|
|
|
|
### Role-based restrictions
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
accounts: {
|
|
default: {
|
|
allowedRoles: ["moderator", "vip"]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
### Allowlist by User ID (most secure)
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
accounts: {
|
|
default: {
|
|
allowFrom: ["123456789", "987654321"]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
### Combined allowlist + roles
|
|
|
|
Users in `allowFrom` bypass role checks:
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
accounts: {
|
|
default: {
|
|
allowFrom: ["123456789"],
|
|
allowedRoles: ["moderator"]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
### Disable @mention requirement
|
|
|
|
By default, `requireMention` is `true`. To disable and respond to all messages:
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
accounts: {
|
|
default: {
|
|
requireMention: false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
## Troubleshooting
|
|
|
|
First, run diagnostic commands:
|
|
|
|
```bash
|
|
clawdbot doctor
|
|
clawdbot channels status --probe
|
|
```
|
|
|
|
### Bot doesn't respond to messages
|
|
|
|
**Check access control:** Temporarily set `allowedRoles: ["all"]` to test.
|
|
|
|
**Check the bot is in the channel:** The bot must join the channel specified in `channel`.
|
|
|
|
### Token issues
|
|
|
|
**"Failed to connect" or authentication errors:**
|
|
- Verify `accessToken` is the OAuth access token value (typically starts with `oauth:` prefix)
|
|
- Check token has `chat:read` and `chat:write` scopes
|
|
- If using token refresh, verify `clientSecret` and `refreshToken` are set
|
|
|
|
### Token refresh not working
|
|
|
|
**Check logs for refresh events:**
|
|
```
|
|
Using env token source for mybot
|
|
Access token refreshed for user 123456 (expires in 14400s)
|
|
```
|
|
|
|
If you see "token refresh disabled (no refresh token)":
|
|
- Ensure `clientSecret` is provided
|
|
- Ensure `refreshToken` is provided
|
|
|
|
## Config
|
|
|
|
**Account config:**
|
|
- `username` - Bot username
|
|
- `accessToken` - OAuth access token with `chat:read` and `chat:write`
|
|
- `clientId` - Twitch Client ID (from Token Generator or your app)
|
|
- `channel` - Channel to join (required)
|
|
- `enabled` - Enable this account (default: `true`)
|
|
- `clientSecret` - Optional: For automatic token refresh
|
|
- `refreshToken` - Optional: For automatic token refresh
|
|
- `expiresIn` - Token expiry in seconds
|
|
- `obtainmentTimestamp` - Token obtained timestamp
|
|
- `allowFrom` - User ID allowlist
|
|
- `allowedRoles` - Role-based access control (`"moderator" | "owner" | "vip" | "subscriber" | "all"`)
|
|
- `requireMention` - Require @mention (default: `true`)
|
|
|
|
**Provider options:**
|
|
- `channels.twitch.enabled` - Enable/disable channel startup
|
|
- `channels.twitch.username` - Bot username (simplified single-account config)
|
|
- `channels.twitch.accessToken` - OAuth access token (simplified single-account config)
|
|
- `channels.twitch.clientId` - Twitch Client ID (simplified single-account config)
|
|
- `channels.twitch.channel` - Channel to join (simplified single-account config)
|
|
- `channels.twitch.accounts.<accountName>` - Multi-account config (all account fields above)
|
|
|
|
Full example:
|
|
|
|
```json5
|
|
{
|
|
channels: {
|
|
twitch: {
|
|
enabled: true,
|
|
username: "clawdbot",
|
|
accessToken: "oauth:abc123...",
|
|
clientId: "xyz789...",
|
|
channel: "vevisk",
|
|
clientSecret: "secret123...",
|
|
refreshToken: "refresh456...",
|
|
allowFrom: ["123456789"],
|
|
allowedRoles: ["moderator", "vip"],
|
|
accounts: {
|
|
default: {
|
|
username: "mybot",
|
|
accessToken: "oauth:abc123...",
|
|
clientId: "xyz789...",
|
|
channel: "your_channel",
|
|
enabled: true,
|
|
clientSecret: "secret123...",
|
|
refreshToken: "refresh456...",
|
|
expiresIn: 14400,
|
|
obtainmentTimestamp: 1706092800000,
|
|
allowFrom: ["123456789", "987654321"],
|
|
allowedRoles: ["moderator"]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
## Tool actions
|
|
|
|
The agent can call `twitch` with action:
|
|
- `send` - Send a message to a channel
|
|
|
|
Example:
|
|
|
|
```json5
|
|
{
|
|
"action": "twitch",
|
|
"params": {
|
|
"message": "Hello Twitch!",
|
|
"to": "#mychannel"
|
|
}
|
|
}
|
|
```
|
|
|
|
## Safety & ops
|
|
|
|
- **Treat tokens like passwords** - Never commit tokens to git
|
|
- **Use automatic token refresh** for long-running bots
|
|
- **Use user ID allowlists** instead of usernames for access control
|
|
- **Monitor logs** for token refresh events and connection status
|
|
- **Scope tokens minimally** - Only request `chat:read` and `chat:write`
|
|
- **If stuck**: Restart the gateway after confirming no other process owns the session
|
|
|
|
## Limits
|
|
|
|
- **500 characters** per message (auto-chunked at word boundaries)
|
|
- Markdown is stripped before chunking
|
|
- No rate limiting (uses Twitch's built-in rate limits)
|