2017-04-02 17:02:41 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
2018-07-12 18:46:02 +02:00
|
|
|
from __future__ import absolute_import, division, unicode_literals
|
2017-04-02 17:02:41 +02:00
|
|
|
from logging import getLogger
|
|
|
|
from threading import Thread
|
|
|
|
from Queue import Empty
|
2018-05-30 07:53:30 +02:00
|
|
|
import xbmc
|
2017-04-02 17:02:41 +02:00
|
|
|
|
2018-07-27 13:38:41 +02:00
|
|
|
from ..plex_api import API
|
|
|
|
from .. import utils, plexdb_functions as plexdb, kodidb_functions as kodidb
|
|
|
|
from .. import itemtypes, artwork, plex_functions as PF, variables as v, state
|
2017-04-02 17:02:41 +02:00
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
2018-04-17 20:18:25 +02:00
|
|
|
LOG = getLogger("PLEX." + __name__)
|
2017-04-02 17:02:41 +02:00
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
2018-06-21 19:24:37 +02:00
|
|
|
@utils.thread_methods(add_suspends=['SUSPEND_LIBRARY_THREAD',
|
|
|
|
'DB_SCAN',
|
|
|
|
'STOP_SYNC',
|
|
|
|
'SUSPEND_SYNC'])
|
2018-04-17 20:18:25 +02:00
|
|
|
class ThreadedProcessFanart(Thread):
|
2017-04-02 17:02:41 +02:00
|
|
|
"""
|
|
|
|
Threaded download of additional fanart in the background
|
|
|
|
|
|
|
|
Input:
|
|
|
|
queue Queue.Queue() object that you will need to fill with
|
|
|
|
dicts of the following form:
|
|
|
|
{
|
|
|
|
'plex_id': the Plex id as a string
|
|
|
|
'plex_type': the Plex media type, e.g. 'movie'
|
|
|
|
'refresh': True/False if True, will overwrite any 3rd party
|
|
|
|
fanart. If False, will only get missing
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
def __init__(self, queue):
|
|
|
|
self.queue = queue
|
|
|
|
Thread.__init__(self)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
"""
|
|
|
|
Do the work
|
|
|
|
"""
|
2018-04-17 20:18:25 +02:00
|
|
|
LOG.debug("---===### Starting FanartSync ###===---")
|
2018-07-27 13:38:41 +02:00
|
|
|
while not self.stopped():
|
2017-04-02 17:02:41 +02:00
|
|
|
# In the event the server goes offline
|
2018-07-27 13:38:41 +02:00
|
|
|
while self.suspended():
|
2017-04-02 17:02:41 +02:00
|
|
|
# Set in service.py
|
2018-07-27 13:38:41 +02:00
|
|
|
if self.stopped():
|
2017-04-02 17:02:41 +02:00
|
|
|
# Abort was requested while waiting. We should exit
|
2018-07-27 13:38:41 +02:00
|
|
|
LOG.debug("---===### Stopped FanartSync ###===---")
|
2017-04-02 17:02:41 +02:00
|
|
|
return
|
2018-05-30 07:53:30 +02:00
|
|
|
xbmc.sleep(1000)
|
2017-04-02 17:02:41 +02:00
|
|
|
# grabs Plex item from queue
|
|
|
|
try:
|
2018-07-27 13:38:41 +02:00
|
|
|
item = self.queue.get(block=False)
|
2017-04-02 17:02:41 +02:00
|
|
|
except Empty:
|
2018-05-30 07:53:30 +02:00
|
|
|
xbmc.sleep(200)
|
2017-04-02 17:02:41 +02:00
|
|
|
continue
|
2018-07-27 13:38:41 +02:00
|
|
|
self.queue.task_done()
|
2018-06-21 19:24:37 +02:00
|
|
|
if isinstance(item, artwork.ArtworkSyncMessage):
|
2018-05-15 20:46:16 +02:00
|
|
|
if state.IMAGE_SYNC_NOTIFICATIONS:
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.dialog('notification',
|
|
|
|
heading=utils.lang(29999),
|
|
|
|
message=item.message,
|
|
|
|
icon='{plex}',
|
|
|
|
sound=False)
|
2018-05-13 15:22:03 +02:00
|
|
|
continue
|
2018-04-17 20:18:25 +02:00
|
|
|
LOG.debug('Get additional fanart for Plex id %s', item['plex_id'])
|
2018-07-27 13:38:41 +02:00
|
|
|
_process(item)
|
2018-04-17 20:18:25 +02:00
|
|
|
LOG.debug("---===### Stopped FanartSync ###===---")
|
2018-07-27 13:38:41 +02:00
|
|
|
|
|
|
|
|
|
|
|
def _process(item):
|
|
|
|
done = False
|
|
|
|
try:
|
|
|
|
artworks = None
|
|
|
|
with plexdb.Get_Plex_DB() as plex_db:
|
|
|
|
db_item = plex_db.getItem_byId(item['plex_id'])
|
|
|
|
try:
|
|
|
|
kodi_id = db_item[0]
|
|
|
|
kodi_type = db_item[4]
|
|
|
|
except TypeError:
|
|
|
|
LOG.error('Could not get Kodi id for plex id %s, abort getfanart',
|
|
|
|
item['plex_id'])
|
|
|
|
return
|
|
|
|
if item['refresh'] is False:
|
|
|
|
with kodidb.GetKodiDB('video') as kodi_db:
|
|
|
|
artworks = kodi_db.get_art(kodi_id, kodi_type)
|
|
|
|
# Check if we even need to get additional art
|
|
|
|
for key in v.ALL_KODI_ARTWORK:
|
|
|
|
if key not in artworks:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
LOG.debug('Already got all fanart for Plex id %s',
|
|
|
|
item['plex_id'])
|
|
|
|
done = True
|
|
|
|
return
|
|
|
|
xml = PF.GetPlexMetadata(item['plex_id'])
|
|
|
|
if xml is None:
|
|
|
|
LOG.error('Could not get metadata for %s. Skipping that item '
|
|
|
|
'for now', item['plex_id'])
|
|
|
|
return
|
|
|
|
elif xml == 401:
|
|
|
|
LOG.error('HTTP 401 returned by PMS. Too much strain? '
|
|
|
|
'Cancelling sync for now')
|
|
|
|
return
|
|
|
|
api = API(xml[0])
|
|
|
|
if artworks is None:
|
|
|
|
artworks = api.artwork()
|
|
|
|
# Get additional missing artwork from fanart artwork sites
|
|
|
|
artworks = api.fanart_artwork(artworks)
|
|
|
|
with getattr(itemtypes,
|
|
|
|
v.ITEMTYPE_FROM_PLEXTYPE[item['plex_type']])() as itm:
|
|
|
|
itm.set_fanart(artworks, kodi_id, kodi_type)
|
|
|
|
# Additional fanart for sets/collections
|
|
|
|
if api.plex_type() == v.PLEX_TYPE_MOVIE:
|
|
|
|
for _, setname in api.collection_list():
|
|
|
|
LOG.debug('Getting artwork for movie set %s', setname)
|
|
|
|
with kodidb.GetKodiDB('video') as kodi_db:
|
|
|
|
setid = kodi_db.create_collection(setname)
|
|
|
|
external_set_artwork = api.set_artwork()
|
|
|
|
if (external_set_artwork and
|
|
|
|
utils.settings('PreferKodiCollectionArt') == 'false'):
|
|
|
|
kodi_artwork = api.artwork(kodi_id=setid,
|
|
|
|
kodi_type=v.KODI_TYPE_SET)
|
|
|
|
for art in kodi_artwork:
|
|
|
|
if art in external_set_artwork:
|
|
|
|
del external_set_artwork[art]
|
|
|
|
with itemtypes.Movies() as movie_db:
|
|
|
|
movie_db.artwork.modify_artwork(external_set_artwork,
|
|
|
|
setid,
|
|
|
|
v.KODI_TYPE_SET,
|
|
|
|
movie_db.kodicursor)
|
|
|
|
done = True
|
|
|
|
finally:
|
|
|
|
if done is True:
|
|
|
|
LOG.debug('Done getting fanart for Plex id %s', item['plex_id'])
|
|
|
|
with plexdb.Get_Plex_DB() as plex_db:
|
|
|
|
plex_db.set_fanart_synched(item['plex_id'])
|