2016-01-22 15:37:20 +01:00
|
|
|
# -*- 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
|
2016-07-22 15:04:42 +02:00
|
|
|
import Queue
|
2016-12-20 16:38:04 +01:00
|
|
|
from socket import SHUT_RDWR
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-12-20 16:38:04 +01:00
|
|
|
from xbmc import sleep
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-09-02 17:20:19 +02:00
|
|
|
from utils import settings, ThreadMethodsAdditionalSuspend, ThreadMethods
|
2016-04-02 16:46:23 +02:00
|
|
|
from plexbmchelper import listener, plexgdm, subscribers, functions, \
|
2016-09-02 17:20:19 +02:00
|
|
|
httppersist, plexsettings
|
2017-01-24 16:57:12 +01:00
|
|
|
from PlexFunctions import ParseContainerKey
|
2016-08-07 15:33:36 +02:00
|
|
|
import player
|
2017-01-02 15:41:38 +01:00
|
|
|
from entrypoint import Plex_Node
|
2017-01-24 16:57:12 +01:00
|
|
|
from variables import KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-09-02 17:20:19 +02:00
|
|
|
###############################################################################
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-09-02 17:20:19 +02:00
|
|
|
log = logging.getLogger("PLEX."+__name__)
|
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
|
|
@ThreadMethodsAdditionalSuspend('plex_serverStatus')
|
|
|
|
@ThreadMethods
|
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()
|
2016-01-22 15:37:20 +01:00
|
|
|
# Start GDM for server/client discovery
|
2016-04-05 10:57:30 +02:00
|
|
|
self.client = plexgdm.plexgdm()
|
|
|
|
self.client.clientDetails(self.settings)
|
2016-09-02 17:20:19 +02:00
|
|
|
log.debug("Registration string is: %s "
|
|
|
|
% 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)
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-07-22 15:04:42 +02:00
|
|
|
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)
|
2016-07-22 15:04:42 +02:00
|
|
|
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-07-22 15:04:42 +02:00
|
|
|
"""
|
2016-09-02 17:20:19 +02:00
|
|
|
log.debug('Processing: %s' % task)
|
2016-07-22 15:04:42 +02:00
|
|
|
data = task['data']
|
|
|
|
|
2017-01-02 15:41:38 +01:00
|
|
|
if (task['action'] == 'playlist' and
|
|
|
|
data.get('address') == 'node.plexapp.com'):
|
|
|
|
# E.g. watch later initiated by Companion
|
|
|
|
thread = Thread(target=Plex_Node,
|
2017-01-02 16:42:07 +01:00
|
|
|
args=('{server}%s' % data.get('key'),
|
2017-01-02 15:41:38 +01:00
|
|
|
data.get('offset'),
|
2017-01-02 16:42:07 +01:00
|
|
|
data.get('type'),
|
|
|
|
True),)
|
2017-01-02 15:41:38 +01:00
|
|
|
thread.setDaemon(True)
|
|
|
|
thread.start()
|
|
|
|
elif task['action'] == 'playlist':
|
2016-12-27 17:33:52 +01:00
|
|
|
# Get the playqueue ID
|
2016-07-22 15:04:42 +02:00
|
|
|
try:
|
2016-12-27 17:33:52 +01:00
|
|
|
_, ID, query = ParseContainerKey(data['containerKey'])
|
2016-07-22 15:04:42 +02:00
|
|
|
except Exception as e:
|
2016-09-02 17:20:19 +02:00
|
|
|
log.error('Exception while processing: %s' % e)
|
2016-07-22 15:04:42 +02:00
|
|
|
import traceback
|
2016-09-02 17:20:19 +02:00
|
|
|
log.error("Traceback:\n%s" % traceback.format_exc())
|
2016-07-22 15:04:42 +02:00
|
|
|
return
|
2016-12-28 13:14:21 +01:00
|
|
|
playqueue = self.mgr.playqueue.get_playqueue_from_type(
|
2017-01-07 20:35:10 +01:00
|
|
|
KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE[data['type']])
|
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'))
|
2016-07-22 15:04:42 +02:00
|
|
|
|
2016-01-22 15:37:20 +01:00
|
|
|
def run(self):
|
2016-04-26 17:15:05 +02:00
|
|
|
httpd = False
|
2016-04-02 16:46:23 +02:00
|
|
|
# Cache for quicker while loops
|
|
|
|
client = self.client
|
|
|
|
threadStopped = self.threadStopped
|
|
|
|
threadSuspended = self.threadSuspended
|
|
|
|
|
|
|
|
# Start up instances
|
|
|
|
requestMgr = httppersist.RequestMgr()
|
2016-04-05 10:57:30 +02:00
|
|
|
jsonClass = functions.jsonClass(requestMgr, self.settings)
|
2016-04-02 16:46:23 +02:00
|
|
|
subscriptionManager = subscribers.SubscriptionManager(
|
2016-12-28 13:14:21 +01:00
|
|
|
jsonClass, requestMgr, self.player, self.mgr)
|
2016-04-02 16:46:23 +02:00
|
|
|
|
2016-07-22 15:04:42 +02:00
|
|
|
queue = Queue.Queue(maxsize=100)
|
|
|
|
|
2016-09-02 17:20:19 +02:00
|
|
|
if settings('plexCompanion') == 'true':
|
2016-04-26 17:15:05 +02:00
|
|
|
# Start up httpd
|
|
|
|
start_count = 0
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
httpd = listener.ThreadedHTTPServer(
|
|
|
|
client,
|
|
|
|
subscriptionManager,
|
|
|
|
jsonClass,
|
|
|
|
self.settings,
|
2016-07-22 15:04:42 +02:00
|
|
|
queue,
|
2016-04-26 17:15:05 +02:00
|
|
|
('', 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-01-22 15:37:20 +01:00
|
|
|
|
2016-12-20 16:38:04 +01:00
|
|
|
sleep(3000)
|
2016-04-26 17:10:11 +02:00
|
|
|
|
2016-04-26 17:15:05 +02:00
|
|
|
if start_count == 3:
|
2016-09-02 17:20:19 +02:00
|
|
|
log.error("Error: Unable to start web helper.")
|
2016-04-26 17:15:05 +02:00
|
|
|
httpd = False
|
|
|
|
break
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-04-26 17:15:05 +02:00
|
|
|
start_count += 1
|
|
|
|
else:
|
2016-09-02 17:20:19 +02:00
|
|
|
log.info('User deactivated Plex Companion')
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-04-02 16:46:23 +02:00
|
|
|
client.start_all()
|
|
|
|
|
2016-01-22 15:37:20 +01:00
|
|
|
message_count = 0
|
2016-08-07 20:52:49 +02:00
|
|
|
if httpd:
|
2016-12-20 16:38:04 +01:00
|
|
|
t = Thread(target=httpd.handle_request)
|
2016-08-07 20:52:49 +02:00
|
|
|
|
2016-04-02 16:46:23 +02:00
|
|
|
while not threadStopped():
|
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
|
2016-04-02 16:46:23 +02:00
|
|
|
while threadSuspended():
|
|
|
|
if threadStopped():
|
2016-03-23 16:07:09 +01:00
|
|
|
break
|
2016-12-20 16:38:04 +01:00
|
|
|
sleep(1000)
|
2016-01-22 15:37:20 +01:00
|
|
|
try:
|
2016-08-07 20:52:49 +02:00
|
|
|
message_count += 1
|
2016-04-26 17:15:05 +02:00
|
|
|
if httpd:
|
2016-08-07 20:52:49 +02:00
|
|
|
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)
|
2016-08-07 20:52:49 +02:00
|
|
|
t.start()
|
2016-04-26 17:15:05 +02:00
|
|
|
|
2016-08-07 20:52:49 +02:00
|
|
|
if message_count == 3000:
|
|
|
|
message_count = 0
|
2016-04-26 17:15:05 +02:00
|
|
|
if client.check_client_registration():
|
2016-09-02 17:20:19 +02:00
|
|
|
log.debug("Client is still registered")
|
2016-04-26 17:15:05 +02:00
|
|
|
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'])
|
2016-04-02 16:46:23 +02:00
|
|
|
# Get and set servers
|
2016-08-07 20:52:49 +02:00
|
|
|
if message_count % 30 == 0:
|
|
|
|
subscriptionManager.serverlist = client.getServerList()
|
|
|
|
subscriptionManager.notify()
|
|
|
|
if not httpd:
|
|
|
|
message_count = 0
|
2016-01-22 15:37:20 +01:00
|
|
|
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())
|
2016-07-22 15:04:42 +02:00
|
|
|
# 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()
|
2016-08-07 20:52:49 +02:00
|
|
|
# Don't sleep
|
|
|
|
continue
|
2016-12-20 16:38:04 +01:00
|
|
|
sleep(20)
|
2016-01-22 15:37:20 +01:00
|
|
|
|
2016-04-02 16:46:23 +02:00
|
|
|
client.stop_all()
|
2016-04-26 17:15:05 +02:00
|
|
|
if httpd:
|
|
|
|
try:
|
2016-12-20 16:38:04 +01:00
|
|
|
httpd.socket.shutdown(SHUT_RDWR)
|
2016-04-26 17:15:05 +02:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
finally:
|
|
|
|
httpd.socket.close()
|
2016-09-02 17:20:19 +02:00
|
|
|
log.info("----===## Plex Companion stopped ##===----")
|