Релиз
All checks were successful
Build And Publish Package / publish (push) Successful in 50s

This commit is contained in:
2026-02-25 09:53:19 +03:00
commit da232d7461
18 changed files with 7443 additions and 0 deletions

58
.gitea/workflows/dev.yaml Normal file
View File

@ -0,0 +1,58 @@
name: Verify Dev Build
run-name: ${{ github.actor }} verifying dev build
on:
push:
branches:
- dev
jobs:
publish:
runs-on: ubuntu-latest
steps:
- name: Checkout code
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Cache uv binary
uses: actions/cache@v4
with:
path: ${{ github.workspace }}/uv
key: uv-${{ runner.os }}
restore-keys: uv-${{ runner.os }}
- name: Cache uv dependencies
uses: actions/cache@v4
with:
path: ${{ github.workspace }}/.cache/uv
key: uv-${{ runner.os }}
restore-keys: uv-${{ runner.os }}
- name: Cache pre-commit
uses: actions/cache@v4
with:
path: ~/.cache/pre-commit
key: pre-commit-cache-${{ runner.os }}-${{ hashFiles('.pre-commit-config.yaml') }}
restore-keys: pre-commit-cache-${{ runner.os }}-
- name: Install uv
uses: astral-sh/setup-uv@v5
with:
version: "0.10.5"
enable-cache: true
cache-local-path: ${{ github.workspace }}/.cache/uv
tool-dir: ${{ github.workspace }}/.cache/uv
tool-bin-dir: ${{ github.workspace }}/.cache/uv
cache-dependency-glob: ""
- name: Set up Python
run: uv python install
- name: Install the project
run: uv sync --all-extras --no-install-project --cache-dir ${{ github.workspace }}/.cache/uv
- name: Linter & Formatter
run: uv run pre-commit run --all-files
- name: Build Package
run: uv build --cache-dir ${{ github.workspace }}/.cache/uv

View File

@ -0,0 +1,64 @@
name: Build And Publish Package
run-name: ${{ github.actor }} builds and publishes package to PyPI
on:
push:
branches:
- latest
jobs:
publish:
runs-on: ubuntu-latest
steps:
- name: Checkout code
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Cache uv binary
uses: actions/cache@v4
with:
path: ${{ github.workspace }}/uv
key: uv-${{ runner.os }}
restore-keys: uv-${{ runner.os }}
- name: Cache uv dependencies
uses: actions/cache@v4
with:
path: ${{ github.workspace }}/.cache/uv
key: uv-${{ runner.os }}
restore-keys: uv-${{ runner.os }}
- name: Cache pre-commit
uses: actions/cache@v4
with:
path: ~/.cache/pre-commit
key: pre-commit-cache-${{ runner.os }}-${{ hashFiles('.pre-commit-config.yaml') }}
restore-keys: pre-commit-cache-${{ runner.os }}-
- name: Install uv
uses: astral-sh/setup-uv@v5
with:
version: "0.10.5"
enable-cache: true
cache-local-path: ${{ github.workspace }}/.cache/uv
tool-dir: ${{ github.workspace }}/.cache/uv
tool-bin-dir: ${{ github.workspace }}/.cache/uv
cache-dependency-glob: ""
- name: Set up Python
run: uv python install
- name: Install the project
run: uv sync --all-extras --no-install-project --cache-dir ${{ github.workspace }}/.cache/uv
- name: Linter & Formatter
run: uv run pre-commit run --all-files
- name: Build Package
run: uv build --cache-dir ${{ github.workspace }}/.cache/uv
- name: Publish to Gitea PyPI
run: |
uv publish \
--index OxideTwitch \
--token ${{ secrets.CI_TOKEN }}

16
.gitignore vendored Normal file
View File

@ -0,0 +1,16 @@
# Python-generated files
__pycache__/
*.py[oc]
build/
dist/
wheels/
*.egg-info
# Virtual environments
.venv
# Ruff
.ruff_cache
# uv
uv.lock

34
.pre-commit-config.yaml Normal file
View File

@ -0,0 +1,34 @@
repos:
- repo: https://github.com/crate-ci/typos
rev: v1.42.0
hooks:
- id: typos
- repo: https://github.com/astral-sh/ruff-pre-commit
rev: v0.15.0
hooks:
- id: ruff
args: [ --fix ]
- id: ruff-format
- repo: https://github.com/RobertCraigie/pyright-python
rev: v1.1.408
hooks:
- id: pyright
- repo: https://github.com/pre-commit/pre-commit-hooks
rev: v6.0.0
hooks:
- id: trailing-whitespace
- id: check-docstring-first
- id: check-added-large-files
- id: check-yaml
- id: debug-statements
- id: check-merge-conflict
- id: double-quote-string-fixer
- id: end-of-file-fixer
- repo: meta
hooks:
- id: check-hooks-apply
- id: check-useless-excludes

81
README.md Normal file
View File

