mirror of
https://github.com/jellyfin/jellyfin-kodi.git
synced 2025-01-12 19:16:10 +00:00
125 lines
3.9 KiB
Python
125 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()
|