PlexKodiConnect/resources/lib/playutils.py

373 lines
14 KiB
Python
Raw Normal View History

#!/usr/bin/env python
2015-12-24 14:07:00 -06:00
# -*- coding: utf-8 -*-
from __future__ import absolute_import, division, unicode_literals
2017-12-09 14:35:08 +01:00
from logging import getLogger
2016-02-07 12:38:50 +01:00
2018-06-21 19:24:37 +02:00
from .downloadutils import DownloadUtils as DU
2018-11-18 14:59:17 +01:00
from . import utils, app
2018-06-21 19:24:37 +02:00
from . import variables as v
2016-01-01 14:40:40 +01:00
2016-09-01 20:02:00 +02:00
###############################################################################
2018-06-21 19:24:37 +02:00
LOG = getLogger('PLEX.playutils')
2016-02-19 20:03:06 +01:00
###############################################################################
2015-12-24 14:07:00 -06:00
class PlayUtils():
2016-02-19 20:03:06 +01:00
2018-01-07 17:50:30 +01:00
def __init__(self, api, playqueue_item):
2018-01-23 19:13:47 +01:00
"""
init with api (PlexAPI wrapper of the PMS xml element) and
playqueue_item (Playlist_Item())
"""
2018-01-07 17:50:30 +01:00
self.api = api
self.item = playqueue_item
2016-01-01 14:40:40 +01:00
2018-01-07 17:50:30 +01:00
def getPlayUrl(self):
2016-01-29 20:07:21 +01:00
"""
Returns the playurl [unicode] for the part or returns None.
2016-02-08 19:40:58 +01:00
(movie might consist of several files)
2016-01-29 20:07:21 +01:00
"""
if self.api.mediastream_number() is None:
return
2016-04-11 18:57:20 +02:00
playurl = self.isDirectPlay()
2016-07-12 19:14:46 +02:00
if playurl is not None:
2018-01-07 17:50:30 +01:00
LOG.info("File is direct playing.")
self.item.playmethod = 'DirectPlay'
2016-04-11 18:57:20 +02:00
elif self.isDirectStream():
2018-01-07 17:50:30 +01:00
LOG.info("File is direct streaming.")
2018-02-11 14:42:49 +01:00
playurl = self.api.transcode_video_path('DirectStream')
2018-01-07 17:50:30 +01:00
self.item.playmethod = 'DirectStream'
else:
2018-01-07 17:50:30 +01:00
LOG.info("File is transcoding.")
2018-02-11 14:42:49 +01:00
playurl = self.api.transcode_video_path(
2016-04-11 18:57:20 +02:00
'Transcode',
quality={
2016-12-20 16:13:19 +01:00
'maxVideoBitrate': self.get_bitrate(),
'videoResolution': self.get_resolution(),
'videoQuality': '100',
2018-06-21 19:24:37 +02:00
'mediaBufferSize': int(
utils.settings('kodi_video_cache')) / 1024,
2018-01-07 17:50:30 +01:00
})
self.item.playmethod = 'Transcode'
LOG.info("The playurl is: %s", playurl)
2018-01-22 11:20:37 +01:00
self.item.file = playurl
2016-02-08 19:40:58 +01:00
return playurl
2015-12-24 14:07:00 -06:00
def isDirectPlay(self):
2016-04-11 18:57:20 +02:00
"""
Returns the path/playurl if we can direct play, None otherwise
2016-04-11 18:57:20 +02:00
"""
2016-04-17 13:36:41 +02:00
# True for e.g. plex.tv watch later
2018-02-11 14:42:49 +01:00
if self.api.should_stream() is True:
2018-01-07 17:50:30 +01:00
LOG.info("Plex item optimized for direct streaming")
2016-07-12 19:14:46 +02:00
return
2017-11-05 12:51:45 +01:00
# Check whether we have a strm file that we need to throw at Kodi 1:1
2018-02-11 14:42:49 +01:00
path = self.api.file_path()
2017-11-05 12:51:45 +01:00
if path is not None and path.endswith('.strm'):
2018-01-07 17:50:30 +01:00
LOG.info('.strm file detected')
2018-02-11 14:42:49 +01:00
playurl = self.api.validate_playurl(path,
self.api.plex_type(),
force_check=True)
2018-01-07 17:50:30 +01:00
return playurl
# set to either 'Direct Stream=1' or 'Transcode=2'
# and NOT to 'Direct Play=0'
2018-06-21 19:24:37 +02:00
if utils.settings('playType') != "0":
2015-12-24 14:07:00 -06:00
# User forcing to play via HTTP
2018-01-07 17:50:30 +01:00
LOG.info("User chose to not direct play")
2016-07-12 19:14:46 +02:00
return
if self.mustTranscode():
2016-07-12 19:14:46 +02:00
return
2018-02-11 14:42:49 +01:00
return self.api.validate_playurl(path,
2018-06-21 19:24:37 +02:00
self.api.plex_type(),
force_check=True)
2015-12-24 14:07:00 -06:00
def mustTranscode(self):
2016-02-07 12:38:50 +01:00
"""
Returns True if we need to transcode because
- codec is in h265
- 10bit video codec
2016-05-15 18:01:13 +02:00
- HEVC codec
2018-02-03 12:45:48 +01:00
- playqueue_item force_transcode is set to True
- state variable FORCE_TRANSCODE set to True
2016-11-06 15:37:22 +01:00
(excepting trailers etc.)
2016-12-20 16:13:19 +01:00
- video bitrate above specified settings bitrate
if the corresponding file settings are set to 'true'
2016-02-07 12:38:50 +01:00
"""
2018-02-11 14:42:49 +01:00
if self.api.plex_type() in (v.PLEX_TYPE_CLIP, v.PLEX_TYPE_SONG):
2018-01-07 17:50:30 +01:00
LOG.info('Plex clip or music track, not transcoding')
2016-12-20 16:13:19 +01:00
return False
2018-02-11 14:42:49 +01:00
videoCodec = self.api.video_codec()
2018-09-22 12:06:02 +02:00
LOG.info("videoCodec: %s", videoCodec)
2018-02-03 12:45:48 +01:00
if self.item.force_transcode is True:
2018-01-07 17:50:30 +01:00
LOG.info('User chose to force-transcode')
2016-12-20 16:13:19 +01:00
return True
codec = videoCodec['videocodec']
if codec is None:
# e.g. trailers. Avoids TypeError with "'h265' in codec"
2018-01-07 17:50:30 +01:00
LOG.info('No codec from PMS, not transcoding.')
return False
2018-06-21 19:24:37 +02:00
if ((utils.settings('transcodeHi10P') == 'true' and
2018-09-22 12:07:30 +02:00
videoCodec['bitDepth'] == '10') and
('h264' in codec)):
2018-01-07 17:50:30 +01:00
LOG.info('Option to transcode 10bit h264 video content enabled.')
return True
2016-12-20 16:13:19 +01:00
try:
bitrate = int(videoCodec['bitrate'])
except (TypeError, ValueError):
2018-01-07 17:50:30 +01:00
LOG.info('No video bitrate from PMS, not transcoding.')
2016-12-20 16:13:19 +01:00
return False
if bitrate > self.get_max_bitrate():
2018-01-07 17:50:30 +01:00
LOG.info('Video bitrate of %s is higher than the maximal video'
2018-09-22 12:06:02 +02:00
'bitrate of %s that the user chose. Transcoding',
bitrate, self.get_max_bitrate())
2016-11-06 15:37:22 +01:00
return True
try:
resolution = int(videoCodec['resolution'])
except (TypeError, ValueError):
2019-04-14 11:37:03 +02:00
if videoCodec['resolution'] == '4k':
resolution = 2160
else:
LOG.info('No video resolution from PMS, not transcoding.')
return False
2016-12-20 16:13:19 +01:00
if 'h265' in codec or 'hevc' in codec:
if resolution >= self.getH265():
2018-09-22 12:06:02 +02:00
LOG.info('Option to transcode h265/HEVC enabled. Resolution '
'of the media: %s, transcoding limit resolution: %s',
resolution, self.getH265())
return True
2016-02-07 12:38:50 +01:00
return False
def isDirectStream(self):
2016-05-08 12:33:13 +02:00
# Never transcode Music
2018-02-11 14:42:49 +01:00
if self.api.plex_type() == 'track':
2016-05-08 12:33:13 +02:00
return True
# set to 'Transcode=2'
2018-06-21 19:24:37 +02:00
if utils.settings('playType') == "2":
# User forcing to play via HTTP
2018-01-07 17:50:30 +01:00
LOG.info("User chose to transcode")
return False
if self.mustTranscode():
2015-12-24 14:07:00 -06:00
return False
return True
2019-02-08 13:52:33 +01:00
@staticmethod
def get_max_bitrate():
2015-12-24 14:07:00 -06:00
# get the addon video quality
2018-06-21 19:24:37 +02:00
videoQuality = utils.settings('maxVideoQualities')
2015-12-24 14:07:00 -06:00
bitrate = {
2016-02-07 12:38:50 +01:00
'0': 320,
'1': 720,
'2': 1500,
2015-12-24 14:07:00 -06:00
'3': 2000,
2016-02-07 12:38:50 +01:00
'4': 3000,
'5': 4000,
'6': 8000,
'7': 10000,
'8': 12000,
'9': 20000,
'10': 40000,
2016-12-20 16:13:19 +01:00
'11': 99999999 # deactivated
2015-12-24 14:07:00 -06:00
}
# max bit rate supported by server (max signed 32bit integer)
return bitrate.get(videoQuality, 2147483)
2019-02-08 13:52:33 +01:00
@staticmethod
def getH265():
"""
Returns the user settings for transcoding h265: boundary resolutions
of 480, 720 or 1080 as an int
OR 9999999 (int) if user chose not to transcode
"""
2016-02-07 12:38:50 +01:00
H265 = {
2016-12-20 16:13:19 +01:00
'0': 99999999,
2016-02-07 12:38:50 +01:00
'1': 480,
'2': 720,
'3': 1080
}
2018-06-21 19:24:37 +02:00
return H265[utils.settings('transcodeH265')]
2016-02-07 12:38:50 +01:00
2019-02-08 13:52:33 +01:00
@staticmethod
def get_bitrate():
2016-12-20 16:13:19 +01:00
"""
Get the desired transcoding bitrate from the settings
"""
2018-06-21 19:24:37 +02:00
videoQuality = utils.settings('transcoderVideoQualities')
2016-12-20 16:13:19 +01:00
bitrate = {
'0': 320,
'1': 720,
'2': 1500,
'3': 2000,
'4': 3000,
'5': 4000,
'6': 8000,
'7': 10000,
'8': 12000,
'9': 20000,
'10': 40000,
}
# max bit rate supported by server (max signed 32bit integer)
return bitrate.get(videoQuality, 2147483)
2019-02-08 13:52:33 +01:00
@staticmethod
def get_resolution():
2016-12-20 16:13:19 +01:00
"""
Get the desired transcoding resolutions from the settings
"""
2018-06-21 19:24:37 +02:00
chosen = utils.settings('transcoderVideoQualities')
2016-02-07 12:38:50 +01:00
res = {
'0': '420x420',
'1': '576x320',
'2': '720x480',
'3': '1024x768',
'4': '1280x720',
'5': '1280x720',
'6': '1920x1080',
'7': '1920x1080',
'8': '1920x1080',
'9': '1920x1080',
'10': '1920x1080',
}
return res[chosen]
2018-01-07 17:50:30 +01:00
def audio_subtitle_prefs(self, listitem):
2017-05-01 19:51:10 +02:00
"""
For transcoding only
2015-12-24 14:07:00 -06:00
2017-05-01 19:51:10 +02:00
Called at the very beginning of play; used to change audio and subtitle
stream by a PUT request to the PMS
"""
# Set media and part where we're at
if (self.api.mediastream is None and
self.api.mediastream_number() is None):
return
2015-12-24 14:07:00 -06:00
try:
2018-01-07 17:50:30 +01:00
mediastreams = self.api.plex_media_streams()
2017-05-01 19:51:10 +02:00
except (TypeError, IndexError):
2018-01-07 17:50:30 +01:00
LOG.error('Could not get media %s, part %s',
self.api.mediastream, self.api.part)
2017-05-01 19:51:10 +02:00
return
part_id = mediastreams.attrib['id']
audio_streams_list = []
audio_streams = []
subtitle_streams_list = []
# No subtitles as an option
2018-06-21 19:24:37 +02:00
subtitle_streams = [utils.lang(39706)]
2017-05-01 19:51:10 +02:00
downloadable_streams = []
download_subs = []
# selectAudioIndex = ""
select_subs_index = ""
audio_numb = 0
2016-02-07 12:38:50 +01:00
# Remember 'no subtitles'
2017-05-01 19:51:10 +02:00
sub_num = 1
default_sub = None
2015-12-24 14:07:00 -06:00
for stream in mediastreams:
2016-09-01 20:02:00 +02:00
# Since Plex returns all possible tracks together, have to sort
# them.
index = stream.attrib.get('id')
2017-05-01 19:51:10 +02:00
typus = stream.attrib.get('streamType')
2016-02-04 20:23:04 +01:00
# Audio
2017-05-01 19:51:10 +02:00
if typus == "2":
codec = stream.attrib.get('codec')
2018-01-07 17:50:30 +01:00
channellayout = stream.attrib.get('audioChannelLayout', "")
2015-12-24 14:07:00 -06:00
try:
2018-06-21 19:24:37 +02:00
track = "%s %s - %s %s" % (audio_numb + 1,
2017-05-01 11:05:51 +02:00
stream.attrib['language'],
codec,
2018-01-07 17:50:30 +01:00
channellayout)
except KeyError:
2018-06-21 19:24:37 +02:00
track = "%s %s - %s %s" % (audio_numb + 1,
utils.lang(39707), # unknown
2017-05-01 19:51:10 +02:00
codec,
2018-01-07 17:50:30 +01:00
channellayout)
2017-05-01 19:51:10 +02:00
audio_streams_list.append(index)
2018-06-21 19:24:37 +02:00
audio_streams.append(utils.try_encode(track))
2017-05-01 19:51:10 +02:00
audio_numb += 1
2015-12-24 14:07:00 -06:00
2016-02-04 20:23:04 +01:00
# Subtitles
2017-05-01 19:51:10 +02:00
elif typus == "3":
2015-12-24 14:07:00 -06:00
try:
2018-06-21 19:24:37 +02:00
track = "%s %s" % (sub_num + 1, stream.attrib['language'])
2017-05-01 19:51:10 +02:00
except KeyError:
2018-06-21 19:24:37 +02:00
track = "%s %s (%s)" % (sub_num + 1,
utils.lang(39707), # unknown
2017-05-01 19:51:10 +02:00
stream.attrib.get('codec'))
2016-02-07 12:38:50 +01:00
default = stream.attrib.get('default')
forced = stream.attrib.get('forced')
downloadable = stream.attrib.get('key')
2015-12-24 14:07:00 -06:00
if default:
2018-06-21 19:24:37 +02:00
track = "%s - %s" % (track, utils.lang(39708)) # Default
2015-12-24 14:07:00 -06:00
if forced:
2018-06-21 19:24:37 +02:00
track = "%s - %s" % (track, utils.lang(39709)) # Forced
if downloadable:
2017-05-01 19:51:10 +02:00
# We do know the language - temporarily download
if 'language' in stream.attrib:
2018-01-07 17:50:30 +01:00
path = self.api.download_external_subtitles(
2017-05-01 19:51:10 +02:00
'{server}%s' % stream.attrib['key'],
"subtitle.%s.%s" % (stream.attrib['languageCode'],
2017-05-01 19:51:10 +02:00
stream.attrib['codec']))
# We don't know the language - no need to download
else:
2018-02-11 14:42:49 +01:00
path = self.api.attach_plex_token_to_url(
2018-11-18 14:59:17 +01:00
"%s%s" % (app.CONN.server,
2017-05-17 16:00:43 +02:00
stream.attrib['key']))
2017-05-01 19:51:10 +02:00
downloadable_streams.append(index)
2018-06-21 19:24:37 +02:00
download_subs.append(utils.try_encode(path))
2016-09-08 14:51:57 +02:00
else:
2018-06-21 19:24:37 +02:00
track = "%s (%s)" % (track, utils.lang(39710)) # burn-in
if stream.attrib.get('selected') == '1' and downloadable:
# Only show subs without asking user if they can be
# turned off
2017-05-01 19:51:10 +02:00
default_sub = index
2015-12-24 14:07:00 -06:00
2017-05-01 19:51:10 +02:00
subtitle_streams_list.append(index)
2018-06-21 19:24:37 +02:00
subtitle_streams.append(utils.try_encode(track))
2017-05-01 19:51:10 +02:00
sub_num += 1
2015-12-24 14:07:00 -06:00
2017-05-01 19:51:10 +02:00
if audio_numb > 1:
2018-06-21 19:24:37 +02:00
resp = utils.dialog('select', utils.lang(33013), audio_streams)
2015-12-24 14:07:00 -06:00
if resp > -1:
2017-05-01 19:51:10 +02:00
# User selected some audio track
args = {
'audioStreamID': audio_streams_list[resp],
'allParts': 1
}
2018-01-23 19:13:47 +01:00
DU().downloadUrl('{server}/library/parts/%s' % part_id,
action_type='PUT',
parameters=args)
2017-05-01 19:51:10 +02:00
if sub_num == 1:
# No subtitles
return
select_subs_index = None
2018-06-21 19:24:37 +02:00
if (utils.settings('pickPlexSubtitles') == 'true' and
2017-05-01 19:51:10 +02:00
default_sub is not None):
2018-01-07 17:50:30 +01:00
LOG.info('Using default Plex subtitle: %s', default_sub)
2017-05-01 19:51:10 +02:00
select_subs_index = default_sub
2017-05-01 11:05:51 +02:00
else:
2018-06-21 19:24:37 +02:00
resp = utils.dialog('select', utils.lang(33014), subtitle_streams)
2017-05-01 19:51:10 +02:00
if resp > 0:
2018-06-21 19:24:37 +02:00
select_subs_index = subtitle_streams_list[resp - 1]
else:
2017-05-01 19:51:10 +02:00
# User selected no subtitles or backed out of dialog
select_subs_index = ''
2018-01-07 17:50:30 +01:00
LOG.debug('Adding external subtitles: %s', download_subs)
2017-05-01 19:51:10 +02:00
# Enable Kodi to switch autonomously to downloadable subtitles
if download_subs:
listitem.setSubtitles(download_subs)
2018-01-07 17:50:30 +01:00
# Don't additionally burn in subtitles
2017-05-01 19:51:10 +02:00
if select_subs_index in downloadable_streams:
select_subs_index = ''
2018-01-07 17:50:30 +01:00
# Now prep the PMS for our choice
2017-05-01 19:51:10 +02:00
args = {
'subtitleStreamID': select_subs_index,
'allParts': 1
}
2018-01-23 19:13:47 +01:00
DU().downloadUrl('{server}/library/parts/%s' % part_id,
action_type='PUT',
parameters=args)