2014-12-29 03:21:58 +00:00
|
|
|
"""
|
|
|
|
PlexAPI Utils
|
|
|
|
"""
|
|
|
|
from datetime import datetime
|
2015-09-05 14:09:15 +00:00
|
|
|
try:
|
|
|
|
from urllib import quote # Python2
|
|
|
|
except ImportError:
|
|
|
|
from urllib.parse import quote # Python3
|
|
|
|
|
2016-03-14 04:19:48 +00:00
|
|
|
# This used to be a simple variable equal to '__NA__'. However, there has been need to
|
|
|
|
# compare NA against None in some use cases. This object allows the internals of PlexAPI
|
|
|
|
# to distinguish between unfetched values and fetched, but non-existent values.
|
|
|
|
# (NA == None results to True; NA is None results to False)
|
|
|
|
class __NA__(object):
|
|
|
|
def __bool__(self): return False # Python3; flake8: noqa
|
|
|
|
def __eq__(self, other): return isinstance(other, __NA__) or other in [None, '__NA__'] # flake8: noqa
|
|
|
|
def __nonzero__(self): return False # Python2; flake8: noqa
|
|
|
|
def __repr__(self): return '__NA__' # flake8: noqa
|
|
|
|
NA = __NA__()
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
class PlexPartialObject(object):
|
|
|
|
""" 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.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, server, data, initpath):
|
|
|
|
self.server = server
|
|
|
|
self.initpath = initpath
|
|
|
|
self._loadData(data)
|
|
|
|
|
|
|
|
def __getattr__(self, attr):
|
|
|
|
if self.isPartialObject():
|
|
|
|
self.reload()
|
2015-11-05 03:49:09 +00:00
|
|
|
return self.__dict__[attr]
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
def __setattr__(self, attr, value):
|
|
|
|
if value != NA:
|
|
|
|
super(PlexPartialObject, self).__setattr__(attr, value)
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
raise Exception('Abstract method not implemented.')
|
|
|
|
|
|
|
|
def isFullObject(self):
|
|
|
|
return self.initpath == self.key
|
|
|
|
|
|
|
|
def isPartialObject(self):
|
|
|
|
return self.initpath != self.key
|
|
|
|
|
|
|
|
def reload(self):
|
|
|
|
data = self.server.query(self.key)
|
|
|
|
self.initpath = self.key
|
|
|
|
self._loadData(data[0])
|
|
|
|
|
|
|
|
|
|
|
|
def cast(func, value):
|
|
|
|
if value not in [None, NA]:
|
|
|
|
if func == bool:
|
|
|
|
value = int(value)
|
|
|
|
value = func(value)
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
def joinArgs(args):
|
|
|
|
if not args: return ''
|
|
|
|
arglist = []
|
|
|
|
for key in sorted(args, key=lambda x:x.lower()):
|
|
|
|
value = str(args[key])
|
2015-09-05 14:09:15 +00:00
|
|
|
arglist.append('%s=%s' % (key, quote(value)))
|
2014-12-29 03:21:58 +00:00
|
|
|
return '?%s' % '&'.join(arglist)
|
|
|
|
|
|
|
|
|
|
|
|
def toDatetime(value, format=None):
|
|
|
|
if value and value != NA:
|
|
|
|
if format: value = datetime.strptime(value, format)
|
|
|
|
else: value = datetime.fromtimestamp(int(value))
|
|
|
|
return value
|