2014-12-29 03:21:58 +00:00
|
|
|
"""
|
|
|
|
PlexVideo
|
|
|
|
"""
|
2015-09-05 14:09:15 +00:00
|
|
|
import re
|
2015-10-28 18:11:23 +00:00
|
|
|
from requests import put
|
2015-02-17 20:35:17 +00:00
|
|
|
from plexapi.client import Client
|
2015-11-05 02:10:10 +00:00
|
|
|
from plexapi.media import Media, TranscodeSession, Country, Director, Genre, Producer, Actor, Writer
|
2015-02-17 20:35:17 +00:00
|
|
|
from plexapi.myplex import MyPlexUser
|
2015-06-04 14:55:30 +00:00
|
|
|
from plexapi.exceptions import NotFound, UnknownType, Unsupported
|
2014-12-29 03:21:58 +00:00
|
|
|
from plexapi.utils import PlexPartialObject, NA
|
|
|
|
from plexapi.utils import cast, toDatetime
|
|
|
|
|
2015-09-05 14:09:15 +00:00
|
|
|
try:
|
|
|
|
from urllib import urlencode # Python2
|
|
|
|
except ImportError:
|
|
|
|
from urllib.parse import urlencode # Python3
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
class Video(PlexPartialObject):
|
|
|
|
TYPE = None
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
self.type = data.attrib.get('type', NA)
|
|
|
|
self.key = data.attrib.get('key', NA)
|
2015-02-17 20:35:17 +00:00
|
|
|
self.librarySectionID = data.attrib.get('librarySectionID', NA)
|
2014-12-29 03:21:58 +00:00
|
|
|
self.ratingKey = data.attrib.get('ratingKey', NA)
|
|
|
|
self.title = data.attrib.get('title', NA)
|
2016-01-19 20:38:08 +00:00
|
|
|
self.originalTitle = data.attrib.get('originalTitle', NA)
|
2014-12-29 03:21:58 +00:00
|
|
|
self.summary = data.attrib.get('summary', NA)
|
|
|
|
self.art = data.attrib.get('art', NA)
|
|
|
|
self.thumb = data.attrib.get('thumb', NA)
|
|
|
|
self.addedAt = toDatetime(data.attrib.get('addedAt', NA))
|
|
|
|
self.updatedAt = toDatetime(data.attrib.get('updatedAt', NA))
|
|
|
|
self.lastViewedAt = toDatetime(data.attrib.get('lastViewedAt', NA))
|
2015-02-17 20:35:17 +00:00
|
|
|
self.sessionKey = cast(int, data.attrib.get('sessionKey', NA))
|
|
|
|
self.user = self._find_user(data) # for active sessions
|
|
|
|
self.player = self._find_player(data) # for active sessions
|
2015-11-05 02:10:10 +00:00
|
|
|
self.transcodeSession = self._find_transcodeSession(data)
|
2014-12-29 03:21:58 +00:00
|
|
|
if self.isFullObject():
|
|
|
|
# These are auto-populated when requested
|
|
|
|
self.media = [Media(self.server, elem, self.initpath, self) for elem in data if elem.tag == Media.TYPE]
|
|
|
|
self.countries = [Country(self.server, elem) for elem in data if elem.tag == Country.TYPE]
|
|
|
|
self.directors = [Director(self.server, elem) for elem in data if elem.tag == Director.TYPE]
|
|
|
|
self.genres = [Genre(self.server, elem) for elem in data if elem.tag == Genre.TYPE]
|
|
|
|
self.producers = [Producer(self.server, elem) for elem in data if elem.tag == Producer.TYPE]
|
|
|
|
self.actors = [Actor(self.server, elem) for elem in data if elem.tag == Actor.TYPE]
|
|
|
|
self.writers = [Writer(self.server, elem) for elem in data if elem.tag == Writer.TYPE]
|
|
|
|
|
2015-02-24 03:42:29 +00:00
|
|
|
@property
|
|
|
|
def thumbUrl(self):
|
|
|
|
return self.server.url(self.thumb)
|
|
|
|
|
2015-02-17 20:35:17 +00:00
|
|
|
def _find_user(self, data):
|
|
|
|
elem = data.find('User')
|
|
|
|
if elem is not None:
|
|
|
|
return MyPlexUser(elem, self.initpath)
|
|
|
|
return None
|
|
|
|
|
|
|
|
def _find_player(self, data):
|
|
|
|
elem = data.find('Player')
|
|
|
|
if elem is not None:
|
|
|
|
return Client(self.server, elem)
|
|
|
|
return None
|
|
|
|
|
2015-11-05 02:10:10 +00:00
|
|
|
def _find_transcodeSession(self, data):
|
|
|
|
elem = data.find('TranscodeSession')
|
|
|
|
if elem is not None:
|
|
|
|
return TranscodeSession(self.server, elem)
|
|
|
|
return None
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
def iter_parts(self):
|
|
|
|
for media in self.media:
|
|
|
|
for part in media.parts:
|
|
|
|
yield part
|
|
|
|
|
|
|
|
def analyze(self):
|
|
|
|
self.server.query('/%s/analyze' % self.key)
|
|
|
|
|
2015-06-04 04:00:37 +00:00
|
|
|
def getStreamUrl(self, offset=0, maxVideoBitrate=None, videoResolution=None, **kwargs):
|
|
|
|
""" Fetch URL to stream video directly.
|
|
|
|
offset: Start time (in seconds) video will initiate from (ex: 300).
|
|
|
|
maxVideoBitrate: Max bitrate video and audio stream (ex: 64).
|
|
|
|
videoResolution: Max resolution of a video stream (ex: 1280x720).
|
|
|
|
params: Dict of additional parameters to include in URL.
|
|
|
|
"""
|
2015-06-04 14:55:30 +00:00
|
|
|
if self.TYPE not in [Movie.TYPE, Episode.TYPE]:
|
|
|
|
raise Unsupported('Cannot get stream URL for %s.' % self.TYPE)
|
2015-06-04 04:00:37 +00:00
|
|
|
params = {}
|
2015-06-08 16:41:47 +00:00
|
|
|
params['path'] = self.key
|
2015-06-04 04:00:37 +00:00
|
|
|
params['offset'] = offset
|
|
|
|
params['copyts'] = kwargs.get('copyts', 1)
|
|
|
|
params['mediaIndex'] = kwargs.get('mediaIndex', 0)
|
|
|
|
params['X-Plex-Platform'] = kwargs.get('platform', 'Chrome')
|
2015-09-25 03:02:42 +00:00
|
|
|
if 'protocol' in kwargs:
|
|
|
|
params['protocol'] = kwargs['protocol']
|
2015-06-04 04:00:37 +00:00
|
|
|
if maxVideoBitrate:
|
|
|
|
params['maxVideoBitrate'] = max(maxVideoBitrate, 64)
|
|
|
|
if videoResolution and re.match('^\d+x\d+$', videoResolution):
|
|
|
|
params['videoResolution'] = videoResolution
|
2015-10-02 13:18:44 +00:00
|
|
|
return self.server.url('/video/:/transcode/universal/start.m3u8?%s' % urlencode(params))
|
2015-06-04 04:00:37 +00:00
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
def markWatched(self):
|
|
|
|
path = '/:/scrobble?key=%s&identifier=com.plexapp.plugins.library' % self.ratingKey
|
|
|
|
self.server.query(path)
|
|
|
|
self.reload()
|
|
|
|
|
|
|
|
def markUnwatched(self):
|
|
|
|
path = '/:/unscrobble?key=%s&identifier=com.plexapp.plugins.library' % self.ratingKey
|
|
|
|
self.server.query(path)
|
|
|
|
self.reload()
|
|
|
|
|
|
|
|
def play(self, client):
|
|
|
|
client.playMedia(self)
|
|
|
|
|
|
|
|
def refresh(self):
|
2015-11-22 11:55:50 +00:00
|
|
|
self.server.query('%s/refresh' % self.key, method=put)
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Movie(Video):
|
|
|
|
TYPE = 'movie'
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
super(Movie, self)._loadData(data)
|
|
|
|
self.studio = data.attrib.get('studio', NA)
|
|
|
|
self.contentRating = data.attrib.get('contentRating', NA)
|
|
|
|
self.rating = data.attrib.get('rating', NA)
|
|
|
|
self.viewCount = cast(int, data.attrib.get('viewCount', 0))
|
|
|
|
self.viewOffset = cast(int, data.attrib.get('viewOffset', 0))
|
|
|
|
self.year = cast(int, data.attrib.get('year', NA))
|
2015-02-17 20:35:17 +00:00
|
|
|
self.summary = data.attrib.get('summary', NA)
|
2014-12-29 03:21:58 +00:00
|
|
|
self.tagline = data.attrib.get('tagline', NA)
|
|
|
|
self.duration = cast(int, data.attrib.get('duration', NA))
|
|
|
|
self.originallyAvailableAt = toDatetime(data.attrib.get('originallyAvailableAt', NA), '%Y-%m-%d')
|
|
|
|
self.primaryExtraKey = data.attrib.get('primaryExtraKey', NA)
|
2015-11-05 03:49:09 +00:00
|
|
|
self.is_watched = bool(self.viewCount > 0) # custom attr
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Show(Video):
|
|
|
|
TYPE = 'show'
|
2015-02-24 03:42:29 +00:00
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
def _loadData(self, data):
|
|
|
|
super(Show, self)._loadData(data)
|
|
|
|
self.studio = data.attrib.get('studio', NA)
|
|
|
|
self.contentRating = data.attrib.get('contentRating', NA)
|
|
|
|
self.rating = data.attrib.get('rating', NA)
|
|
|
|
self.year = cast(int, data.attrib.get('year', NA))
|
|
|
|
self.banner = data.attrib.get('banner', NA)
|
|
|
|
self.theme = data.attrib.get('theme', NA)
|
|
|
|
self.duration = cast(int, data.attrib.get('duration', NA))
|
|
|
|
self.originallyAvailableAt = toDatetime(data.attrib.get('originallyAvailableAt', NA), '%Y-%m-%d')
|
|
|
|
self.leafCount = cast(int, data.attrib.get('leafCount', NA))
|
|
|
|
self.viewedLeafCount = cast(int, data.attrib.get('viewedLeafCount', NA))
|
|
|
|
self.childCount = cast(int, data.attrib.get('childCount', NA))
|
|
|
|
|
|
|
|
def seasons(self):
|
|
|
|
path = '/library/metadata/%s/children' % self.ratingKey
|
|
|
|
return list_items(self.server, path, Season.TYPE)
|
|
|
|
|
|
|
|
def season(self, title):
|
|
|
|
path = '/library/metadata/%s/children' % self.ratingKey
|
|
|
|
return find_item(self.server, path, title)
|
|
|
|
|
2015-06-02 02:55:20 +00:00
|
|
|
def episodes(self, watched=None):
|
2014-12-29 03:21:58 +00:00
|
|
|
leavesKey = '/library/metadata/%s/allLeaves' % self.ratingKey
|
2015-06-02 02:55:20 +00:00
|
|
|
return list_items(self.server, leavesKey, watched=watched)
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
def episode(self, title):
|
|
|
|
path = '/library/metadata/%s/allLeaves' % self.ratingKey
|
|
|
|
return find_item(self.server, path, title)
|
|
|
|
|
2015-06-02 02:55:20 +00:00
|
|
|
def watched(self):
|
|
|
|
return self.episodes(watched=True)
|
|
|
|
|
|
|
|
def unwatched(self):
|
|
|
|
return self.episodes(watched=False)
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
def get(self, title):
|
|
|
|
return self.episode(title)
|
|
|
|
|
2015-06-02 02:27:43 +00:00
|
|
|
def refresh(self):
|
|
|
|
self.server.query('/library/metadata/%s/refresh' % self.ratingKey)
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
class Season(Video):
|
|
|
|
TYPE = 'season'
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
super(Season, self)._loadData(data)
|
|
|
|
self.librarySectionID = data.attrib.get('librarySectionID', NA)
|
|
|
|
self.librarySectionTitle = data.attrib.get('librarySectionTitle', NA)
|
|
|
|
self.parentRatingKey = data.attrib.get('parentRatingKey', NA)
|
|
|
|
self.parentKey = data.attrib.get('parentKey', NA)
|
|
|
|
self.parentTitle = data.attrib.get('parentTitle', NA)
|
|
|
|
self.parentSummary = data.attrib.get('parentSummary', NA)
|
|
|
|
self.index = data.attrib.get('index', NA)
|
|
|
|
self.parentIndex = data.attrib.get('parentIndex', NA)
|
|
|
|
self.parentThumb = data.attrib.get('parentThumb', NA)
|
|
|
|
self.parentTheme = data.attrib.get('parentTheme', NA)
|
|
|
|
self.leafCount = cast(int, data.attrib.get('leafCount', NA))
|
|
|
|
self.viewedLeafCount = cast(int, data.attrib.get('viewedLeafCount', NA))
|
|
|
|
|
2015-06-02 02:55:20 +00:00
|
|
|
def episodes(self, watched=None):
|
2014-12-29 03:21:58 +00:00
|
|
|
childrenKey = '/library/metadata/%s/children' % self.ratingKey
|
2015-06-02 02:55:20 +00:00
|
|
|
return list_items(self.server, childrenKey, watched=watched)
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
def episode(self, title):
|
|
|
|
path = '/library/metadata/%s/children' % self.ratingKey
|
|
|
|
return find_item(self.server, path, title)
|
|
|
|
|
|
|
|
def get(self, title):
|
|
|
|
return self.episode(title)
|
|
|
|
|
|
|
|
def show(self):
|
|
|
|
return list_items(self.server, self.parentKey)[0]
|
|
|
|
|
2015-06-02 02:55:20 +00:00
|
|
|
def watched(self):
|
|
|
|
return self.episodes(watched=True)
|
|
|
|
|
|
|
|
def unwatched(self):
|
|
|
|
return self.episodes(watched=False)
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
|
|
|
|
class Episode(Video):
|
|
|
|
TYPE = 'episode'
|
|
|
|
|
|
|
|
def _loadData(self, data):
|
|
|
|
super(Episode, self)._loadData(data)
|
|
|
|
self.librarySectionID = data.attrib.get('librarySectionID', NA)
|
|
|
|
self.librarySectionTitle = data.attrib.get('librarySectionTitle', NA)
|
|
|
|
self.grandparentKey = data.attrib.get('grandparentKey', NA)
|
|
|
|
self.grandparentTitle = data.attrib.get('grandparentTitle', NA)
|
|
|
|
self.grandparentThumb = data.attrib.get('grandparentThumb', NA)
|
|
|
|
self.parentKey = data.attrib.get('parentKey', NA)
|
|
|
|
self.parentIndex = data.attrib.get('parentIndex', NA)
|
|
|
|
self.parentThumb = data.attrib.get('parentThumb', NA)
|
|
|
|
self.contentRating = data.attrib.get('contentRating', NA)
|
|
|
|
self.index = data.attrib.get('index', NA)
|
|
|
|
self.rating = data.attrib.get('rating', NA)
|
|
|
|
self.viewCount = cast(int, data.attrib.get('viewCount', 0))
|
|
|
|
self.viewOffset = cast(int, data.attrib.get('viewOffset', 0))
|
|
|
|
self.year = cast(int, data.attrib.get('year', NA))
|
|
|
|
self.duration = cast(int, data.attrib.get('duration', NA))
|
|
|
|
self.originallyAvailableAt = toDatetime(data.attrib.get('originallyAvailableAt', NA), '%Y-%m-%d')
|
2015-11-05 03:49:09 +00:00
|
|
|
self.is_watched = bool(self.viewCount > 0) # custom attr
|
2014-12-29 03:21:58 +00:00
|
|
|
|
2015-02-24 03:42:29 +00:00
|
|
|
@property
|
|
|
|
def thumbUrl(self):
|
|
|
|
return self.server.url(self.grandparentThumb)
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
def season(self):
|
|
|
|
return list_items(self.server, self.parentKey)[0]
|
|
|
|
|
|
|
|
def show(self):
|
|
|
|
return list_items(self.server, self.grandparentKey)[0]
|
|
|
|
|
|
|
|
|
|
|
|
def build_item(server, elem, initpath):
|
|
|
|
VIDEOCLS = {Movie.TYPE:Movie, Show.TYPE:Show, Season.TYPE:Season, Episode.TYPE:Episode}
|
|
|
|
vtype = elem.attrib.get('type')
|
|
|
|
if vtype in VIDEOCLS:
|
|
|
|
cls = VIDEOCLS[vtype]
|
|
|
|
return cls(server, elem, initpath)
|
|
|
|
raise UnknownType('Unknown video type: %s' % vtype)
|
|
|
|
|
|
|
|
|
2015-06-02 02:28:50 +00:00
|
|
|
def find_key(server, key):
|
|
|
|
path = '/library/metadata/{0}'.format(key)
|
|
|
|
try:
|
|
|
|
# Video seems to be the first sub element
|
|
|
|
elem = server.query(path)[0]
|
|
|
|
return build_item(server, elem, path)
|
|
|
|
except:
|
|
|
|
raise NotFound('Unable to find key: %s' % key)
|
|
|
|
|
|
|
|
|
2014-12-29 03:21:58 +00:00
|
|
|
def find_item(server, path, title):
|
|
|
|
for elem in server.query(path):
|
|
|
|
if elem.attrib.get('title').lower() == title.lower():
|
|
|
|
return build_item(server, elem, path)
|
|
|
|
raise NotFound('Unable to find title: %s' % title)
|
|
|
|
|
|
|
|
|
2015-06-02 02:55:20 +00:00
|
|
|
def list_items(server, path, videotype=None, watched=None):
|
2014-12-29 03:21:58 +00:00
|
|
|
items = []
|
|
|
|
for elem in server.query(path):
|
2015-06-02 02:55:20 +00:00
|
|
|
if videotype and elem.attrib.get('type') != videotype: continue
|
|
|
|
if watched is True and elem.attrib.get('viewCount', 0) == 0: continue
|
|
|
|
if watched is False and elem.attrib.get('viewCount', 0) >= 1: continue
|
|
|
|
try:
|
|
|
|
items.append(build_item(server, elem, path))
|
|
|
|
except UnknownType:
|
|
|
|
pass
|
2014-12-29 03:21:58 +00:00
|
|
|
return items
|
|
|
|
|
|
|
|
|
|
|
|
def search_type(videotype):
|
|
|
|
if videotype == Movie.TYPE: return 1
|
|
|
|
elif videotype == Show.TYPE: return 2
|
|
|
|
elif videotype == Season.TYPE: return 3
|
|
|
|
elif videotype == Episode.TYPE: return 4
|
|
|
|
raise NotFound('Unknown videotype: %s' % videotype)
|