PlexKodiConnect/resources/lib/PlexCompanion.py

292 lines
11 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2016-09-02 17:20:19 +02:00
import logging
2016-12-20 16:38:04 +01:00
from threading import Thread
import Queue
2016-12-20 16:38:04 +01:00
from socket import SHUT_RDWR
from urllib import urlencode
from xbmc import sleep, executebuiltin
2017-05-17 13:55:24 +02:00
from utils import settings, thread_methods
from plexbmchelper import listener, plexgdm, subscribers, functions, \
2016-09-02 17:20:19 +02:00
httppersist, plexsettings
from PlexFunctions import ParseContainerKey, GetPlexMetadata
from PlexAPI import API
from playlist_func import get_pms_playqueue, get_plextype_from_xml
2016-08-07 15:33:36 +02:00
import player
2017-03-05 16:51:13 +01:00
import variables as v
2017-05-17 20:22:16 +02:00
import state
2017-03-04 17:54:24 +01:00
2016-09-02 17:20:19 +02:00
###############################################################################
2016-09-02 17:20:19 +02:00
log = logging.getLogger("PLEX."+__name__)
###############################################################################
2017-05-17 13:55:24 +02:00
@thread_methods(add_suspends=['PMS_STATUS'])
2016-12-20 16:38:04 +01:00
class PlexCompanion(Thread):
2016-07-20 18:36:31 +02:00
"""
"""
2016-12-27 17:33:52 +01:00
def __init__(self, callback=None):
2016-09-02 17:20:19 +02:00
log.info("----===## Starting PlexCompanion ##===----")
2016-12-27 17:33:52 +01:00
if callback is not None:
self.mgr = callback
2016-09-02 17:20:19 +02:00
self.settings = plexsettings.getSettings()
# Start GDM for server/client discovery
self.client = plexgdm.plexgdm()
self.client.clientDetails(self.settings)
2017-02-19 17:07:42 +01:00
log.debug("Registration string is:\n%s"
2016-09-02 17:20:19 +02:00
% self.client.getClientDetails())
2016-07-23 18:06:47 +02:00
# kodi player instance
2016-08-07 15:33:36 +02:00
self.player = player.Player()
2016-07-23 18:06:47 +02:00
2016-12-20 16:38:04 +01:00
Thread.__init__(self)
def _getStartItem(self, string):
"""
Grabs the Plex id from e.g. '/library/metadata/12987'
and returns the tuple (typus, id) where typus is either 'queueId' or
'plexId' and id is the corresponding id as a string
"""
typus = 'plexId'
if string.startswith('/library/metadata'):
try:
string = string.split('/')[3]
except IndexError:
string = ''
else:
2016-09-02 17:20:19 +02:00
log.error('Unknown string! %s' % string)
return typus, string
def processTasks(self, task):
"""
2016-12-28 13:14:21 +01:00
Processes tasks picked up e.g. by Companion listener, e.g.
{'action': 'playlist',
'data': {'address': 'xyz.plex.direct',
'commandID': '7',
'containerKey': '/playQueues/6669?own=1&repeat=0&window=200',
'key': '/library/metadata/220493',
'machineIdentifier': 'xyz',
'offset': '0',
'port': '32400',
'protocol': 'https',
'token': 'transient-cd2527d1-0484-48e0-a5f7-f5caa7d591bd',
'type': 'video'}}
"""
2016-09-02 17:20:19 +02:00
log.debug('Processing: %s' % task)
data = task['data']
2017-05-17 20:22:16 +02:00
# Get the token of the user flinging media (might be different one)
token = data.get('token')
2017-03-05 17:51:58 +01:00
if task['action'] == 'alexa':
# e.g. Alexa
xml = GetPlexMetadata(data['key'])
try:
xml[0].attrib
except (AttributeError, IndexError, TypeError):
log.error('Could not download Plex metadata')
return
api = API(xml[0])
if api.getType() == v.PLEX_TYPE_ALBUM:
log.debug('Plex music album detected')
queue = self.mgr.playqueue.init_playqueue_from_plex_children(
2017-03-05 17:51:58 +01:00
api.getRatingKey())
queue.plex_transient_token = token
2017-03-05 17:51:58 +01:00
else:
state.PLEX_TRANSIENT_TOKEN = token
params = {
'mode': 'plex_node',
'key': '{server}%s' % data.get('key'),
'view_offset': data.get('offset'),
'play_directly': 'true',
'node': 'false'
}
executebuiltin('RunPlugin(plugin://%s?%s)'
% (v.ADDON_ID, urlencode(params)))
2017-03-05 17:51:58 +01:00
elif (task['action'] == 'playlist' and
2017-01-02 15:41:38 +01:00
data.get('address') == 'node.plexapp.com'):
# E.g. watch later initiated by Companion
state.PLEX_TRANSIENT_TOKEN = token
params = {
'mode': 'plex_node',
'key': '{server}%s' % data.get('key'),
'view_offset': data.get('offset'),
'play_directly': 'true'
}
executebuiltin('RunPlugin(plugin://%s?%s)'
% (v.ADDON_ID, urlencode(params)))
2017-03-05 17:51:58 +01:00
2017-01-02 15:41:38 +01:00
elif task['action'] == 'playlist':
2016-12-27 17:33:52 +01:00
# Get the playqueue ID
try:
2017-03-04 17:54:24 +01:00
typus, ID, query = ParseContainerKey(data['containerKey'])
except Exception as e:
2016-09-02 17:20:19 +02:00
log.error('Exception while processing: %s' % e)
import traceback
2016-09-02 17:20:19 +02:00
log.error("Traceback:\n%s" % traceback.format_exc())
return
try:
playqueue = self.mgr.playqueue.get_playqueue_from_type(
2017-03-05 16:51:13 +01:00
v.KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE[data['type']])
except KeyError:
# E.g. Plex web does not supply the media type
# Still need to figure out the type (video vs. music vs. pix)
xml = GetPlexMetadata(data['key'])
2017-03-05 16:51:13 +01:00
try:
xml[0].attrib
except (AttributeError, IndexError, TypeError):
log.error('Could not download Plex metadata')
return
api = API(xml[0])
playqueue = self.mgr.playqueue.get_playqueue_from_type(
2017-03-05 16:51:13 +01:00
v.KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE[api.getType()])
2017-01-02 14:07:24 +01:00
self.mgr.playqueue.update_playqueue_from_PMS(
playqueue,
ID,
repeat=query.get('repeat'),
offset=data.get('offset'))
playqueue.plex_transient_token = token
elif task['action'] == 'refreshPlayQueue':
# example data: {'playQueueID': '8475', 'commandID': '11'}
xml = get_pms_playqueue(data['playQueueID'])
if xml is None:
return
if len(xml) == 0:
log.debug('Empty playqueue received - clearing playqueue')
plex_type = get_plextype_from_xml(xml)
if plex_type is None:
return
playqueue = self.mgr.playqueue.get_playqueue_from_type(
v.KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE[plex_type])
playqueue.clear()
return
playqueue = self.mgr.playqueue.get_playqueue_from_type(
v.KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE[xml[0].attrib['type']])
self.mgr.playqueue.update_playqueue_from_PMS(
playqueue,
data['playQueueID'])
def run(self):
# Ensure that sockets will be closed no matter what
try:
self.__run()
finally:
try:
self.httpd.socket.shutdown(SHUT_RDWR)
except AttributeError:
pass
finally:
try:
self.httpd.socket.close()
except AttributeError:
pass
log.info("----===## Plex Companion stopped ##===----")
def __run(self):
self.httpd = False
httpd = self.httpd
# Cache for quicker while loops
client = self.client
thread_stopped = self.thread_stopped
thread_suspended = self.thread_suspended
# Start up instances
requestMgr = httppersist.RequestMgr()
jsonClass = functions.jsonClass(requestMgr, self.settings)
subscriptionManager = subscribers.SubscriptionManager(
2016-12-28 13:14:21 +01:00
jsonClass, requestMgr, self.player, self.mgr)
queue = Queue.Queue(maxsize=100)
2017-03-04 17:54:24 +01:00
self.queue = queue
2016-09-02 17:20:19 +02:00
if settings('plexCompanion') == 'true':
# Start up httpd
start_count = 0
while True:
try:
httpd = listener.ThreadedHTTPServer(
client,
subscriptionManager,
jsonClass,
self.settings,
queue,
('', self.settings['myport']),
listener.MyHandler)
httpd.timeout = 0.95
break
except:
2016-09-02 17:20:19 +02:00
log.error("Unable to start PlexCompanion. Traceback:")
2016-12-20 16:38:04 +01:00
import traceback
2016-09-02 17:20:19 +02:00
log.error(traceback.print_exc())
2016-12-20 16:38:04 +01:00
sleep(3000)
if start_count == 3:
2016-09-02 17:20:19 +02:00
log.error("Error: Unable to start web helper.")
httpd = False
break
start_count += 1
else:
2016-09-02 17:20:19 +02:00
log.info('User deactivated Plex Companion')
client.start_all()
message_count = 0
if httpd:
2016-12-20 16:38:04 +01:00
t = Thread(target=httpd.handle_request)
while not thread_stopped():
2016-03-10 16:02:46 +01:00
# If we are not authorized, sleep
# Otherwise, we trigger a download which leads to a
# re-authorizations
while thread_suspended():
if thread_stopped():
break
2016-12-20 16:38:04 +01:00
sleep(1000)
try:
message_count += 1
if httpd:
if not t.isAlive():
2016-08-10 19:03:37 +02:00
# Use threads cause the method will stall
2016-12-20 16:38:04 +01:00
t = Thread(target=httpd.handle_request)
t.start()
if message_count == 3000:
message_count = 0
if client.check_client_registration():
2016-09-02 17:20:19 +02:00
log.debug("Client is still registered")
else:
2017-02-03 12:27:59 +01:00
log.debug("Client is no longer registered. "
"Plex Companion still running on port %s"
% self.settings['myport'])
2017-02-19 17:07:42 +01:00
client.register_as_client()
# Get and set servers
if message_count % 30 == 0:
subscriptionManager.serverlist = client.getServerList()
subscriptionManager.notify()
if not httpd:
message_count = 0
except:
2016-09-02 17:20:19 +02:00
log.warn("Error in loop, continuing anyway. Traceback:")
2016-12-20 16:38:04 +01:00
import traceback
2016-09-02 17:20:19 +02:00
log.warn(traceback.format_exc())
# See if there's anything we need to process
try:
task = queue.get(block=False)
except Queue.Empty:
pass
else:
# Got instructions, process them
self.processTasks(task)
queue.task_done()
# Don't sleep
continue
2017-03-05 16:43:06 +01:00
sleep(50)
client.stop_all()