2018-07-12 18:46:02 +02:00
|
|
|
#!/usr/bin/env python
|
2016-03-19 17:57:57 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2017-12-09 14:35:08 +01:00
|
|
|
from logging import getLogger
|
2021-03-17 21:13:11 +01:00
|
|
|
import json
|
2016-03-19 17:57:57 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
from . import websocket
|
|
|
|
from . import backgroundthread, app, variables as v, utils, companion
|
2016-09-04 16:41:05 +02:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
log = getLogger('PLEX.websocket')
|
2016-03-19 17:57:57 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
PMS_PATH = '/:/websockets/notifications'
|
2016-03-19 17:57:57 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
PMS_INTERESTING_MESSAGE_TYPES = ('playing', 'timeline', 'activity')
|
|
|
|
SETTINGS_STRING = '_status'
|
2016-03-19 17:57:57 +01:00
|
|
|
|
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def get_pms_uri():
|
|
|
|
uri = app.CONN.server
|
|
|
|
if not uri:
|
|
|
|
return
|
|
|
|
# Get the appropriate prefix for the websocket
|
|
|
|
if uri.startswith('https'):
|
|
|
|
uri = "wss%s" % uri[5:]
|
|
|
|
else:
|
|
|
|
uri = "ws%s" % uri[4:]
|
|
|
|
uri += PMS_PATH
|
|
|
|
log.debug('uri to connect pms websocket: %s', uri)
|
|
|
|
if app.ACCOUNT.pms_token:
|
|
|
|
uri += '?X-Plex-Token=' + app.ACCOUNT.pms_token
|
|
|
|
return uri
|
2018-02-09 15:10:20 +01:00
|
|
|
|
2016-03-19 17:57:57 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def get_alexa_uri():
|
|
|
|
if not app.ACCOUNT.plex_token:
|
|
|
|
return
|
|
|
|
return (f'wss://pubsub.plex.tv/sub/websockets/{app.ACCOUNT.plex_user_id}/'
|
|
|
|
f'{v.PKC_MACHINE_IDENTIFIER}?'
|
|
|
|
f'X-Plex-Token={app.ACCOUNT.plex_token}')
|
2020-05-07 07:48:50 +02:00
|
|
|
|
2016-03-21 17:15:22 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def pms_on_message(ws, message):
|
|
|
|
"""
|
|
|
|
Called when we receive a message from the PMS, e.g. when a new library
|
|
|
|
item has been added.
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
message = json.loads(message)
|
|
|
|
except ValueError as err:
|
|
|
|
log.error('Error decoding PMS websocket message: %s', err)
|
|
|
|
log.error('message: %s', message)
|
|
|
|
return
|
|
|
|
try:
|
|
|
|
message = message['NotificationContainer']
|
|
|
|
typus = message['type']
|
|
|
|
except KeyError:
|
|
|
|
log.error('Could not parse PMS message: %s', message)
|
|
|
|
return
|
|
|
|
# Triage
|
|
|
|
if typus not in PMS_INTERESTING_MESSAGE_TYPES:
|
|
|
|
# Drop everything we're not interested in
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
# Put PMS message on queue and let libsync take care of it
|
|
|
|
app.APP.websocket_queue.put(message)
|
|
|
|
|
|
|
|
|
|
|
|
def alexa_on_message(ws, message):
|
|
|
|
"""
|
|
|
|
Called when we receive a message from Alexa
|
|
|
|
"""
|
|
|
|
log.debug('alexa message received: %s', message)
|
|
|
|
try:
|
|
|
|
message = utils.etree.fromstring(message)
|
|
|
|
except Exception as err:
|
|
|
|
log.error('Error decoding message from Alexa: %s %s', type(err), err)
|
|
|
|
log.error('message from Alexa: %s', message)
|
|
|
|
return
|
|
|
|
try:
|
|
|
|
if message.attrib['command'] == 'processRemoteControlCommand':
|
|
|
|
message = message[0]
|
|
|
|
else:
|
|
|
|
log.error('Unknown Alexa message received: %s', message)
|
|
|
|
return
|
|
|
|
companion.process_command(message.attrib['path'][1:], message.attrib)
|
|
|
|
except Exception as err:
|
|
|
|
log.exception('Could not parse Alexa message, error: %s %s',
|
|
|
|
type(err), err)
|
|
|
|
log.error('message: %s', message)
|
|
|
|
|
2016-03-19 17:57:57 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def on_error(ws, error):
|
|
|
|
status = ws.name + SETTINGS_STRING
|
|
|
|
if isinstance(error, IOError):
|
|
|
|
# We are probably offline
|
|
|
|
log.debug('%s: IOError connecting', ws.name)
|
|
|
|
# Status = IOError - not connected
|
|
|
|
utils.settings(status, value=utils.lang(39092))
|
|
|
|
ws.sleep_cycle()
|
|
|
|
elif isinstance(error, websocket.WebSocketTimeoutException):
|
|
|
|
log.debug('%s: WebSocketTimeoutException', ws.name)
|
|
|
|
# Status = 'Timeout - not connected'
|
|
|
|
utils.settings(status, value=utils.lang(39091))
|
|
|
|
ws.sleep_cycle()
|
|
|
|
elif isinstance(error, websocket.WebSocketConnectionClosedException):
|
|
|
|
log.debug('%s: WebSocketConnectionClosedException', ws.name)
|
|
|
|
# Status = Not connected
|
|
|
|
utils.settings(ws.name + SETTINGS_STRING, value=utils.lang(15208))
|
|
|
|
elif isinstance(error, websocket.WebSocketBadStatusException):
|
|
|
|
# Most likely Alexa not connecting, throwing a 403
|
|
|
|
log.debug('%s: got a bad HTTP status: %s', ws.name, error)
|
|
|
|
# Status = <value of exception>
|
|
|
|
utils.settings(status, value=str(error))
|
|
|
|
ws.sleep_cycle()
|
|
|
|
elif isinstance(error, websocket.WebSocketException):
|
|
|
|
log.error('%s: got another websocket exception %s: %s',
|
|
|
|
ws.name, type(error), error)
|
|
|
|
# Status = Error
|
|
|
|
utils.settings(status, value=utils.lang(257))
|
|
|
|
ws.sleep_cycle()
|
|
|
|
elif isinstance(error, SystemExit):
|
|
|
|
log.debug('%s: SystemExit detected', ws.name)
|
|
|
|
# Status = Not connected
|
|
|
|
utils.settings(ws.name + SETTINGS_STRING, value=utils.lang(15208))
|
|
|
|
else:
|
|
|
|
log.exception('%s: got an unexpected exception of type %s: %s',
|
|
|
|
ws.name, type(error), error)
|
|
|
|
# Status = Error
|
|
|
|
utils.settings(status, value=utils.lang(257))
|
|
|
|
raise RuntimeError
|
2016-03-19 17:57:57 +01:00
|
|
|
|
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def on_close(ws):
|
|
|
|
"""
|
|
|
|
This does not seem to get called by our websocket client :-(
|
|
|
|
"""
|
|
|
|
log.debug('%s: connection closed', ws.name)
|
|
|
|
# Status = Not connected
|
|
|
|
utils.settings(ws.name + SETTINGS_STRING, value=utils.lang(15208))
|
|
|
|
|
2016-03-19 17:57:57 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def on_open(ws):
|
|
|
|
log.debug('%s: connected', ws.name)
|
|
|
|
# Status = Connected
|
|
|
|
utils.settings(ws.name + SETTINGS_STRING, value=utils.lang(13296))
|
|
|
|
ws.sleeptime = 0
|
|
|
|
|
|
|
|
|
|
|
|
class PlexWebSocketApp(websocket.WebSocketApp,
|
|
|
|
backgroundthread.KillableThread):
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
self.sleeptime = 0
|
|
|
|
backgroundthread.KillableThread.__init__(self)
|
|
|
|
websocket.WebSocketApp.__init__(self, self.get_uri(), **kwargs)
|
|
|
|
|
|
|
|
def sleep_cycle(self):
|
2019-08-11 15:31:09 +02:00
|
|
|
"""
|
|
|
|
Sleeps for 2^self.sleeptime where sleeping period will be doubled with
|
|
|
|
each unsuccessful connection attempt.
|
|
|
|
Will sleep at most 64 seconds
|
|
|
|
"""
|
2019-11-28 17:49:48 +01:00
|
|
|
self.sleep(2 ** self.sleeptime)
|
2019-08-11 15:31:09 +02:00
|
|
|
if self.sleeptime < 6:
|
2021-03-17 21:13:11 +01:00
|
|
|
self.sleeptime += 1
|
|
|
|
|
|
|
|
def suspend(self, block=False, timeout=None):
|
|
|
|
"""
|
|
|
|
Call this method from another thread to suspend this websocket thread
|
|
|
|
"""
|
|
|
|
self.close()
|
|
|
|
backgroundthread.KillableThread.suspend(self, block, timeout)
|
|
|
|
|
|
|
|
def cancel(self):
|
|
|
|
"""
|
|
|
|
Call this method from another thread to cancel this websocket thread
|
|
|
|
"""
|
|
|
|
self.close()
|
|
|
|
backgroundthread.KillableThread.cancel(self)
|
2019-08-11 15:31:09 +02:00
|
|
|
|
2020-05-07 07:27:30 +02:00
|
|
|
def run(self):
|
2021-03-17 21:13:11 +01:00
|
|
|
"""
|
|
|
|
Ensure that sockets will be closed no matter what
|
|
|
|
"""
|
|
|
|
log.info("----===## Starting %s ##===----", self.name)
|
2020-05-07 07:27:30 +02:00
|
|
|
app.APP.register_thread(self)
|
|
|
|
try:
|
|
|
|
self._run()
|
2021-03-17 21:13:11 +01:00
|
|
|
except RuntimeError:
|
|
|
|
pass
|
|
|
|
except Exception as err:
|
|
|
|
log.exception('Exception of type %s occured: %s', type(err), err)
|
2020-05-07 07:27:30 +02:00
|
|
|
finally:
|
2021-03-17 21:13:11 +01:00
|
|
|
self.close()
|
|
|
|
# Status = Not connected
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(15208))
|
2020-05-07 07:27:30 +02:00
|
|
|
app.APP.deregister_thread(self)
|
2021-03-17 21:13:11 +01:00
|
|
|
log.info("----===## %s stopped ##===----", self.name)
|
2020-05-07 07:27:30 +02:00
|
|
|
|
2019-02-21 08:47:36 +01:00
|
|
|
def _run(self):
|
2019-11-28 17:49:48 +01:00
|
|
|
while not self.should_cancel():
|
2016-03-29 18:44:13 +02:00
|
|
|
# In the event the server goes offline
|
2021-03-17 21:13:11 +01:00
|
|
|
while self.should_suspend():
|
|
|
|
# We will be caught in this loop if either another thread
|
|
|
|
# called the suspend() method, thus setting _suspended = True
|
|
|
|
# OR if there any other conditions to not open a websocket
|
|
|
|
# connection - see methods should_suspend() below
|
|
|
|
# Status = Suspended - not connected
|
|
|
|
self.set_suspension_settings_status()
|
2019-01-30 20:36:52 +01:00
|
|
|
if self.wait_while_suspended():
|
2016-03-29 18:44:13 +02:00
|
|
|
# Abort was requested while waiting. We should exit
|
|
|
|
return
|
2021-03-17 21:13:11 +01:00
|
|
|
if not self._suspended:
|
|
|
|
# because wait_while_suspended will return instantly if
|
|
|
|
# this thread did not get suspended from another thread
|
|
|
|
self.sleep_cycle()
|
|
|
|
self.url = self.get_uri()
|
|
|
|
if not self.url:
|
|
|
|
self.sleep_cycle()
|
|
|
|
continue
|
|
|
|
self.run_forever()
|
|
|
|
|
|
|
|
|
|
|
|
class PMSWebsocketApp(PlexWebSocketApp):
|
|
|
|
name = 'pms_websocket'
|
|
|
|
|
|
|
|
def get_uri(self):
|
|
|
|
return get_pms_uri()
|
2021-03-14 13:54:40 +01:00
|
|
|
|
2020-05-07 07:27:30 +02:00
|
|
|
def should_suspend(self):
|
|
|
|
"""
|
2021-03-17 21:13:11 +01:00
|
|
|
Returns True if the thread needs to suspend.
|
2020-05-07 07:27:30 +02:00
|
|
|
"""
|
2021-03-17 21:13:11 +01:00
|
|
|
return (self._suspended or
|
|
|
|
utils.settings('enableBackgroundSync') != 'true')
|
2017-03-04 17:54:24 +01:00
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
def set_suspension_settings_status(self):
|
|
|
|
if utils.settings('enableBackgroundSync') != 'true':
|
|
|
|
# Status = Disabled
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(24023))
|
2017-09-08 12:06:31 +02:00
|
|
|
else:
|
2021-03-17 21:13:11 +01:00
|
|
|
# Status = 'Suspended - not connected'
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(39093))
|
2017-03-04 17:54:24 +01:00
|
|
|
|
|
|
|
|
2021-03-17 21:13:11 +01:00
|
|
|
class AlexaWebsocketApp(PlexWebSocketApp):
|
|
|
|
name = 'alexa_websocket'
|
|
|
|
|
|
|
|
def get_uri(self):
|
|
|
|
return get_alexa_uri()
|
2021-03-14 13:54:40 +01:00
|
|
|
|
2020-05-07 07:27:30 +02:00
|
|
|
def should_suspend(self):
|
|
|
|
"""
|
2021-03-17 21:13:11 +01:00
|
|
|
Returns True if the thread needs to suspend.
|
2020-05-07 07:27:30 +02:00
|
|
|
"""
|
2021-03-17 21:13:11 +01:00
|
|
|
return self._suspended or \
|
|
|
|
utils.settings('enable_alexa') != 'true' or \
|
2020-05-07 09:30:35 +02:00
|
|
|
app.ACCOUNT.restricted_user or \
|
|
|
|
not app.ACCOUNT.plex_token
|
2021-03-17 21:13:11 +01:00
|
|
|
|
|
|
|
def set_suspension_settings_status(self):
|
|
|
|
if utils.settings('enable_alexa') != 'true':
|
|
|
|
# Status = Disabled
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(24023))
|
|
|
|
elif app.ACCOUNT.restricted_user:
|
|
|
|
# Status = Managed Plex User - not connected
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(39094))
|
|
|
|
elif not app.ACCOUNT.plex_token:
|
|
|
|
# Status = Not logged in to plex.tv
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(39226))
|
2019-02-05 12:37:01 +01:00
|
|
|
else:
|
2021-03-17 21:13:11 +01:00
|
|
|
# Status = 'Suspended - not connected'
|
|
|
|
utils.settings(self.name + SETTINGS_STRING,
|
|
|
|
value=utils.lang(39093))
|
|
|
|
|
|
|
|
|
|
|
|
def get_pms_websocketapp():
|
|
|
|
return PMSWebsocketApp(on_open=on_open,
|
|
|
|
on_message=pms_on_message,
|
|
|
|
on_error=on_error,
|
|
|
|
on_close=on_close)
|
|
|
|
|
|
|
|
|
|
|
|
def get_alexa_websocketapp():
|
|
|
|
return AlexaWebsocketApp(on_open=on_open,
|
|
|
|
on_message=alexa_on_message,
|
|
|
|
on_error=on_error,
|
|
|
|
on_close=on_close)
|