@ -0,0 +1,81 @@
# 🟣 OxideTwitch
**OxideTwitch** is a specialized Twitch API client built on the [OxideHTTP](https://git.miwory.dev/OxideHTTP/OxideHTTP) core. It combines the speed of Rust with an easy-to-use Pythonic interface, specifically tuned for the rigors of the Twitch Developer ecosystem.
---
## 🔥 Why OxideTwitch?
Twitch's API has strict rate limits and requires constant token management. OxideTwitch handles the heavy lifting for you:
* **⚡ Oxide Core:** Uses `pyreqwest` (Rust) for underlying HTTP calls.
* **🛡️ Distributed Rate Limiting:** Uses your OxideHTTP Redis integration to ensure your bot never hits a `429 Too Many Requests` even across multiple instances.
* **💾 Intelligent Caching:** Automatically caches common lookups (like User IDs or Stream Status) to save your API quota.
* **🏗️ Type-Safe Models:** Fully validated responses using Pydantic models.
---
## 📦 Installation
Ensure you have your Gitea index configured in `uv`, then run:
```bash
uv add oxidetwitch
```
---
## 🛠 Quick Start
### Basic User Lookup
OxideTwitch automatically handles the `base_url` and header injection for you.
```python
import asyncio
from oxidetwitch.api import TwitchAPIClient
async def main():
async with TwitchClient(
client_id="your_id",
client_secret="your_client_secret",
redis_url="redis://localhost:6379",
) as twitch:
# Get user data (automatically cached if configured)
users = await twitch.get_users(access_token="access_token", login="Miwowy")
user = users.data[0]
print(f"User ID: {user.id} | Description: {user.description}")
asyncio.run(main())
```
### Handling Streams with Rate Limiting
If you are polling 100+ streams, OxideTwitch spaces out the requests using the **GCRA algorithm** to keep your token healthy.
```python
async def poll_streams(channels):
async with TwitchClient(...) as twitch:
# These will be executed as fast as the rate limiter allows
tasks = [twitch.get_stream(user_login=name) for name in channels]
streams = await asyncio.gather(*tasks)
return [s for s in streams if s.is_live]
```
---
## ⚙️ Advanced: Using with `uv` and Gitea
Since **OxideTwitch** depends on **OxideHTTP**, ensure your `pyproject.toml` is configured to find both in your private registry:
```toml
[[tool.uv.index]]
name = "OxideTwitch"
url = "https://git.miwory.dev/api/packages/OxideHTTP/pypi/simple"
```
---

104
pyproject.toml Normal file
View File

@ -0,0 +1,104 @@
[project]
name = "oxidetwitch"
version = "1.0.0"
description = "Client for Twitch API"
readme = "README.md"
authors = [{ name = "Miwory", email = "miwory.uwu@gmail.com" }]
requires-python = ">=3.14"
dependencies = ["oxidehttp>=1.0.3,<=2.0.0", "pydantic>=2.12,<=2.13"]
[build-system]
requires = ["uv_build>=0.9.2,<0.10.0"]
build-backend = "uv_build"
[dependency-groups]
dev = [
"ty>=0.0.17",
"ruff>=0.15.0",
"pyright>=1.1.408",
"poethepoet>=0.40.0",
"pre-commit>=4.5.1",
]
[[tool.uv.index]]
name = "OxideHTTP"
url = "https://git.miwory.dev/api/packages/OxideHTTP/pypi/simple"
[[tool.uv.index]]
name = "OxideTwitch"
url = "https://git.miwory.dev/api/packages/OxideHTTP/pypi/simple"
publish-url = "https://git.miwory.dev/api/packages/OxideHTTP/pypi"
explicit = true
[tool.poe.tasks]
_git = "git add ."
_lint = "pre-commit run --all-files"
lint = ["_git", "_lint"]
check = "uv pip ls --outdated"
major = "uv version --bump major"
minor = "uv version --bump minor"
patch = "uv version --bump patch"
[tool.pyright]
venvPath = "."
venv = ".venv"
strictListInference = true
strictDictionaryInference = true
strictSetInference = true
deprecateTypingAliases = true
typeCheckingMode = "strict"
pythonPlatform = "All"
stubPath = "typings"
[tool.ruff]
target-version = "py313"
line-length = 79
fix = true
[tool.ruff.lint]
preview = true
select = [
"E",
"W",
"F",
"UP",
"A",
"B",
"C4",
"SIM",
"I",
"S",
"G",
"FAST",
"ASYNC",
"BLE",
"INT",
"ISC",
"ICN",
"PYI",
"INP",
"RSE",
"PIE",
"SLOT",
"TID",
"LOG",
"FBT",
"DTZ",
"EM",
"PERF",
"RUF",
]
ignore = ["RUF002", "RUF029", "S101", "S104", "W505"]
[tool.ruff.lint.pydoclint]
ignore-one-line-docstrings = true
[tool.ruff.format]
quote-style = "single"
indent-style = "space"
docstring-code-format = true
[tool.uv.sources]
aiohttpx = { index = "Miwory" }

View File

@ -0,0 +1 @@
__version__ = '0.1.0'

4412
src/oxidetwitch/api.py Normal file

File diff suppressed because it is too large Load Diff

131
src/oxidetwitch/auth.py Normal file
View File

@ -0,0 +1,131 @@
from urllib.parse import urlencode
from oxidehttp import status as st
from oxidehttp.client import OxideHTTP
from . import schema as s
from . import scopes
class TwitchAuthClient(OxideHTTP):
def __init__(
self,
client_id: str,
client_secret: str,
redirect_uri: str,
redis_url: str | None = None,
proxy_url: str | None = None,
) -> None:
self.base_uri = 'https://id.twitch.tv/oauth2'
self.client_id = client_id
self.client_secret = client_secret
self.redirect_uri = redirect_uri
super().__init__(
base_url=self.base_uri,
redis_url=redis_url,
ratelimit_key='twitch' if redis_url else None,
ratelimit_limit=700 if redis_url else None,
proxy_url=proxy_url,
)
async def create_authorization_code_grant_flow_url(
self,
scope: list[scopes.Any],
*,
force_verify: bool = False,
state: str | None = None,
) -> str:
url = 'https://id.twitch.tv/oauth2/authorize?'
query = self.clean_dict(
{
'client_id': self.client_id,
'redirect_uri': self.redirect_uri,
'response_type': 'code',
'scope': ' '.join(scope),
'state': state,
'force_verify': force_verify,
}
)
return url + urlencode(query)
async def app_access_token(self) -> s.AppAccessToken:
req = await self.post(
'/token',
json={
'client_id': self.client_id,
'client_secret': self.client_secret,
'grant_type': 'client_credentials',
},
)
match req.status_code:
case st.OK:
return s.AppAccessToken.model_validate(req.json())
case _:
raise s.InternalError(req.status_code, 'Internal Server Error')
async def user_access_token(self, code: str) -> s.UserAccessToken | None:
req = await self.post(
'/token',
json={
'client_id': self.client_id,
'client_secret': self.client_secret,
'redirect_uri': self.redirect_uri,
'grant_type': 'authorization_code',
'code': code,
},
)
match req.status_code:
case st.OK:
return s.UserAccessToken.model_validate(req.json())
case st.BAD_REQUEST:
return None
case _:
raise s.InternalError(req.status_code, 'Internal Server Error')
async def validate_access_token(
self, access_token: str
) -> s.AccessTokenValidation | None:
req = await self.get(
'/validate',
headers={'Authorization': f'OAuth {access_token}'},
)
match req.status_code:
case st.OK:
return s.AccessTokenValidation.model_validate(req.json())
case st.UNAUTHORIZED:
return None
case _:
raise s.InternalError(req.status_code, 'Internal Server Error')
async def refresh_access_token(
self, refresh_token: str
) -> s.UserAccessToken | None:
req = await self.post(
'/token',
json={
'client_id': self.client_id,
'client_secret': self.client_secret,
'grant_type': 'refresh_token',
'refresh_token': refresh_token,
},
)
match req.status_code:
case st.OK:
return s.UserAccessToken.model_validate(req.json())
case st.BAD_REQUEST:
return None
case _:
raise s.InternalError(req.status_code, 'Internal Server Error')

View File

View File

@ -0,0 +1,51 @@
from pydantic import BaseModel
class User(BaseModel):
user_id: str
class Broadcaster(BaseModel):
broadcaster_user_id: str
class ToBroadcaster(BaseModel):
to_broadcaster_user_id: str
class FromBroadcaster(BaseModel):
from_broadcaster_user_id: str
class BroadcasterUser(Broadcaster, User):
pass
class BroadcasterModerator(Broadcaster):
moderator_user_id: str
class ClientID(BaseModel):
client_id: str
class Organization(BaseModel):
organization_id: int
category_id: int | None = None
campaign_id: int | None = None
class ExtensionClientID(BaseModel):
extension_client_id: str
Any = (
Broadcaster
| ToBroadcaster
| FromBroadcaster
| BroadcasterUser
| BroadcasterModerator
| ClientID
| Organization
| ExtensionClientID
)

View File

@ -0,0 +1,43 @@
from typing import Literal
ENABLED = Literal['enabled']
PENDING = Literal['webhook_callback_verification_pending']
FAILED = Literal['webhook_callback_verification_failed']
EXCEEDED = Literal['notification_failures_exceeded']
REVOKED = Literal['authorization_revoked']
MODERATOR_REMOVED = Literal['moderator_removed']
USER_REMOVED = Literal['user_removed']
CHAT_USER_BANNED = Literal['chat_user_banned']
VERSION_REMOVED = Literal['version_removed']
BETA_MAINTENANCE = Literal['beta_maintenance']
WEBSOCKET_DISCONNECTED = Literal['websocket_disconnected']
WEBSOCKET_FAILED_PING_PONG = Literal['websocket_failed_ping_pong']
WEBSOCKET_RECEIVED_INBOUND_TRAFFIC = Literal[
'websocket_received_inbound_traffic'
]
WEBSOCKET_CONNECTION_UNUSED = Literal['websocket_connection_unused']
WEBSOCKET_INTERNAL_ERROR = Literal['websocket_internal_error']
WEBSOCKET_NETWORK_TIMEOUT = Literal['websocket_network_timeout']
WEBSOCKET_NETWORK_ERROR = Literal['websocket_network_error']
WEBSOCKET_FAILED_TO_RECONNECT = Literal['websocket_failed_to_reconnect']
Any = (
ENABLED
| PENDING
| FAILED
| EXCEEDED
| REVOKED
| MODERATOR_REMOVED
| USER_REMOVED
| CHAT_USER_BANNED
| VERSION_REMOVED
| BETA_MAINTENANCE
| WEBSOCKET_DISCONNECTED
| WEBSOCKET_FAILED_PING_PONG
| WEBSOCKET_RECEIVED_INBOUND_TRAFFIC
| WEBSOCKET_CONNECTION_UNUSED
| WEBSOCKET_INTERNAL_ERROR
| WEBSOCKET_NETWORK_TIMEOUT
| WEBSOCKET_NETWORK_ERROR
| WEBSOCKET_FAILED_TO_RECONNECT
)

View File

@ -0,0 +1,564 @@
from datetime import datetime
from pydantic import BaseModel
from . import conditions as c
from . import statuses as s
from . import transports as t
from . import types as st
class BaseSubscription(BaseModel):
id: str
version: int
status: s.Any
cost: int
created_at: datetime
transport: t.Any
default_version: int = 1
class AutomodMessageHold(BaseSubscription):
type: st.AUTOMOD_MESSAGE_HOLD
condition: c.BroadcasterModerator
default_version: int = 2
class AutomodMessageUpdate(BaseSubscription):
type: st.AUTOMOD_MESSAGE_UPDATE
condition: c.BroadcasterModerator
default_version: int = 2
class AutomodSettingsUpdate(BaseSubscription):
type: st.AUTOMOD_SETTINGS_UPDATE
condition: c.BroadcasterModerator
default_version: int = 1
class AutomodTermsUpdate(BaseSubscription):
type: st.AUTOMOD_TERMS_UPDATE
condition: c.BroadcasterModerator
default_version: int = 1
class ChannelBitsUse(BaseSubscription):
type: st.CHANNEL_BITS_USE
condition: c.Broadcaster
default_version: int = 1
class ChannelUpdate(BaseSubscription):
type: st.CHANNEL_UPDATE
condition: c.Broadcaster
default_version: int = 2
class ChannelFollow(BaseSubscription):
type: st.CHANNEL_FOLLOW
condition: c.BroadcasterModerator
default_version: int = 2
class ChannelAdBreakBegin(BaseSubscription):
type: st.CHANNEL_AD_BREAK_BEGIN
condition: c.Broadcaster
default_version: int = 1
class ChannelChatClear(BaseSubscription):
type: st.CHANNEL_CHAT_CLEAR
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatClearUserMessages(BaseSubscription):
type: st.CHANNEL_CHAT_CLEAR_USER_MESSAGES
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatMessage(BaseSubscription):
type: st.CHANNEL_CHAT_MESSAGE
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatMessageDelete(BaseSubscription):
type: st.CHANNEL_CHAT_MESSAGE_DELETE
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatNotification(BaseSubscription):
type: st.CHANNEL_CHAT_NOTIFICATION
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatSettingsUpdate(BaseSubscription):
type: st.CHANNEL_CHAT_SETTINGS_UPDATE
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatUserMessageHold(BaseSubscription):
type: st.CHANNEL_CHAT_USER_MESSAGE_HOLD
condition: c.BroadcasterUser
default_version: int = 1
class ChannelChatUserMessageUpdate(BaseSubscription):
type: st.CHANNEL_CHAT_USER_MESSAGE_UPDATE
condition: c.BroadcasterUser
default_version: int = 1
class ChannelSharedChatSessionBegin(BaseSubscription):
type: st.CHANNEL_SHARED_CHAT_SESSION_BEGIN
condition: c.Broadcaster
default_version: int = 1
class ChannelSharedChatSessionUpdate(BaseSubscription):
type: st.CHANNEL_SHARED_CHAT_SESSION_UPDATE
condition: c.Broadcaster
default_version: int = 1
class ChannelSharedChatSessionEnd(BaseSubscription):
type: st.CHANNEL_SHARED_CHAT_SESSION_END
condition: c.Broadcaster
default_version: int = 1
class ChannelSubscribe(BaseSubscription):
type: st.CHANNEL_SUBSCRIBE
condition: c.Broadcaster
default_version: int = 1
class ChannelSubscriptionEnd(BaseSubscription):
type: st.CHANNEL_SUBSCRIPTION_END
condition: c.Broadcaster
default_version: int = 1
class ChannelSubscriptionGift(BaseSubscription):
type: st.CHANNEL_SUBSCRIPTION_GIFT
condition: c.Broadcaster
default_version: int = 1
class ChannelSubscriptionMessage(BaseSubscription):
type: st.CHANNEL_SUBSCRIPTION_MESSAGE
condition: c.Broadcaster
default_version: int = 1
class ChannelCheer(BaseSubscription):
type: st.CHANNEL_CHEER
condition: c.Broadcaster
default_version: int = 1
class ChannelRaid(BaseSubscription):
type: st.CHANNEL_RAID
condition: c.ToBroadcaster
default_version: int = 1
class ChannelBan(BaseSubscription):
type: st.CHANNEL_BAN
condition: c.Broadcaster
default_version: int = 1
class ChannelUnban(BaseSubscription):
type: st.CHANNEL_UNBAN
condition: c.Broadcaster
default_version: int = 1
class ChannelUnbanRequestCreate(BaseSubscription):
type: st.CHANNEL_UNBAN_REQUEST_CREATE
condition: c.BroadcasterModerator
default_version: int = 1
class ChannelUnbanRequestResolve(BaseSubscription):
type: st.CHANNEL_UNBAN_REQUEST_RESOLVE
condition: c.BroadcasterModerator
default_version: int = 1
class ChannelModerate(BaseSubscription):
type: st.CHANNEL_MODERATE
condition: c.BroadcasterModerator
default_version: int = 2
class ChannelModeratorAdd(BaseSubscription):
type: st.CHANNEL_MODERATOR_ADD
condition: c.Broadcaster
default_version: int = 1
class ChannelModeratorRemove(BaseSubscription):
type: st.CHANNEL_MODERATOR_REMOVE
condition: c.Broadcaster
default_version: int = 1
# TODO: Complete when subscription version is released
# class ChannelGuestStarSessionBegin(BaseSubscription):
# type: st.CHANNEL_GUEST_STAR_SESSION_BEGIN
# condition: c.Broadcaster
# default_version: int = 1
# TODO: Complete when subscription version is released
# class ChannelGuestStarSessionEnd(BaseSubscription):
# type: st.CHANNEL_GUEST_STAR_SESSION_END
# condition: c.Broadcaster
# default_version: int = 1
# TODO: Complete when subscription version is released
# class ChannelGuestStarGuestUpdate(BaseSubscription):
# type: st.CHANNEL_GUEST_STAR_GUEST_UPDATE
# condition: c.Broadcaster
# default_version: int = 1
# TODO: Complete when subscription version is released
# class ChannelGuestStarSettingsUpdate(BaseSubscription):
# type: st.CHANNEL_GUEST_STAR_SETTINGS_UPDATE
# condition: c.Broadcaster
# default_version: int = 1
class ChannelPointsAutomaticRewardRedemptionAdd(BaseSubscription):
type: st.CHANNEL_POINTS_AUTOMATIC_REWARD_REDEMPTION_ADD
condition: c.Broadcaster
default_version: int = 2
class ChannelPointsCustomRewardAdd(BaseSubscription):
type: st.CHANNEL_POINTS_CUSTOM_REWARD_ADD
condition: c.Broadcaster
default_version: int = 1
class ChannelPointsCustomRewardUpdate(BaseSubscription):
type: st.CHANNEL_POINTS_CUSTOM_REWARD_UPDATE
condition: c.Broadcaster
default_version: int = 1
class ChannelPointsCustomRewardRemove(BaseSubscription):
type: st.CHANNEL_POINTS_CUSTOM_REWARD_REMOVE
condition: c.Broadcaster
default_version: int = 1
class ChannelPointsCustomRewardRedemptionAdd(BaseSubscription):
type: st.CHANNEL_POINTS_CUSTOM_REWARD_REDEMPTION_ADD
condition: c.Broadcaster
default_version: int = 1
class ChannelPointsCustomRewardRedemptionUpdate(BaseSubscription):
type: st.CHANNEL_POINTS_CUSTOM_REWARD_REDEMPTION_UPDATE
condition: c.Broadcaster
default_version: int = 1
class ChannelPollBegin(BaseSubscription):
type: st.CHANNEL_POLL_BEGIN
condition: c.Broadcaster
default_version: int = 1
class ChannelPollProgress(BaseSubscription):
type: st.CHANNEL_POLL_PROGRESS
condition: c.Broadcaster
default_version: int = 1
class ChannelPollEnd(BaseSubscription):
type: st.CHANNEL_POLL_END
condition: c.Broadcaster
default_version: int = 1
class ChannelPredictionBegin(BaseSubscription):
type: st.CHANNEL_PREDICTION_BEGIN
condition: c.Broadcaster
default_version: int = 1
class ChannelPredictionProgress(BaseSubscription):
type: st.CHANNEL_PREDICTION_PROGRESS
condition: c.Broadcaster
default_version: int = 1
class ChannelPredictionLock(BaseSubscription):
type: st.CHANNEL_PREDICTION_LOCK
condition: c.Broadcaster
default_version: int = 1
class ChannelPredictionEnd(BaseSubscription):
type: st.CHANNEL_PREDICTION_END
condition: c.Broadcaster
default_version: int = 1
class ChannelSuspiciousUserMessage(BaseSubscription):
type: st.CHANNEL_SUSPICIOUS_USER_MESSAGE
condition: c.BroadcasterModerator
default_version: int = 1
class ChannelSuspiciousUserUpdate(BaseSubscription):
type: st.CHANNEL_SUSPICIOUS_USER_UPDATE
condition: c.BroadcasterModerator
default_version: int = 1
class ChannelVIPAdd(BaseSubscription):
type: st.CHANNEL_VIP_ADD
condition: c.Broadcaster
default_version: int = 1
class ChannelVIPRemove(BaseSubscription):
type: st.CHANNEL_VIP_REMOVE
condition: c.Broadcaster
default_version: int = 1
class ChannelWarningAcknowledgement(BaseSubscription):
type: st.CHANNEL_WARNING_ACKNOWLEDGEMENT
condition: c.BroadcasterModerator
default_version: int = 1
class ChannelWarningSend(BaseSubscription):
type: st.CHANNEL_WARNING_SEND
condition: c.BroadcasterModerator
default_version: int = 1
class CharityDonation(BaseSubscription):
type: st.CHARITY_DONATION
condition: c.Broadcaster
default_version: int = 1
class CharityCampaignStart(BaseSubscription):
type: st.CHARITY_CAMPAIGN_START
condition: c.Broadcaster
default_version: int = 1
class CharityCampaignProgress(BaseSubscription):
type: st.CHARITY_CAMPAIGN_PROGRESS
condition: c.Broadcaster
default_version: int = 1
class CharityCampaignStop(BaseSubscription):
type: st.CHARITY_CAMPAIGN_STOP
condition: c.Broadcaster
default_version: int = 1
class ConduitShardDisabled(BaseSubscription):
type: st.CONDUIT_SHARD_DISABLED
condition: c.ClientID
default_version: int = 1
class DropEntitlementGrant(BaseSubscription):
type: st.DROP_ENTITLEMENT_GRANT
condition: c.Organization
default_version: int = 1
class ExtensionBitsTransactionCreate(BaseSubscription):
type: st.EXTENSION_BITS_TRANSACTION_CREATE
condition: c.ExtensionClientID
default_version: int = 1
class GoalBegin(BaseSubscription):
type: st.GOAL_BEGIN
condition: c.Broadcaster
default_version: int = 1
class GoalProgress(BaseSubscription):
type: st.GOAL_PROGRESS
condition: c.Broadcaster
default_version: int = 1
class GoalEnd(BaseSubscription):
type: st.GOAL_END
condition: c.Broadcaster
default_version: int = 1
class HypeTrainBegin(BaseSubscription):
type: st.HYPE_TRAIN_BEGIN
condition: c.Broadcaster
default_version: int = 2
class HypeTrainProgress(BaseSubscription):
type: st.HYPE_TRAIN_PROGRESS
condition: c.Broadcaster
default_version: int = 2
class HypeTrainEnd(BaseSubscription):
type: st.HYPE_TRAIN_END
condition: c.Broadcaster
default_version: int = 2
class ShieldModeBegin(BaseSubscription):
type: st.SHIELD_MODE_BEGIN
condition: c.BroadcasterModerator
default_version: int = 1
class ShieldModeEnd(BaseSubscription):
type: st.SHIELD_MODE_END
condition: c.BroadcasterModerator
default_version: int = 1
class ShoutoutCreate(BaseSubscription):
type: st.SHOUTOUT_CREATE
condition: c.BroadcasterModerator
default_version: int = 1
class ShoutoutReceived(BaseSubscription):
type: st.SHOUTOUT_RECEIVED
condition: c.BroadcasterModerator
default_version: int = 1
class StreamOnline(BaseSubscription):
type: st.STREAM_ONLINE
condition: c.Broadcaster
default_version: int = 1
class StreamOffline(BaseSubscription):
type: st.STREAM_OFFLINE
condition: c.Broadcaster
default_version: int = 1
class UserAuthorizationGrant(BaseSubscription):
type: st.USER_AUTHORIZATION_GRANT
condition: c.ClientID
default_version: int = 1
class UserAuthorizationRevoke(BaseSubscription):
type: st.USER_AUTHORIZATION_REVOKE
condition: c.ClientID
default_version: int = 1
class UserUpdate(BaseSubscription):
type: st.USER_UPDATE
condition: c.User
default_version: int = 1
class WhisperReceived(BaseSubscription):
type: st.WHISPER_RECEIVED
condition: c.User
default_version: int = 1
Any = (
AutomodMessageHold
| AutomodMessageUpdate
| AutomodSettingsUpdate
| AutomodTermsUpdate
| ChannelBitsUse
| ChannelUpdate
| ChannelFollow
| ChannelAdBreakBegin
| ChannelChatClear
| ChannelChatClearUserMessages
| ChannelChatMessage
| ChannelChatMessageDelete
| ChannelChatNotification
| ChannelChatSettingsUpdate
| ChannelChatUserMessageHold
| ChannelChatUserMessageUpdate
| ChannelSharedChatSessionBegin
| ChannelSharedChatSessionUpdate
| ChannelSharedChatSessionEnd
| ChannelSubscribe
| ChannelSubscriptionEnd
| ChannelSubscriptionGift
| ChannelSubscriptionMessage
| ChannelCheer
| ChannelRaid
| ChannelBan
| ChannelUnban
| ChannelUnbanRequestCreate
| ChannelUnbanRequestResolve
| ChannelModerate
| ChannelModeratorAdd
| ChannelModeratorRemove
| ChannelPointsAutomaticRewardRedemptionAdd
| ChannelPointsCustomRewardAdd
| ChannelPointsCustomRewardUpdate
| ChannelPointsCustomRewardRemove
| ChannelPointsCustomRewardRedemptionAdd
| ChannelPointsCustomRewardRedemptionUpdate
| ChannelPollBegin
| ChannelPollProgress
| ChannelPollEnd
| ChannelPredictionBegin
| ChannelPredictionProgress
| ChannelPredictionLock
| ChannelPredictionEnd
| ChannelSuspiciousUserMessage
| ChannelSuspiciousUserUpdate
| ChannelVIPAdd
| ChannelVIPRemove
| ChannelWarningAcknowledgement
| ChannelWarningSend
| CharityDonation
| CharityCampaignStart
| CharityCampaignProgress
| CharityCampaignStop
| ConduitShardDisabled
| DropEntitlementGrant
| ExtensionBitsTransactionCreate
| GoalBegin
| GoalProgress
| GoalEnd
| HypeTrainBegin
| HypeTrainProgress
| HypeTrainEnd
| ShoutoutCreate
| ShoutoutReceived
| StreamOnline
| StreamOffline
| UserAuthorizationGrant
| UserAuthorizationRevoke
| UserUpdate
| WhisperReceived
)

View File

@ -0,0 +1,30 @@
from datetime import datetime
from typing import Literal
from pydantic import BaseModel
class BaseWebhookTransport(BaseModel):
method: Literal['webhook']
callback: str
class BaseWebsocketConnectedTransport(BaseModel):
method: Literal['websocket']
session_id: str
connected_at: datetime
class BaseWebsocketDisconnectedTransport(BaseWebsocketConnectedTransport):
disconnected_at: datetime
class SubscribeWebhookTransport(BaseWebhookTransport):
secret: str
Any = (
BaseWebhookTransport
| BaseWebsocketConnectedTransport
| BaseWebsocketDisconnectedTransport
)

View File

@ -0,0 +1,201 @@
from typing import Literal
# Automod types
AUTOMOD_MESSAGE_HOLD = Literal['automod.message.hold']
AUTOMOD_MESSAGE_UPDATE = Literal['automod.message.update']
AUTOMOD_SETTINGS_UPDATE = Literal['automod.settings.update']
AUTOMOD_TERMS_UPDATE = Literal['automod.terms.update']
# Channel types
CHANNEL_BITS_USE = Literal['channel.bits.use']
CHANNEL_UPDATE = Literal['channel.update']
CHANNEL_FOLLOW = Literal['channel.follow']
CHANNEL_AD_BREAK_BEGIN = Literal['channel.ad_break.begin']
CHANNEL_CHAT_CLEAR = Literal['channel.chat.clear']
CHANNEL_CHAT_CLEAR_USER_MESSAGES = Literal['channel.chat.clear_user_messages']
CHANNEL_CHAT_MESSAGE = Literal['channel.chat.message']
CHANNEL_CHAT_MESSAGE_DELETE = Literal['channel.chat.message_delete']
CHANNEL_CHAT_NOTIFICATION = Literal['channel.chat.notification']
CHANNEL_CHAT_SETTINGS_UPDATE = Literal['channel.chat_settings.update']
CHANNEL_CHAT_USER_MESSAGE_HOLD = Literal['channel.chat.user_message_hold']
CHANNEL_CHAT_USER_MESSAGE_UPDATE = Literal['channel.chat.user_message_update']
CHANNEL_SHARED_CHAT_SESSION_BEGIN = Literal['channel.shared_chat.begin']
CHANNEL_SHARED_CHAT_SESSION_UPDATE = Literal['channel.shared_chat.update']
CHANNEL_SHARED_CHAT_SESSION_END = Literal['channel.shared_chat.end']
CHANNEL_SUBSCRIBE = Literal['channel.subscribe']
CHANNEL_SUBSCRIPTION_END = Literal['channel.subscription.end']
CHANNEL_SUBSCRIPTION_GIFT = Literal['channel.subscription.gift']
CHANNEL_SUBSCRIPTION_MESSAGE = Literal['channel.subscription.message']
CHANNEL_CHEER = Literal['channel.cheer']
CHANNEL_RAID = Literal['channel.raid']
CHANNEL_BAN = Literal['channel.ban']
CHANNEL_UNBAN = Literal['channel.unban']
CHANNEL_UNBAN_REQUEST_CREATE = Literal['channel.unban_request.create']
CHANNEL_UNBAN_REQUEST_RESOLVE = Literal['channel.unban_request.resolve']
CHANNEL_MODERATE = Literal['channel.moderate']
CHANNEL_MODERATOR_ADD = Literal['channel.moderator.add']
CHANNEL_MODERATOR_REMOVE = Literal['channel.moderator.remove']
CHANNEL_GUEST_STAR_SESSION_BEGIN = Literal['channel.guest_star_session.begin']
CHANNEL_GUEST_STAR_SESSION_END = Literal['channel.guest_star_session.end']
CHANNEL_GUEST_STAR_GUEST_UPDATE = Literal['channel.guest_star_guest.update']
CHANNEL_GUEST_STAR_SETTINGS_UPDATE = Literal[
'channel.guest_star_settings.update'
]
CHANNEL_POINTS_AUTOMATIC_REWARD_REDEMPTION_ADD = Literal[
'channel.channel_points_automatic_reward_redemption.add'
]
CHANNEL_POINTS_CUSTOM_REWARD_ADD = Literal[
'channel.channel_points_custom_reward.add'
]
CHANNEL_POINTS_CUSTOM_REWARD_UPDATE = Literal[
'channel.channel_points_custom_reward.update'
]
CHANNEL_POINTS_CUSTOM_REWARD_REMOVE = Literal[
'channel.channel_points_custom_reward.remove'
]
CHANNEL_POINTS_CUSTOM_REWARD_REDEMPTION_ADD = Literal[
'channel.channel_points_custom_reward_redemption.add'
]
CHANNEL_POINTS_CUSTOM_REWARD_REDEMPTION_UPDATE = Literal[
'channel.channel_points_custom_reward_redemption.update'
]
CHANNEL_POLL_BEGIN = Literal['channel.poll.begin']
CHANNEL_POLL_PROGRESS = Literal['channel.poll.progress']
CHANNEL_POLL_END = Literal['channel.poll.end']
CHANNEL_PREDICTION_BEGIN = Literal['channel.prediction.begin']
CHANNEL_PREDICTION_PROGRESS = Literal['channel.prediction.progress']
CHANNEL_PREDICTION_LOCK = Literal['channel.prediction.lock']
CHANNEL_PREDICTION_END = Literal['channel.prediction.end']
CHANNEL_SUSPICIOUS_USER_MESSAGE = Literal['channel.suspicious_user.message']
CHANNEL_SUSPICIOUS_USER_UPDATE = Literal['channel.suspicious_user.update']
CHANNEL_VIP_ADD = Literal['channel.vip.add']
CHANNEL_VIP_REMOVE = Literal['channel.vip.remove']
CHANNEL_WARNING_ACKNOWLEDGEMENT = Literal['channel.warning.acknowledge']
CHANNEL_WARNING_SEND = Literal['channel.warning.send']
# Charity types
CHARITY_DONATION = Literal['channel.charity_campaign.donate']
CHARITY_CAMPAIGN_START = Literal['channel.charity_campaign.start']
CHARITY_CAMPAIGN_PROGRESS = Literal['channel.charity_campaign.progress']
CHARITY_CAMPAIGN_STOP = Literal['channel.charity_campaign.stop']
# Conduit types
CONDUIT_SHARD_DISABLED = Literal['conduit.shard.disabled']
# Drop types
DROP_ENTITLEMENT_GRANT = Literal['drop.entitlement.grant']
# Extension types
EXTENSION_BITS_TRANSACTION_CREATE = Literal[
'extension.bits_transaction.create'
]
# Goal types
GOAL_BEGIN = Literal['channel.goal.begin']
GOAL_PROGRESS = Literal['channel.goal.progress']
GOAL_END = Literal['channel.goal.end']
# Hype Train types
HYPE_TRAIN_BEGIN = Literal['channel.hype_train.begin']
HYPE_TRAIN_PROGRESS = Literal['channel.hype_train.progress']
HYPE_TRAIN_END = Literal['channel.hype_train.end']
# Shield Mode types
SHIELD_MODE_BEGIN = Literal['channel.shield_mode.begin']
SHIELD_MODE_END = Literal['channel.shield_mode.end']
# Shoutout types
SHOUTOUT_CREATE = Literal['channel.shoutout.create']
SHOUTOUT_RECEIVED = Literal['channel.shoutout.receive']
# Stream types
STREAM_ONLINE = Literal['stream.online']
STREAM_OFFLINE = Literal['stream.offline']
# User types
USER_AUTHORIZATION_GRANT = Literal['user.authorization.grant']
USER_AUTHORIZATION_REVOKE = Literal['user.authorization.revoke']
USER_UPDATE = Literal['user.update']
WHISPER_RECEIVED = Literal['user.whisper.message']
# Union type for all events
Any = (
AUTOMOD_MESSAGE_HOLD
| AUTOMOD_MESSAGE_UPDATE
| AUTOMOD_SETTINGS_UPDATE
| AUTOMOD_TERMS_UPDATE
| CHANNEL_BITS_USE
| CHANNEL_UPDATE
| CHANNEL_FOLLOW
| CHANNEL_AD_BREAK_BEGIN
| CHANNEL_CHAT_CLEAR
| CHANNEL_CHAT_CLEAR_USER_MESSAGES
| CHANNEL_CHAT_MESSAGE
| CHANNEL_CHAT_MESSAGE_DELETE
| CHANNEL_CHAT_NOTIFICATION
| CHANNEL_CHAT_SETTINGS_UPDATE
| CHANNEL_CHAT_USER_MESSAGE_HOLD
| CHANNEL_CHAT_USER_MESSAGE_UPDATE
| CHANNEL_SHARED_CHAT_SESSION_BEGIN
| CHANNEL_SHARED_CHAT_SESSION_UPDATE
| CHANNEL_SHARED_CHAT_SESSION_END
| CHANNEL_SUBSCRIBE
| CHANNEL_SUBSCRIPTION_END
| CHANNEL_SUBSCRIPTION_GIFT
| CHANNEL_SUBSCRIPTION_MESSAGE
| CHANNEL_CHEER
| CHANNEL_RAID
| CHANNEL_BAN
| CHANNEL_UNBAN
| CHANNEL_UNBAN_REQUEST_CREATE
| CHANNEL_UNBAN_REQUEST_RESOLVE
| CHANNEL_MODERATE
| CHANNEL_MODERATOR_ADD
| CHANNEL_MODERATOR_REMOVE
| CHANNEL_GUEST_STAR_SESSION_BEGIN
| CHANNEL_GUEST_STAR_SESSION_END
| CHANNEL_GUEST_STAR_GUEST_UPDATE
| CHANNEL_GUEST_STAR_SETTINGS_UPDATE
| CHANNEL_POINTS_AUTOMATIC_REWARD_REDEMPTION_ADD
| CHANNEL_POINTS_CUSTOM_REWARD_ADD
| CHANNEL_POINTS_CUSTOM_REWARD_UPDATE
| CHANNEL_POINTS_CUSTOM_REWARD_REMOVE
| CHANNEL_POINTS_CUSTOM_REWARD_REDEMPTION_ADD
| CHANNEL_POINTS_CUSTOM_REWARD_REDEMPTION_UPDATE
| CHANNEL_POLL_BEGIN
| CHANNEL_POLL_PROGRESS
| CHANNEL_POLL_END
| CHANNEL_PREDICTION_BEGIN
| CHANNEL_PREDICTION_PROGRESS
| CHANNEL_PREDICTION_LOCK
| CHANNEL_PREDICTION_END
| CHANNEL_SUSPICIOUS_USER_MESSAGE
| CHANNEL_SUSPICIOUS_USER_UPDATE
| CHANNEL_VIP_ADD
| CHANNEL_VIP_REMOVE
| CHANNEL_WARNING_ACKNOWLEDGEMENT
| CHANNEL_WARNING_SEND
| CHARITY_DONATION
| CHARITY_CAMPAIGN_START
| CHARITY_CAMPAIGN_PROGRESS
| CHARITY_CAMPAIGN_STOP
| CONDUIT_SHARD_DISABLED
| DROP_ENTITLEMENT_GRANT
| EXTENSION_BITS_TRANSACTION_CREATE
| GOAL_BEGIN
| GOAL_PROGRESS
| GOAL_END
| HYPE_TRAIN_BEGIN
| HYPE_TRAIN_PROGRESS
| HYPE_TRAIN_END
| SHIELD_MODE_BEGIN
| SHIELD_MODE_END
| SHOUTOUT_CREATE
| SHOUTOUT_RECEIVED
| STREAM_ONLINE
| STREAM_OFFLINE
| USER_AUTHORIZATION_GRANT
| USER_AUTHORIZATION_REVOKE
| USER_UPDATE
| WHISPER_RECEIVED
)

0
src/oxidetwitch/py.typed Normal file
View File

1487
src/oxidetwitch/schema.py Normal file

File diff suppressed because it is too large Load Diff

166
src/oxidetwitch/scopes.py Normal file
View File

@ -0,0 +1,166 @@
from typing import Literal
# Analytics
ANALYTICS_READ_EXTENSIONS = 'analytics:read:extensions'
ANALYTICS_READ_GAMES = 'analytics:read:games'
# Bits
BITS_READ = 'bits:read'
# Channel
CHANNEL_MANAGE_ADS = 'channel:manage:ads'
CHANNEL_READ_ADS = 'channel:read:ads'
CHANNEL_MANAGE_BROADCAST = 'channel:manage:broadcast'
CHANNEL_READ_CHARITY = 'channel:read:charity'
CHANNEL_EDIT_COMMERCIAL = 'channel:edit:commercial'
CHANNEL_READ_EDITORS = 'channel:read:editors'
CHANNEL_MANAGE_EXTENSIONS = 'channel:manage:extensions'
CHANNEL_READ_GOALS = 'channel:read:goals'
CHANNEL_READ_GUEST_STAR = 'channel:read:guest_star'
CHANNEL_MANAGE_GUEST_STAR = 'channel:manage:guest_star'
CHANNEL_READ_HYPE_TRAIN = 'channel:read:hype_train'
CHANNEL_MANAGE_MODERATORS = 'channel:manage:moderators'
CHANNEL_READ_POLLS = 'channel:read:polls'
CHANNEL_MANAGE_POLLS = 'channel:manage:polls'
CHANNEL_READ_PREDICTIONS = 'channel:read:predictions'
CHANNEL_MANAGE_PREDICTIONS = 'channel:manage:predictions'
CHANNEL_MANAGE_RAIDS = 'channel:manage:raids'
CHANNEL_READ_REDEMPTIONS = 'channel:read:redemptions'
CHANNEL_MANAGE_REDEMPTIONS = 'channel:manage:redemptions'
CHANNEL_MANAGE_SCHEDULE = 'channel:manage:schedule'
CHANNEL_READ_STREAM_KEY = 'channel:read:stream_key'
CHANNEL_READ_SUBSCRIPTIONS = 'channel:read:subscriptions'
CHANNEL_MANAGE_VIDEOS = 'channel:manage:videos'
CHANNEL_READ_VIPS = 'channel:read:vips'
CHANNEL_MANAGE_VIPS = 'channel:manage:vips'
CHANNEL_BOT = 'channel:bot'
CHANNEL_MODERATE = 'channel:moderate'
# clips
CLIPS_EDIT = 'clips:edit'
# Moderation
MODERATION_READ = 'moderation:read'
MODERATOR_MANAGE_ANNOUNCEMENTS = 'moderator:manage:announcements'
MODERATOR_MANAGE_AUTOMOD = 'moderator:manage:automod'
MODERATOR_READ_AUTOMOD_SETTINGS = 'moderator:read:automod_settings'
MODERATOR_MANAGE_AUTOMOD_SETTINGS = 'moderator:manage:automod_settings'
MODERATOR_MANAGE_BANNED_USERS = 'moderator:manage:banned_users'
MODERATOR_READ_BLOCKED_TERMS = 'moderator:read:blocked_terms'
MODERATOR_MANAGE_BLOCKED_TERMS = 'moderator:manage:blocked_terms'
MODERATOR_MANAGE_CHAT_MESSAGES = 'moderator:manage:chat_messages'
MODERATOR_READ_CHAT_SETTINGS = 'moderator:read:chat_settings'
MODERATOR_MANAGE_CHAT_SETTINGS = 'moderator:manage:chat_settings'
MODERATOR_READ_CHATTERS = 'moderator:read:chatters'
MODERATOR_READ_FOLLOWERS = 'moderator:read:followers'
MODERATOR_READ_GUEST_STAR = 'moderator:read:guest_star'
MODERATOR_MANAGE_GUEST_STAR = 'moderator:manage:guest_star'
MODERATOR_READ_SHIELD_MODE = 'moderator:read:shield_mode'
MODERATOR_MANAGE_SHIELD_MODE = 'moderator:manage:shield_mode'
MODERATOR_READ_SHOUTOUTS = 'moderator:read:shoutouts'
MODERATOR_MANAGE_SHOUTOUTS = 'moderator:manage:shoutouts'
MODERATOR_READ_UNBAN_REQUESTS = 'moderator:read:unban_requests'
MODERATOR_MANAGE_UNBAN_REQUESTS = 'moderator:manage:unban_requests'
MODERATOR_READ_WARNINGS = 'moderator:read:warnings'
MODERATOR_MANAGE_WARNINGS = 'moderator:manage:warnings'
# User
USER_EDIT = 'user:edit'
USER_EDIT_FOLLOWS = 'user:edit:follows'
USER_READ_BLOCKED_USERS = 'user:read:blocked_users'
USER_MANAGE_BLOCKED_USERS = 'user:manage:blocked_users'
USER_READ_BROADCAST = 'user:read:broadcast'
USER_MANAGE_CHAT_COLOR = 'user:manage:chat_color'
USER_READ_EMAIL = 'user:read:email'
USER_READ_EMOTES = 'user:read:emotes'
USER_READ_FOLLOWS = 'user:read:follows'
USER_READ_MODERATED_CHANNELS = 'user:read:moderated_channels'
USER_READ_SUBSCRIPTIONS = 'user:read:subscriptions'
USER_MANAGE_WHISPERS = 'user:manage:whispers'
USER_BOT = 'user:bot'
USER_READ_CHAT = 'user:read:chat'
USER_WRITE_CHAT = 'user:write:chat'
# Chat
CHAT_EDIT = 'chat:edit'
CHAT_READ = 'chat:read'
# Whispers
WHISPERS_READ = 'whispers:read'
WHISPERS_EDIT = 'whispers:edit'
type Any = Literal[
'analytics:read:extensions',
'analytics:read:games',
'bits:read',
'channel:manage:ads',
'channel:read:ads',
'channel:manage:broadcast',
'channel:read:charity',
'channel:edit:commercial',
'channel:read:editors',
'channel:manage:extensions',
'channel:read:goals',
'channel:read:guest_star',
'channel:manage:guest_star',
'channel:read:hype_train',
'channel:manage:moderators',
'channel:read:polls',
'channel:manage:polls',
'channel:read:predictions',
'channel:manage:predictions',
'channel:manage:raids',
'channel:read:redemptions',
'channel:manage:redemptions',
'channel:manage:schedule',
'channel:read:stream_key',
'channel:read:subscriptions',
'channel:manage:videos',
'channel:read:vips',
'channel:manage:vips',
'channel:bot',
'channel:moderate',
'clips:edit',
'moderation:read',
'moderator:manage:announcements',
'moderator:manage:automod',
'moderator:read:automod_settings',
'moderator:manage:automod_settings',
'moderator:manage:banned_users',
'moderator:read:blocked_terms',
'moderator:manage:blocked_terms',
'moderator:manage:chat_messages',
'moderator:read:chat_settings',
'moderator:manage:chat_settings',
'moderator:read:chatters',
'moderator:read:followers',
'moderator:read:guest_star',
'moderator:manage:guest_star',
'moderator:read:shield_mode',
'moderator:manage:shield_mode',
'moderator:read:shoutouts',
'moderator:manage:shoutouts',
'moderator:read:unban_requests',
'moderator:manage:unban_requests',
'moderator:read:warnings',
'moderator:manage:warnings',
'user:edit',
'user:edit:follows',
'user:read:blocked_users',
'user:manage:blocked_users',
'user:read:broadcast',
'user:manage:chat_color',
'user:read:email',
'user:read:emotes',
'user:read:follows',
'user:read:moderated_channels',
'user:read:subscriptions',
'user:manage:whispers',
'user:bot',
'user:read:chat',
'user:write:chat',
'chat:edit',
'chat:read',
'whispers:read',
'whispers:edit',
]