mirror of
https://github.com/Tautulli/Tautulli.git
synced 2025-03-12 04:35:40 -07:00
2233 lines
114 KiB
Python
2233 lines
114 KiB
Python
# This file is part of PlexPy.
|
|
#
|
|
# PlexPy is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# PlexPy is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with PlexPy. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
import threading
|
|
import urllib
|
|
from urlparse import urlparse
|
|
|
|
import plexpy
|
|
import common
|
|
import database
|
|
import helpers
|
|
import http_handler
|
|
import libraries
|
|
import logger
|
|
import session
|
|
import users
|
|
|
|
|
|
def get_server_friendly_name():
|
|
logger.info(u"PlexPy Pmsconnect :: Requesting name from server...")
|
|
server_name = PmsConnect().get_server_pref(pref='FriendlyName')
|
|
|
|
# If friendly name is blank
|
|
if not server_name:
|
|
servers_info = PmsConnect().get_servers_info()
|
|
for server in servers_info:
|
|
if server['machine_identifier'] == plexpy.CONFIG.PMS_IDENTIFIER:
|
|
server_name = server['name']
|
|
break
|
|
|
|
if server_name and server_name != plexpy.CONFIG.PMS_NAME:
|
|
plexpy.CONFIG.__setattr__('PMS_NAME', server_name)
|
|
plexpy.CONFIG.write()
|
|
logger.info(u"PlexPy Pmsconnect :: Server name retrieved.")
|
|
|
|
return server_name
|
|
|
|
|
|
def refresh_libraries():
|
|
logger.info(u"PlexPy Pmsconnect :: Requesting libraries list refresh...")
|
|
|
|
server_id = plexpy.CONFIG.PMS_IDENTIFIER
|
|
if not server_id:
|
|
logger.error(u"PlexPy Pmsconnect :: No PMS identifier, cannot refresh libraries. Verify server in settings.")
|
|
return
|
|
|
|
library_sections = PmsConnect().get_library_details()
|
|
|
|
if library_sections:
|
|
monitor_db = database.MonitorDatabase()
|
|
|
|
library_keys = []
|
|
new_keys = []
|
|
|
|
for section in library_sections:
|
|
section_keys = {'server_id': server_id,
|
|
'section_id': section['section_id']}
|
|
section_values = {'server_id': server_id,
|
|
'section_id': section['section_id'],
|
|
'section_name': section['section_name'],
|
|
'section_type': section['section_type'],
|
|
'thumb': section['thumb'],
|
|
'art': section['art'],
|
|
'count': section['count'],
|
|
'parent_count': section.get('parent_count', None),
|
|
'child_count': section.get('child_count', None),
|
|
}
|
|
|
|
result = monitor_db.upsert('library_sections', key_dict=section_keys, value_dict=section_values)
|
|
|
|
library_keys.append(section['section_id'])
|
|
|
|
if result == 'insert':
|
|
new_keys.append(section['section_id'])
|
|
|
|
if plexpy.CONFIG.HOME_LIBRARY_CARDS == ['first_run_wizard']:
|
|
plexpy.CONFIG.__setattr__('HOME_LIBRARY_CARDS', library_keys)
|
|
plexpy.CONFIG.write()
|
|
else:
|
|
new_keys = plexpy.CONFIG.HOME_LIBRARY_CARDS + new_keys
|
|
plexpy.CONFIG.__setattr__('HOME_LIBRARY_CARDS', new_keys)
|
|
plexpy.CONFIG.write()
|
|
|
|
if plexpy.CONFIG.UPDATE_SECTION_IDS == 1 or plexpy.CONFIG.UPDATE_SECTION_IDS == -1:
|
|
# Start library section_id update on it's own thread
|
|
threading.Thread(target=libraries.update_section_ids).start()
|
|
|
|
if plexpy.CONFIG.UPDATE_LABELS == 1 or plexpy.CONFIG.UPDATE_LABELS == -1:
|
|
# Start library labels update on it's own thread
|
|
threading.Thread(target=libraries.update_labels).start()
|
|
|
|
logger.info(u"PlexPy Pmsconnect :: Libraries list refreshed.")
|
|
return True
|
|
else:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to refresh libraries list.")
|
|
return False
|
|
|
|
|
|
class PmsConnect(object):
|
|
"""
|
|
Retrieve data from Plex Server
|
|
"""
|
|
|
|
def __init__(self, token=None):
|
|
if plexpy.CONFIG.PMS_URL:
|
|
url_parsed = urlparse(plexpy.CONFIG.PMS_URL)
|
|
hostname = url_parsed.hostname
|
|
port = url_parsed.port
|
|
self.protocol = url_parsed.scheme
|
|
else:
|
|
hostname = plexpy.CONFIG.PMS_IP
|
|
port = plexpy.CONFIG.PMS_PORT
|
|
self.protocol = 'http'
|
|
|
|
if not token:
|
|
# Check if we should use the admin token, or the guest server token
|
|
if session.get_session_user_id():
|
|
user_data = users.Users()
|
|
user_tokens = user_data.get_tokens(user_id=session.get_session_user_id())
|
|
self.token = user_tokens['server_token']
|
|
else:
|
|
self.token = plexpy.CONFIG.PMS_TOKEN
|
|
else:
|
|
self.token = token
|
|
|
|
self.request_handler = http_handler.HTTPHandler(host=hostname,
|
|
port=port,
|
|
token=self.token)
|
|
|
|
def get_sessions(self, output_format=''):
|
|
"""
|
|
Return current sessions.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/status/sessions'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_metadata(self, rating_key='', output_format=''):
|
|
"""
|
|
Return metadata for request item.
|
|
|
|
Parameters required: rating_key { Plex ratingKey }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/metadata/' + rating_key
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_metadata_children(self, rating_key='', output_format=''):
|
|
"""
|
|
Return metadata for children of the request item.
|
|
|
|
Parameters required: rating_key { Plex ratingKey }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/metadata/' + rating_key + '/children'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_recently_added(self, start='0', count='0', output_format=''):
|
|
"""
|
|
Return list of recently added items.
|
|
|
|
Parameters required: count { number of results to return }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/recentlyAdded?X-Plex-Container-Start=%s&X-Plex-Container-Size=%s' % (start, count)
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_library_recently_added(self, section_id='', start='0', count='0', output_format=''):
|
|
"""
|
|
Return list of recently added items.
|
|
|
|
Parameters required: count { number of results to return }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/sections/%s/recentlyAdded?X-Plex-Container-Start=%s&X-Plex-Container-Size=%s' % (section_id, start, count)
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_children_list(self, rating_key='', output_format=''):
|
|
"""
|
|
Return list of children in requested library item.
|
|
|
|
Parameters required: rating_key { ratingKey of parent }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/metadata/' + rating_key + '/children'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_childrens_list(self, rating_key='', output_format=''):
|
|
"""
|
|
Return list of children in requested library item.
|
|
|
|
Parameters required: rating_key { ratingKey of parent }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/metadata/' + rating_key + '/allLeaves'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_server_list(self, output_format=''):
|
|
"""
|
|
Return list of local servers.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/servers'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_server_prefs(self, output_format=''):
|
|
"""
|
|
Return the local servers preferences.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/:/prefs'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_local_server_identity(self, output_format=''):
|
|
"""
|
|
Return the local server identity.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/identity'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_libraries_list(self, output_format=''):
|
|
"""
|
|
Return list of libraries on server.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/sections'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_library_list(self, section_id='', list_type='all', count='0', sort_type='', label_key='', output_format=''):
|
|
"""
|
|
Return list of items in library on server.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
count = '&X-Plex-Container-Size=' + count if count else ''
|
|
label_key = '&label=' + label_key if label_key else ''
|
|
|
|
uri = '/library/sections/' + section_id + '/' + list_type + '?X-Plex-Container-Start=0' + count + sort_type + label_key
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_library_labels(self, section_id='', output_format=''):
|
|
"""
|
|
Return list of labels for a library on server.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/library/sections/' + section_id + '/label'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_sync_item(self, sync_id=None, output_format=''):
|
|
"""
|
|
Return sync item details.
|
|
|
|
Parameters required: sync_id { unique sync id for item }
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/sync/items/' + sync_id
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_sync_transcode_queue(self, output_format=''):
|
|
"""
|
|
Return sync transcode queue.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/sync/transcodeQueue'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_search(self, query='', track='', output_format=''):
|
|
"""
|
|
Return search results.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/search?query=' + urllib.quote(query.encode('utf8')) + track
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_account(self, output_format=''):
|
|
"""
|
|
Return account details.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/myplex/account'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def put_refresh_reachability(self):
|
|
"""
|
|
Refresh Plex remote access port mapping.
|
|
|
|
Optional parameters: None
|
|
|
|
Output: None
|
|
"""
|
|
uri = '/myplex/refreshReachability'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='PUT')
|
|
|
|
return request
|
|
|
|
def put_updater(self, output_format=''):
|
|
"""
|
|
Refresh updater status.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/updater/check?download=0'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='PUT',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_updater(self, output_format=''):
|
|
"""
|
|
Return updater status.
|
|
|
|
Optional parameters: output_format { dict, json }
|
|
|
|
Output: array
|
|
"""
|
|
uri = '/updater/status'
|
|
request = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
output_format=output_format)
|
|
|
|
return request
|
|
|
|
def get_recently_added_details(self, section_id='', start='0', count='0'):
|
|
"""
|
|
Return processed and validated list of recently added items.
|
|
|
|
Parameters required: count { number of results to return }
|
|
|
|
Output: array
|
|
"""
|
|
if section_id:
|
|
recent = self.get_library_recently_added(section_id, start, count, output_format='xml')
|
|
else:
|
|
recent = self.get_recently_added(start, count, output_format='xml')
|
|
|
|
try:
|
|
xml_head = recent.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_recently_added: %s." % e)
|
|
return []
|
|
|
|
recents_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
output = {'recently_added': []}
|
|
return output
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
recents_main = a.getElementsByTagName('Directory')
|
|
for item in recents_main:
|
|
recent_items = {'media_type': helpers.get_xml_attr(item, 'type'),
|
|
'rating_key': helpers.get_xml_attr(item, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(item, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(item, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(item, 'title'),
|
|
'parent_title': helpers.get_xml_attr(item, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(item, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(item, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(item, 'parentIndex'),
|
|
'section_id': section_id if section_id else helpers.get_xml_attr(item, 'librarySectionID'),
|
|
'library_name': helpers.get_xml_attr(item, 'librarySectionTitle'),
|
|
'year': helpers.get_xml_attr(item, 'year'),
|
|
'thumb': helpers.get_xml_attr(item, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(item, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(item, 'grandparentThumb'),
|
|
'added_at': helpers.get_xml_attr(item, 'addedAt')
|
|
}
|
|
recents_list.append(recent_items)
|
|
|
|
if a.getElementsByTagName('Video'):
|
|
recents_main = a.getElementsByTagName('Video')
|
|
for item in recents_main:
|
|
recent_items = {'media_type': helpers.get_xml_attr(item, 'type'),
|
|
'rating_key': helpers.get_xml_attr(item, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(item, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(item, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(item, 'title'),
|
|
'parent_title': helpers.get_xml_attr(item, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(item, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(item, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(item, 'parentIndex'),
|
|
'section_id': section_id if section_id else helpers.get_xml_attr(item, 'librarySectionID'),
|
|
'library_name': helpers.get_xml_attr(item, 'librarySectionTitle'),
|
|
'year': helpers.get_xml_attr(item, 'year'),
|
|
'thumb': helpers.get_xml_attr(item, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(item, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(item, 'grandparentThumb'),
|
|
'added_at': helpers.get_xml_attr(item, 'addedAt')
|
|
}
|
|
recents_list.append(recent_items)
|
|
|
|
output = {'recently_added': sorted(recents_list, key=lambda k: k['added_at'], reverse=True)}
|
|
|
|
return output
|
|
|
|
def get_metadata_details(self, rating_key='', get_media_info=False):
|
|
"""
|
|
Return processed and validated metadata list for requested item.
|
|
|
|
Parameters required: rating_key { Plex ratingKey }
|
|
|
|
Output: array
|
|
"""
|
|
metadata = self.get_metadata(str(rating_key), output_format='xml')
|
|
|
|
try:
|
|
xml_head = metadata.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_metadata: %s." % e)
|
|
return []
|
|
|
|
metadata_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') != '1':
|
|
return metadata_list
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
metadata_main = a.getElementsByTagName('Directory')[0]
|
|
metadata_type = helpers.get_xml_attr(metadata_main, 'type')
|
|
elif a.getElementsByTagName('Video'):
|
|
metadata_main = a.getElementsByTagName('Video')[0]
|
|
metadata_type = helpers.get_xml_attr(metadata_main, 'type')
|
|
elif a.getElementsByTagName('Track'):
|
|
metadata_main = a.getElementsByTagName('Track')[0]
|
|
metadata_type = helpers.get_xml_attr(metadata_main, 'type')
|
|
else:
|
|
logger.debug(u"PlexPy Pmsconnect :: Metadata failed")
|
|
return None
|
|
|
|
section_id = helpers.get_xml_attr(a, 'librarySectionID')
|
|
library_name = helpers.get_xml_attr(a, 'librarySectionTitle')
|
|
|
|
directors = []
|
|
writers = []
|
|
actors = []
|
|
genres = []
|
|
labels = []
|
|
|
|
if metadata_main.getElementsByTagName('Director'):
|
|
for director in metadata_main.getElementsByTagName('Director'):
|
|
directors.append(helpers.get_xml_attr(director, 'tag'))
|
|
|
|
if metadata_main.getElementsByTagName('Writer'):
|
|
for writer in metadata_main.getElementsByTagName('Writer'):
|
|
writers.append(helpers.get_xml_attr(writer, 'tag'))
|
|
|
|
if metadata_main.getElementsByTagName('Role'):
|
|
for actor in metadata_main.getElementsByTagName('Role'):
|
|
actors.append(helpers.get_xml_attr(actor, 'tag'))
|
|
|
|
if metadata_main.getElementsByTagName('Genre'):
|
|
for genre in metadata_main.getElementsByTagName('Genre'):
|
|
genres.append(helpers.get_xml_attr(genre, 'tag'))
|
|
|
|
if metadata_main.getElementsByTagName('Label'):
|
|
for label in metadata_main.getElementsByTagName('Label'):
|
|
labels.append(helpers.get_xml_attr(label, 'tag'))
|
|
|
|
if metadata_type == 'movie':
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': helpers.get_xml_attr(metadata_main, 'studio'),
|
|
'content_rating': helpers.get_xml_attr(metadata_main, 'contentRating'),
|
|
'summary': helpers.get_xml_attr(metadata_main, 'summary'),
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': helpers.get_xml_attr(metadata_main, 'duration'),
|
|
'year': helpers.get_xml_attr(metadata_main, 'year'),
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': directors,
|
|
'writers': writers,
|
|
'actors': actors,
|
|
'genres': genres,
|
|
'labels': labels
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
elif metadata_type == 'show':
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': helpers.get_xml_attr(metadata_main, 'studio'),
|
|
'content_rating': helpers.get_xml_attr(metadata_main, 'contentRating'),
|
|
'summary': helpers.get_xml_attr(metadata_main, 'summary'),
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': helpers.get_xml_attr(metadata_main, 'duration'),
|
|
'year': helpers.get_xml_attr(metadata_main, 'year'),
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': directors,
|
|
'writers': writers,
|
|
'actors': actors,
|
|
'genres': genres,
|
|
'labels': labels
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
elif metadata_type == 'season':
|
|
parent_rating_key = helpers.get_xml_attr(metadata_main, 'parentRatingKey')
|
|
show_details = self.get_metadata_details(parent_rating_key)
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': show_details['metadata']['studio'],
|
|
'content_rating': show_details['metadata']['content_rating'],
|
|
'summary': show_details['metadata']['summary'],
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': show_details['metadata']['duration'],
|
|
'year': helpers.get_xml_attr(metadata_main, 'year'),
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': show_details['metadata']['directors'],
|
|
'writers': show_details['metadata']['writers'],
|
|
'actors': show_details['metadata']['actors'],
|
|
'genres': show_details['metadata']['genres'],
|
|
'labels': show_details['metadata']['labels']
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
elif metadata_type == 'episode':
|
|
grandparent_rating_key = helpers.get_xml_attr(metadata_main, 'grandparentRatingKey')
|
|
show_details = self.get_metadata_details(grandparent_rating_key)
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': show_details['metadata']['studio'],
|
|
'content_rating': helpers.get_xml_attr(metadata_main, 'contentRating'),
|
|
'summary': helpers.get_xml_attr(metadata_main, 'summary'),
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': helpers.get_xml_attr(metadata_main, 'duration'),
|
|
'year': helpers.get_xml_attr(metadata_main, 'year'),
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': directors,
|
|
'writers': writers,
|
|
'actors': show_details['metadata']['actors'],
|
|
'genres': show_details['metadata']['genres'],
|
|
'labels': show_details['metadata']['labels']
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
elif metadata_type == 'artist':
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': helpers.get_xml_attr(metadata_main, 'studio'),
|
|
'content_rating': helpers.get_xml_attr(metadata_main, 'contentRating'),
|
|
'summary': helpers.get_xml_attr(metadata_main, 'summary'),
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': helpers.get_xml_attr(metadata_main, 'duration'),
|
|
'year': helpers.get_xml_attr(metadata_main, 'year'),
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': directors,
|
|
'writers': writers,
|
|
'actors': actors,
|
|
'genres': genres,
|
|
'labels': labels
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
elif metadata_type == 'album':
|
|
parent_rating_key = helpers.get_xml_attr(metadata_main, 'parentRatingKey')
|
|
artist_details = self.get_metadata_details(parent_rating_key)
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': helpers.get_xml_attr(metadata_main, 'studio'),
|
|
'content_rating': helpers.get_xml_attr(metadata_main, 'contentRating'),
|
|
'summary': artist_details['metadata']['summary'],
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': helpers.get_xml_attr(metadata_main, 'duration'),
|
|
'year': helpers.get_xml_attr(metadata_main, 'year'),
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': directors,
|
|
'writers': writers,
|
|
'actors': actors,
|
|
'genres': genres,
|
|
'labels': labels
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
elif metadata_type == 'track':
|
|
parent_rating_key = helpers.get_xml_attr(metadata_main, 'parentRatingKey')
|
|
album_details = self.get_metadata_details(parent_rating_key)
|
|
metadata = {'media_type': metadata_type,
|
|
'section_id': section_id,
|
|
'library_name': library_name,
|
|
'rating_key': helpers.get_xml_attr(metadata_main, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(metadata_main, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(metadata_main, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(metadata_main, 'title'),
|
|
'parent_title': helpers.get_xml_attr(metadata_main, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(metadata_main, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(metadata_main, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(metadata_main, 'parentIndex'),
|
|
'studio': helpers.get_xml_attr(metadata_main, 'studio'),
|
|
'content_rating': helpers.get_xml_attr(metadata_main, 'contentRating'),
|
|
'summary': helpers.get_xml_attr(metadata_main, 'summary'),
|
|
'tagline': helpers.get_xml_attr(metadata_main, 'tagline'),
|
|
'rating': helpers.get_xml_attr(metadata_main, 'rating'),
|
|
'duration': helpers.get_xml_attr(metadata_main, 'duration'),
|
|
'year': album_details['metadata']['year'],
|
|
'thumb': helpers.get_xml_attr(metadata_main, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(metadata_main, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(metadata_main, 'grandparentThumb'),
|
|
'art': helpers.get_xml_attr(metadata_main, 'art'),
|
|
'originally_available_at': helpers.get_xml_attr(metadata_main, 'originallyAvailableAt'),
|
|
'added_at': helpers.get_xml_attr(metadata_main, 'addedAt'),
|
|
'updated_at': helpers.get_xml_attr(metadata_main, 'updatedAt'),
|
|
'last_viewed_at': helpers.get_xml_attr(metadata_main, 'lastViewedAt'),
|
|
'guid': helpers.get_xml_attr(metadata_main, 'guid'),
|
|
'directors': directors,
|
|
'writers': writers,
|
|
'actors': actors,
|
|
'genres': album_details['metadata']['genres'],
|
|
'labels': album_details['metadata']['labels']
|
|
}
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
else:
|
|
return None
|
|
|
|
if get_media_info:
|
|
item_media = metadata_main.getElementsByTagName('Media')
|
|
for media in item_media:
|
|
media_info = {'container': helpers.get_xml_attr(media, 'container'),
|
|
'bitrate': helpers.get_xml_attr(media, 'bitrate'),
|
|
'video_codec': helpers.get_xml_attr(media, 'videoCodec'),
|
|
'video_resolution': helpers.get_xml_attr(media, 'videoResolution'),
|
|
'video_framerate': helpers.get_xml_attr(media, 'videoFrameRate'),
|
|
'audio_codec': helpers.get_xml_attr(media, 'audioCodec'),
|
|
'audio_channels': helpers.get_xml_attr(media, 'audioChannels'),
|
|
'file': helpers.get_xml_attr(media.getElementsByTagName('Part')[0], 'file'),
|
|
'file_size': helpers.get_xml_attr(media.getElementsByTagName('Part')[0], 'size'),
|
|
}
|
|
metadata.update(media_info)
|
|
|
|
return metadata_list
|
|
|
|
def get_metadata_children_details(self, rating_key='', get_children=False, get_media_info=False):
|
|
"""
|
|
Return processed and validated metadata list for all children of requested item.
|
|
|
|
Parameters required: rating_key { Plex ratingKey }
|
|
|
|
Output: array
|
|
"""
|
|
metadata = self.get_metadata_children(str(rating_key), output_format='xml')
|
|
|
|
try:
|
|
xml_head = metadata.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_metadata_children: %s." % e)
|
|
return {'metadata': []}
|
|
|
|
metadata_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
metadata_list = {'metadata': []}
|
|
return metadata_list
|
|
|
|
if a.getElementsByTagName('Video'):
|
|
metadata_main = a.getElementsByTagName('Video')
|
|
for item in metadata_main:
|
|
child_rating_key = helpers.get_xml_attr(item, 'ratingKey')
|
|
metadata = self.get_metadata_details(str(child_rating_key), get_media_info)
|
|
if metadata:
|
|
metadata_list.append(metadata['metadata'])
|
|
|
|
elif a.getElementsByTagName('Track'):
|
|
metadata_main = a.getElementsByTagName('Track')
|
|
for item in metadata_main:
|
|
child_rating_key = helpers.get_xml_attr(item, 'ratingKey')
|
|
metadata = self.get_metadata_details(str(child_rating_key), get_media_info)
|
|
if metadata:
|
|
metadata_list.append(metadata['metadata'])
|
|
|
|
elif get_children and a.getElementsByTagName('Directory'):
|
|
dir_main = a.getElementsByTagName('Directory')
|
|
metadata_main = [d for d in dir_main if helpers.get_xml_attr(d, 'ratingKey')]
|
|
for item in metadata_main:
|
|
child_rating_key = helpers.get_xml_attr(item, 'ratingKey')
|
|
metadata = self.get_metadata_children_details(str(child_rating_key), get_children, get_media_info)
|
|
if metadata:
|
|
metadata_list.extend(metadata['metadata'])
|
|
|
|
output = {'metadata': metadata_list}
|
|
return output
|
|
|
|
def get_library_metadata_details(self, section_id=''):
|
|
"""
|
|
Return processed and validated metadata list for requested library.
|
|
|
|
Parameters required: section_id { Plex library key }
|
|
|
|
Output: array
|
|
"""
|
|
libraries_data = self.get_libraries_list(output_format='xml')
|
|
|
|
try:
|
|
xml_head = libraries_data.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_library_metadata_details: %s." % e)
|
|
return []
|
|
|
|
metadata_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
metadata_list = {'metadata': None}
|
|
return metadata_list
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
result_data = a.getElementsByTagName('Directory')
|
|
for result in result_data:
|
|
key = helpers.get_xml_attr(result, 'key')
|
|
if key == section_id:
|
|
metadata = {'media_type': 'library',
|
|
'section_id': helpers.get_xml_attr(result, 'key'),
|
|
'library': helpers.get_xml_attr(result, 'type'),
|
|
'title': helpers.get_xml_attr(result, 'title'),
|
|
'art': helpers.get_xml_attr(result, 'art'),
|
|
'thumb': helpers.get_xml_attr(result, 'thumb')
|
|
}
|
|
if metadata['library'] == 'movie':
|
|
metadata['section_type'] = 'movie'
|
|
elif metadata['library'] == 'show':
|
|
metadata['section_type'] = 'episode'
|
|
elif metadata['library'] == 'artist':
|
|
metadata['section_type'] = 'track'
|
|
|
|
metadata_list = {'metadata': metadata}
|
|
|
|
return metadata_list
|
|
|
|
def get_current_activity(self):
|
|
"""
|
|
Return processed and validated session list.
|
|
|
|
Output: array
|
|
"""
|
|
session_data = self.get_sessions(output_format='xml')
|
|
|
|
try:
|
|
xml_head = session_data.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_current_activity: %s." % e)
|
|
return []
|
|
|
|
session_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
session_list = {'stream_count': '0',
|
|
'sessions': []
|
|
}
|
|
return session_list
|
|
|
|
if a.getElementsByTagName('Track'):
|
|
session_data = a.getElementsByTagName('Track')
|
|
session_type = 'track'
|
|
for session_ in session_data:
|
|
session_output = self.get_session_each(session_type, session_)
|
|
session_list.append(session_output)
|
|
if a.getElementsByTagName('Video'):
|
|
session_data = a.getElementsByTagName('Video')
|
|
session_type = 'video'
|
|
for session_ in session_data:
|
|
session_output = self.get_session_each(session_type, session_)
|
|
session_list.append(session_output)
|
|
if a.getElementsByTagName('Photo'):
|
|
session_data = a.getElementsByTagName('Photo')
|
|
session_type = 'photo'
|
|
for session_ in session_data:
|
|
session_output = self.get_session_each(session_type, session_)
|
|
session_list.append(session_output)
|
|
|
|
session_list = sorted(session_list, key=lambda k: k['session_key'])
|
|
|
|
output = {'stream_count': helpers.get_xml_attr(xml_head[0], 'size'),
|
|
'sessions': session.mask_session_info(session_list)
|
|
}
|
|
|
|
return output
|
|
|
|
def get_session_each(self, stream_type='', session=None):
|
|
"""
|
|
Return selected data from current sessions.
|
|
This function processes and validates session data
|
|
|
|
Parameters required: stream_type { track or video }
|
|
session { the session dictionary }
|
|
Output: dict
|
|
"""
|
|
session_output = None
|
|
user_data = users.Users()
|
|
|
|
if stream_type == 'track':
|
|
media_info = session.getElementsByTagName('Media')[0]
|
|
audio_decision = 'direct play'
|
|
audio_channels = helpers.get_xml_attr(media_info, 'audioChannels')
|
|
audio_codec = helpers.get_xml_attr(media_info, 'audioCodec')
|
|
container = helpers.get_xml_attr(media_info, 'container')
|
|
bitrate = helpers.get_xml_attr(media_info, 'bitrate')
|
|
duration = helpers.get_xml_attr(media_info, 'duration')
|
|
progress = helpers.get_xml_attr(session, 'viewOffset')
|
|
|
|
if session.getElementsByTagName('TranscodeSession'):
|
|
transcode_session = session.getElementsByTagName('TranscodeSession')[0]
|
|
transcode_key = helpers.get_xml_attr(transcode_session, 'key')
|
|
throttled = helpers.get_xml_attr(transcode_session, 'throttled')
|
|
transcode_progress = helpers.get_xml_attr(transcode_session, 'progress')
|
|
transcode_speed = helpers.get_xml_attr(transcode_session, 'speed')
|
|
audio_decision = helpers.get_xml_attr(transcode_session, 'audioDecision')
|
|
transcode_audio_channels = helpers.get_xml_attr(transcode_session, 'audioChannels')
|
|
transcode_audio_codec = helpers.get_xml_attr(transcode_session, 'audioCodec')
|
|
transcode_container = helpers.get_xml_attr(transcode_session, 'container')
|
|
transcode_protocol = helpers.get_xml_attr(transcode_session, 'protocol')
|
|
duration = helpers.get_xml_attr(transcode_session, 'duration')
|
|
else:
|
|
transcode_key = ''
|
|
throttled = '0'
|
|
transcode_progress = '0'
|
|
transcode_speed = ''
|
|
transcode_audio_channels = ''
|
|
transcode_audio_codec = ''
|
|
transcode_container = ''
|
|
transcode_protocol = ''
|
|
|
|
# Generate a combined transcode decision value
|
|
transcode_decision = audio_decision
|
|
|
|
user_details = user_data.get_details(
|
|
user=helpers.get_xml_attr(session.getElementsByTagName('User')[0], 'title'))
|
|
|
|
if helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier').endswith('_Track'):
|
|
machine_id = helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier')[:-6]
|
|
else:
|
|
machine_id = helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier')
|
|
|
|
labels = []
|
|
if session.getElementsByTagName('Label'):
|
|
for label in session.getElementsByTagName('Label'):
|
|
labels.append(helpers.get_xml_attr(label, 'tag'))
|
|
|
|
session_output = {'session_key': helpers.get_xml_attr(session, 'sessionKey'),
|
|
'section_id': helpers.get_xml_attr(session, 'librarySectionID'),
|
|
'media_index': helpers.get_xml_attr(session, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(session, 'parentIndex'),
|
|
'art': helpers.get_xml_attr(session, 'art'),
|
|
'parent_thumb': helpers.get_xml_attr(session, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(session, 'grandparentThumb'),
|
|
'thumb': helpers.get_xml_attr(session, 'thumb'),
|
|
'bif_thumb': '',
|
|
'user': user_details['username'],
|
|
'user_id': user_details['user_id'],
|
|
'friendly_name': user_details['friendly_name'],
|
|
'user_thumb': user_details['user_thumb'],
|
|
'ip_address': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'address').split('::ffff:')[-1],
|
|
'player': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'title'),
|
|
'platform': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'platform'),
|
|
'machine_id': machine_id,
|
|
'state': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'state'),
|
|
'grandparent_title': helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
'parent_title': helpers.get_xml_attr(session, 'parentTitle'),
|
|
'title': helpers.get_xml_attr(session, 'title'),
|
|
'full_title': '%s - %s' % (helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
helpers.get_xml_attr(session, 'title')),
|
|
'year': helpers.get_xml_attr(session, 'year'),
|
|
'rating_key': helpers.get_xml_attr(session, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(session, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(session, 'grandparentRatingKey'),
|
|
'content_rating': helpers.get_xml_attr(session, 'contentRating'),
|
|
'labels': labels,
|
|
'transcode_key': transcode_key,
|
|
'throttled': throttled,
|
|
'transcode_progress': int(round(helpers.cast_to_float(transcode_progress), 0)),
|
|
'transcode_speed': str(round(helpers.cast_to_float(transcode_speed), 1)),
|
|
'audio_decision': audio_decision,
|
|
'audio_channels': audio_channels,
|
|
'audio_codec': audio_codec,
|
|
'video_decision': '',
|
|
'video_codec': '',
|
|
'height': '',
|
|
'width': '',
|
|
'container': container,
|
|
'bitrate': bitrate,
|
|
'video_resolution': '',
|
|
'video_framerate': '',
|
|
'aspect_ratio': '',
|
|
'transcode_decision': transcode_decision,
|
|
'transcode_audio_channels': transcode_audio_channels,
|
|
'transcode_audio_codec': transcode_audio_codec,
|
|
'transcode_video_codec': '',
|
|
'transcode_width': '',
|
|
'transcode_height': '',
|
|
'transcode_container': transcode_container,
|
|
'transcode_protocol': transcode_protocol,
|
|
'duration': duration,
|
|
'view_offset': progress,
|
|
'progress_percent': str(helpers.get_percent(progress, duration)),
|
|
'media_type': 'track',
|
|
'indexes': 0
|
|
}
|
|
|
|
elif stream_type == 'video':
|
|
media_info = session.getElementsByTagName('Media')[0]
|
|
audio_decision = 'direct play'
|
|
audio_channels = helpers.get_xml_attr(media_info, 'audioChannels')
|
|
audio_codec = helpers.get_xml_attr(media_info, 'audioCodec')
|
|
video_decision = 'direct play'
|
|
video_codec = helpers.get_xml_attr(media_info, 'videoCodec')
|
|
container = helpers.get_xml_attr(media_info, 'container')
|
|
bitrate = helpers.get_xml_attr(media_info, 'bitrate')
|
|
video_resolution = helpers.get_xml_attr(media_info, 'videoResolution')
|
|
video_framerate = helpers.get_xml_attr(media_info, 'videoFrameRate')
|
|
aspect_ratio = helpers.get_xml_attr(media_info, 'aspectRatio')
|
|
width = helpers.get_xml_attr(media_info, 'width')
|
|
height = helpers.get_xml_attr(media_info, 'height')
|
|
duration = helpers.get_xml_attr(media_info, 'duration')
|
|
progress = helpers.get_xml_attr(session, 'viewOffset')
|
|
|
|
if session.getElementsByTagName('TranscodeSession'):
|
|
transcode_session = session.getElementsByTagName('TranscodeSession')[0]
|
|
transcode_key = helpers.get_xml_attr(transcode_session, 'key')
|
|
throttled = helpers.get_xml_attr(transcode_session, 'throttled')
|
|
transcode_progress = helpers.get_xml_attr(transcode_session, 'progress')
|
|
transcode_speed = helpers.get_xml_attr(transcode_session, 'speed')
|
|
audio_decision = helpers.get_xml_attr(transcode_session, 'audioDecision')
|
|
transcode_audio_channels = helpers.get_xml_attr(transcode_session, 'audioChannels')
|
|
transcode_audio_codec = helpers.get_xml_attr(transcode_session, 'audioCodec')
|
|
video_decision = helpers.get_xml_attr(transcode_session, 'videoDecision')
|
|
transcode_video_codec = helpers.get_xml_attr(transcode_session, 'videoCodec')
|
|
transcode_width = helpers.get_xml_attr(transcode_session, 'width') or width
|
|
transcode_height = helpers.get_xml_attr(transcode_session, 'height') or height
|
|
transcode_container = helpers.get_xml_attr(transcode_session, 'container')
|
|
transcode_protocol = helpers.get_xml_attr(transcode_session, 'protocol')
|
|
else:
|
|
transcode_key = ''
|
|
throttled = '0'
|
|
transcode_progress = '0'
|
|
transcode_speed = ''
|
|
transcode_audio_channels = ''
|
|
transcode_audio_codec = ''
|
|
transcode_video_codec = ''
|
|
transcode_width = ''
|
|
transcode_height = ''
|
|
transcode_container = ''
|
|
transcode_protocol = ''
|
|
|
|
# Generate a combined transcode decision value
|
|
if video_decision == 'transcode' or audio_decision == 'transcode':
|
|
transcode_decision = 'transcode'
|
|
elif video_decision == 'copy' or audio_decision == 'copy':
|
|
transcode_decision = 'copy'
|
|
else:
|
|
transcode_decision = 'direct play'
|
|
|
|
if media_info.getElementsByTagName('Part'):
|
|
indexes = helpers.get_xml_attr(media_info.getElementsByTagName('Part')[0], 'indexes')
|
|
part_id = helpers.get_xml_attr(media_info.getElementsByTagName('Part')[0], 'id')
|
|
if indexes == 'sd':
|
|
bif_thumb = '/library/parts/' + part_id + '/indexes/sd/' + progress
|
|
else:
|
|
bif_thumb = ''
|
|
else:
|
|
indexes = ''
|
|
bif_thumb = ''
|
|
|
|
if plexpy.CONFIG.PMS_USE_BIF and indexes == 'sd':
|
|
use_indexes = 1
|
|
else:
|
|
use_indexes = 0
|
|
|
|
user_details = user_data.get_details(
|
|
user=helpers.get_xml_attr(session.getElementsByTagName('User')[0], 'title'))
|
|
|
|
if helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier').endswith('_Video'):
|
|
machine_id = helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier')[:-6]
|
|
else:
|
|
machine_id = helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier')
|
|
|
|
labels = []
|
|
if session.getElementsByTagName('Label'):
|
|
for label in session.getElementsByTagName('Label'):
|
|
labels.append(helpers.get_xml_attr(label, 'tag'))
|
|
|
|
if helpers.get_xml_attr(session, 'type') == 'episode':
|
|
session_output = {'session_key': helpers.get_xml_attr(session, 'sessionKey'),
|
|
'section_id': helpers.get_xml_attr(session, 'librarySectionID'),
|
|
'media_index': helpers.get_xml_attr(session, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(session, 'parentIndex'),
|
|
'art': helpers.get_xml_attr(session, 'art'),
|
|
'parent_thumb': helpers.get_xml_attr(session, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(session, 'grandparentThumb'),
|
|
'thumb': helpers.get_xml_attr(session, 'thumb'),
|
|
'bif_thumb': bif_thumb,
|
|
'user': user_details['username'],
|
|
'user_id': user_details['user_id'],
|
|
'friendly_name': user_details['friendly_name'],
|
|
'user_thumb': user_details['user_thumb'],
|
|
'ip_address': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'address').split('::ffff:')[-1],
|
|
'player': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'title'),
|
|
'platform': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'platform'),
|
|
'machine_id': machine_id,
|
|
'state': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'state'),
|
|
'grandparent_title': helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
'parent_title': helpers.get_xml_attr(session, 'parentTitle'),
|
|
'title': helpers.get_xml_attr(session, 'title'),
|
|
'full_title': '%s - %s' % (helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
helpers.get_xml_attr(session, 'title')),
|
|
'year': helpers.get_xml_attr(session, 'year'),
|
|
'rating_key': helpers.get_xml_attr(session, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(session, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(session, 'grandparentRatingKey'),
|
|
'content_rating': helpers.get_xml_attr(session, 'contentRating'),
|
|
'labels': labels,
|
|
'transcode_key': transcode_key,
|
|
'throttled': throttled,
|
|
'transcode_progress': int(round(helpers.cast_to_float(transcode_progress), 0)),
|
|
'transcode_speed': str(round(helpers.cast_to_float(transcode_speed), 1)),
|
|
'audio_decision': audio_decision,
|
|
'audio_channels': audio_channels,
|
|
'audio_codec': audio_codec,
|
|
'video_decision': video_decision,
|
|
'video_codec': video_codec,
|
|
'height': height,
|
|
'width': width,
|
|
'container': container,
|
|
'bitrate': bitrate,
|
|
'video_resolution': video_resolution,
|
|
'video_framerate': video_framerate,
|
|
'aspect_ratio': aspect_ratio,
|
|
'transcode_decision': transcode_decision,
|
|
'transcode_audio_channels': transcode_audio_channels,
|
|
'transcode_audio_codec': transcode_audio_codec,
|
|
'transcode_video_codec': transcode_video_codec,
|
|
'transcode_width': transcode_width,
|
|
'transcode_height': transcode_height,
|
|
'transcode_container': transcode_container,
|
|
'transcode_protocol': transcode_protocol,
|
|
'duration': duration,
|
|
'view_offset': progress,
|
|
'progress_percent': str(helpers.get_percent(progress, duration)),
|
|
'indexes': use_indexes
|
|
}
|
|
if helpers.get_xml_attr(session, 'ratingKey').isdigit():
|
|
session_output['media_type'] = helpers.get_xml_attr(session, 'type')
|
|
else:
|
|
session_output['media_type'] = 'clip'
|
|
|
|
elif helpers.get_xml_attr(session, 'type') == 'movie':
|
|
session_output = {'session_key': helpers.get_xml_attr(session, 'sessionKey'),
|
|
'section_id': helpers.get_xml_attr(session, 'librarySectionID'),
|
|
'media_index': helpers.get_xml_attr(session, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(session, 'parentIndex'),
|
|
'art': helpers.get_xml_attr(session, 'art'),
|
|
'thumb': helpers.get_xml_attr(session, 'thumb'),
|
|
'bif_thumb': bif_thumb,
|
|
'parent_thumb': helpers.get_xml_attr(session, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(session, 'grandparentThumb'),
|
|
'user': user_details['username'],
|
|
'user_id': user_details['user_id'],
|
|
'friendly_name': user_details['friendly_name'],
|
|
'user_thumb': user_details['user_thumb'],
|
|
'ip_address': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'address').split('::ffff:')[-1],
|
|
'player': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'title'),
|
|
'platform': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'platform'),
|
|
'machine_id': machine_id,
|
|
'state': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'state'),
|
|
'grandparent_title': helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
'parent_title': helpers.get_xml_attr(session, 'parentTitle'),
|
|
'title': helpers.get_xml_attr(session, 'title'),
|
|
'full_title': helpers.get_xml_attr(session, 'title'),
|
|
'year': helpers.get_xml_attr(session, 'year'),
|
|
'rating_key': helpers.get_xml_attr(session, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(session, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(session, 'grandparentRatingKey'),
|
|
'content_rating': helpers.get_xml_attr(session, 'contentRating'),
|
|
'labels': labels,
|
|
'transcode_key': transcode_key,
|
|
'throttled': throttled,
|
|
'transcode_progress': int(round(helpers.cast_to_float(transcode_progress), 0)),
|
|
'transcode_speed': str(round(helpers.cast_to_float(transcode_speed), 1)),
|
|
'audio_decision': audio_decision,
|
|
'audio_channels': audio_channels,
|
|
'audio_codec': audio_codec,
|
|
'video_decision': video_decision,
|
|
'video_codec': video_codec,
|
|
'height': height,
|
|
'width': width,
|
|
'container': container,
|
|
'bitrate': bitrate,
|
|
'video_resolution': video_resolution,
|
|
'video_framerate': video_framerate,
|
|
'aspect_ratio': aspect_ratio,
|
|
'transcode_decision': transcode_decision,
|
|
'transcode_audio_channels': transcode_audio_channels,
|
|
'transcode_audio_codec': transcode_audio_codec,
|
|
'transcode_video_codec': transcode_video_codec,
|
|
'transcode_width': transcode_width,
|
|
'transcode_height': transcode_height,
|
|
'transcode_container': transcode_container,
|
|
'transcode_protocol': transcode_protocol,
|
|
'duration': duration,
|
|
'view_offset': progress,
|
|
'progress_percent': str(helpers.get_percent(progress, duration)),
|
|
'indexes': use_indexes
|
|
}
|
|
if helpers.get_xml_attr(session, 'ratingKey').isdigit():
|
|
session_output['media_type'] = helpers.get_xml_attr(session, 'type')
|
|
else:
|
|
session_output['media_type'] = 'clip'
|
|
|
|
elif helpers.get_xml_attr(session, 'type') == 'clip':
|
|
session_output = {'session_key': helpers.get_xml_attr(session, 'sessionKey'),
|
|
'section_id': helpers.get_xml_attr(session, 'librarySectionID'),
|
|
'media_index': helpers.get_xml_attr(session, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(session, 'parentIndex'),
|
|
'art': helpers.get_xml_attr(session, 'art'),
|
|
'thumb': helpers.get_xml_attr(session, 'thumb'),
|
|
'bif_thumb': bif_thumb,
|
|
'parent_thumb': helpers.get_xml_attr(session, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(session, 'grandparentThumb'),
|
|
'user': user_details['username'],
|
|
'user_id': user_details['user_id'],
|
|
'friendly_name': user_details['friendly_name'],
|
|
'user_thumb': user_details['user_thumb'],
|
|
'ip_address': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'address').split('::ffff:')[-1],
|
|
'player': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'title'),
|
|
'platform': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'platform'),
|
|
'machine_id': machine_id,
|
|
'state': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'state'),
|
|
'grandparent_title': helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
'parent_title': helpers.get_xml_attr(session, 'parentTitle'),
|
|
'title': helpers.get_xml_attr(session, 'title'),
|
|
'full_title': helpers.get_xml_attr(session, 'title'),
|
|
'year': helpers.get_xml_attr(session, 'year'),
|
|
'rating_key': helpers.get_xml_attr(session, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(session, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(session, 'grandparentRatingKey'),
|
|
'content_rating': helpers.get_xml_attr(session, 'contentRating'),
|
|
'labels': labels,
|
|
'transcode_key': transcode_key,
|
|
'throttled': throttled,
|
|
'transcode_progress': int(round(helpers.cast_to_float(transcode_progress), 0)),
|
|
'transcode_speed': str(round(helpers.cast_to_float(transcode_speed), 1)),
|
|
'audio_decision': audio_decision,
|
|
'audio_channels': audio_channels,
|
|
'audio_codec': audio_codec,
|
|
'video_decision': video_decision,
|
|
'video_codec': video_codec,
|
|
'height': height,
|
|
'width': width,
|
|
'container': container,
|
|
'bitrate': bitrate,
|
|
'video_resolution': video_resolution,
|
|
'video_framerate': video_framerate,
|
|
'aspect_ratio': aspect_ratio,
|
|
'transcode_decision': transcode_decision,
|
|
'transcode_audio_channels': transcode_audio_channels,
|
|
'transcode_audio_codec': transcode_audio_codec,
|
|
'transcode_video_codec': transcode_video_codec,
|
|
'transcode_width': transcode_width,
|
|
'transcode_height': transcode_height,
|
|
'transcode_container': transcode_container,
|
|
'transcode_protocol': transcode_protocol,
|
|
'duration': duration,
|
|
'view_offset': progress,
|
|
'progress_percent': str(helpers.get_percent(progress, duration)),
|
|
'media_type': helpers.get_xml_attr(session, 'type'),
|
|
'indexes': 0
|
|
}
|
|
|
|
elif stream_type == 'photo':
|
|
media_info = session.getElementsByTagName('Media')[0]
|
|
video_decision = 'direct play'
|
|
container = helpers.get_xml_attr(media_info, 'container')
|
|
aspect_ratio = helpers.get_xml_attr(media_info, 'aspectRatio')
|
|
width = helpers.get_xml_attr(media_info, 'width')
|
|
height = helpers.get_xml_attr(media_info, 'height')
|
|
|
|
if session.getElementsByTagName('TranscodeSession'):
|
|
transcode_session = session.getElementsByTagName('TranscodeSession')[0]
|
|
transcode_key = helpers.get_xml_attr(transcode_session, 'key')
|
|
throttled = helpers.get_xml_attr(transcode_session, 'throttled')
|
|
transcode_progress = helpers.get_xml_attr(transcode_session, 'progress')
|
|
transcode_speed = helpers.get_xml_attr(transcode_session, 'speed')
|
|
video_decision = helpers.get_xml_attr(transcode_session, 'videoDecision')
|
|
transcode_video_codec = helpers.get_xml_attr(transcode_session, 'videoCodec')
|
|
transcode_width = helpers.get_xml_attr(transcode_session, 'width') or width
|
|
transcode_height = helpers.get_xml_attr(transcode_session, 'height') or height
|
|
transcode_container = helpers.get_xml_attr(transcode_session, 'container')
|
|
transcode_protocol = helpers.get_xml_attr(transcode_session, 'protocol')
|
|
else:
|
|
transcode_key = ''
|
|
throttled = '0'
|
|
transcode_progress = '0'
|
|
transcode_speed = ''
|
|
transcode_video_codec = ''
|
|
transcode_width = ''
|
|
transcode_height = ''
|
|
transcode_container = ''
|
|
transcode_protocol = ''
|
|
|
|
# Generate a combined transcode decision value
|
|
transcode_decision = video_decision
|
|
|
|
user_details = user_data.get_details(
|
|
user=helpers.get_xml_attr(session.getElementsByTagName('User')[0], 'title'))
|
|
|
|
if helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier').endswith('_Photo'):
|
|
machine_id = helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier')[:-6]
|
|
else:
|
|
machine_id = helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'machineIdentifier')
|
|
|
|
labels = []
|
|
if session.getElementsByTagName('Label'):
|
|
for label in session.getElementsByTagName('Label'):
|
|
labels.append(helpers.get_xml_attr(label, 'tag'))
|
|
|
|
session_output = {'session_key': helpers.get_xml_attr(session, 'sessionKey'),
|
|
'section_id': helpers.get_xml_attr(session, 'librarySectionID'),
|
|
'media_index': helpers.get_xml_attr(session, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(session, 'parentIndex'),
|
|
'art': helpers.get_xml_attr(session, 'art'),
|
|
'parent_thumb': helpers.get_xml_attr(session, 'parentThumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(session, 'grandparentThumb'),
|
|
'thumb': helpers.get_xml_attr(session, 'thumb'),
|
|
'bif_thumb': '',
|
|
'user': user_details['username'],
|
|
'user_id': user_details['user_id'],
|
|
'friendly_name': user_details['friendly_name'],
|
|
'user_thumb': user_details['user_thumb'],
|
|
'ip_address': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'address').split('::ffff:')[-1],
|
|
'player': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'title'),
|
|
'platform': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'platform'),
|
|
'machine_id': machine_id,
|
|
'state': helpers.get_xml_attr(session.getElementsByTagName('Player')[0], 'state'),
|
|
'grandparent_title': helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
'parent_title': helpers.get_xml_attr(session, 'parentTitle'),
|
|
'title': helpers.get_xml_attr(session, 'title'),
|
|
'full_title': '%s - %s' % (helpers.get_xml_attr(session, 'grandparentTitle'),
|
|
helpers.get_xml_attr(session, 'title')),
|
|
'year': helpers.get_xml_attr(session, 'year'),
|
|
'rating_key': helpers.get_xml_attr(session, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(session, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(session, 'grandparentRatingKey'),
|
|
'content_rating': helpers.get_xml_attr(session, 'contentRating'),
|
|
'labels': labels,
|
|
'transcode_key': transcode_key,
|
|
'throttled': throttled,
|
|
'transcode_progress': int(round(helpers.cast_to_float(transcode_progress), 0)),
|
|
'transcode_speed': str(round(helpers.cast_to_float(transcode_speed), 1)),
|
|
'audio_decision': '',
|
|
'audio_channels': '',
|
|
'audio_codec': '',
|
|
'video_decision': video_decision,
|
|
'video_codec': '',
|
|
'height': height,
|
|
'width': width,
|
|
'container': container,
|
|
'bitrate': '',
|
|
'video_resolution': '',
|
|
'video_framerate': '',
|
|
'aspect_ratio': aspect_ratio,
|
|
'transcode_decision': transcode_decision,
|
|
'transcode_audio_channels': '',
|
|
'transcode_audio_codec': '',
|
|
'transcode_video_codec': transcode_video_codec,
|
|
'transcode_width': transcode_width,
|
|
'transcode_height': transcode_height,
|
|
'transcode_container': transcode_container,
|
|
'transcode_protocol': transcode_protocol,
|
|
'duration': '',
|
|
'view_offset': '',
|
|
'progress_percent': '100',
|
|
'media_type': 'photo',
|
|
'indexes': 0
|
|
}
|
|
|
|
else:
|
|
logger.warn(u"PlexPy Pmsconnect :: No known stream types found in session list.")
|
|
|
|
# Rename Mystery platform names
|
|
session_output['platform'] = common.PLATFORM_NAME_OVERRIDES.get(session_output['platform'],
|
|
session_output['platform'])
|
|
|
|
return session_output
|
|
|
|
def get_item_children(self, rating_key=''):
|
|
"""
|
|
Return processed and validated children list.
|
|
|
|
Output: array
|
|
"""
|
|
children_data = self.get_children_list(rating_key, output_format='xml')
|
|
|
|
try:
|
|
xml_head = children_data.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_children_list: %s." % e)
|
|
return []
|
|
|
|
children_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
logger.debug(u"PlexPy Pmsconnect :: No children data.")
|
|
children_list = {'children_count': '0',
|
|
'children_list': []
|
|
}
|
|
return children_list
|
|
|
|
result_data = []
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
result_data = a.getElementsByTagName('Directory')
|
|
if a.getElementsByTagName('Video'):
|
|
result_data = a.getElementsByTagName('Video')
|
|
if a.getElementsByTagName('Track'):
|
|
result_data = a.getElementsByTagName('Track')
|
|
|
|
section_id = helpers.get_xml_attr(a, 'librarySectionID')
|
|
|
|
if result_data:
|
|
for result in result_data:
|
|
children_output = {'section_id': section_id,
|
|
'rating_key': helpers.get_xml_attr(result, 'ratingKey'),
|
|
'media_index': helpers.get_xml_attr(result, 'index'),
|
|
'title': helpers.get_xml_attr(result, 'title'),
|
|
'thumb': helpers.get_xml_attr(result, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(a, 'thumb'),
|
|
'duration': helpers.get_xml_attr(result, 'duration')
|
|
}
|
|
children_list.append(children_output)
|
|
|
|
output = {'children_count': helpers.get_xml_attr(xml_head[0], 'size'),
|
|
'children_type': helpers.get_xml_attr(xml_head[0], 'viewGroup'),
|
|
'title': helpers.get_xml_attr(xml_head[0], 'title2'),
|
|
'children_list': children_list
|
|
}
|
|
|
|
return output
|
|
|
|
def get_servers_info(self):
|
|
"""
|
|
Return the list of local servers.
|
|
|
|
Output: array
|
|
"""
|
|
recent = self.get_server_list(output_format='xml')
|
|
|
|
try:
|
|
xml_head = recent.getElementsByTagName('Server')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_server_list: %s." % e)
|
|
return []
|
|
|
|
server_info = []
|
|
for a in xml_head:
|
|
output = {"name": helpers.get_xml_attr(a, 'name'),
|
|
"machine_identifier": helpers.get_xml_attr(a, 'machineIdentifier'),
|
|
"host": helpers.get_xml_attr(a, 'host'),
|
|
"port": helpers.get_xml_attr(a, 'port'),
|
|
"version": helpers.get_xml_attr(a, 'version')
|
|
}
|
|
|
|
server_info.append(output)
|
|
|
|
return server_info
|
|
|
|
def get_server_identity(self):
|
|
"""
|
|
Return the local machine identity.
|
|
|
|
Output: dict
|
|
"""
|
|
identity = self.get_local_server_identity(output_format='xml')
|
|
|
|
try:
|
|
xml_head = identity.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_local_server_identity: %s." % e)
|
|
return {}
|
|
|
|
server_identity = {}
|
|
for a in xml_head:
|
|
server_identity = {"machine_identifier": helpers.get_xml_attr(a, 'machineIdentifier'),
|
|
"version": helpers.get_xml_attr(a, 'version')
|
|
}
|
|
|
|
return server_identity
|
|
|
|
def get_server_pref(self, pref=None):
|
|
"""
|
|
Return a specified server preference.
|
|
|
|
Parameters required: pref { name of preference }
|
|
|
|
Output: string
|
|
"""
|
|
if pref:
|
|
prefs = self.get_server_prefs(output_format='xml')
|
|
|
|
try:
|
|
xml_head = prefs.getElementsByTagName('Setting')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_local_server_name: %s." % e)
|
|
return ''
|
|
|
|
pref_value = 'None'
|
|
for a in xml_head:
|
|
if helpers.get_xml_attr(a, 'id') == pref:
|
|
pref_value = helpers.get_xml_attr(a, 'value')
|
|
break
|
|
|
|
return pref_value
|
|
else:
|
|
logger.debug(u"PlexPy Pmsconnect :: Server preferences queried but no parameter received.")
|
|
return None
|
|
|
|
def get_server_children(self):
|
|
"""
|
|
Return processed and validated server libraries list.
|
|
|
|
Output: array
|
|
"""
|
|
libraries_data = self.get_libraries_list(output_format='xml')
|
|
|
|
try:
|
|
xml_head = libraries_data.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_libraries_list: %s." % e)
|
|
return []
|
|
|
|
libraries_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
logger.debug(u"PlexPy Pmsconnect :: No libraries data.")
|
|
libraries_list = {'libraries_count': '0',
|
|
'libraries_list': []
|
|
}
|
|
return libraries_list
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
result_data = a.getElementsByTagName('Directory')
|
|
for result in result_data:
|
|
libraries_output = {'section_id': helpers.get_xml_attr(result, 'key'),
|
|
'section_type': helpers.get_xml_attr(result, 'type'),
|
|
'section_name': helpers.get_xml_attr(result, 'title'),
|
|
'thumb': helpers.get_xml_attr(result, 'thumb'),
|
|
'art': helpers.get_xml_attr(result, 'art')
|
|
}
|
|
libraries_list.append(libraries_output)
|
|
|
|
output = {'libraries_count': helpers.get_xml_attr(xml_head[0], 'size'),
|
|
'title': helpers.get_xml_attr(xml_head[0], 'title1'),
|
|
'libraries_list': libraries_list
|
|
}
|
|
|
|
return output
|
|
|
|
def get_library_children_details(self, section_id='', section_type='', list_type='all', count='',
|
|
rating_key='', label_key='', get_media_info=False):
|
|
"""
|
|
Return processed and validated server library items list.
|
|
|
|
Parameters required: section_type { movie, show, episode, artist }
|
|
section_id { unique library key }
|
|
|
|
Output: array
|
|
"""
|
|
|
|
if section_type == 'movie':
|
|
sort_type = '&type=1'
|
|
elif section_type == 'show':
|
|
sort_type = '&type=2'
|
|
elif section_type == 'season':
|
|
sort_type = '&type=3'
|
|
elif section_type == 'episode':
|
|
sort_type = '&type=4'
|
|
elif section_type == 'artist':
|
|
sort_type = '&type=8'
|
|
elif section_type == 'album':
|
|
sort_type = '&type=9'
|
|
elif section_type == 'track':
|
|
sort_type = '&type=10'
|
|
elif section_type == 'photo':
|
|
sort_type = ''
|
|
elif section_type == 'photoAlbum':
|
|
sort_type = '&type=14'
|
|
elif section_type == 'picture':
|
|
sort_type = '&type=13'
|
|
else:
|
|
sort_type = ''
|
|
|
|
if str(section_id).isdigit():
|
|
library_data = self.get_library_list(str(section_id), list_type, count, sort_type, label_key, output_format='xml')
|
|
elif str(rating_key).isdigit():
|
|
library_data = self.get_children_list(str(rating_key), output_format='xml')
|
|
else:
|
|
logger.warn(u"PlexPy Pmsconnect :: get_library_children called by invalid section_id or rating_key provided.")
|
|
return []
|
|
|
|
try:
|
|
xml_head = library_data.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_library_children_details: %s." % e)
|
|
return []
|
|
|
|
childern_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
logger.debug(u"PlexPy Pmsconnect :: No library data.")
|
|
childern_list = {'library_count': '0',
|
|
'childern_list': []
|
|
}
|
|
return childern_list
|
|
|
|
if rating_key:
|
|
library_count = helpers.get_xml_attr(xml_head[0], 'size')
|
|
else:
|
|
library_count = helpers.get_xml_attr(xml_head[0], 'totalSize')
|
|
|
|
# Get show/season info from xml_head
|
|
|
|
item_main = []
|
|
if a.getElementsByTagName('Directory'):
|
|
dir_main = a.getElementsByTagName('Directory')
|
|
item_main += [d for d in dir_main if helpers.get_xml_attr(d, 'ratingKey')]
|
|
if a.getElementsByTagName('Video'):
|
|
item_main += a.getElementsByTagName('Video')
|
|
if a.getElementsByTagName('Track'):
|
|
item_main += a.getElementsByTagName('Track')
|
|
if a.getElementsByTagName('Photo'):
|
|
item_main += a.getElementsByTagName('Photo')
|
|
|
|
for item in item_main:
|
|
item_info = {'section_id': helpers.get_xml_attr(a, 'librarySectionID'),
|
|
'media_type': helpers.get_xml_attr(item, 'type'),
|
|
'rating_key': helpers.get_xml_attr(item, 'ratingKey'),
|
|
'parent_rating_key': helpers.get_xml_attr(item, 'parentRatingKey'),
|
|
'grandparent_rating_key': helpers.get_xml_attr(a, 'grandparentRatingKey'),
|
|
'title': helpers.get_xml_attr(item, 'title'),
|
|
'parent_title': helpers.get_xml_attr(a, 'parentTitle'),
|
|
'grandparent_title': helpers.get_xml_attr(a, 'grandparentTitle'),
|
|
'media_index': helpers.get_xml_attr(item, 'index'),
|
|
'parent_media_index': helpers.get_xml_attr(a, 'parentIndex'),
|
|
'year': helpers.get_xml_attr(item, 'year'),
|
|
'thumb': helpers.get_xml_attr(item, 'thumb'),
|
|
'parent_thumb': helpers.get_xml_attr(a, 'thumb'),
|
|
'grandparent_thumb': helpers.get_xml_attr(a, 'grandparentThumb'),
|
|
'added_at': helpers.get_xml_attr(item, 'addedAt')
|
|
}
|
|
|
|
if get_media_info:
|
|
item_media = item.getElementsByTagName('Media')
|
|
for media in item_media:
|
|
media_info = {'container': helpers.get_xml_attr(media, 'container'),
|
|
'bitrate': helpers.get_xml_attr(media, 'bitrate'),
|
|
'video_codec': helpers.get_xml_attr(media, 'videoCodec'),
|
|
'video_resolution': helpers.get_xml_attr(media, 'videoResolution'),
|
|
'video_framerate': helpers.get_xml_attr(media, 'videoFrameRate'),
|
|
'audio_codec': helpers.get_xml_attr(media, 'audioCodec'),
|
|
'audio_channels': helpers.get_xml_attr(media, 'audioChannels'),
|
|
'file': helpers.get_xml_attr(media.getElementsByTagName('Part')[0], 'file'),
|
|
'file_size': helpers.get_xml_attr(media.getElementsByTagName('Part')[0], 'size'),
|
|
}
|
|
item_info.update(media_info)
|
|
|
|
childern_list.append(item_info)
|
|
|
|
output = {'library_count': library_count,
|
|
'childern_list': childern_list
|
|
}
|
|
|
|
return output
|
|
|
|
def get_library_details(self):
|
|
"""
|
|
Return processed and validated library statistics.
|
|
|
|
Output: array
|
|
"""
|
|
server_libraries = self.get_server_children()
|
|
|
|
server_library_stats = []
|
|
|
|
if server_libraries and server_libraries['libraries_count'] != '0':
|
|
libraries_list = server_libraries['libraries_list']
|
|
|
|
for library in libraries_list:
|
|
section_type = library['section_type']
|
|
section_id = library['section_id']
|
|
children_list = self.get_library_children_details(section_id=section_id, section_type=section_type, count='1')
|
|
|
|
if children_list:
|
|
library_stats = {'section_id': section_id,
|
|
'section_name': library['section_name'],
|
|
'section_type': section_type,
|
|
'thumb': library['thumb'],
|
|
'art': library['art'],
|
|
'count': children_list['library_count']
|
|
}
|
|
|
|
if section_type == 'show':
|
|
parent_list = self.get_library_children_details(section_id=section_id, section_type='season', count='1')
|
|
if parent_list:
|
|
parent_stats = {'parent_count': parent_list['library_count']}
|
|
library_stats.update(parent_stats)
|
|
|
|
child_list = self.get_library_children_details(section_id=section_id, section_type='episode', count='1')
|
|
if child_list:
|
|
child_stats = {'child_count': child_list['library_count']}
|
|
library_stats.update(child_stats)
|
|
|
|
if section_type == 'artist':
|
|
parent_list = self.get_library_children_details(section_id=section_id, section_type='album', count='1')
|
|
if parent_list:
|
|
parent_stats = {'parent_count': parent_list['library_count']}
|
|
library_stats.update(parent_stats)
|
|
|
|
child_list = self.get_library_children_details(section_id=section_id, section_type='track', count='1')
|
|
if child_list:
|
|
child_stats = {'child_count': child_list['library_count']}
|
|
library_stats.update(child_stats)
|
|
|
|
if section_type == 'photo':
|
|
parent_list = self.get_library_children_details(section_id=section_id, section_type='photoAlbum', count='1')
|
|
if parent_list:
|
|
parent_stats = {'parent_count': parent_list['library_count']}
|
|
library_stats.update(parent_stats)
|
|
|
|
child_list = self.get_library_children_details(section_id=section_id, section_type='picture', count='1')
|
|
if child_list:
|
|
child_stats = {'child_count': child_list['library_count']}
|
|
library_stats.update(child_stats)
|
|
|
|
server_library_stats.append(library_stats)
|
|
|
|
return server_library_stats
|
|
|
|
def get_library_label_details(self, section_id=''):
|
|
labels_data = self.get_library_labels(section_id=str(section_id), output_format='xml')
|
|
|
|
try:
|
|
xml_head = labels_data.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_library_label_details: %s." % e)
|
|
return None
|
|
|
|
labels_list = []
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
logger.debug(u"PlexPy Pmsconnect :: No labels data.")
|
|
return labels_list
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
labels_main = a.getElementsByTagName('Directory')
|
|
for item in labels_main:
|
|
label = {'label_key': helpers.get_xml_attr(item, 'key'),
|
|
'label_title': helpers.get_xml_attr(item, 'title')
|
|
}
|
|
labels_list.append(label)
|
|
|
|
return labels_list
|
|
|
|
def get_image(self, img=None, width='1000', height='1500'):
|
|
"""
|
|
Return image data as array.
|
|
Array contains the image content type and image binary
|
|
|
|
Parameters required: img { Plex image location }
|
|
Optional parameters: width { the image width }
|
|
height { the image height }
|
|
Output: array
|
|
"""
|
|
|
|
if img:
|
|
params = {'url': 'http://127.0.0.1:32400%s?%s' % (img, urllib.urlencode({'X-Plex-Token': self.token}))}
|
|
if width.isdigit() and height.isdigit():
|
|
params['width'] = width
|
|
params['height'] = height
|
|
|
|
uri = '/photo/:/transcode?%s' % urllib.urlencode(params)
|
|
result = self.request_handler.make_request(uri=uri,
|
|
proto=self.protocol,
|
|
request_type='GET',
|
|
return_type=True)
|
|
|
|
if result is None:
|
|
return
|
|
else:
|
|
return result[0], result[1]
|
|
|
|
else:
|
|
logger.error(u"PlexPy Pmsconnect :: Image proxy queried but no input received.")
|
|
|
|
def get_search_results(self, query=''):
|
|
"""
|
|
Return processed list of search results.
|
|
|
|
Output: array
|
|
"""
|
|
search_results = self.get_search(query=query, output_format='xml')
|
|
search_results_tracks = self.get_search(query=query, track='&type=10', output_format='xml')
|
|
|
|
xml_head = []
|
|
try:
|
|
try:
|
|
xml_head += search_results.getElementsByTagName('MediaContainer')
|
|
except:
|
|
pass
|
|
try:
|
|
xml_head += search_results_tracks.getElementsByTagName('MediaContainer')
|
|
except:
|
|
pass
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_search_result_details: %s." % e)
|
|
return []
|
|
|
|
search_results_count = 0
|
|
search_results_list = {'movie': [],
|
|
'show': [],
|
|
'season': [],
|
|
'episode': [],
|
|
'artist': [],
|
|
'album': [],
|
|
'track': []
|
|
}
|
|
|
|
totalSize = 0
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
totalSize += int(a.getAttribute('size'))
|
|
if totalSize == 0:
|
|
logger.debug(u"PlexPy Pmsconnect :: No search results.")
|
|
search_results_list = {'results_count': search_results_count,
|
|
'results_list': []
|
|
}
|
|
return search_results_list
|
|
|
|
for a in xml_head:
|
|
if a.getElementsByTagName('Video'):
|
|
result_data = a.getElementsByTagName('Video')
|
|
for result in result_data:
|
|
rating_key = helpers.get_xml_attr(result, 'ratingKey')
|
|
metadata = self.get_metadata_details(rating_key=rating_key)
|
|
if metadata['metadata']['media_type'] == 'movie':
|
|
search_results_list['movie'].append(metadata['metadata'])
|
|
elif metadata['metadata']['media_type'] == 'episode':
|
|
search_results_list['episode'].append(metadata['metadata'])
|
|
search_results_count += 1
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
result_data = a.getElementsByTagName('Directory')
|
|
for result in result_data:
|
|
rating_key = helpers.get_xml_attr(result, 'ratingKey')
|
|
metadata = self.get_metadata_details(rating_key=rating_key)
|
|
if metadata['metadata']['media_type'] == 'show':
|
|
search_results_list['show'].append(metadata['metadata'])
|
|
|
|
show_seasons = self.get_item_children(rating_key=metadata['metadata']['rating_key'])
|
|
if show_seasons['children_count'] != '0':
|
|
for season in show_seasons['children_list']:
|
|
if season['rating_key']:
|
|
rating_key = season['rating_key']
|
|
metadata = self.get_metadata_details(rating_key=rating_key)
|
|
search_results_list['season'].append(metadata['metadata'])
|
|
search_results_count += 1
|
|
|
|
elif metadata['metadata']['media_type'] == 'artist':
|
|
search_results_list['artist'].append(metadata['metadata'])
|
|
elif metadata['metadata']['media_type'] == 'album':
|
|
search_results_list['album'].append(metadata['metadata'])
|
|
search_results_count += 1
|
|
|
|
if a.getElementsByTagName('Track'):
|
|
result_data = a.getElementsByTagName('Track')
|
|
for result in result_data:
|
|
rating_key = helpers.get_xml_attr(result, 'ratingKey')
|
|
metadata = self.get_metadata_details(rating_key=rating_key)
|
|
search_results_list['track'].append(metadata['metadata'])
|
|
search_results_count += 1
|
|
|
|
output = {'results_count': search_results_count,
|
|
'results_list': {k: v for k, v in search_results_list.iteritems()}
|
|
}
|
|
|
|
return output
|
|
|
|
def get_rating_keys_list(self, rating_key='', media_type=''):
|
|
"""
|
|
Return processed list of grandparent/parent/child rating keys.
|
|
|
|
Output: array
|
|
"""
|
|
|
|
if media_type == 'movie':
|
|
key_list = {0: {'rating_key': int(rating_key)}}
|
|
return key_list
|
|
|
|
if media_type == 'artist' or media_type == 'album' or media_type == 'track':
|
|
match_type = 'title'
|
|
else:
|
|
match_type = 'index'
|
|
|
|
section_id = None
|
|
library_name = None
|
|
|
|
# get grandparent rating key
|
|
if media_type == 'season' or media_type == 'album':
|
|
try:
|
|
metadata = self.get_metadata_details(rating_key=rating_key)
|
|
rating_key = metadata['metadata']['parent_rating_key']
|
|
section_id = metadata['metadata']['section_id']
|
|
library_name = metadata['metadata']['library_name']
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to get parent_rating_key for get_rating_keys_list: %s." % e)
|
|
return {}
|
|
|
|
elif media_type == 'episode' or media_type == 'track':
|
|
try:
|
|
metadata = self.get_metadata_details(rating_key=rating_key)
|
|
rating_key = metadata['metadata']['grandparent_rating_key']
|
|
section_id = metadata['metadata']['section_id']
|
|
library_name = metadata['metadata']['library_name']
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to get grandparent_rating_key for get_rating_keys_list: %s." % e)
|
|
return {}
|
|
|
|
# get parent_rating_keys
|
|
metadata = self.get_metadata_children(str(rating_key), output_format='xml')
|
|
|
|
try:
|
|
xml_head = metadata.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_rating_keys_list: %s." % e)
|
|
return {}
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
return {}
|
|
|
|
title = helpers.get_xml_attr(a, 'title2')
|
|
|
|
if a.getElementsByTagName('Directory'):
|
|
parents_metadata = a.getElementsByTagName('Directory')
|
|
else:
|
|
parents_metadata = []
|
|
|
|
parents = {}
|
|
for item in parents_metadata:
|
|
parent_rating_key = helpers.get_xml_attr(item, 'ratingKey')
|
|
parent_index = helpers.get_xml_attr(item, 'index')
|
|
parent_title = helpers.get_xml_attr(item, 'title')
|
|
|
|
if parent_rating_key:
|
|
# get rating_keys
|
|
metadata = self.get_metadata_children(str(parent_rating_key), output_format='xml')
|
|
|
|
try:
|
|
xml_head = metadata.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_rating_keys_list: %s." % e)
|
|
return {}
|
|
|
|
for a in xml_head:
|
|
if a.getAttribute('size'):
|
|
if a.getAttribute('size') == '0':
|
|
return {}
|
|
|
|
if a.getElementsByTagName('Video'):
|
|
children_metadata = a.getElementsByTagName('Video')
|
|
elif a.getElementsByTagName('Track'):
|
|
children_metadata = a.getElementsByTagName('Track')
|
|
else:
|
|
children_metadata = []
|
|
|
|
children = {}
|
|
for item in children_metadata:
|
|
child_rating_key = helpers.get_xml_attr(item, 'ratingKey')
|
|
child_index = helpers.get_xml_attr(item, 'index')
|
|
child_title = helpers.get_xml_attr(item, 'title')
|
|
|
|
if child_rating_key:
|
|
key = int(child_index)
|
|
children.update({key: {'rating_key': int(child_rating_key)}})
|
|
|
|
key = int(parent_index) if match_type == 'index' else parent_title
|
|
parents.update({key:
|
|
{'rating_key': int(parent_rating_key),
|
|
'children': children}
|
|
})
|
|
|
|
key = 0 if match_type == 'index' else title
|
|
key_list = {key: {'rating_key': int(rating_key),
|
|
'children': parents},
|
|
'section_id': section_id,
|
|
'library_name': library_name
|
|
}
|
|
|
|
return key_list
|
|
|
|
def get_server_response(self):
|
|
# Refresh Plex remote access port mapping first
|
|
self.put_refresh_reachability()
|
|
account_data = self.get_account(output_format='xml')
|
|
|
|
try:
|
|
xml_head = account_data.getElementsByTagName('MyPlex')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_server_response: %s." % e)
|
|
return None
|
|
|
|
server_response = {}
|
|
|
|
for a in xml_head:
|
|
server_response = {'mapping_state': helpers.get_xml_attr(a, 'mappingState'),
|
|
'mapping_error': helpers.get_xml_attr(a, 'mappingError'),
|
|
'public_address': helpers.get_xml_attr(a, 'publicAddress'),
|
|
'public_port': helpers.get_xml_attr(a, 'publicPort')
|
|
}
|
|
|
|
return server_response
|
|
|
|
def get_update_staus(self):
|
|
# Refresh the Plex updater status first
|
|
self.put_updater()
|
|
updater_status = self.get_updater(output_format='xml')
|
|
|
|
try:
|
|
xml_head = updater_status.getElementsByTagName('MediaContainer')
|
|
except Exception as e:
|
|
logger.warn(u"PlexPy Pmsconnect :: Unable to parse XML for get_update_staus: %s." % e)
|
|
|
|
# Catch the malformed XML on certain PMX version.
|
|
# XML parser helper returns empty list if there is an error parsing XML
|
|
if updater_status == []:
|
|
logger.warn(u"Plex API updater XML is broken on the current PMS version. Please update your PMS manually.")
|
|
logger.info(u"PlexPy is unable to check for Plex updates. Disabling check for Plex updates.")
|
|
|
|
# Disable check for Plex updates
|
|
plexpy.CONFIG.MONITOR_PMS_UPDATES = 0
|
|
plexpy.initialize_scheduler()
|
|
plexpy.CONFIG.write()
|
|
|
|
return {}
|
|
|
|
updater_info = {}
|
|
for a in xml_head:
|
|
if a.getElementsByTagName('Release'):
|
|
release = a.getElementsByTagName('Release')
|
|
for item in release:
|
|
updater_info = {'can_install': helpers.get_xml_attr(a, 'canInstall'),
|
|
'download_url': helpers.get_xml_attr(a, 'downloadURL'),
|
|
'version': helpers.get_xml_attr(item, 'version'),
|
|
'state': helpers.get_xml_attr(item, 'state'),
|
|
'changelog': helpers.get_xml_attr(item, 'fixed')
|
|
}
|
|
|
|
return updater_info
|
|
|
|
def set_server_version(self):
|
|
identity = self.get_server_identity()
|
|
version = identity.get('version', plexpy.CONFIG.PMS_VERSION)
|
|
|
|
plexpy.CONFIG.__setattr__('PMS_VERSION', version)
|
|
plexpy.CONFIG.write() |