mirror of
https://github.com/onyx-and-iris/streamlabs-socketio-py
synced 2025-01-18 17:00:48 +00:00
110 lines
3.5 KiB
Python
110 lines
3.5 KiB
Python
import logging
|
|
from pathlib import Path
|
|
from typing import Optional
|
|
|
|
import socketio
|
|
from observable import Observable
|
|
|
|
from .error import SteamlabsSIOConnectionError
|
|
from .models import as_dataclass
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class Client:
|
|
def __init__(self, token=None, raw=False):
|
|
self.logger = logger.getChild(self.__class__.__name__)
|
|
self.token = token or self._token_from_toml()
|
|
self._raw = raw
|
|
self.sio = socketio.Client()
|
|
self.sio.on("connect", self.connect_handler)
|
|
self.sio.on("event", self.event_handler)
|
|
self.sio.on("disconnect", self.disconnect_handler)
|
|
self.obs = Observable()
|
|
self.streamlabs = ("donation",)
|
|
self.twitch = ("follow", "subscription", "host", "bits", "raid")
|
|
self.youtube = ("follow", "subscription", "superchat")
|
|
|
|
def __enter__(self):
|
|
try:
|
|
self.sio.connect(f"https://sockets.streamlabs.com?token={self.token}")
|
|
except socketio.exceptions.ConnectionError as e:
|
|
self.logger.exception(f"{type(e).__name__}: {e}")
|
|
raise SteamlabsSIOConnectionError(
|
|
"no connection could be established to the Streamlabs SIO server"
|
|
) from e
|
|
self.log_mode()
|
|
return self
|
|
|
|
@property
|
|
def raw(self):
|
|
return self._raw
|
|
|
|
@raw.setter
|
|
def raw(self, val):
|
|
self._raw = val
|
|
self.log_mode()
|
|
|
|
def log_mode(self):
|
|
info = (f"Running client in {'raw' if self.raw else 'normal'} mode.",)
|
|
if self.raw:
|
|
info += ("raw JSON messages will be passed to callbacks",)
|
|
else:
|
|
info += ("event data objects will be passed to callbacks",)
|
|
self.logger.info(" ".join(info))
|
|
|
|
def _token_from_toml(self) -> str:
|
|
try:
|
|
import tomllib
|
|
except ModuleNotFoundError:
|
|
import tomli as tomllib
|
|
|
|
def get_filepath() -> Optional[Path]:
|
|
filepaths = (
|
|
Path.cwd() / "config.toml",
|
|
Path.home() / ".config" / "streamlabsio" / "config.toml",
|
|
)
|
|
for filepath in filepaths:
|
|
if filepath.exists():
|
|
return filepath
|
|
|
|
try:
|
|
filepath = get_filepath()
|
|
if not filepath:
|
|
raise FileNotFoundError("config.toml was not found")
|
|
with open(filepath, "rb") as f:
|
|
conn = tomllib.load(f)
|
|
assert (
|
|
"streamlabs" in conn and "token" in conn["streamlabs"]
|
|
), "expected [streamlabs][token] in config.toml"
|
|
return conn["streamlabs"]["token"]
|
|
except (FileNotFoundError, tomllib.TOMLDecodeError) as e:
|
|
self.logger.error(f"{type(e).__name__}: {e}")
|
|
raise
|
|
|
|
def connect_handler(self):
|
|
self.logger.info("Connected to Streamlabs Socket API")
|
|
|
|
def event_handler(self, data):
|
|
if "for" in data and data["type"] in set(
|
|
self.streamlabs + self.twitch + self.youtube
|
|
):
|
|
message = data["message"][0]
|
|
self.obs.trigger(
|
|
data["for"],
|
|
data["type"],
|
|
message if self.raw else as_dataclass(data["type"], message),
|
|
)
|
|
self.logger.debug(data)
|
|
|
|
def disconnect_handler(self):
|
|
self.logger.info("Disconnected from Streamlabs Socket API")
|
|
|
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
self.sio.disconnect()
|
|
|
|
|
|
def connect(**kwargs):
|
|
SIO_cls = Client
|
|
return SIO_cls(**kwargs)
|