2015-12-24 13:51:47 -06:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2016-02-19 20:03:06 +01:00
|
|
|
###############################################################################
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-09-01 19:53:04 +02:00
|
|
|
import logging
|
2015-12-24 13:51:47 -06:00
|
|
|
import json
|
2016-02-11 10:30:29 +01:00
|
|
|
from urllib import urlencode
|
2016-08-07 15:33:36 +02:00
|
|
|
from threading import Lock
|
|
|
|
from functools import wraps
|
2015-12-24 13:51:47 -06:00
|
|
|
|
|
|
|
import xbmc
|
|
|
|
|
|
|
|
import embydb_functions as embydb
|
2016-09-01 19:53:04 +02:00
|
|
|
from utils import window, tryEncode
|
2016-05-08 13:17:04 +02:00
|
|
|
import playbackutils
|
2016-02-11 10:30:29 +01:00
|
|
|
import PlexFunctions
|
|
|
|
import PlexAPI
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-02-19 20:03:06 +01:00
|
|
|
###############################################################################
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-09-01 19:53:04 +02:00
|
|
|
log = logging.getLogger("PLEX."+__name__)
|
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
class lockMethod:
|
2016-08-08 18:40:35 +02:00
|
|
|
"""
|
|
|
|
Decorator for class methods to lock hem completely. Same lock is used for
|
|
|
|
every single decorator and instance used!
|
|
|
|
|
|
|
|
Here only used for Playlist()
|
|
|
|
"""
|
2016-08-07 15:33:36 +02:00
|
|
|
lock = Lock()
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def decorate(cls, func):
|
|
|
|
@wraps(func)
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
with cls.lock:
|
|
|
|
result = func(*args, **kwargs)
|
|
|
|
return result
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
2015-12-24 13:51:47 -06:00
|
|
|
class Playlist():
|
2016-06-26 16:10:32 +02:00
|
|
|
"""
|
|
|
|
Initiate with Playlist(typus='video' or 'music')
|
|
|
|
"""
|
2016-08-07 15:33:36 +02:00
|
|
|
# Borg - multiple instances, shared state
|
|
|
|
_shared_state = {}
|
|
|
|
|
2016-08-11 22:11:00 +02:00
|
|
|
typus = None
|
|
|
|
queueId = None
|
|
|
|
playQueueVersion = None
|
|
|
|
guid = None
|
|
|
|
playlistId = None
|
|
|
|
player = xbmc.Player()
|
|
|
|
# "interal" PKC playlist
|
|
|
|
items = []
|
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
|
|
|
def __init__(self, typus=None):
|
|
|
|
# Borg
|
|
|
|
self.__dict__ = self._shared_state
|
|
|
|
|
2016-09-01 19:53:04 +02:00
|
|
|
self.userid = window('currUserId')
|
|
|
|
self.server = window('pms_server')
|
2016-07-20 18:36:31 +02:00
|
|
|
# Construct the Kodi playlist instance
|
2016-08-11 22:11:00 +02:00
|
|
|
if self.typus == typus:
|
|
|
|
return
|
2016-06-26 16:10:32 +02:00
|
|
|
if typus == 'video':
|
|
|
|
self.playlist = xbmc.PlayList(xbmc.PLAYLIST_VIDEO)
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = 'video'
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Initiated video playlist')
|
2016-06-26 16:10:32 +02:00
|
|
|
elif typus == 'music':
|
|
|
|
self.playlist = xbmc.PlayList(xbmc.PLAYLIST_MUSIC)
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = 'music'
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Initiated music playlist')
|
2016-06-26 16:10:32 +02:00
|
|
|
else:
|
|
|
|
self.playlist = None
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = None
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Empty playlist initiated')
|
2016-07-19 18:22:34 +02:00
|
|
|
if self.playlist is not None:
|
|
|
|
self.playlistId = self.playlist.getPlayListId()
|
2016-08-11 22:11:00 +02:00
|
|
|
|
|
|
|
@lockMethod.decorate
|
|
|
|
def getQueueIdFromPosition(self, playlistPosition):
|
|
|
|
return self.items[playlistPosition]['playQueueItemID']
|
|
|
|
|
|
|
|
@lockMethod.decorate
|
|
|
|
def Typus(self, value=None):
|
|
|
|
if value:
|
|
|
|
self.typus = value
|
|
|
|
else:
|
|
|
|
return self.typus
|
|
|
|
|
|
|
|
@lockMethod.decorate
|
|
|
|
def PlayQueueVersion(self, value=None):
|
|
|
|
if value:
|
|
|
|
self.playQueueVersion = value
|
|
|
|
else:
|
|
|
|
return self.playQueueVersion
|
|
|
|
|
|
|
|
@lockMethod.decorate
|
|
|
|
def QueueId(self, value=None):
|
|
|
|
if value:
|
|
|
|
self.queueId = value
|
|
|
|
else:
|
|
|
|
return self.queueId
|
|
|
|
|
|
|
|
@lockMethod.decorate
|
|
|
|
def Guid(self, value=None):
|
|
|
|
if value:
|
|
|
|
self.guid = value
|
|
|
|
else:
|
|
|
|
return self.guid
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2016-07-20 18:36:31 +02:00
|
|
|
def clear(self):
|
|
|
|
"""
|
2016-08-11 22:11:00 +02:00
|
|
|
Empties current Kodi playlist and associated variables
|
2016-07-20 18:36:31 +02:00
|
|
|
"""
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Clearing playlist')
|
2016-07-20 18:36:31 +02:00
|
|
|
self.playlist.clear()
|
|
|
|
self.items = []
|
2016-08-11 22:11:00 +02:00
|
|
|
self.queueId = None
|
|
|
|
self.playQueueVersion = None
|
|
|
|
self.guid = None
|
2016-07-20 18:36:31 +02:00
|
|
|
|
|
|
|
def _initiatePlaylist(self):
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Initiating playlist')
|
2016-06-26 16:10:32 +02:00
|
|
|
playlist = None
|
|
|
|
with embydb.GetEmbyDB() as emby_db:
|
2016-07-20 18:36:31 +02:00
|
|
|
for item in self.items:
|
|
|
|
itemid = item['plexId']
|
2016-06-26 16:10:32 +02:00
|
|
|
embydb_item = emby_db.getItem_byId(itemid)
|
|
|
|
try:
|
|
|
|
mediatype = embydb_item[4]
|
|
|
|
except TypeError:
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Couldnt find item %s in Kodi db' % itemid)
|
2016-06-26 16:10:32 +02:00
|
|
|
item = PlexFunctions.GetPlexMetadata(itemid)
|
|
|
|
if item in (None, 401):
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Couldnt find item %s on PMS, trying next'
|
|
|
|
% itemid)
|
2016-06-26 16:10:32 +02:00
|
|
|
continue
|
|
|
|
if PlexAPI.API(item[0]).getType() == 'track':
|
|
|
|
playlist = xbmc.PlayList(xbmc.PLAYLIST_MUSIC)
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Music playlist initiated')
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = 'music'
|
2016-06-26 16:10:32 +02:00
|
|
|
else:
|
|
|
|
playlist = xbmc.PlayList(xbmc.PLAYLIST_VIDEO)
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Video playlist initiated')
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = 'video'
|
2016-06-26 16:10:32 +02:00
|
|
|
else:
|
|
|
|
if mediatype == 'song':
|
|
|
|
playlist = xbmc.PlayList(xbmc.PLAYLIST_MUSIC)
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Music playlist initiated')
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = 'music'
|
2016-06-26 16:10:32 +02:00
|
|
|
else:
|
|
|
|
playlist = xbmc.PlayList(xbmc.PLAYLIST_VIDEO)
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Video playlist initiated')
|
2016-07-20 18:36:31 +02:00
|
|
|
self.typus = 'video'
|
2016-06-26 16:10:32 +02:00
|
|
|
break
|
|
|
|
self.playlist = playlist
|
2016-07-19 18:22:34 +02:00
|
|
|
if self.playlist is not None:
|
|
|
|
self.playlistId = self.playlist.getPlayListId()
|
2016-06-26 16:10:32 +02:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
def _processItems(self, startitem, startPlayer=False):
|
2016-07-23 17:32:57 +02:00
|
|
|
startpos = None
|
2016-02-10 13:50:04 +01:00
|
|
|
with embydb.GetEmbyDB() as emby_db:
|
2016-07-20 18:36:31 +02:00
|
|
|
for pos, item in enumerate(self.items):
|
|
|
|
kodiId = None
|
|
|
|
plexId = item['plexId']
|
|
|
|
embydb_item = emby_db.getItem_byId(plexId)
|
2016-02-10 13:50:04 +01:00
|
|
|
try:
|
2016-07-20 18:36:31 +02:00
|
|
|
kodiId = embydb_item[0]
|
2016-02-10 13:50:04 +01:00
|
|
|
mediatype = embydb_item[4]
|
|
|
|
except TypeError:
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Couldnt find item %s in Kodi db' % plexId)
|
2016-07-20 18:36:31 +02:00
|
|
|
xml = PlexFunctions.GetPlexMetadata(plexId)
|
|
|
|
if xml in (None, 401):
|
2016-09-01 19:53:04 +02:00
|
|
|
log.error('Could not download plexId %s' % plexId)
|
2016-04-07 18:29:23 +02:00
|
|
|
else:
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug('Downloaded xml metadata, adding now')
|
2016-07-20 18:36:31 +02:00
|
|
|
self._addtoPlaylist_xbmc(xml[0])
|
2016-02-10 13:50:04 +01:00
|
|
|
else:
|
|
|
|
# Add to playlist
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug("Adding %s PlexId %s, KodiId %s to playlist."
|
|
|
|
% (mediatype, plexId, kodiId))
|
2016-08-07 15:33:36 +02:00
|
|
|
self._addtoPlaylist(kodiId, mediatype)
|
2016-07-20 18:36:31 +02:00
|
|
|
# Add the kodiId
|
|
|
|
if kodiId is not None:
|
|
|
|
item['kodiId'] = str(kodiId)
|
2016-07-23 17:32:57 +02:00
|
|
|
if (startpos is None and startitem[1] == item[startitem[0]]):
|
|
|
|
startpos = pos
|
|
|
|
|
|
|
|
if startPlayer is True and len(self.playlist) > 0:
|
|
|
|
if startpos is not None:
|
2016-07-23 17:38:45 +02:00
|
|
|
self.player.play(self.playlist, startpos=startpos)
|
2016-07-23 17:32:57 +02:00
|
|
|
else:
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info('Never received a starting item for playlist, '
|
|
|
|
'starting with the first entry')
|
2016-07-23 17:38:45 +02:00
|
|
|
self.player.play(self.playlist)
|
2016-07-20 18:36:31 +02:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2016-07-20 18:36:31 +02:00
|
|
|
def playAll(self, items, startitem, offset):
|
|
|
|
"""
|
|
|
|
items: list of dicts of the form
|
|
|
|
{
|
2016-08-11 22:11:00 +02:00
|
|
|
'playQueueItemID': Plex playQueueItemID, e.g. '29175'
|
2016-07-20 18:36:31 +02:00
|
|
|
'plexId': Plex ratingKey, e.g. '125'
|
|
|
|
'kodiId': Kodi's db id of the same item
|
|
|
|
}
|
2016-02-10 13:50:04 +01:00
|
|
|
|
2016-08-11 22:11:00 +02:00
|
|
|
startitem: tuple (typus, id), where typus is either
|
|
|
|
'playQueueItemID' or 'plexId' and id is the corresponding
|
|
|
|
id as a string
|
2016-07-20 18:36:31 +02:00
|
|
|
offset: First item's time offset to play in Kodi time (an int)
|
|
|
|
"""
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info("---*** PLAY ALL ***---")
|
|
|
|
log.debug('Startitem: %s, offset: %s, items: %s'
|
|
|
|
% (startitem, offset, items))
|
2016-07-20 18:36:31 +02:00
|
|
|
self.items = items
|
2016-06-26 16:10:32 +02:00
|
|
|
if self.playlist is None:
|
2016-07-20 18:36:31 +02:00
|
|
|
self._initiatePlaylist()
|
2016-06-26 16:10:32 +02:00
|
|
|
if self.playlist is None:
|
2016-09-01 19:53:04 +02:00
|
|
|
log.error('Could not create playlist, abort')
|
2016-06-26 16:10:32 +02:00
|
|
|
return
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-09-01 19:53:04 +02:00
|
|
|
window('plex_customplaylist', value="true")
|
2016-07-20 18:36:31 +02:00
|
|
|
if offset != 0:
|
2016-06-26 16:10:32 +02:00
|
|
|
# Seek to the starting position
|
2016-09-01 19:53:04 +02:00
|
|
|
window('plex_customplaylist.seektime', str(offset))
|
2016-08-07 15:33:36 +02:00
|
|
|
self._processItems(startitem, startPlayer=True)
|
2016-07-20 18:36:31 +02:00
|
|
|
# Log playlist
|
2016-08-07 15:33:36 +02:00
|
|
|
self._verifyPlaylist()
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug('Internal playlist: %s' % self.items)
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2015-12-24 13:51:47 -06:00
|
|
|
def modifyPlaylist(self, itemids):
|
2016-09-01 19:53:04 +02:00
|
|
|
log.info("---*** MODIFY PLAYLIST ***---")
|
|
|
|
log.debug("Items: %s" % itemids)
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-07-19 18:22:34 +02:00
|
|
|
self._initiatePlaylist(itemids)
|
2016-08-07 15:33:36 +02:00
|
|
|
self._processItems(itemids, startPlayer=True)
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
self._verifyPlaylist()
|
2016-04-26 14:41:58 +02:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2015-12-24 13:51:47 -06:00
|
|
|
def addtoPlaylist(self, dbid=None, mediatype=None, url=None):
|
2016-06-26 16:10:32 +02:00
|
|
|
"""
|
|
|
|
mediatype: Kodi type: 'movie', 'episode', 'musicvideo', 'artist',
|
|
|
|
'album', 'song', 'genre'
|
|
|
|
"""
|
2016-08-11 22:11:00 +02:00
|
|
|
self._addtoPlaylist(dbid=None, mediatype=None, url=None)
|
|
|
|
|
|
|
|
def _addtoPlaylist(self, dbid=None, mediatype=None, url=None):
|
2015-12-24 13:51:47 -06:00
|
|
|
pl = {
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'id': 1,
|
|
|
|
'method': "Playlist.Add",
|
|
|
|
'params': {
|
2016-07-19 18:22:34 +02:00
|
|
|
'playlistid': self.playlistId
|
2015-12-24 13:51:47 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if dbid is not None:
|
2016-09-01 19:53:04 +02:00
|
|
|
pl['params']['item'] = {'%sid' % tryEncode(mediatype): int(dbid)}
|
2015-12-24 13:51:47 -06:00
|
|
|
else:
|
|
|
|
pl['params']['item'] = {'file': url}
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug(xbmc.executeJSONRPC(json.dumps(pl)))
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-07-19 18:22:34 +02:00
|
|
|
def _addtoPlaylist_xbmc(self, item):
|
|
|
|
API = PlexAPI.API(item)
|
2016-02-11 10:30:29 +01:00
|
|
|
params = {
|
|
|
|
'mode': "play",
|
2016-05-08 13:17:04 +02:00
|
|
|
'dbid': 999999999,
|
|
|
|
'id': API.getRatingKey(),
|
|
|
|
'filename': API.getKey()
|
2016-02-11 10:30:29 +01:00
|
|
|
}
|
2016-05-08 13:17:04 +02:00
|
|
|
playurl = "plugin://plugin.video.plexkodiconnect.movies/?%s" \
|
|
|
|
% urlencode(params)
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-05-08 13:17:04 +02:00
|
|
|
listitem = API.CreateListItemFromPlexItem()
|
2016-07-19 18:22:34 +02:00
|
|
|
playbackutils.PlaybackUtils(item).setArtwork(listitem)
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-06-26 16:10:32 +02:00
|
|
|
self.playlist.add(playurl, listitem)
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2016-09-01 19:53:04 +02:00
|
|
|
def insertintoPlaylist(self,
|
|
|
|
position,
|
|
|
|
dbid=None,
|
|
|
|
mediatype=None,
|
|
|
|
url=None):
|
2015-12-24 13:51:47 -06:00
|
|
|
pl = {
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'id': 1,
|
|
|
|
'method': "Playlist.Insert",
|
|
|
|
'params': {
|
2016-07-19 18:22:34 +02:00
|
|
|
'playlistid': self.playlistId,
|
2015-12-24 13:51:47 -06:00
|
|
|
'position': position
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if dbid is not None:
|
2016-09-01 19:53:04 +02:00
|
|
|
pl['params']['item'] = {'%sid' % tryEncode(mediatype): int(dbid)}
|
2015-12-24 13:51:47 -06:00
|
|
|
else:
|
|
|
|
pl['params']['item'] = {'file': url}
|
|
|
|
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug(xbmc.executeJSONRPC(json.dumps(pl)))
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2015-12-24 13:51:47 -06:00
|
|
|
def verifyPlaylist(self):
|
2016-08-07 15:33:36 +02:00
|
|
|
self._verifyPlaylist()
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
def _verifyPlaylist(self):
|
2015-12-24 13:51:47 -06:00
|
|
|
pl = {
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'id': 1,
|
|
|
|
'method': "Playlist.GetItems",
|
|
|
|
'params': {
|
2016-07-19 18:22:34 +02:00
|
|
|
'playlistid': self.playlistId,
|
2016-02-07 13:26:28 +01:00
|
|
|
'properties': ['title', 'file']
|
2015-12-24 13:51:47 -06:00
|
|
|
}
|
|
|
|
}
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug(xbmc.executeJSONRPC(json.dumps(pl)))
|
2015-12-24 13:51:47 -06:00
|
|
|
|
2016-08-07 15:33:36 +02:00
|
|
|
@lockMethod.decorate
|
2015-12-24 13:51:47 -06:00
|
|
|
def removefromPlaylist(self, position):
|
|
|
|
pl = {
|
|
|
|
'jsonrpc': "2.0",
|
|
|
|
'id': 1,
|
|
|
|
'method': "Playlist.Remove",
|
|
|
|
'params': {
|
2016-07-19 18:22:34 +02:00
|
|
|
'playlistid': self.playlistId,
|
2015-12-24 13:51:47 -06:00
|
|
|
'position': position
|
|
|
|
}
|
|
|
|
}
|
2016-09-01 19:53:04 +02:00
|
|
|
log.debug(xbmc.executeJSONRPC(json.dumps(pl)))
|