PlexKodiConnect/resources/lib/playqueue.py

238 lines
9.8 KiB
Python
Raw Normal View History

2017-12-08 19:43:06 +01:00
"""
Monitors the Kodi playqueue and adjusts the Plex playqueue accordingly
"""
2017-12-09 14:35:08 +01:00
from logging import getLogger
2017-01-02 14:07:24 +01:00
from threading import RLock, Thread
2016-12-27 17:33:52 +01:00
2017-01-02 14:07:24 +01:00
from xbmc import sleep, Player, PlayList, PLAYLIST_MUSIC, PLAYLIST_VIDEO
2016-12-27 17:33:52 +01:00
2017-05-17 13:55:24 +02:00
from utils import window, thread_methods
2016-12-27 17:33:52 +01:00
import playlist_func as PL
2017-03-05 17:51:58 +01:00
from PlexFunctions import ConvertPlexToKodiTime, GetAllPlexChildren
from PlexAPI import API
2017-01-09 19:56:57 +01:00
from playbackutils import PlaybackUtils
2017-12-08 19:43:06 +01:00
import json_rpc as js
2017-03-05 17:51:58 +01:00
import variables as v
2016-12-27 17:33:52 +01:00
###############################################################################
2017-12-09 14:35:08 +01:00
LOG = getLogger("PLEX." + __name__)
2016-12-27 17:33:52 +01:00
2017-12-08 19:43:06 +01:00
# lock used for playqueue manipulations
LOCK = RLock()
PLUGIN = 'plugin://%s' % v.ADDON_ID
2016-12-27 17:33:52 +01:00
###############################################################################
2017-05-17 13:55:24 +02:00
@thread_methods(add_suspends=['PMS_STATUS'])
2016-12-27 17:33:52 +01:00
class Playqueue(Thread):
"""
Monitors Kodi's playqueues for changes on the Kodi side
"""
# Borg - multiple instances, shared state
__shared_state = {}
playqueues = None
def __init__(self, callback=None):
self.__dict__ = self.__shared_state
if self.playqueues is not None:
2017-12-08 19:43:06 +01:00
LOG.debug('Playqueue thread has already been initialized')
2017-03-05 17:59:56 +01:00
Thread.__init__(self)
2016-12-27 17:33:52 +01:00
return
self.mgr = callback
# Initialize Kodi playqueues
2017-12-08 19:43:06 +01:00
with LOCK:
2017-01-02 14:07:24 +01:00
self.playqueues = []
2017-12-08 19:43:06 +01:00
for queue in js.get_playlists():
2017-01-02 14:07:24 +01:00
playqueue = PL.Playqueue_Object()
playqueue.playlistid = queue['playlistid']
playqueue.type = queue['type']
# Initialize each Kodi playlist
if playqueue.type == 'audio':
playqueue.kodi_pl = PlayList(PLAYLIST_MUSIC)
elif playqueue.type == 'video':
playqueue.kodi_pl = PlayList(PLAYLIST_VIDEO)
else:
# Currently, only video or audio playqueues available
playqueue.kodi_pl = PlayList(PLAYLIST_VIDEO)
# Overwrite 'picture' with 'photo'
playqueue.type = v.KODI_TYPE_PHOTO
2017-01-02 14:07:24 +01:00
self.playqueues.append(playqueue)
# sort the list by their playlistid, just in case
self.playqueues = sorted(
self.playqueues, key=lambda i: i.playlistid)
LOG.debug('Initialized the Kodi play queues: %s', self.playqueues)
2017-01-02 15:41:38 +01:00
Thread.__init__(self)
2016-12-27 17:33:52 +01:00
2016-12-28 13:14:21 +01:00
def get_playqueue_from_type(self, typus):
"""
Returns the playqueue according to the typus ('video', 'audio',
'picture') passed in
"""
2017-12-08 19:43:06 +01:00
with LOCK:
2017-01-02 14:07:24 +01:00
for playqueue in self.playqueues:
if playqueue.type == typus:
break
else:
raise ValueError('Wrong playlist type passed in: %s' % typus)
return playqueue
2016-12-29 15:41:14 +01:00
2018-01-01 13:28:39 +01:00
def init_playqueue_from_plex_children(self, plex_id, transient_token=None):
2017-03-05 17:51:58 +01:00
"""
Init a new playqueue e.g. from an album. Alexa does this
Returns the Playlist_Object
2017-03-05 17:51:58 +01:00
"""
xml = GetAllPlexChildren(plex_id)
try:
xml[0].attrib
except (TypeError, IndexError, AttributeError):
LOG.error('Could not download the PMS xml for %s', plex_id)
2017-03-05 17:51:58 +01:00
return
playqueue = self.get_playqueue_from_type(
v.KODI_PLAYLIST_TYPE_FROM_PLEX_TYPE[xml[0].attrib['type']])
playqueue.clear()
for i, child in enumerate(xml):
api = API(child)
PL.add_item_to_playlist(playqueue, i, plex_id=api.getRatingKey())
2018-01-01 13:28:39 +01:00
playqueue.plex_transient_token = transient_token
2017-12-08 19:43:06 +01:00
LOG.debug('Firing up Kodi player')
2017-03-05 18:28:30 +01:00
Player().play(playqueue.kodi_pl, None, False, 0)
return playqueue
2017-03-05 17:51:58 +01:00
2016-12-28 13:14:21 +01:00
def update_playqueue_from_PMS(self,
playqueue,
playqueue_id=None,
2016-12-28 14:57:10 +01:00
repeat=None,
offset=None,
transient_token=None):
2016-12-28 13:14:21 +01:00
"""
Completely updates the Kodi playqueue with the new Plex playqueue. Pass
in playqueue_id if we need to fetch a new playqueue
repeat = 0, 1, 2
2017-01-02 14:07:24 +01:00
offset = time offset in Plextime (milliseconds)
2016-12-28 14:48:23 +01:00
"""
2017-12-08 19:43:06 +01:00
LOG.info('New playqueue %s received from Plex companion with offset '
'%s, repeat %s', playqueue_id, offset, repeat)
2018-01-01 13:28:39 +01:00
# Safe transient token from being deleted
if transient_token is None:
transient_token = playqueue.plex_transient_token
2017-12-08 19:43:06 +01:00
with LOCK:
2017-01-09 20:00:37 +01:00
xml = PL.get_PMS_playlist(playqueue, playqueue_id)
playqueue.clear()
try:
PL.get_playlist_details_from_xml(playqueue, xml)
except KeyError:
LOG.error('Could not get playqueue ID %s', playqueue_id)
2017-01-09 20:00:37 +01:00
return
2017-01-02 14:07:24 +01:00
playqueue.repeat = 0 if not repeat else int(repeat)
2018-01-01 13:28:39 +01:00
playqueue.plex_transient_token = transient_token
2017-12-28 21:31:05 +01:00
PlaybackUtils(xml, playqueue).play_all()
2017-01-02 14:07:24 +01:00
window('plex_customplaylist', value="true")
if offset not in (None, "0"):
window('plex_customplaylist.seektime',
str(ConvertPlexToKodiTime(offset)))
for startpos, item in enumerate(playqueue.items):
2017-12-08 19:43:06 +01:00
if item.id == playqueue.selectedItemID:
2017-01-02 14:07:24 +01:00
break
else:
2017-01-09 19:56:57 +01:00
startpos = 0
2017-01-02 14:07:24 +01:00
# Start playback. Player does not return in time
LOG.debug('Playqueues after Plex Companion update are now: %s',
self.playqueues)
2017-01-09 19:56:57 +01:00
thread = Thread(target=Player().play,
args=(playqueue.kodi_pl,
None,
False,
startpos))
2017-01-02 14:07:24 +01:00
thread.setDaemon(True)
thread.start()
2016-12-28 14:48:23 +01:00
2016-12-27 17:33:52 +01:00
def _compare_playqueues(self, playqueue, new):
"""
Used to poll the Kodi playqueue and update the Plex playqueue if needed
"""
2017-01-02 14:07:24 +01:00
old = list(playqueue.items)
2016-12-28 14:48:23 +01:00
index = list(range(0, len(old)))
LOG.debug('Comparing new Kodi playqueue %s with our play queue %s',
new, old)
2017-05-17 15:27:32 +02:00
if self.thread_stopped():
# Chances are that we got an empty Kodi playlist due to
# Kodi exit
return
2016-12-27 17:33:52 +01:00
for i, new_item in enumerate(new):
if (new_item['file'].startswith('plugin://') and
not new_item['file'].startswith(PLUGIN)):
# Ignore new media added by other addons
continue
2016-12-27 17:33:52 +01:00
for j, old_item in enumerate(old):
2017-05-17 15:23:39 +02:00
try:
if (old_item.file.startswith('plugin://') and
not old_item['file'].startswith(PLUGIN)):
# Ignore media by other addons
continue
2017-05-20 18:49:40 +02:00
except (TypeError, AttributeError):
2017-05-17 15:23:39 +02:00
# were not passed a filename; ignore
pass
2017-01-02 14:07:24 +01:00
if new_item.get('id') is None:
identical = old_item.file == new_item['file']
2016-12-27 17:33:52 +01:00
else:
2017-01-02 14:07:24 +01:00
identical = (old_item.kodi_id == new_item['id'] and
old_item.kodi_type == new_item['type'])
2016-12-27 17:33:52 +01:00
if j == 0 and identical:
del old[j], index[j]
break
elif identical:
LOG.debug('Detected playqueue item %s moved to position %s',
i+j, i)
2016-12-27 17:33:52 +01:00
PL.move_playlist_item(playqueue, i + j, i)
2016-12-28 14:48:23 +01:00
del old[j], index[j]
2016-12-27 17:33:52 +01:00
break
2017-01-02 14:07:24 +01:00
else:
LOG.debug('Detected new Kodi element at position %s: %s ',
i, new_item)
2017-12-08 19:43:06 +01:00
if playqueue.id is None:
2017-01-02 14:07:24 +01:00
PL.init_Plex_playlist(playqueue,
kodi_item=new_item)
2016-12-27 17:33:52 +01:00
else:
2017-01-02 14:07:24 +01:00
PL.add_item_to_PMS_playlist(playqueue,
i,
kodi_item=new_item)
for j in range(i, len(index)):
index[j] += 1
2017-01-02 14:07:24 +01:00
for i in reversed(index):
LOG.debug('Detected deletion of playqueue element at pos %s', i)
2017-01-02 14:07:24 +01:00
PL.delete_playlist_item_from_PMS(playqueue, i)
2017-12-08 19:43:06 +01:00
LOG.debug('Done comparing playqueues')
2016-12-28 19:38:43 +01:00
def run(self):
thread_stopped = self.thread_stopped
thread_suspended = self.thread_suspended
2017-12-08 19:43:06 +01:00
LOG.info("----===## Starting PlayQueue client ##===----")
2016-12-28 19:38:43 +01:00
# Initialize the playqueues, if Kodi already got items in them
2017-01-02 14:07:24 +01:00
for playqueue in self.playqueues:
2017-12-21 09:28:06 +01:00
for i, item in enumerate(js.playlist_get_items(playqueue.id)):
2017-01-02 14:07:24 +01:00
if i == 0:
PL.init_Plex_playlist(playqueue, kodi_item=item)
else:
PL.add_item_to_PMS_playlist(playqueue, i, kodi_item=item)
while not thread_stopped():
while thread_suspended():
if thread_stopped():
2016-12-27 17:33:52 +01:00
break
2017-01-02 14:07:24 +01:00
sleep(1000)
2017-12-21 09:28:06 +01:00
# with LOCK:
# for playqueue in self.playqueues:
# kodi_playqueue = js.playlist_get_items(playqueue.id)
# if playqueue.old_kodi_pl != kodi_playqueue:
# # compare old and new playqueue
# self._compare_playqueues(playqueue, kodi_playqueue)
# playqueue.old_kodi_pl = list(kodi_playqueue)
# # Still sleep a bit so Kodi does not become
# # unresponsive
# sleep(10)
# continue
sleep(200)
2017-12-08 19:43:06 +01:00
LOG.info("----===## PlayQueue client stopped ##===----")