2017-02-04 19:46:51 +00:00
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
import re
|
2020-12-06 06:39:13 +00:00
|
|
|
|
import weakref
|
2022-07-21 03:03:20 +00:00
|
|
|
|
from urllib.parse import urlencode
|
2021-05-15 18:25:26 +00:00
|
|
|
|
from xml.etree import ElementTree
|
2017-07-18 15:59:23 +00:00
|
|
|
|
|
2017-02-06 04:52:10 +00:00
|
|
|
|
from plexapi import log, utils
|
2017-02-10 23:16:23 +00:00
|
|
|
|
from plexapi.exceptions import BadRequest, NotFound, UnknownType, Unsupported
|
2017-02-09 06:54:38 +00:00
|
|
|
|
|
2021-05-30 23:25:25 +00:00
|
|
|
|
USER_DONT_RELOAD_FOR_KEYS = set()
|
2022-07-21 03:03:20 +00:00
|
|
|
|
_DONT_RELOAD_FOR_KEYS = {'key'}
|
2017-02-09 06:54:38 +00:00
|
|
|
|
OPERATORS = {
|
2017-02-10 23:16:23 +00:00
|
|
|
|
'exact': lambda v, q: v == q,
|
|
|
|
|
'iexact': lambda v, q: v.lower() == q.lower(),
|
|
|
|
|
'contains': lambda v, q: q in v,
|
|
|
|
|
'icontains': lambda v, q: q.lower() in v.lower(),
|
2017-10-13 23:46:09 +00:00
|
|
|
|
'ne': lambda v, q: v != q,
|
2017-02-10 23:16:23 +00:00
|
|
|
|
'in': lambda v, q: v in q,
|
|
|
|
|
'gt': lambda v, q: v > q,
|
|
|
|
|
'gte': lambda v, q: v >= q,
|
|
|
|
|
'lt': lambda v, q: v < q,
|
|
|
|
|
'lte': lambda v, q: v <= q,
|
|
|
|
|
'startswith': lambda v, q: v.startswith(q),
|
|
|
|
|
'istartswith': lambda v, q: v.lower().startswith(q),
|
|
|
|
|
'endswith': lambda v, q: v.endswith(q),
|
|
|
|
|
'iendswith': lambda v, q: v.lower().endswith(q),
|
2017-02-13 03:38:56 +00:00
|
|
|
|
'exists': lambda v, q: v is not None if q else v is None,
|
2017-02-10 23:16:23 +00:00
|
|
|
|
'regex': lambda v, q: re.match(q, v),
|
|
|
|
|
'iregex': lambda v, q: re.match(q, v, flags=re.IGNORECASE),
|
2017-02-09 06:54:38 +00:00
|
|
|
|
}
|
2017-02-04 19:46:51 +00:00
|
|
|
|
|
|
|
|
|
|
2022-05-30 16:05:00 +00:00
|
|
|
|
class PlexObject:
|
2017-02-07 06:20:49 +00:00
|
|
|
|
""" Base class for all Plex objects.
|
|
|
|
|
|
2017-02-13 02:55:55 +00:00
|
|
|
|
Parameters:
|
|
|
|
|
server (:class:`~plexapi.server.PlexServer`): PlexServer this client is connected to (optional)
|
|
|
|
|
data (ElementTree): Response from PlexServer used to build this object (optional).
|
|
|
|
|
initpath (str): Relative path requested when retrieving specified `data` (optional).
|
2020-12-06 06:57:25 +00:00
|
|
|
|
parent (:class:`~plexapi.base.PlexObject`): The parent object that this object is built from (optional).
|
2017-02-13 02:55:55 +00:00
|
|
|
|
"""
|
|
|
|
|
TAG = None # xml element tag
|
|
|
|
|
TYPE = None # xml element type
|
|
|
|
|
key = None # plex relative url
|
|
|
|
|
|
2020-12-06 06:39:13 +00:00
|
|
|
|
def __init__(self, server, data, initpath=None, parent=None):
|
2017-02-13 02:55:55 +00:00
|
|
|
|
self._server = server
|
|
|
|
|
self._data = data
|
|
|
|
|
self._initpath = initpath or self.key
|
2021-06-06 20:51:59 +00:00
|
|
|
|
self._parent = weakref.ref(parent) if parent is not None else None
|
2021-02-05 02:56:21 +00:00
|
|
|
|
self._details_key = None
|
2022-05-30 15:53:27 +00:00
|
|
|
|
self._overwriteNone = True # Allow overwriting previous attribute values with `None` when manually reloading
|
|
|
|
|
self._autoReload = True # Automatically reload the object when accessing a missing attribute
|
|
|
|
|
self._edits = None # Save batch edits for a single API call
|
2017-02-13 06:37:23 +00:00
|
|
|
|
if data is not None:
|
|
|
|
|
self._loadData(data)
|
2020-11-21 20:08:27 +00:00
|
|
|
|
self._details_key = self._buildDetailsKey()
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
|
|
|
|
def __repr__(self):
|
2017-02-15 05:13:22 +00:00
|
|
|
|
uid = self._clean(self.firstAttr('_baseurl', 'key', 'id', 'playQueueID', 'uri'))
|
2017-02-23 06:33:30 +00:00
|
|
|
|
name = self._clean(self.firstAttr('title', 'name', 'username', 'product', 'tag', 'value'))
|
2017-02-15 05:13:22 +00:00
|
|
|
|
return '<%s>' % ':'.join([p for p in [self.__class__.__name__, uid, name] if p])
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
|
|
|
|
def __setattr__(self, attr, value):
|
2022-05-17 03:03:06 +00:00
|
|
|
|
overwriteNone = self.__dict__.get('_overwriteNone')
|
|
|
|
|
# Don't overwrite an attr with None unless it's a private variable or overwrite None is True
|
|
|
|
|
if value is not None or attr.startswith('_') or attr not in self.__dict__ or overwriteNone:
|
2017-02-07 06:20:49 +00:00
|
|
|
|
self.__dict__[attr] = value
|
|
|
|
|
|
2017-02-15 05:13:22 +00:00
|
|
|
|
def _clean(self, value):
|
|
|
|
|
""" Clean attr value for display in __repr__. """
|
|
|
|
|
if value:
|
2017-02-15 05:43:48 +00:00
|
|
|
|
value = str(value).replace('/library/metadata/', '')
|
|
|
|
|
value = value.replace('/children', '')
|
2021-01-03 00:44:02 +00:00
|
|
|
|
value = value.replace('/accounts/', '')
|
2021-01-03 02:04:46 +00:00
|
|
|
|
value = value.replace('/devices/', '')
|
2017-02-15 05:13:22 +00:00
|
|
|
|
return value.replace(' ', '-')[:20]
|
|
|
|
|
|
2017-02-13 02:55:55 +00:00
|
|
|
|
def _buildItem(self, elem, cls=None, initpath=None):
|
|
|
|
|
""" Factory function to build objects based on registered PLEXOBJECTS. """
|
|
|
|
|
# cls is specified, build the object and return
|
2017-02-07 06:58:29 +00:00
|
|
|
|
initpath = initpath or self._initpath
|
2017-02-13 02:55:55 +00:00
|
|
|
|
if cls is not None:
|
2020-12-06 06:39:13 +00:00
|
|
|
|
return cls(self._server, elem, initpath, parent=self)
|
2017-02-13 02:55:55 +00:00
|
|
|
|
# cls is not specified, try looking it up in PLEXOBJECTS
|
2021-01-24 20:48:38 +00:00
|
|
|
|
etype = elem.attrib.get('streamType', elem.attrib.get('tagType', elem.attrib.get('type')))
|
2017-02-13 02:55:55 +00:00
|
|
|
|
ehash = '%s.%s' % (elem.tag, etype) if etype else elem.tag
|
2022-07-21 03:03:20 +00:00
|
|
|
|
if initpath == '/status/sessions':
|
|
|
|
|
ehash = '%s.%s' % (ehash, 'session')
|
2017-02-13 02:55:55 +00:00
|
|
|
|
ecls = utils.PLEXOBJECTS.get(ehash, utils.PLEXOBJECTS.get(elem.tag))
|
2017-02-20 05:37:00 +00:00
|
|
|
|
# log.debug('Building %s as %s', elem.tag, ecls.__name__)
|
2017-02-13 02:55:55 +00:00
|
|
|
|
if ecls is not None:
|
|
|
|
|
return ecls(self._server, elem, initpath)
|
|
|
|
|
raise UnknownType("Unknown library type <%s type='%s'../>" % (elem.tag, etype))
|
|
|
|
|
|
|
|
|
|
def _buildItemOrNone(self, elem, cls=None, initpath=None):
|
2020-11-23 03:06:30 +00:00
|
|
|
|
""" Calls :func:`~plexapi.base.PlexObject._buildItem` but returns
|
2017-02-07 06:20:49 +00:00
|
|
|
|
None if elem is an unknown type.
|
|
|
|
|
"""
|
|
|
|
|
try:
|
2017-02-13 02:55:55 +00:00
|
|
|
|
return self._buildItem(elem, cls, initpath)
|
2017-02-07 06:20:49 +00:00
|
|
|
|
except UnknownType:
|
|
|
|
|
return None
|
|
|
|
|
|
2020-11-21 20:08:27 +00:00
|
|
|
|
def _buildDetailsKey(self, **kwargs):
|
2020-11-21 03:51:02 +00:00
|
|
|
|
""" Builds the details key with the XML include parameters.
|
|
|
|
|
All parameters are included by default with the option to override each parameter
|
|
|
|
|
or disable each parameter individually by setting it to False or 0.
|
|
|
|
|
"""
|
2020-11-21 20:08:27 +00:00
|
|
|
|
details_key = self.key
|
2020-12-07 02:55:45 +00:00
|
|
|
|
if details_key and hasattr(self, '_INCLUDES'):
|
2020-11-21 03:51:02 +00:00
|
|
|
|
includes = {}
|
2020-11-22 03:52:01 +00:00
|
|
|
|
for k, v in self._INCLUDES.items():
|
2020-11-21 03:51:02 +00:00
|
|
|
|
value = kwargs.get(k, v)
|
|
|
|
|
if value not in [False, 0, '0']:
|
|
|
|
|
includes[k] = 1 if value is True else value
|
2020-11-21 20:08:27 +00:00
|
|
|
|
if includes:
|
2020-11-22 04:02:31 +00:00
|
|
|
|
details_key += '?' + urlencode(sorted(includes.items()))
|
2020-11-21 20:08:27 +00:00
|
|
|
|
return details_key
|
2020-11-21 03:51:02 +00:00
|
|
|
|
|
2021-01-24 20:21:56 +00:00
|
|
|
|
def _isChildOf(self, **kwargs):
|
|
|
|
|
""" Returns True if this object is a child of the given attributes.
|
|
|
|
|
This will search the parent objects all the way to the top.
|
2021-01-24 23:13:22 +00:00
|
|
|
|
|
2020-12-06 06:39:13 +00:00
|
|
|
|
Parameters:
|
2021-01-24 20:21:56 +00:00
|
|
|
|
**kwargs (dict): The attributes and values to search for in the parent objects.
|
|
|
|
|
See all possible `**kwargs*` in :func:`~plexapi.base.PlexObject.fetchItem`.
|
2020-12-06 06:39:13 +00:00
|
|
|
|
"""
|
|
|
|
|
obj = self
|
2021-02-07 04:31:07 +00:00
|
|
|
|
while obj and obj._parent is not None:
|
2020-12-06 06:39:13 +00:00
|
|
|
|
obj = obj._parent()
|
2021-02-07 04:31:07 +00:00
|
|
|
|
if obj and obj._checkAttrs(obj._data, **kwargs):
|
2021-01-24 20:21:56 +00:00
|
|
|
|
return True
|
|
|
|
|
return False
|
2020-12-06 06:39:13 +00:00
|
|
|
|
|
2021-05-15 18:25:26 +00:00
|
|
|
|
def _manuallyLoadXML(self, xml, cls=None):
|
|
|
|
|
""" Manually load an XML string as a :class:`~plexapi.base.PlexObject`.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
xml (str): The XML string to load.
|
|
|
|
|
cls (:class:`~plexapi.base.PlexObject`): If you know the class of the
|
|
|
|
|
items to be fetched, passing this in will help the parser ensure
|
|
|
|
|
it only returns those items. By default we convert the xml elements
|
|
|
|
|
with the best guess PlexObjects based on tag and type attrs.
|
|
|
|
|
"""
|
|
|
|
|
elem = ElementTree.fromstring(xml)
|
|
|
|
|
return self._buildItemOrNone(elem, cls)
|
|
|
|
|
|
2017-02-13 02:55:55 +00:00
|
|
|
|
def fetchItem(self, ekey, cls=None, **kwargs):
|
2017-02-07 06:20:49 +00:00
|
|
|
|
""" Load the specified key to find and build the first item with the
|
|
|
|
|
specified tag and attrs. If no tag or attrs are specified then
|
|
|
|
|
the first item in the result set is returned.
|
2017-02-09 06:54:38 +00:00
|
|
|
|
|
|
|
|
|
Parameters:
|
2017-08-13 05:50:40 +00:00
|
|
|
|
ekey (str or int): Path in Plex to fetch items from. If an int is passed
|
2017-02-09 06:54:38 +00:00
|
|
|
|
in, the key will be translated to /library/metadata/<key>. This allows
|
|
|
|
|
fetching an item only knowing its key-id.
|
|
|
|
|
cls (:class:`~plexapi.base.PlexObject`): If you know the class of the
|
2017-02-10 23:16:23 +00:00
|
|
|
|
items to be fetched, passing this in will help the parser ensure
|
2017-02-09 06:54:38 +00:00
|
|
|
|
it only returns those items. By default we convert the xml elements
|
2017-02-13 02:55:55 +00:00
|
|
|
|
with the best guess PlexObjects based on tag and type attrs.
|
|
|
|
|
etag (str): Only fetch items with the specified tag.
|
2021-03-12 17:18:36 +00:00
|
|
|
|
**kwargs (dict): Optionally add XML attribute to filter the items.
|
|
|
|
|
See :func:`~plexapi.base.PlexObject.fetchItems` for more details
|
|
|
|
|
on how this is used.
|
2017-02-07 06:20:49 +00:00
|
|
|
|
"""
|
2020-04-11 13:30:05 +00:00
|
|
|
|
if ekey is None:
|
|
|
|
|
raise BadRequest('ekey was not provided')
|
2017-02-13 02:55:55 +00:00
|
|
|
|
if isinstance(ekey, int):
|
|
|
|
|
ekey = '/library/metadata/%s' % ekey
|
2022-07-21 03:03:20 +00:00
|
|
|
|
|
2022-02-27 04:17:23 +00:00
|
|
|
|
data = self._server.query(ekey)
|
2022-07-21 03:03:20 +00:00
|
|
|
|
item = self.findItem(data, cls, ekey, **kwargs)
|
|
|
|
|
|
|
|
|
|
if item:
|
|
|
|
|
librarySectionID = utils.cast(int, data.attrib.get('librarySectionID'))
|
|
|
|
|
if librarySectionID:
|
|
|
|
|
item.librarySectionID = librarySectionID
|
|
|
|
|
return item
|
|
|
|
|
|
2017-02-13 02:55:55 +00:00
|
|
|
|
clsname = cls.__name__ if cls else 'None'
|
|
|
|
|
raise NotFound('Unable to find elem: cls=%s, attrs=%s' % (clsname, kwargs))
|
|
|
|
|
|
2020-04-27 16:22:10 +00:00
|
|
|
|
def fetchItems(self, ekey, cls=None, container_start=None, container_size=None, **kwargs):
|
2017-02-09 06:54:38 +00:00
|
|
|
|
""" Load the specified key to find and build all items with the specified tag
|
2021-03-12 17:18:36 +00:00
|
|
|
|
and attrs.
|
2020-04-26 19:18:52 +00:00
|
|
|
|
|
2020-04-27 16:22:10 +00:00
|
|
|
|
Parameters:
|
2021-03-12 17:18:36 +00:00
|
|
|
|
ekey (str): API URL path in Plex to fetch items from.
|
|
|
|
|
cls (:class:`~plexapi.base.PlexObject`): If you know the class of the
|
|
|
|
|
items to be fetched, passing this in will help the parser ensure
|
|
|
|
|
it only returns those items. By default we convert the xml elements
|
|
|
|
|
with the best guess PlexObjects based on tag and type attrs.
|
|
|
|
|
etag (str): Only fetch items with the specified tag.
|
2020-04-27 16:22:10 +00:00
|
|
|
|
container_start (None, int): offset to get a subset of the data
|
|
|
|
|
container_size (None, int): How many items in data
|
2021-03-12 17:18:36 +00:00
|
|
|
|
**kwargs (dict): Optionally add XML attribute to filter the items.
|
|
|
|
|
See the details below for more info.
|
|
|
|
|
|
|
|
|
|
**Filtering XML Attributes**
|
|
|
|
|
|
|
|
|
|
Any XML attribute can be filtered when fetching results. Filtering is done before
|
|
|
|
|
the Python objects are built to help keep things speedy. For example, passing in
|
|
|
|
|
``viewCount=0`` will only return matching items where the view count is ``0``.
|
2022-02-27 03:26:08 +00:00
|
|
|
|
Note that case matters when specifying attributes. Attributes further down in the XML
|
2021-03-12 17:18:36 +00:00
|
|
|
|
tree can be filtered by *prepending* the attribute with each element tag ``Tag__``.
|
|
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
fetchItem(ekey, viewCount=0)
|
|
|
|
|
fetchItem(ekey, contentRating="PG")
|
|
|
|
|
fetchItem(ekey, Genre__tag="Animation")
|
|
|
|
|
fetchItem(ekey, Media__videoCodec="h265")
|
|
|
|
|
fetchItem(ekey, Media__Part__container="mp4)
|
|
|
|
|
|
|
|
|
|
Note that because some attribute names are already used as arguments to this
|
|
|
|
|
function, such as ``tag``, you may still reference the attr tag by prepending an
|
|
|
|
|
underscore. For example, passing in ``_tag='foobar'`` will return all items where
|
|
|
|
|
``tag='foobar'``.
|
|
|
|
|
|
|
|
|
|
**Using PlexAPI Operators**
|
|
|
|
|
|
|
|
|
|
Optionally, PlexAPI operators can be specified by *appending* it to the end of the
|
|
|
|
|
attribute for more complex lookups. For example, passing in ``viewCount__gte=0``
|
|
|
|
|
will return all items where ``viewCount >= 0``.
|
|
|
|
|
|
|
|
|
|
List of Available Operators:
|
|
|
|
|
|
|
|
|
|
* ``__contains``: Value contains specified arg.
|
|
|
|
|
* ``__endswith``: Value ends with specified arg.
|
|
|
|
|
* ``__exact``: Value matches specified arg.
|
|
|
|
|
* ``__exists`` (*bool*): Value is or is not present in the attrs.
|
|
|
|
|
* ``__gt``: Value is greater than specified arg.
|
|
|
|
|
* ``__gte``: Value is greater than or equal to specified arg.
|
2022-02-27 03:26:08 +00:00
|
|
|
|
* ``__icontains``: Case insensitive value contains specified arg.
|
|
|
|
|
* ``__iendswith``: Case insensitive value ends with specified arg.
|
|
|
|
|
* ``__iexact``: Case insensitive value matches specified arg.
|
2021-03-12 17:18:36 +00:00
|
|
|
|
* ``__in``: Value is in a specified list or tuple.
|
2022-02-27 03:26:08 +00:00
|
|
|
|
* ``__iregex``: Case insensitive value matches the specified regular expression.
|
|
|
|
|
* ``__istartswith``: Case insensitive value starts with specified arg.
|
2021-03-12 17:18:36 +00:00
|
|
|
|
* ``__lt``: Value is less than specified arg.
|
|
|
|
|
* ``__lte``: Value is less than or equal to specified arg.
|
|
|
|
|
* ``__regex``: Value matches the specified regular expression.
|
|
|
|
|
* ``__startswith``: Value starts with specified arg.
|
|
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
fetchItem(ekey, viewCount__gte=0)
|
|
|
|
|
fetchItem(ekey, Media__container__in=["mp4", "mkv"])
|
|
|
|
|
fetchItem(ekey, guid__iregex=r"(imdb:\/\/|themoviedb:\/\/)")
|
|
|
|
|
fetchItem(ekey, Media__Part__file__startswith="D:\\Movies")
|
2020-04-27 16:22:10 +00:00
|
|
|
|
|
2017-02-07 06:20:49 +00:00
|
|
|
|
"""
|
2022-07-21 03:03:20 +00:00
|
|
|
|
if ekey is None:
|
|
|
|
|
raise BadRequest('ekey was not provided')
|
|
|
|
|
|
|
|
|
|
params = {}
|
2020-04-27 16:22:10 +00:00
|
|
|
|
if container_start is not None:
|
2022-07-21 03:03:20 +00:00
|
|
|
|
params["X-Plex-Container-Start"] = container_start
|
2020-04-27 16:22:10 +00:00
|
|
|
|
if container_size is not None:
|
2022-07-21 03:03:20 +00:00
|
|
|
|
params["X-Plex-Container-Size"] = container_size
|
2020-04-26 19:18:52 +00:00
|
|
|
|
|
2022-07-21 03:03:20 +00:00
|
|
|
|
data = self._server.query(ekey, params=params)
|
2018-09-14 18:27:24 +00:00
|
|
|
|
items = self.findItems(data, cls, ekey, **kwargs)
|
2020-04-26 19:18:52 +00:00
|
|
|
|
|
2021-03-11 20:20:56 +00:00
|
|
|
|
librarySectionID = utils.cast(int, data.attrib.get('librarySectionID'))
|
2018-09-14 18:27:24 +00:00
|
|
|
|
if librarySectionID:
|
|
|
|
|
for item in items:
|
|
|
|
|
item.librarySectionID = librarySectionID
|
|
|
|
|
return items
|
2017-02-13 02:55:55 +00:00
|
|
|
|
|
2022-07-21 03:03:20 +00:00
|
|
|
|
def findItem(self, data, cls=None, initpath=None, rtag=None, **kwargs):
|
|
|
|
|
""" Load the specified data to find and build the first items with the specified tag
|
|
|
|
|
and attrs. See :func:`~plexapi.base.PlexObject.fetchItem` for more details
|
|
|
|
|
on how this is used.
|
|
|
|
|
"""
|
|
|
|
|
# filter on cls attrs if specified
|
|
|
|
|
if cls and cls.TAG and 'tag' not in kwargs:
|
|
|
|
|
kwargs['etag'] = cls.TAG
|
|
|
|
|
if cls and cls.TYPE and 'type' not in kwargs:
|
|
|
|
|
kwargs['type'] = cls.TYPE
|
|
|
|
|
# rtag to iter on a specific root tag
|
|
|
|
|
if rtag:
|
|
|
|
|
data = next(data.iter(rtag), [])
|
|
|
|
|
# loop through all data elements to find matches
|
|
|
|
|
for elem in data:
|
|
|
|
|
if self._checkAttrs(elem, **kwargs):
|
|
|
|
|
item = self._buildItemOrNone(elem, cls, initpath)
|
|
|
|
|
return item
|
|
|
|
|
|
2021-05-25 00:28:11 +00:00
|
|
|
|
def findItems(self, data, cls=None, initpath=None, rtag=None, **kwargs):
|
2017-02-13 02:55:55 +00:00
|
|
|
|
""" Load the specified data to find and build all items with the specified tag
|
|
|
|
|
and attrs. See :func:`~plexapi.base.PlexObject.fetchItem` for more details
|
|
|
|
|
on how this is used.
|
|
|
|
|
"""
|
|
|
|
|
# filter on cls attrs if specified
|
|
|
|
|
if cls and cls.TAG and 'tag' not in kwargs:
|
|
|
|
|
kwargs['etag'] = cls.TAG
|
|
|
|
|
if cls and cls.TYPE and 'type' not in kwargs:
|
|
|
|
|
kwargs['type'] = cls.TYPE
|
2022-05-17 02:46:10 +00:00
|
|
|
|
# rtag to iter on a specific root tag using breadth-first search
|
2021-05-25 00:28:11 +00:00
|
|
|
|
if rtag:
|
2022-05-17 02:46:10 +00:00
|
|
|
|
data = next(utils.iterXMLBFS(data, rtag), [])
|
2017-02-13 02:55:55 +00:00
|
|
|
|
# loop through all data elements to find matches
|
2017-02-07 06:20:49 +00:00
|
|
|
|
items = []
|
2017-02-13 02:55:55 +00:00
|
|
|
|
for elem in data:
|
|
|
|
|
if self._checkAttrs(elem, **kwargs):
|
|
|
|
|
item = self._buildItemOrNone(elem, cls, initpath)
|
|
|
|
|
if item is not None:
|
|
|
|
|
items.append(item)
|
|
|
|
|
return items
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
2017-02-15 05:13:22 +00:00
|
|
|
|
def firstAttr(self, *attrs):
|
|
|
|
|
""" Return the first attribute in attrs that is not None. """
|
|
|
|
|
for attr in attrs:
|
2021-02-15 03:58:03 +00:00
|
|
|
|
value = getattr(self, attr, None)
|
2017-02-15 05:13:22 +00:00
|
|
|
|
if value is not None:
|
|
|
|
|
return value
|
|
|
|
|
|
2021-06-14 04:57:07 +00:00
|
|
|
|
def listAttrs(self, data, attr, rtag=None, **kwargs):
|
2020-12-24 06:29:26 +00:00
|
|
|
|
""" Return a list of values from matching attribute. """
|
2017-02-13 03:15:47 +00:00
|
|
|
|
results = []
|
2022-05-17 02:46:10 +00:00
|
|
|
|
# rtag to iter on a specific root tag using breadth-first search
|
2021-06-14 04:57:07 +00:00
|
|
|
|
if rtag:
|
2022-05-17 02:46:10 +00:00
|
|
|
|
data = next(utils.iterXMLBFS(data, rtag), [])
|
2017-02-13 03:15:47 +00:00
|
|
|
|
for elem in data:
|
|
|
|
|
kwargs['%s__exists' % attr] = True
|
|
|
|
|
if self._checkAttrs(elem, **kwargs):
|
|
|
|
|
results.append(elem.attrib.get(attr))
|
|
|
|
|
return results
|
|
|
|
|
|
2021-06-02 15:59:52 +00:00
|
|
|
|
def reload(self, key=None, **kwargs):
|
2020-11-21 03:51:02 +00:00
|
|
|
|
""" Reload the data for this object from self.key.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
2020-11-22 03:52:33 +00:00
|
|
|
|
key (string, optional): Override the key to reload.
|
|
|
|
|
**kwargs (dict): A dictionary of XML include parameters to exclude or override.
|
|
|
|
|
All parameters are included by default with the option to override each parameter
|
|
|
|
|
or disable each parameter individually by setting it to False or 0.
|
|
|
|
|
See :class:`~plexapi.base.PlexPartialObject` for all the available include parameters.
|
|
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
from plexapi.server import PlexServer
|
|
|
|
|
plex = PlexServer('http://localhost:32400', token='xxxxxxxxxxxxxxxxxxxx')
|
|
|
|
|
movie = plex.library.section('Movies').get('Cars')
|
|
|
|
|
|
|
|
|
|
# Partial reload of the movie without the `checkFiles` parameter.
|
|
|
|
|
# Excluding `checkFiles` will prevent the Plex server from reading the
|
|
|
|
|
# file to check if the file still exists and is accessible.
|
|
|
|
|
# The movie object will remain as a partial object.
|
|
|
|
|
movie.reload(checkFiles=False)
|
|
|
|
|
movie.isPartialObject() # Returns True
|
|
|
|
|
|
|
|
|
|
# Full reload of the movie with all include parameters.
|
|
|
|
|
# The movie object will be a full object.
|
|
|
|
|
movie.reload()
|
|
|
|
|
movie.isFullObject() # Returns True
|
|
|
|
|
|
2020-11-21 03:51:02 +00:00
|
|
|
|
"""
|
2021-06-02 15:59:52 +00:00
|
|
|
|
return self._reload(key=key, **kwargs)
|
|
|
|
|
|
2022-05-17 03:03:06 +00:00
|
|
|
|
def _reload(self, key=None, _overwriteNone=True, **kwargs):
|
2021-06-02 15:59:52 +00:00
|
|
|
|
""" Perform the actual reload. """
|
2020-11-21 20:08:27 +00:00
|
|
|
|
details_key = self._buildDetailsKey(**kwargs) if kwargs else self._details_key
|
|
|
|
|
key = key or details_key or self.key
|
2017-02-27 22:16:02 +00:00
|
|
|
|
if not key:
|
2017-02-07 06:20:49 +00:00
|
|
|
|
raise Unsupported('Cannot reload an object not built from a URL.')
|
2017-02-27 22:16:02 +00:00
|
|
|
|
self._initpath = key
|
|
|
|
|
data = self._server.query(key)
|
2022-05-17 03:03:06 +00:00
|
|
|
|
self._overwriteNone = _overwriteNone
|
2017-02-07 06:20:49 +00:00
|
|
|
|
self._loadData(data[0])
|
2022-05-17 03:03:06 +00:00
|
|
|
|
self._overwriteNone = True
|
2017-02-08 07:00:43 +00:00
|
|
|
|
return self
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
2017-02-09 06:54:38 +00:00
|
|
|
|
def _checkAttrs(self, elem, **kwargs):
|
2017-02-09 21:29:23 +00:00
|
|
|
|
attrsFound = {}
|
|
|
|
|
for attr, query in kwargs.items():
|
|
|
|
|
attr, op, operator = self._getAttrOperator(attr)
|
|
|
|
|
values = self._getAttrValue(elem, attr)
|
2017-02-13 02:55:55 +00:00
|
|
|
|
# special case query in (None, 0, '') to include missing attr
|
|
|
|
|
if op == 'exact' and not values and query in (None, 0, ''):
|
2017-02-09 06:54:38 +00:00
|
|
|
|
return True
|
|
|
|
|
# return if attr were looking for is missing
|
2017-02-09 21:29:23 +00:00
|
|
|
|
attrsFound[attr] = False
|
|
|
|
|
for value in values:
|
2017-02-13 02:55:55 +00:00
|
|
|
|
value = self._castAttrValue(op, query, value)
|
2017-02-09 21:29:23 +00:00
|
|
|
|
if operator(value, query):
|
|
|
|
|
attrsFound[attr] = True
|
|
|
|
|
break
|
2017-02-20 05:37:00 +00:00
|
|
|
|
# log.debug('Checking %s for %s found: %s', elem.tag, kwargs, attrsFound)
|
2017-02-09 21:29:23 +00:00
|
|
|
|
return all(attrsFound.values())
|
|
|
|
|
|
|
|
|
|
def _getAttrOperator(self, attr):
|
|
|
|
|
for op, operator in OPERATORS.items():
|
|
|
|
|
if attr.endswith('__%s' % op):
|
|
|
|
|
attr = attr.rsplit('__', 1)[0]
|
|
|
|
|
return attr, op, operator
|
|
|
|
|
# default to exact match
|
|
|
|
|
return attr, 'exact', OPERATORS['exact']
|
|
|
|
|
|
|
|
|
|
def _getAttrValue(self, elem, attrstr, results=None):
|
2017-02-20 05:37:00 +00:00
|
|
|
|
# log.debug('Fetching %s in %s', attrstr, elem.tag)
|
2017-02-09 21:29:23 +00:00
|
|
|
|
parts = attrstr.split('__', 1)
|
|
|
|
|
attr = parts[0]
|
|
|
|
|
attrstr = parts[1] if len(parts) == 2 else None
|
|
|
|
|
if attrstr:
|
|
|
|
|
results = [] if results is None else results
|
|
|
|
|
for child in [c for c in elem if c.tag.lower() == attr.lower()]:
|
|
|
|
|
results += self._getAttrValue(child, attrstr, results)
|
|
|
|
|
return [r for r in results if r is not None]
|
2017-02-13 02:55:55 +00:00
|
|
|
|
# check were looking for the tag
|
|
|
|
|
if attr.lower() == 'etag':
|
|
|
|
|
return [elem.tag]
|
2022-02-27 03:26:08 +00:00
|
|
|
|
# loop through attrs so we can perform case-insensitive match
|
2017-02-09 21:29:23 +00:00
|
|
|
|
for _attr, value in elem.attrib.items():
|
|
|
|
|
if attr.lower() == _attr.lower():
|
|
|
|
|
return [value]
|
|
|
|
|
return []
|
2017-02-09 06:54:38 +00:00
|
|
|
|
|
2017-02-13 02:55:55 +00:00
|
|
|
|
def _castAttrValue(self, op, query, value):
|
|
|
|
|
if op == 'exists':
|
|
|
|
|
return value
|
|
|
|
|
if isinstance(query, bool):
|
|
|
|
|
return bool(int(value))
|
|
|
|
|
if isinstance(query, int) and '.' in value:
|
|
|
|
|
return float(value)
|
|
|
|
|
if isinstance(query, int):
|
|
|
|
|
return int(value)
|
|
|
|
|
if isinstance(query, float):
|
|
|
|
|
return float(value)
|
|
|
|
|
return value
|
|
|
|
|
|
2017-02-09 06:54:38 +00:00
|
|
|
|
def _loadData(self, data):
|
2017-02-09 06:59:14 +00:00
|
|
|
|
raise NotImplementedError('Abstract method not implemented.')
|
2017-02-09 06:54:38 +00:00
|
|
|
|
|
2022-02-27 05:40:51 +00:00
|
|
|
|
@property
|
|
|
|
|
def _searchType(self):
|
|
|
|
|
return self.TYPE
|
|
|
|
|
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
|
|
|
|
class PlexPartialObject(PlexObject):
|
|
|
|
|
""" Not all objects in the Plex listings return the complete list of elements
|
|
|
|
|
for the object. This object will allow you to assume each object is complete,
|
|
|
|
|
and if the specified value you request is None it will fetch the full object
|
|
|
|
|
automatically and update itself.
|
|
|
|
|
"""
|
2020-11-22 03:52:01 +00:00
|
|
|
|
_INCLUDES = {
|
|
|
|
|
'checkFiles': 1,
|
|
|
|
|
'includeAllConcerts': 1,
|
|
|
|
|
'includeBandwidths': 1,
|
|
|
|
|
'includeChapters': 1,
|
|
|
|
|
'includeChildren': 1,
|
|
|
|
|
'includeConcerts': 1,
|
|
|
|
|
'includeExternalMedia': 1,
|
|
|
|
|
'includeExtras': 1,
|
2020-11-22 04:02:31 +00:00
|
|
|
|
'includeFields': 'thumbBlurHash,artBlurHash',
|
2020-11-22 03:52:01 +00:00
|
|
|
|
'includeGeolocation': 1,
|
|
|
|
|
'includeLoudnessRamps': 1,
|
|
|
|
|
'includeMarkers': 1,
|
|
|
|
|
'includeOnDeck': 1,
|
|
|
|
|
'includePopularLeaves': 1,
|
|
|
|
|
'includePreferences': 1,
|
|
|
|
|
'includeRelated': 1,
|
|
|
|
|
'includeRelatedCount': 1,
|
|
|
|
|
'includeReviews': 1,
|
|
|
|
|
'includeStations': 1
|
|
|
|
|
}
|
2017-07-16 20:46:03 +00:00
|
|
|
|
|
2017-02-07 06:20:49 +00:00
|
|
|
|
def __eq__(self, other):
|
2020-10-31 05:13:19 +00:00
|
|
|
|
return other not in [None, []] and self.key == other.key
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
2017-10-05 21:21:14 +00:00
|
|
|
|
def __hash__(self):
|
|
|
|
|
return hash(repr(self))
|
|
|
|
|
|
2017-10-09 14:07:09 +00:00
|
|
|
|
def __iter__(self):
|
|
|
|
|
yield self
|
|
|
|
|
|
2017-02-07 06:20:49 +00:00
|
|
|
|
def __getattribute__(self, attr):
|
2017-02-08 05:36:22 +00:00
|
|
|
|
# Dragons inside.. :-/
|
2017-10-07 20:10:23 +00:00
|
|
|
|
value = super(PlexPartialObject, self).__getattribute__(attr)
|
2022-02-27 03:26:08 +00:00
|
|
|
|
# Check a few cases where we don't want to reload
|
2021-05-30 23:25:25 +00:00
|
|
|
|
if attr in _DONT_RELOAD_FOR_KEYS: return value
|
2021-06-07 15:55:04 +00:00
|
|
|
|
if attr in USER_DONT_RELOAD_FOR_KEYS: return value
|
2019-11-10 03:35:33 +00:00
|
|
|
|
if attr.startswith('_'): return value
|
2017-02-07 06:20:49 +00:00
|
|
|
|
if value not in (None, []): return value
|
|
|
|
|
if self.isFullObject(): return value
|
2022-07-21 03:03:20 +00:00
|
|
|
|
if isinstance(self, PlexSession): return value
|
2022-05-17 03:03:06 +00:00
|
|
|
|
if self._autoReload is False: return value
|
2018-03-02 16:08:10 +00:00
|
|
|
|
# Log the reload.
|
2017-02-07 06:20:49 +00:00
|
|
|
|
clsname = self.__class__.__name__
|
|
|
|
|
title = self.__dict__.get('title', self.__dict__.get('name'))
|
|
|
|
|
objname = "%s '%s'" % (clsname, title) if title else clsname
|
2021-02-24 17:55:53 +00:00
|
|
|
|
log.debug("Reloading %s for attr '%s'", objname, attr)
|
2017-02-07 06:20:49 +00:00
|
|
|
|
# Reload and return the value
|
2022-05-19 19:48:01 +00:00
|
|
|
|
self._reload(_overwriteNone=False)
|
2017-10-07 20:10:23 +00:00
|
|
|
|
return super(PlexPartialObject, self).__getattribute__(attr)
|
|
|
|
|
|
2017-02-09 20:01:23 +00:00
|
|
|
|
def analyze(self):
|
|
|
|
|
""" Tell Plex Media Server to performs analysis on it this item to gather
|
|
|
|
|
information. Analysis includes:
|
|
|
|
|
|
|
|
|
|
* Gather Media Properties: All of the media you add to a Library has
|
|
|
|
|
properties that are useful to know–whether it's a video file, a
|
|
|
|
|
music track, or one of your photos (container, codec, resolution, etc).
|
|
|
|
|
* Generate Default Artwork: Artwork will automatically be grabbed from a
|
|
|
|
|
video file. A background image will be pulled out as well as a
|
|
|
|
|
smaller image to be used for poster/thumbnail type purposes.
|
|
|
|
|
* Generate Video Preview Thumbnails: Video preview thumbnails are created,
|
|
|
|
|
if you have that feature enabled. Video preview thumbnails allow
|
|
|
|
|
graphical seeking in some Apps. It's also used in the Plex Web App Now
|
|
|
|
|
Playing screen to show a graphical representation of where playback
|
|
|
|
|
is. Video preview thumbnails creation is a CPU-intensive process akin
|
|
|
|
|
to transcoding the file.
|
2020-05-25 02:55:52 +00:00
|
|
|
|
* Generate intro video markers: Detects show intros, exposing the
|
|
|
|
|
'Skip Intro' button in clients.
|
2017-02-09 20:01:23 +00:00
|
|
|
|
"""
|
|
|
|
|
key = '/%s/analyze' % self.key.lstrip('/')
|
|
|
|
|
self._server.query(key, method=self._server._session.put)
|
|
|
|
|
|
2017-02-07 06:20:49 +00:00
|
|
|
|
def isFullObject(self):
|
2021-06-06 21:54:15 +00:00
|
|
|
|
""" Returns True if this is already a full object. A full object means all attributes
|
2017-02-07 06:20:49 +00:00
|
|
|
|
were populated from the api path representing only this item. For example, the
|
|
|
|
|
search result for a movie often only contain a portion of the attributes a full
|
2020-11-21 04:34:41 +00:00
|
|
|
|
object (main url) for that movie would contain.
|
2017-02-07 06:20:49 +00:00
|
|
|
|
"""
|
2020-11-21 04:34:41 +00:00
|
|
|
|
return not self.key or (self._details_key or self.key) == self._initpath
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
|
|
|
|
def isPartialObject(self):
|
|
|
|
|
""" Returns True if this is not a full object. """
|
|
|
|
|
return not self.isFullObject()
|
|
|
|
|
|
2021-09-13 00:56:21 +00:00
|
|
|
|
def _edit(self, **kwargs):
|
|
|
|
|
""" Actually edit an object. """
|
2022-02-27 05:40:51 +00:00
|
|
|
|
if isinstance(self._edits, dict):
|
|
|
|
|
self._edits.update(kwargs)
|
|
|
|
|
return self
|
|
|
|
|
|
2021-09-13 00:56:21 +00:00
|
|
|
|
if 'id' not in kwargs:
|
|
|
|
|
kwargs['id'] = self.ratingKey
|
|
|
|
|
if 'type' not in kwargs:
|
2022-02-27 05:40:51 +00:00
|
|
|
|
kwargs['type'] = utils.searchType(self._searchType)
|
2021-09-13 00:56:21 +00:00
|
|
|
|
|
2022-02-27 05:40:51 +00:00
|
|
|
|
part = '/library/sections/%s/all%s' % (self.librarySectionID,
|
|
|
|
|
utils.joinArgs(kwargs))
|
2021-09-13 00:56:21 +00:00
|
|
|
|
self._server.query(part, method=self._server._session.put)
|
2022-02-27 05:40:51 +00:00
|
|
|
|
return self
|
2021-09-13 00:56:21 +00:00
|
|
|
|
|
2017-07-16 20:46:03 +00:00
|
|
|
|
def edit(self, **kwargs):
|
2017-07-30 04:31:45 +00:00
|
|
|
|
""" Edit an object.
|
2022-02-27 05:40:51 +00:00
|
|
|
|
Note: This is a low level method and you need to know all the field/tag keys.
|
|
|
|
|
See :class:`~plexapi.mixins.EditFieldMixin` and :class:`~plexapi.mixins.EditTagsMixin`
|
|
|
|
|
for individual field and tag editing methods.
|
2017-07-16 20:46:03 +00:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
kwargs (dict): Dict of settings to edit.
|
|
|
|
|
|
|
|
|
|
Example:
|
2022-02-27 05:40:51 +00:00
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
edits = {
|
|
|
|
|
'type': 1,
|
|
|
|
|
'id': movie.ratingKey,
|
|
|
|
|
'title.value': 'A new title',
|
|
|
|
|
'title.locked': 1,
|
|
|
|
|
'summary.value': 'This is a summary.',
|
|
|
|
|
'summary.locked': 1,
|
|
|
|
|
'collection[0].tag.tag': 'A tag',
|
|
|
|
|
'collection.locked': 1}
|
|
|
|
|
}
|
|
|
|
|
movie.edit(**edits)
|
|
|
|
|
|
2017-07-16 20:46:03 +00:00
|
|
|
|
"""
|
2022-02-27 05:40:51 +00:00
|
|
|
|
return self._edit(**kwargs)
|
|
|
|
|
|
|
|
|
|
def batchEdits(self):
|
|
|
|
|
""" Enable batch editing mode to save API calls.
|
|
|
|
|
Must call :func:`~plexapi.base.PlexPartialObject.saveEdits` at the end to save all the edits.
|
|
|
|
|
See :class:`~plexapi.mixins.EditFieldMixin` and :class:`~plexapi.mixins.EditTagsMixin`
|
|
|
|
|
for individual field and tag editing methods.
|
|
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
# Batch editing multiple fields and tags in a single API call
|
|
|
|
|
Movie.batchEdits()
|
|
|
|
|
Movie.editTitle('A New Title').editSummary('A new summary').editTagline('A new tagline') \\
|
|
|
|
|
.addCollection('New Collection').removeGenre('Action').addLabel('Favorite')
|
|
|
|
|
Movie.saveEdits()
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
self._edits = {}
|
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
def saveEdits(self):
|
|
|
|
|
""" Save all the batch edits and automatically reload the object.
|
|
|
|
|
See :func:`~plexapi.base.PlexPartialObject.batchEdits` for details.
|
|
|
|
|
"""
|
|
|
|
|
if not isinstance(self._edits, dict):
|
|
|
|
|
raise BadRequest('Batch editing mode not enabled. Must call `batchEdits()` first.')
|
|
|
|
|
|
|
|
|
|
edits = self._edits
|
|
|
|
|
self._edits = None
|
|
|
|
|
self._edit(**edits)
|
|
|
|
|
return self.reload()
|
2017-07-16 20:46:03 +00:00
|
|
|
|
|
2017-02-09 20:01:23 +00:00
|
|
|
|
def refresh(self):
|
|
|
|
|
""" Refreshing a Library or individual item causes the metadata for the item to be
|
|
|
|
|
refreshed, even if it already has metadata. You can think of refreshing as
|
|
|
|
|
"update metadata for the requested item even if it already has some". You should
|
|
|
|
|
refresh a Library or individual item if:
|
|
|
|
|
|
|
|
|
|
* You've changed the Library Metadata Agent.
|
|
|
|
|
* You've added "Local Media Assets" (such as artwork, theme music, external
|
|
|
|
|
subtitle files, etc.)
|
|
|
|
|
* You want to freshen the item posters, summary, etc.
|
|
|
|
|
* There's a problem with the poster image that's been downloaded.
|
|
|
|
|
* Items are missing posters or other downloaded information. This is possible if
|
|
|
|
|
the refresh process is interrupted (the Server is turned off, internet
|
|
|
|
|
connection dies, etc).
|
|
|
|
|
"""
|
|
|
|
|
key = '%s/refresh' % self.key
|
|
|
|
|
self._server.query(key, method=self._server._session.put)
|
|
|
|
|
|
|
|
|
|
def section(self):
|
|
|
|
|
""" Returns the :class:`~plexapi.library.LibrarySection` this item belongs to. """
|
|
|
|
|
return self._server.library.sectionByID(self.librarySectionID)
|
|
|
|
|
|
2017-02-10 23:16:23 +00:00
|
|
|
|
def delete(self):
|
2017-07-30 04:31:45 +00:00
|
|
|
|
""" Delete a media element. This has to be enabled under settings > server > library in plex webui. """
|
2017-02-10 23:16:23 +00:00
|
|
|
|
try:
|
|
|
|
|
return self._server.query(self.key, method=self._server._session.delete)
|
|
|
|
|
except BadRequest: # pragma: no cover
|
2017-07-30 04:31:45 +00:00
|
|
|
|
log.error('Failed to delete %s. This could be because you '
|
2021-02-24 17:55:53 +00:00
|
|
|
|
'have not allowed items to be deleted', self.key)
|
2017-02-10 23:16:23 +00:00
|
|
|
|
raise
|
|
|
|
|
|
2019-11-20 11:50:25 +00:00
|
|
|
|
def history(self, maxresults=9999999, mindate=None):
|
|
|
|
|
""" Get Play History for a media item.
|
2021-08-03 03:37:17 +00:00
|
|
|
|
|
2019-11-20 11:50:25 +00:00
|
|
|
|
Parameters:
|
|
|
|
|
maxresults (int): Only return the specified number of results (optional).
|
|
|
|
|
mindate (datetime): Min datetime to return results from.
|
|
|
|
|
"""
|
|
|
|
|
return self._server.history(maxresults=maxresults, mindate=mindate, ratingKey=self.ratingKey)
|
2019-11-14 17:21:49 +00:00
|
|
|
|
|
2021-09-26 22:23:09 +00:00
|
|
|
|
def _getWebURL(self, base=None):
|
|
|
|
|
""" Get the Plex Web URL with the correct parameters.
|
|
|
|
|
Private method to allow overriding parameters from subclasses.
|
2021-08-03 03:37:17 +00:00
|
|
|
|
"""
|
2021-09-26 22:23:09 +00:00
|
|
|
|
return self._server._buildWebURL(base=base, endpoint='details', key=self.key)
|
2021-08-03 03:37:17 +00:00
|
|
|
|
|
|
|
|
|
def getWebURL(self, base=None):
|
2021-09-26 22:23:09 +00:00
|
|
|
|
""" Returns the Plex Web URL for a media item.
|
2021-08-03 03:37:17 +00:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
base (str): The base URL before the fragment (``#!``).
|
|
|
|
|
Default is https://app.plex.tv/desktop.
|
|
|
|
|
"""
|
2021-09-26 22:23:09 +00:00
|
|
|
|
return self._getWebURL(base=base)
|
2021-08-03 03:37:17 +00:00
|
|
|
|
|
2017-02-07 06:20:49 +00:00
|
|
|
|
|
2022-05-30 16:05:00 +00:00
|
|
|
|
class Playable:
|
2017-02-04 19:46:51 +00:00
|
|
|
|
""" This is a general place to store functions specific to media that is Playable.
|
|
|
|
|
Things were getting mixed up a bit when dealing with Shows, Season, Artists,
|
|
|
|
|
Albums which are all not playable.
|
|
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
|
viewedAt (datetime): Datetime item was last viewed (history).
|
2021-03-22 19:40:13 +00:00
|
|
|
|
accountID (int): The associated :class:`~plexapi.server.SystemAccount` ID.
|
|
|
|
|
deviceID (int): The associated :class:`~plexapi.server.SystemDevice` ID.
|
2017-02-14 04:32:27 +00:00
|
|
|
|
playlistItemID (int): Playlist item ID (only populated for :class:`~plexapi.playlist.Playlist` items).
|
2020-09-11 21:23:27 +00:00
|
|
|
|
playQueueItemID (int): PlayQueue item ID (only populated for :class:`~plexapi.playlist.PlayQueue` items).
|
2017-02-04 19:46:51 +00:00
|
|
|
|
"""
|
2017-07-16 20:46:03 +00:00
|
|
|
|
|
2017-02-04 19:46:51 +00:00
|
|
|
|
def _loadData(self, data):
|
2017-02-13 06:37:23 +00:00
|
|
|
|
self.viewedAt = utils.toDatetime(data.attrib.get('viewedAt')) # history
|
2019-06-03 04:50:02 +00:00
|
|
|
|
self.accountID = utils.cast(int, data.attrib.get('accountID')) # history
|
2021-03-22 19:40:13 +00:00
|
|
|
|
self.deviceID = utils.cast(int, data.attrib.get('deviceID')) # history
|
2017-02-13 06:37:23 +00:00
|
|
|
|
self.playlistItemID = utils.cast(int, data.attrib.get('playlistItemID')) # playlist
|
2020-09-11 21:23:27 +00:00
|
|
|
|
self.playQueueItemID = utils.cast(int, data.attrib.get('playQueueItemID')) # playqueue
|
2017-02-04 19:46:51 +00:00
|
|
|
|
|
|
|
|
|
def getStreamURL(self, **params):
|
|
|
|
|
""" Returns a stream url that may be used by external applications such as VLC.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
**params (dict): optional parameters to manipulate the playback when accessing
|
|
|
|
|
the stream. A few known parameters include: maxVideoBitrate, videoResolution
|
|
|
|
|
offset, copyts, protocol, mediaIndex, platform.
|
|
|
|
|
|
|
|
|
|
Raises:
|
2020-11-23 20:20:56 +00:00
|
|
|
|
:exc:`~plexapi.exceptions.Unsupported`: When the item doesn't support fetching a stream URL.
|
2017-02-04 19:46:51 +00:00
|
|
|
|
"""
|
2021-06-07 00:50:35 +00:00
|
|
|
|
if self.TYPE not in ('movie', 'episode', 'track', 'clip'):
|
2017-02-04 19:46:51 +00:00
|
|
|
|
raise Unsupported('Fetching stream URL for %s is unsupported.' % self.TYPE)
|
|
|
|
|
mvb = params.get('maxVideoBitrate')
|
|
|
|
|
vr = params.get('videoResolution', '')
|
|
|
|
|
params = {
|
|
|
|
|
'path': self.key,
|
|
|
|
|
'offset': params.get('offset', 0),
|
|
|
|
|
'copyts': params.get('copyts', 1),
|
|
|
|
|
'protocol': params.get('protocol'),
|
|
|
|
|
'mediaIndex': params.get('mediaIndex', 0),
|
|
|
|
|
'X-Plex-Platform': params.get('platform', 'Chrome'),
|
|
|
|
|
'maxVideoBitrate': max(mvb, 64) if mvb else None,
|
2020-12-13 20:36:43 +00:00
|
|
|
|
'videoResolution': vr if re.match(r'^\d+x\d+$', vr) else None
|
2017-02-04 19:46:51 +00:00
|
|
|
|
}
|
|
|
|
|
# remove None values
|
|
|
|
|
params = {k: v for k, v in params.items() if v is not None}
|
|
|
|
|
streamtype = 'audio' if self.TYPE in ('track', 'album') else 'video'
|
|
|
|
|
# sort the keys since the randomness fucks with my tests..
|
|
|
|
|
sorted_params = sorted(params.items(), key=lambda val: val[0])
|
2017-02-09 04:29:17 +00:00
|
|
|
|
return self._server.url('/%s/:/transcode/universal/start.m3u8?%s' %
|
2018-01-05 02:44:35 +00:00
|
|
|
|
(streamtype, urlencode(sorted_params)), includeToken=True)
|
2017-02-04 19:46:51 +00:00
|
|
|
|
|
|
|
|
|
def iterParts(self):
|
|
|
|
|
""" Iterates over the parts of this media item. """
|
|
|
|
|
for item in self.media:
|
|
|
|
|
for part in item.parts:
|
|
|
|
|
yield part
|
|
|
|
|
|
|
|
|
|
def play(self, client):
|
|
|
|
|
""" Start playback on the specified client.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
client (:class:`~plexapi.client.PlexClient`): Client to start playing on.
|
|
|
|
|
"""
|
|
|
|
|
client.playMedia(self)
|
|
|
|
|
|
2019-01-07 13:04:53 +00:00
|
|
|
|
def download(self, savepath=None, keep_original_name=False, **kwargs):
|
2021-11-20 22:16:58 +00:00
|
|
|
|
""" Downloads the media item to the specified location. Returns a list of
|
2017-02-04 19:46:51 +00:00
|
|
|
|
filepaths that have been saved to disk.
|
2017-02-10 23:16:23 +00:00
|
|
|
|
|
2017-02-04 19:46:51 +00:00
|
|
|
|
Parameters:
|
2021-11-20 22:16:58 +00:00
|
|
|
|
savepath (str): Defaults to current working dir.
|
|
|
|
|
keep_original_name (bool): True to keep the original filename otherwise
|
|
|
|
|
a friendlier filename is generated. See filenames below.
|
|
|
|
|
**kwargs (dict): Additional options passed into :func:`~plexapi.audio.Track.getStreamURL`
|
|
|
|
|
to download a transcoded stream, otherwise the media item will be downloaded
|
|
|
|
|
as-is and saved to disk.
|
|
|
|
|
|
|
|
|
|
**Filenames**
|
|
|
|
|
|
|
|
|
|
* Movie: ``<title> (<year>)``
|
|
|
|
|
* Episode: ``<show title> - s00e00 - <episode title>``
|
|
|
|
|
* Track: ``<artist title> - <album title> - 00 - <track title>``
|
|
|
|
|
* Photo: ``<photoalbum title> - <photo/clip title>`` or ``<photo/clip title>``
|
2017-02-04 19:46:51 +00:00
|
|
|
|
"""
|
|
|
|
|
filepaths = []
|
2021-11-20 22:16:58 +00:00
|
|
|
|
parts = [i for i in self.iterParts() if i]
|
|
|
|
|
|
|
|
|
|
for part in parts:
|
|
|
|
|
if not keep_original_name:
|
|
|
|
|
filename = utils.cleanFilename('%s.%s' % (self._prettyfilename(), part.container))
|
|
|
|
|
else:
|
|
|
|
|
filename = part.file
|
|
|
|
|
|
2017-02-04 19:46:51 +00:00
|
|
|
|
if kwargs:
|
2022-02-27 03:26:08 +00:00
|
|
|
|
# So this seems to be a a lot slower but allows transcode.
|
2017-02-04 19:46:51 +00:00
|
|
|
|
download_url = self.getStreamURL(**kwargs)
|
|
|
|
|
else:
|
2021-11-20 22:16:58 +00:00
|
|
|
|
download_url = self._server.url('%s?download=1' % part.key)
|
|
|
|
|
|
|
|
|
|
filepath = utils.download(
|
|
|
|
|
download_url,
|
|
|
|
|
self._server._token,
|
|
|
|
|
filename=filename,
|
|
|
|
|
savepath=savepath,
|
|
|
|
|
session=self._server._session
|
|
|
|
|
)
|
|
|
|
|
|
2017-02-04 19:46:51 +00:00
|
|
|
|
if filepath:
|
|
|
|
|
filepaths.append(filepath)
|
2021-11-20 22:16:58 +00:00
|
|
|
|
|
2017-02-04 19:46:51 +00:00
|
|
|
|
return filepaths
|
2017-02-26 22:31:09 +00:00
|
|
|
|
|
2018-02-22 11:29:39 +00:00
|
|
|
|
def updateProgress(self, time, state='stopped'):
|
|
|
|
|
""" Set the watched progress for this video.
|
|
|
|
|
|
|
|
|
|
Note that setting the time to 0 will not work.
|
|
|
|
|
Use `markWatched` or `markUnwatched` to achieve
|
|
|
|
|
that goal.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
time (int): milliseconds watched
|
|
|
|
|
state (string): state of the video, default 'stopped'
|
|
|
|
|
"""
|
|
|
|
|
key = '/:/progress?key=%s&identifier=com.plexapp.plugins.library&time=%d&state=%s' % (self.ratingKey,
|
|
|
|
|
time, state)
|
|
|
|
|
self._server.query(key)
|
2022-05-17 03:03:06 +00:00
|
|
|
|
self._reload(_overwriteNone=False)
|
2019-12-31 13:06:56 +00:00
|
|
|
|
|
2018-05-13 08:50:19 +00:00
|
|
|
|
def updateTimeline(self, time, state='stopped', duration=None):
|
2018-05-12 19:29:17 +00:00
|
|
|
|
""" Set the timeline progress for this video.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
time (int): milliseconds watched
|
|
|
|
|
state (string): state of the video, default 'stopped'
|
2018-05-13 09:40:15 +00:00
|
|
|
|
duration (int): duration of the item
|
2018-05-12 19:29:17 +00:00
|
|
|
|
"""
|
2018-05-18 08:58:06 +00:00
|
|
|
|
durationStr = '&duration='
|
2018-09-08 15:26:59 +00:00
|
|
|
|
if duration is not None:
|
2018-05-18 08:58:06 +00:00
|
|
|
|
durationStr = durationStr + str(duration)
|
|
|
|
|
else:
|
|
|
|
|
durationStr = durationStr + str(self.duration)
|
2018-09-08 15:26:59 +00:00
|
|
|
|
key = '/:/timeline?ratingKey=%s&key=%s&identifier=com.plexapp.plugins.library&time=%d&state=%s%s'
|
|
|
|
|
key %= (self.ratingKey, self.key, time, state, durationStr)
|
2018-05-12 19:29:17 +00:00
|
|
|
|
self._server.query(key)
|
2022-05-17 03:03:06 +00:00
|
|
|
|
self._reload(_overwriteNone=False)
|
2018-02-22 11:29:39 +00:00
|
|
|
|
|
2017-07-18 15:59:23 +00:00
|
|
|
|
|
2022-07-21 03:03:20 +00:00
|
|
|
|
class PlexSession(object):
|
|
|
|
|
""" This is a general place to store functions specific to media that is a Plex Session.
|
|
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
|
live (bool): True if this is a live tv session.
|
|
|
|
|
player (:class:`~plexapi.client.PlexClient`): PlexClient object for the session.
|
|
|
|
|
session (:class:`~plexapi.media.Session`): Session object for the session
|
|
|
|
|
if the session is using bandwidth (None otherwise).
|
|
|
|
|
sessionKey (int): The session key for the session.
|
|
|
|
|
transcodeSession (:class:`~plexapi.media.TranscodeSession`): TranscodeSession object
|
|
|
|
|
if item is being transcoded (None otherwise).
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
|
self.live = utils.cast(bool, data.attrib.get('live', '0'))
|
|
|
|
|
self.player = self.findItem(data, etag='Player')
|
|
|
|
|
self.session = self.findItem(data, etag='Session')
|
|
|
|
|
self.sessionKey = utils.cast(int, data.attrib.get('sessionKey'))
|
|
|
|
|
self.transcodeSession = self.findItem(data, etag='TranscodeSession')
|
|
|
|
|
|
|
|
|
|
user = data.find('User')
|
|
|
|
|
self._username = user.attrib.get('title')
|
|
|
|
|
self._userId = utils.cast(int, user.attrib.get('id'))
|
|
|
|
|
self._user = None # Cache for user object
|
|
|
|
|
|
|
|
|
|
# For backwards compatibility
|
|
|
|
|
self.players = [self.player] if self.player else []
|
|
|
|
|
self.sessions = [self.session] if self.session else []
|
|
|
|
|
self.transcodeSessions = [self.transcodeSession] if self.transcodeSession else []
|
|
|
|
|
self.usernames = [self._username] if self._username else []
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def user(self):
|
|
|
|
|
""" Returns the :class:`~plexapi.myplex.MyPlexAccount` object (for admin)
|
|
|
|
|
or :class:`~plexapi.myplex.MyPlexUser` object (for users) for this session.
|
|
|
|
|
"""
|
|
|
|
|
if self._user is None:
|
|
|
|
|
myPlexAccount = self._server.myPlexAccount()
|
|
|
|
|
if self._userId == 1:
|
|
|
|
|
self._user = myPlexAccount
|
|
|
|
|
else:
|
|
|
|
|
self._user = myPlexAccount.user(self._username)
|
|
|
|
|
return self._user
|
|
|
|
|
|
|
|
|
|
def reload(self):
|
|
|
|
|
""" Reload the data for the session.
|
|
|
|
|
Note: This will return the object as-is if the session is no longer active.
|
|
|
|
|
"""
|
|
|
|
|
return self._reload()
|
|
|
|
|
|
|
|
|
|
def _reload(self, _autoReload=False, **kwargs):
|
|
|
|
|
""" Perform the actual reload. """
|
|
|
|
|
# Do not auto reload sessions
|
|
|
|
|
if _autoReload:
|
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
key = self._initpath
|
|
|
|
|
data = self._server.query(key)
|
|
|
|
|
for elem in data:
|
|
|
|
|
if elem.attrib.get('sessionKey') == str(self.sessionKey):
|
|
|
|
|
self._loadData(elem)
|
|
|
|
|
break
|
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
def source(self):
|
|
|
|
|
""" Return the source media object for the session. """
|
|
|
|
|
return self.fetchItem(self._details_key)
|
|
|
|
|
|
|
|
|
|
def stop(self, reason=''):
|
|
|
|
|
""" Stop playback for the session.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
reason (str): Message displayed to the user for stopping playback.
|
|
|
|
|
"""
|
|
|
|
|
params = {
|
|
|
|
|
'sessionId': self.session.id,
|
|
|
|
|
'reason': reason,
|
|
|
|
|
}
|
|
|
|
|
key = '/status/sessions/terminate'
|
|
|
|
|
return self._server.query(key, params=params)
|
|
|
|
|
|
|
|
|
|
|
2020-07-15 20:09:05 +00:00
|
|
|
|
class MediaContainer(PlexObject):
|
|
|
|
|
""" Represents a single MediaContainer.
|
|
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
|
TAG (str): 'MediaContainer'
|
2021-05-28 13:30:24 +00:00
|
|
|
|
allowSync (int): Sync/Download is allowed/disallowed for feature.
|
|
|
|
|
augmentationKey (str): API URL (/library/metadata/augmentations/<augmentationKey>).
|
|
|
|
|
identifier (str): "com.plexapp.plugins.library"
|
|
|
|
|
librarySectionID (int): :class:`~plexapi.library.LibrarySection` ID.
|
|
|
|
|
librarySectionTitle (str): :class:`~plexapi.library.LibrarySection` title.
|
|
|
|
|
librarySectionUUID (str): :class:`~plexapi.library.LibrarySection` UUID.
|
|
|
|
|
mediaTagPrefix (str): "/system/bundle/media/flags/"
|
|
|
|
|
mediaTagVersion (int): Unknown
|
|
|
|
|
size (int): The number of items in the hub.
|
2020-07-15 20:09:05 +00:00
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
TAG = 'MediaContainer'
|
|
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
|
self._data = data
|
|
|
|
|
self.allowSync = utils.cast(int, data.attrib.get('allowSync'))
|
|
|
|
|
self.augmentationKey = data.attrib.get('augmentationKey')
|
|
|
|
|
self.identifier = data.attrib.get('identifier')
|
2021-05-28 13:30:14 +00:00
|
|
|
|
self.librarySectionID = utils.cast(int, data.attrib.get('librarySectionID'))
|
|
|
|
|
self.librarySectionTitle = data.attrib.get('librarySectionTitle')
|
|
|
|
|
self.librarySectionUUID = data.attrib.get('librarySectionUUID')
|
2020-07-15 20:09:05 +00:00
|
|
|
|
self.mediaTagPrefix = data.attrib.get('mediaTagPrefix')
|
|
|
|
|
self.mediaTagVersion = data.attrib.get('mediaTagVersion')
|
|
|
|
|
self.size = utils.cast(int, data.attrib.get('size'))
|