2015-12-24 14:07:00 -06:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2016-03-16 17:02:22 +01:00
|
|
|
###############################################################################
|
2017-12-09 14:35:08 +01:00
|
|
|
from logging import getLogger
|
2018-03-11 18:54:05 +01:00
|
|
|
import copy
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2018-03-22 17:26:11 +01:00
|
|
|
import xbmc
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2018-03-11 20:10:02 +01:00
|
|
|
import kodidb_functions as kodidb
|
|
|
|
import plexdb_functions as plexdb
|
2018-01-21 18:31:49 +01:00
|
|
|
from downloadutils import DownloadUtils as DU
|
|
|
|
from plexbmchelper.subscribers import LOCKER
|
2018-03-11 20:10:02 +01:00
|
|
|
from utils import kodi_time_to_millis, unix_date_to_kodi, unix_timestamp
|
2017-01-24 16:04:42 +01:00
|
|
|
import variables as v
|
2017-05-17 20:22:16 +02:00
|
|
|
import state
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-03-16 17:02:22 +01:00
|
|
|
###############################################################################
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2017-12-09 14:35:08 +01:00
|
|
|
LOG = getLogger("PLEX." + __name__)
|
2016-09-01 19:41:55 +02:00
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2018-01-25 17:15:38 +01:00
|
|
|
@LOCKER.lockthis
|
2018-03-15 10:40:15 +01:00
|
|
|
def playback_cleanup(ended=False):
|
2018-01-25 17:15:38 +01:00
|
|
|
"""
|
2018-03-15 10:40:15 +01: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-25 17:15:38 +01:00
|
|
|
"""
|
2018-03-11 18:54:05 +01:00
|
|
|
LOG.debug('playback_cleanup called')
|
2018-01-25 17:15:38 +01: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
|
2018-03-11 18:57:00 +01:00
|
|
|
state.OLD_PLAYER_STATES[playerid] = copy.deepcopy(status)
|
2018-01-25 17:15:38 +01:00
|
|
|
# Stop transcoding
|
|
|
|
if status['playmethod'] == 'Transcode':
|
2018-02-03 14:05:18 +01:00
|
|
|
LOG.debug('Tell the PMS to stop transcoding')
|
2018-01-25 17:15:38 +01:00
|
|
|
DU().downloadUrl(
|
|
|
|
'{server}/video/:/transcode/universal/stop',
|
|
|
|
parameters={'session': v.PKC_MACHINE_IDENTIFIER})
|
2018-03-11 20:10:02 +01:00
|
|
|
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
|
2018-03-15 10:40:15 +01:00
|
|
|
_record_playstate(status, ended)
|
2018-01-25 17:15:38 +01:00
|
|
|
# Reset the player's status
|
2018-03-31 18:51:03 +02:00
|
|
|
state.PLAYER_STATES[playerid] = copy.deepcopy(state.PLAYSTATE)
|
2018-01-25 17:15:38 +01:00
|
|
|
# As all playback has halted, reset the players that have been active
|
|
|
|
state.ACTIVE_PLAYERS = []
|
2018-02-03 14:05:18 +01:00
|
|
|
LOG.debug('Finished PKC playback cleanup')
|
2018-01-25 17:15:38 +01:00
|
|
|
|
|
|
|
|
2018-03-15 10:40:15 +01:00
|
|
|
def _record_playstate(status, ended):
|
2018-03-11 20:10:02 +01:00
|
|
|
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
|
2018-03-15 10:40:15 +01:00
|
|
|
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)
|
2018-03-11 20:10:02 +01:00
|
|
|
playcount = status['playcount']
|
2018-03-15 13:12:33 +01:00
|
|
|
last_played = unix_date_to_kodi(unix_timestamp())
|
2018-03-11 20:10:02 +01:00
|
|
|
if playcount is None:
|
2018-03-31 15:37:05 +02:00
|
|
|
LOG.debug('playcount not found, looking it up in the Kodi DB')
|
2018-03-11 20:10:02 +01:00
|
|
|
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
|
2018-03-18 15:08:55 +01:00
|
|
|
playcount = None
|
2018-03-15 13:12:33 +01:00
|
|
|
last_played = None
|
2018-03-11 20:10:02 +01:00
|
|
|
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,
|
2018-03-15 13:12:33 +01:00
|
|
|
last_played)
|
2018-03-22 17:26:11 +01:00
|
|
|
# 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()')
|
2018-03-11 20:10:02 +01:00
|
|
|
|
|
|
|
|
2018-03-22 17:26:11 +01:00
|
|
|
class PKC_Player(xbmc.Player):
|
2015-12-24 14:07:00 -06:00
|
|
|
def __init__(self):
|
2018-03-22 17:26:11 +01:00
|
|
|
xbmc.Player.__init__(self)
|
2017-12-08 19:43:06 +01:00
|
|
|
LOG.info("Started playback monitor.")
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-02-17 02:13:37 -06:00
|
|
|
def onPlayBackStarted(self):
|
2016-03-16 17:02:22 +01:00
|
|
|
"""
|
2016-09-01 19:41:55 +02:00
|
|
|
Will be called when xbmc starts playing a file.
|
2016-03-16 17:02:22 +01:00
|
|
|
"""
|
2018-01-21 18:31:49 +01:00
|
|
|
pass
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-02-20 17:21:39 -06:00
|
|
|
def onPlayBackPaused(self):
|
2018-01-21 18:31:49 +01:00
|
|
|
"""
|
|
|
|
Will be called when playback is paused
|
|
|
|
"""
|
|
|
|
pass
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-02-20 17:21:39 -06:00
|
|
|
def onPlayBackResumed(self):
|
2018-01-21 18:31:49 +01:00
|
|
|
"""
|
|
|
|
Will be called when playback is resumed
|
|
|
|
"""
|
|
|
|
pass
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-02-20 17:21:39 -06:00
|
|
|
def onPlayBackSeek(self, time, seekOffset):
|
2018-01-21 18:31:49 +01:00
|
|
|
"""
|
|
|
|
Will be called when user seeks to a certain time during playback
|
|
|
|
"""
|
|
|
|
pass
|
2016-05-31 19:02:11 +02:00
|
|
|
|
2016-02-20 17:21:39 -06:00
|
|
|
def onPlayBackStopped(self):
|
2018-01-21 18:31:49 +01:00
|
|
|
"""
|
|
|
|
Will be called when playback is stopped by the user
|
|
|
|
"""
|
2018-02-03 14:05:18 +01:00
|
|
|
LOG.debug("ONPLAYBACK_STOPPED")
|
2018-03-27 18:20:36 +02:00
|
|
|
playback_cleanup()
|
2016-03-16 18:01:07 +01:00
|
|
|
|
2018-01-21 18:31:49 +01:00
|
|
|
def onPlayBackEnded(self):
|
|
|
|
"""
|
|
|
|
Will be called when playback ends due to the media file being finished
|
|
|
|
"""
|
2018-02-03 14:05:18 +01:00
|
|
|
LOG.debug("ONPLAYBACK_ENDED")
|
2018-03-27 18:20:36 +02:00
|
|
|
if state.PKC_CAUSED_STOP is True:
|
|
|
|
state.PKC_CAUSED_STOP = False
|
|
|
|
LOG.debug('PKC caused this playback stop - ignoring')
|
|
|
|
else:
|
2018-03-31 15:37:05 +02:00
|
|
|
playback_cleanup(ended=True)
|