PlexKodiConnect/resources/lib/api.py

416 lines
12 KiB
Python
Raw Normal View History

2015-12-24 14:07:00 -06:00
# -*- coding: utf-8 -*-
2016-02-01 19:29:43 -06:00
# Read an api response and convert more complex cases
2015-12-24 14:07:00 -06:00
##################################################################################################
import clientinfo
import utils
##################################################################################################
class API():
def __init__(self, item):
2016-02-01 19:29:43 -06:00
# item is the api response
2015-12-24 14:07:00 -06:00
self.item = item
2016-02-01 19:29:43 -06:00
2015-12-24 14:07:00 -06:00
self.clientinfo = clientinfo.ClientInfo()
self.addonName = self.clientinfo.getAddonName()
def logMsg(self, msg, lvl=1):
className = self.__class__.__name__
utils.logMsg("%s %s" % (self.addonName, className), msg, lvl)
def getUserData(self):
# Default
favorite = False
2016-01-11 19:26:38 +01:00
likes = None
2015-12-24 14:07:00 -06:00
playcount = None
played = False
lastPlayedDate = None
resume = 0
userrating = 0
2015-12-24 14:07:00 -06:00
try:
userdata = self.item['UserData']
except KeyError: # No userdata found.
pass
else:
favorite = userdata['IsFavorite']
likes = userdata.get('Likes')
# Userrating is based on likes and favourite
2015-12-24 14:07:00 -06:00
if favorite:
userrating = 5
2015-12-24 14:07:00 -06:00
elif likes:
userrating = 3
2015-12-24 14:07:00 -06:00
elif likes == False:
userrating = 0
2015-12-24 14:07:00 -06:00
else:
userrating = 1
2015-12-24 14:07:00 -06:00
lastPlayedDate = userdata.get('LastPlayedDate')
if lastPlayedDate:
lastPlayedDate = lastPlayedDate.split('.')[0].replace('T', " ")
if userdata['Played']:
# Playcount is tied to the watch status
played = True
playcount = userdata['PlayCount']
if playcount == 0:
playcount = 1
if lastPlayedDate is None:
lastPlayedDate = self.getDateCreated()
playbackPosition = userdata.get('PlaybackPositionTicks')
if playbackPosition:
resume = playbackPosition / 10000000.0
return {
'Favorite': favorite,
'Likes': likes,
2015-12-24 14:07:00 -06:00
'PlayCount': playcount,
'Played': played,
'LastPlayedDate': lastPlayedDate,
'Resume': resume,
'UserRating': userrating
2015-12-24 14:07:00 -06:00
}
def getPeople(self):
# Process People
director = []
writer = []
cast = []
try:
people = self.item['People']
except KeyError:
pass
else:
for person in people:
type = person['Type']
name = person['Name']
if "Director" in type:
director.append(name)
elif "Actor" in type:
cast.append(name)
elif type in ("Writing", "Writer"):
writer.append(name)
return {
'Director': director,
'Writer': writer,
'Cast': cast
}
def getMediaStreams(self):
item = self.item
videotracks = []
audiotracks = []
subtitlelanguages = []
try:
media_streams = item['MediaSources'][0]['MediaStreams']
except KeyError:
if not item.get("MediaStreams"): return None
2015-12-24 14:07:00 -06:00
media_streams = item['MediaStreams']
for media_stream in media_streams:
# Sort through Video, Audio, Subtitle
stream_type = media_stream['Type']
codec = media_stream.get('Codec', "").lower()
profile = media_stream.get('Profile', "").lower()
if stream_type == "Video":
# Height, Width, Codec, AspectRatio, AspectFloat, 3D
track = {
'codec': codec,
2015-12-24 14:07:00 -06:00
'height': media_stream.get('Height'),
'width': media_stream.get('Width'),
'video3DFormat': item.get('Video3DFormat'),
'aspect': 1.85
2015-12-24 14:07:00 -06:00
}
try:
container = item['MediaSources'][0]['Container'].lower()
except:
container = ""
# Sort codec vs container/profile
if "msmpeg4" in codec:
track['codec'] = "divx"
2015-12-24 14:07:00 -06:00
elif "mpeg4" in codec:
if "simple profile" in profile or not profile:
track['codec'] = "xvid"
2015-12-24 14:07:00 -06:00
elif "h264" in codec:
if container in ("mp4", "mov", "m4v"):
track['codec'] = "avc1"
2015-12-24 14:07:00 -06:00
# Aspect ratio
if item.get('AspectRatio'):
# Metadata AR
aspect = item['AspectRatio']
2015-12-24 14:07:00 -06:00
else: # File AR
aspect = media_stream.get('AspectRatio', "0")
2015-12-24 14:07:00 -06:00
try:
aspectwidth, aspectheight = aspect.split(':')
track['aspect'] = round(float(aspectwidth) / float(aspectheight), 6)
2015-12-24 14:07:00 -06:00
2015-12-24 18:02:31 -06:00
except (ValueError, ZeroDivisionError):
width = track.get('width')
height = track.get('height')
2015-12-24 14:07:00 -06:00
if width and height:
track['aspect'] = round(float(width / height), 6)
2015-12-24 18:02:31 -06:00
else:
track['aspect'] = 1.85
if item.get("RunTimeTicks"):
track['duration'] = item.get("RunTimeTicks") / 10000000.0
2015-12-24 14:07:00 -06:00
videotracks.append(track)
elif stream_type == "Audio":
# Codec, Channels, language
track = {
'codec': codec,
2015-12-24 14:07:00 -06:00
'channels': media_stream.get('Channels'),
'language': media_stream.get('Language')
2015-12-24 14:07:00 -06:00
}
if "dca" in codec and "dts-hd ma" in profile:
track['codec'] = "dtshd_ma"
2015-12-24 14:07:00 -06:00
audiotracks.append(track)
elif stream_type == "Subtitle":
# Language
subtitlelanguages.append(media_stream.get('Language', "Unknown"))
return {
'video': videotracks,
'audio': audiotracks,
'subtitle': subtitlelanguages
}
def getRuntime(self):
item = self.item
try:
runtime = item['RunTimeTicks'] / 10000000.0
except KeyError:
runtime = item.get('CumulativeRunTimeTicks', 0) / 10000000.0
return runtime
def adjustResume(self, resume_seconds):
resume = 0
if resume_seconds:
resume = round(float(resume_seconds), 6)
jumpback = int(utils.settings('resumeJumpBack'))
if resume > jumpback:
# To avoid negative bookmark
resume = resume - jumpback
return resume
def getStudios(self):
# Process Studios
item = self.item
studios = []
try:
studio = item['SeriesStudio']
studios.append(self.verifyStudio(studio))
except KeyError:
studioList = item['Studios']
for studio in studioList:
name = studio['Name']
studios.append(self.verifyStudio(name))
return studios
def verifyStudio(self, studioName):
# Convert studio for Kodi to properly detect them
studios = {
'abc (us)': "ABC",
'fox (us)': "FOX",
'mtv (us)': "MTV",
'showcase (ca)': "Showcase",
'wgn america': "WGN"
}
return studios.get(studioName.lower(), studioName)
def getChecksum(self):
# Use the etags checksum and userdata
item = self.item
userdata = item['UserData']
checksum = "%s%s%s%s%s%s%s" % (
2015-12-24 14:07:00 -06:00
item['Etag'],
userdata['Played'],
userdata['IsFavorite'],
userdata.get('Likes',''),
2015-12-24 14:07:00 -06:00
userdata['PlaybackPositionTicks'],
userdata.get('UnplayedItemCount', ""),
userdata.get('LastPlayedDate', "")
)
return checksum
def getGenres(self):
item = self.item
all_genres = ""
genres = item.get('Genres', item.get('SeriesGenres'))
if genres:
all_genres = " / ".join(genres)
return all_genres
def getDateCreated(self):
try:
dateadded = self.item['DateCreated']
dateadded = dateadded.split('.')[0].replace('T', " ")
except KeyError:
dateadded = None
return dateadded
def getPremiereDate(self):
try:
premiere = self.item['PremiereDate']
premiere = premiere.split('.')[0].replace('T', " ")
except KeyError:
premiere = None
return premiere
def getOverview(self):
try:
overview = self.item['Overview']
overview = overview.replace("\"", "\'")
overview = overview.replace("\n", " ")
overview = overview.replace("\r", " ")
except KeyError:
overview = ""
return overview
def getTagline(self):
try:
tagline = self.item['Taglines'][0]
except IndexError:
tagline = None
return tagline
def getProvider(self, providername):
try:
provider = self.item['ProviderIds'][providername]
except KeyError:
provider = None
return provider
def getMpaa(self):
# Convert more complex cases
mpaa = self.item.get('OfficialRating', "")
if mpaa in ("NR", "UR"):
# Kodi seems to not like NR, but will accept Not Rated
mpaa = "Not Rated"
return mpaa
def getCountry(self):
try:
country = self.item['ProductionLocations'][0]
except IndexError:
country = None
return country
def getFilePath(self):
item = self.item
try:
filepath = item['Path']
except KeyError:
filepath = ""
else:
if "\\\\" in filepath:
# append smb protocol
filepath = filepath.replace("\\\\", "smb://")
filepath = filepath.replace("\\", "/")
if item.get('VideoType'):
videotype = item['VideoType']
# Specific format modification
if 'Dvd'in videotype:
filepath = "%s/VIDEO_TS/VIDEO_TS.IFO" % filepath
2016-02-01 19:29:43 -06:00
elif 'BluRay' in videotype:
2015-12-24 14:07:00 -06:00
filepath = "%s/BDMV/index.bdmv" % filepath
if "\\" in filepath:
# Local path scenario, with special videotype
filepath = filepath.replace("/", "\\")
return filepath
def updateUserRating(self, itemid, like=None, favourite=None, deletelike=False):
#updates the userrating to Emby
2016-02-01 19:29:43 -06:00
# This should be moved to read_embyserver.py
import downloadutils
doUtils = downloadutils.DownloadUtils()
if favourite != None and favourite==True:
url = "{server}/emby/Users/{UserId}/FavoriteItems/%s?format=json" % itemid
doUtils.downloadUrl(url, type="POST")
elif favourite != None and favourite==False:
url = "{server}/emby/Users/{UserId}/FavoriteItems/%s?format=json" % itemid
doUtils.downloadUrl(url, type="DELETE")
if not deletelike and like != None and like==True:
url = "{server}/emby/Users/{UserId}/Items/%s/Rating?Likes=true&format=json" % itemid
doUtils.downloadUrl(url, type="POST")
if not deletelike and like != None and like==False:
url = "{server}/emby/Users/{UserId}/Items/%s/Rating?Likes=false&format=json" % itemid
doUtils.downloadUrl(url, type="POST")
if deletelike:
url = "{server}/emby/Users/{UserId}/Items/%s/Rating?format=json" % itemid
doUtils.downloadUrl(url, type="DELETE")
2016-01-11 19:26:38 +01:00
self.logMsg( "updateUserRating on embyserver for embyId: %s - like: %s - favourite: %s - deletelike: %s" %(itemid, like, favourite, deletelike))