jellyfin-kodi/jellyfin_kodi/jellyfin/ws_client.py

126 lines
3.9 KiB
Python

# -*- coding: utf-8 -*-
from __future__ import division, absolute_import, print_function, unicode_literals
#################################################################################################
import json
import threading
import time
import xbmc
from ..helper import LazyLogger, settings
# If numpy is installed, the websockets library tries to use it, and then
# kodi hard crashes for reasons I don't even want to pretend to understand
import sys # noqa: E402,I100
sys.modules["numpy"] = None
import websocket # noqa: E402,I201
##################################################################################################
LOG = LazyLogger(__name__)
##################################################################################################
class WSClient(threading.Thread):
wsc = None
stop = False
def __init__(self, client):
LOG.debug("WSClient initializing...")
self.client = client
threading.Thread.__init__(self)
def send(self, message, data=""):
if self.wsc is None:
raise ValueError("The websocket client is not started.")
self.wsc.send(json.dumps({"MessageType": message, "Data": data}))
def run(self):
monitor = xbmc.Monitor()
token = self.client.config.data["auth.token"]
device_id = self.client.config.data["app.device_id"]
server = self.client.config.data["auth.server"]
server = (
server.replace("https://", "wss://")
if server.startswith("https")
else server.replace("http://", "ws://")
)
wsc_url = "%s/socket?api_key=%s&device_id=%s" % (server, token, device_id)
LOG.info("Websocket url: %s", wsc_url)
self.wsc = websocket.WebSocketApp(
wsc_url,
on_open=lambda ws: self.on_open(ws),
on_message=lambda ws, message: self.on_message(ws, message),
on_error=lambda ws, error: self.on_error(ws, error),
)
while not self.stop:
self.wsc.run_forever(ping_interval=10, reconnect=10)
if not self.stop and monitor.waitForAbort(5):
break
def on_error(self, ws, error):
LOG.error(error)
def on_open(self, ws):
LOG.info("--->[ websocket opened ]")
# Avoid a timing issue where the capabilities are not correctly registered
time.sleep(5)
if settings("remoteControl.bool"):
self.client.jellyfin.post_capabilities(
{
"PlayableMediaTypes": "Audio,Video",
"SupportsMediaControl": True,
"SupportedCommands": (
"MoveUp,MoveDown,MoveLeft,MoveRight,Select,"
"Back,ToggleContextMenu,ToggleFullscreen,ToggleOsdMenu,"
"GoHome,PageUp,NextLetter,GoToSearch,"
"GoToSettings,PageDown,PreviousLetter,TakeScreenshot,"
"VolumeUp,VolumeDown,ToggleMute,SendString,DisplayMessage,"
"SetAudioStreamIndex,SetSubtitleStreamIndex,"
"SetRepeatMode,Mute,Unmute,SetVolume,"
"Play,Playstate,PlayNext,PlayMediaSource"
),
}
)
else:
self.client.jellyfin.post_capabilities(
{"PlayableMediaTypes": "Audio, Video", "SupportsMediaControl": False}
)
def on_message(self, ws, message):
message = json.loads(message)
data = message.get("Data", {})
if message["MessageType"] in ("RefreshProgress",):
LOG.debug("Ignoring %s", message)
return
if not self.client.config.data["app.default"]:
data["ServerId"] = self.client.auth.server_id
self.client.callback(message["MessageType"], data)
def stop_client(self):
self.stop = True
if self.wsc is not None:
self.wsc.close()