python-plexapi/plexapi/playlist.py

215 lines
9 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
from plexapi import utils
from plexapi.base import PlexPartialObject, Playable
2018-09-08 15:25:16 +00:00
from plexapi.exceptions import BadRequest, Unsupported
2017-02-07 06:58:29 +00:00
from plexapi.playqueue import PlayQueue
2016-02-03 18:07:53 +00:00
from plexapi.utils import cast, toDatetime
2018-09-08 15:25:16 +00:00
from plexapi.compat import quote_plus
2016-02-03 18:07:53 +00:00
@utils.registerPlexObject
class Playlist(PlexPartialObject, Playable):
2017-02-20 05:37:00 +00:00
""" Represents a single Playlist object.
# TODO: Document attributes
"""
TAG = 'Playlist'
2016-02-03 18:07:53 +00:00
TYPE = 'playlist'
def _loadData(self, data):
""" Load attribute values from Plex XML response. """
Playable._loadData(self, data)
2017-02-04 17:43:50 +00:00
self.addedAt = toDatetime(data.attrib.get('addedAt'))
self.composite = data.attrib.get('composite') # url to thumbnail
self.duration = cast(int, data.attrib.get('duration'))
self.durationInSeconds = cast(int, data.attrib.get('durationInSeconds'))
self.guid = data.attrib.get('guid')
self.key = data.attrib.get('key')
2016-04-11 03:49:23 +00:00
self.key = self.key.replace('/items', '') if self.key else self.key # FIX_BUG_50
2017-02-04 17:43:50 +00:00
self.leafCount = cast(int, data.attrib.get('leafCount'))
self.playlistType = data.attrib.get('playlistType')
self.ratingKey = cast(int, data.attrib.get('ratingKey'))
self.smart = cast(bool, data.attrib.get('smart'))
self.summary = data.attrib.get('summary')
self.title = data.attrib.get('title')
self.type = data.attrib.get('type')
self.updatedAt = toDatetime(data.attrib.get('updatedAt'))
2018-09-08 15:25:16 +00:00
self.allowSync = cast(bool, data.attrib.get('allowSync'))
2017-10-09 13:58:44 +00:00
self._items = None # cache for self.items
2017-10-05 20:24:49 +00:00
2017-10-25 22:01:42 +00:00
def __len__(self): # pragma: no cover
2017-10-05 20:24:49 +00:00
return len(self.items())
2018-09-08 15:25:16 +00:00
@property
def metadataType(self):
if self.isVideo:
return 'movie'
elif self.isAudio:
return 'track'
elif self.isPhoto:
return 'photo'
else:
raise Unsupported('Unexpected playlist type')
@property
def isVideo(self):
return self.playlistType == 'video'
@property
def isAudio(self):
return self.playlistType == 'audio'
@property
def isPhoto(self):
return self.playlistType == 'photo'
2017-10-25 22:01:42 +00:00
def __contains__(self, other): # pragma: no cover
2017-10-05 20:24:49 +00:00
return any(i.key == other.key for i in self.items())
2017-10-25 22:01:42 +00:00
def __getitem__(self, key): # pragma: no cover
2017-10-05 20:24:49 +00:00
return self.items()[key]
2016-02-03 18:07:53 +00:00
def items(self):
""" Returns a list of all items in the playlist. """
2017-10-05 20:24:49 +00:00
if self._items is None:
key = '%s/items' % self.key
items = self.fetchItems(key)
self._items = items
return self._items
2016-12-21 13:17:28 +00:00
2016-04-11 03:49:23 +00:00
def addItems(self, items):
""" Add items to a playlist. """
2016-04-11 03:49:23 +00:00
if not isinstance(items, (list, tuple)):
items = [items]
ratingKeys = []
for item in items:
2017-10-25 22:01:42 +00:00
if item.listType != self.playlistType: # pragma: no cover
2017-02-20 05:37:00 +00:00
raise BadRequest('Can not mix media types when building a playlist: %s and %s' %
(self.playlistType, item.listType))
2017-01-09 14:21:54 +00:00
ratingKeys.append(str(item.ratingKey))
uuid = items[0].section().uuid
2017-02-02 14:09:34 +00:00
ratingKeys = ','.join(ratingKeys)
key = '%s/items%s' % (self.key, utils.joinArgs({
'uri': 'library://%s/directory//library/metadata/%s' % (uuid, ratingKeys)
2016-04-11 03:49:23 +00:00
}))
result = self._server.query(key, method=self._server._session.put)
self.reload()
return result
2016-04-11 03:49:23 +00:00
def removeItem(self, item):
""" Remove a file from a playlist. """
key = '%s/items/%s' % (self.key, item.playlistItemID)
result = self._server.query(key, method=self._server._session.delete)
self.reload()
return result
2016-04-11 03:49:23 +00:00
def moveItem(self, item, after=None):
""" Move a to a new position in playlist. """
key = '%s/items/%s/move' % (self.key, item.playlistItemID)
2017-01-02 21:06:40 +00:00
if after:
key += '?after=%s' % after.playlistItemID
result = self._server.query(key, method=self._server._session.put)
self.reload()
return result
2016-12-21 13:17:28 +00:00
2016-04-11 03:49:23 +00:00
def edit(self, title=None, summary=None):
""" Edit playlist. """
2017-02-20 05:37:00 +00:00
key = '/library/metadata/%s%s' % (self.ratingKey, utils.joinArgs({'title': title, 'summary': summary}))
2017-10-11 20:40:10 +00:00
result = self._server.query(key, method=self._server._session.put)
self.reload()
return result
2016-12-21 13:17:28 +00:00
2016-04-11 03:49:23 +00:00
def delete(self):
""" Delete playlist. """
return self._server.query(self.key, method=self._server._session.delete)
2016-12-21 13:17:28 +00:00
2017-02-07 06:58:29 +00:00
def playQueue(self, *args, **kwargs):
""" Create a playqueue from this playlist. """
return PlayQueue.create(self._server, self, *args, **kwargs)
2017-02-07 06:58:29 +00:00
2016-04-11 03:49:23 +00:00
@classmethod
def create(cls, server, title, items):
""" Create a playlist. """
2016-04-11 03:49:23 +00:00
if not isinstance(items, (list, tuple)):
items = [items]
ratingKeys = []
for item in items:
2017-10-25 22:01:42 +00:00
if item.listType != items[0].listType: # pragma: no cover
2016-04-11 03:49:23 +00:00
raise BadRequest('Can not mix media types when building a playlist')
2016-12-21 13:17:28 +00:00
ratingKeys.append(str(item.ratingKey))
ratingKeys = ','.join(ratingKeys)
2016-04-13 03:52:47 +00:00
uuid = items[0].section().uuid
key = '/playlists%s' % utils.joinArgs({
'uri': 'library://%s/directory//library/metadata/%s' % (uuid, ratingKeys),
'type': items[0].listType,
2016-04-11 03:49:23 +00:00
'title': title,
'smart': 0
})
data = server.query(key, method=server._session.post)[0]
return cls(server, data, initpath=key)
2017-07-17 14:11:03 +00:00
def copyToUser(self, user):
""" Copy playlist to another user account. """
2017-07-17 14:11:03 +00:00
from plexapi.server import PlexServer
myplex = self._server.myPlexAccount()
user = myplex.user(user)
# Get the token for your machine.
token = user.get_token(self._server.machineIdentifier)
# Login to your server using your friends credentials.
user_server = PlexServer(self._server._baseurl, token)
return self.create(user_server, self.title, self.items())
2018-09-08 15:25:16 +00:00
def sync(self, videoQuality=None, photoResolution=None, audioBitrate=None, client=None, clientId=None, limit=None,
unwatched=False, title=None):
""" Add current playlist as sync item for specified device.
See :func:`plexapi.myplex.MyPlexAccount.sync()` for possible exceptions.
Parameters:
videoQuality (int): idx of quality of the video, one of VIDEO_QUALITY_* values defined in
:mod:`plexapi.sync` module. Used only when playlist contains video.
photoResolution (str): maximum allowed resolution for synchronized photos, see PHOTO_QUALITY_* values in
the module :mod:`plexapi.sync`. Used only when playlist contains photos.
audioBitrate (int): maximum bitrate for synchronized music, better use one of MUSIC_BITRATE_* values from the
module :mod:`plexapi.sync`. Used only when playlist contains audio.
client (:class:`plexapi.myplex.MyPlexDevice`): sync destination, see
:func:`plexapi.myplex.MyPlexAccount.sync`.
clientId (str): sync destination, see :func:`plexapi.myplex.MyPlexAccount.sync`.
limit (int): maximum count of items to sync, unlimited if `None`.
unwatched (bool): if `True` watched videos wouldn't be synced.
title (str): descriptive title for the new :class:`plexapi.sync.SyncItem`, if empty the value would be
generated from metadata of current photo.
Raises:
:class:`plexapi.exceptions.BadRequest`: when playlist is not allowed to sync.
:class:`plexapi.exceptions.Unsupported`: when playlist content is unsupported.
Returns:
:class:`plexapi.sync.SyncItem`: an instance of created syncItem.
"""
if not self.allowSync:
raise BadRequest('The playlist is not allowed to sync')
from plexapi.sync import SyncItem, Policy, MediaSettings
myplex = self._server.myPlexAccount()
sync_item = SyncItem(self._server, None)
sync_item.title = title if title else self.title
sync_item.rootTitle = self.title
sync_item.contentType = self.playlistType
sync_item.metadataType = self.metadataType
sync_item.machineIdentifier = self._server.machineIdentifier
sync_item.location = 'playlist:///%s' % quote_plus(self.guid)
sync_item.policy = Policy.create(limit, unwatched)
if self.isVideo:
sync_item.mediaSettings = MediaSettings.createVideo(videoQuality)
elif self.isAudio:
sync_item.mediaSettings = MediaSettings.createMusic(audioBitrate)
elif self.isPhoto:
sync_item.mediaSettings = MediaSettings.createPhoto(photoResolution)
else:
raise Unsupported('Unsupported playlist content')
return myplex.sync(sync_item, client=client, clientId=clientId)