PlexKodiConnect/resources/lib/player.py

164 lines
5.8 KiB
Python
Raw Normal View History

2015-12-25 07:07:00 +11:00
# -*- coding: utf-8 -*-
###############################################################################
2017-12-10 00:35:08 +11:00
from logging import getLogger
import copy
2015-12-25 07:07:00 +11:00
import xbmc
2015-12-25 07:07:00 +11:00
import kodidb_functions as kodidb
import plexdb_functions as plexdb
2018-01-22 04:31:49 +11:00
from downloadutils import DownloadUtils as DU
from plexbmchelper.subscribers import LOCKER
from utils import kodi_time_to_millis, unix_date_to_kodi, unix_timestamp
import variables as v
2017-05-18 04:22:16 +10:00
import state
2015-12-25 07:07:00 +11:00
###############################################################################
2015-12-25 07:07:00 +11:00
2017-12-10 00:35:08 +11:00
LOG = getLogger("PLEX." + __name__)
2016-09-02 03:41:55 +10:00
###############################################################################
2015-12-25 07:07:00 +11:00
2018-01-26 03:15:38 +11:00
@LOCKER.lockthis
def playback_cleanup(ended=False):
2018-01-26 03:15:38 +11:00
"""
PKC cleanup after playback ends/is stopped. Pass ended=True if Kodi
completely finished playing an item (because we will get and use wrong
timing data otherwise)
2018-01-26 03:15:38 +11:00
"""
LOG.debug('playback_cleanup called. Active players: %s',
state.ACTIVE_PLAYERS)
2018-01-26 03:15:38 +11:00
# We might have saved a transient token from a user flinging media via
# Companion (if we could not use the playqueue to store the token)
state.PLEX_TRANSIENT_TOKEN = None
for playerid in state.ACTIVE_PLAYERS:
status = state.PLAYER_STATES[playerid]
# Remember the last played item later
state.OLD_PLAYER_STATES[playerid] = copy.deepcopy(status)
2018-01-26 03:15:38 +11:00
# Stop transcoding
if status['playmethod'] == 'Transcode':
2018-02-04 00:05:18 +11:00
LOG.debug('Tell the PMS to stop transcoding')
2018-01-26 03:15:38 +11:00
DU().downloadUrl(
'{server}/video/:/transcode/universal/stop',
parameters={'session': v.PKC_MACHINE_IDENTIFIER})
if playerid == 1:
# Bookmarks might not be pickup up correctly, so let's do them
# manually. Applies to addon paths, but direct paths might have
# started playback via PMS
_record_playstate(status, ended)
2018-01-26 03:15:38 +11:00
# Reset the player's status
state.PLAYER_STATES[playerid] = copy.deepcopy(state.PLAYSTATE)
2018-01-26 03:15:38 +11:00
# As all playback has halted, reset the players that have been active
state.ACTIVE_PLAYERS = []
2018-02-04 00:05:18 +11:00
LOG.debug('Finished PKC playback cleanup')
2018-01-26 03:15:38 +11:00
def _record_playstate(status, ended):
if not status['plex_id']:
LOG.debug('No Plex id found to record playstate for status %s', status)
return
with plexdb.Get_Plex_DB() as plex_db:
kodi_db_item = plex_db.getItem_byId(status['plex_id'])
if kodi_db_item is None:
# Item not (yet) in Kodi library
LOG.debug('No playstate update due to Plex id not found: %s', status)
return
totaltime = float(kodi_time_to_millis(status['totaltime'])) / 1000
if ended:
progress = 0.99
time = v.IGNORE_SECONDS_AT_START + 1
else:
time = float(kodi_time_to_millis(status['time'])) / 1000
try:
progress = time / totaltime
except ZeroDivisionError:
progress = 0.0
LOG.debug('Playback progress %s (%s of %s seconds)',
progress, time, totaltime)
playcount = status['playcount']
last_played = unix_date_to_kodi(unix_timestamp())
if playcount is None:
LOG.debug('playcount not found, looking it up in the Kodi DB')
with kodidb.GetKodiDB('video') as kodi_db:
playcount = kodi_db.get_playcount(kodi_db_item[1])
playcount = 0 if playcount is None else playcount
if time < v.IGNORE_SECONDS_AT_START:
LOG.debug('Ignoring playback less than %s seconds',
v.IGNORE_SECONDS_AT_START)
# Annoying Plex bug - it'll reset an already watched video to unwatched
playcount = None
last_played = None
time = 0
elif progress >= v.MARK_PLAYED_AT:
LOG.debug('Recording entirely played video since progress > %s',
v.MARK_PLAYED_AT)
playcount += 1
time = 0
with kodidb.GetKodiDB('video') as kodi_db:
kodi_db.addPlaystate(kodi_db_item[1],
time,
totaltime,
playcount,
last_played)
# Hack to force "in progress" widget to appear if it wasn't visible before
if (state.FORCE_RELOAD_SKIN and
xbmc.getCondVisibility('Window.IsVisible(Home.xml)')):
LOG.debug('Refreshing skin to update widgets')
xbmc.executebuiltin('ReloadSkin()')
if (state.DIRECT_PATHS and
status['playmethod'] in ('DirectStream', 'Transcode')):
LOG.debug('Start cleaning Kodi files table')
with kodidb.GetKodiDB('video') as kodi_db:
kodi_db.clean_file_table()
class PKC_Player(xbmc.Player):
2015-12-25 07:07:00 +11:00
def __init__(self):
xbmc.Player.__init__(self)
2017-12-09 05:43:06 +11:00
LOG.info("Started playback monitor.")
2015-12-25 07:07:00 +11:00
2016-02-17 19:13:37 +11:00
def onPlayBackStarted(self):
"""
2016-09-02 03:41:55 +10:00
Will be called when xbmc starts playing a file.
"""
2018-01-22 04:31:49 +11:00
pass
2015-12-25 07:07:00 +11:00
2016-02-21 10:21:39 +11:00
def onPlayBackPaused(self):
2018-01-22 04:31:49 +11:00
"""
Will be called when playback is paused
"""
pass
2015-12-25 07:07:00 +11:00
2016-02-21 10:21:39 +11:00
def onPlayBackResumed(self):
2018-01-22 04:31:49 +11:00
"""
Will be called when playback is resumed
"""
pass
2015-12-25 07:07:00 +11:00
2016-02-21 10:21:39 +11:00
def onPlayBackSeek(self, time, seekOffset):
2018-01-22 04:31:49 +11:00
"""
Will be called when user seeks to a certain time during playback
"""
pass
2016-02-21 10:21:39 +11:00
def onPlayBackStopped(self):
2018-01-22 04:31:49 +11:00
"""
Will be called when playback is stopped by the user
"""
2018-02-04 00:05:18 +11:00
LOG.debug("ONPLAYBACK_STOPPED")
playback_cleanup()
2016-03-17 04:01:07 +11:00
2018-01-22 04:31:49 +11:00
def onPlayBackEnded(self):
"""
Will be called when playback ends due to the media file being finished
"""
2018-02-04 00:05:18 +11:00
LOG.debug("ONPLAYBACK_ENDED")
if state.PKC_CAUSED_STOP is True:
state.PKC_CAUSED_STOP = False
state.PKC_CAUSED_STOP_DONE = True
LOG.debug('PKC caused this playback stop - ignoring')
else:
playback_cleanup(ended=True)