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
|
2018-10-20 14:49:04 +02:00
|
|
|
import xbmcgui
|
2017-04-02 17:02:41 +02:00
|
|
|
|
2018-11-09 09:08:04 +01:00
|
|
|
from cProfile import Profile
|
|
|
|
from pstats import Stats
|
|
|
|
from StringIO import StringIO
|
|
|
|
|
2018-10-20 14:49:04 +02:00
|
|
|
from . import common
|
2018-11-09 15:05:17 +01:00
|
|
|
from .. import backgroundthread, utils, variables as v
|
2017-04-02 17:02:41 +02:00
|
|
|
|
2018-11-01 15:43:43 +01:00
|
|
|
LOG = getLogger('PLEX.sync.process_metadata')
|
2017-04-02 17:02:41 +02:00
|
|
|
|
|
|
|
|
2018-10-21 12:03:21 +02:00
|
|
|
class InitNewSection(object):
|
|
|
|
"""
|
|
|
|
Throw this into the queue used for ProcessMetadata to tell it which
|
|
|
|
Plex library section we're looking at
|
|
|
|
|
|
|
|
context: itemtypes.Movie, itemtypes.Episode, etc.
|
|
|
|
"""
|
|
|
|
def __init__(self, context, total_number_of_items, section_name,
|
2018-11-09 09:10:22 +01:00
|
|
|
section_id, plex_type):
|
2018-10-21 12:03:21 +02:00
|
|
|
self.context = context
|
|
|
|
self.total = total_number_of_items
|
|
|
|
self.name = section_name
|
|
|
|
self.id = section_id
|
2018-11-09 09:10:22 +01:00
|
|
|
self.plex_type = plex_type
|
2018-10-21 12:03:21 +02:00
|
|
|
|
|
|
|
|
2018-11-09 11:19:32 +01:00
|
|
|
class UpdateLastSync(object):
|
|
|
|
def __init__(self, plex_id):
|
|
|
|
self.plex_id = plex_id
|
|
|
|
|
|
|
|
|
2018-11-25 17:21:32 +01:00
|
|
|
class DeleteItem(object):
|
|
|
|
def __init__(self, plex_id):
|
|
|
|
self.plex_id = plex_id
|
|
|
|
|
|
|
|
|
2018-10-20 14:49:04 +02:00
|
|
|
class ProcessMetadata(backgroundthread.KillableThread, common.libsync_mixin):
|
2017-04-02 17:02:41 +02:00
|
|
|
"""
|
|
|
|
Not yet implemented for more than 1 thread - if ever. Only to be called by
|
|
|
|
ONE thread!
|
|
|
|
Processes the XML metadata in the queue
|
|
|
|
|
|
|
|
Input:
|
|
|
|
queue: Queue.Queue() object that you'll need to fill up with
|
|
|
|
the downloaded XML eTree objects
|
2018-04-17 20:18:25 +02:00
|
|
|
item_class: as used to call functions in itemtypes.py e.g. 'Movies' =>
|
2017-04-02 17:02:41 +02:00
|
|
|
itemtypes.Movies()
|
|
|
|
"""
|
2018-10-24 07:08:32 +02:00
|
|
|
def __init__(self, queue, last_sync, show_dialog):
|
2018-10-24 18:08:00 +02:00
|
|
|
self._canceled = False
|
2017-04-02 17:02:41 +02:00
|
|
|
self.queue = queue
|
2018-10-21 12:03:21 +02:00
|
|
|
self.last_sync = last_sync
|
2018-10-24 07:08:32 +02:00
|
|
|
self.show_dialog = show_dialog
|
2018-10-21 12:03:21 +02:00
|
|
|
self.total = 0
|
2018-11-09 09:05:14 +01:00
|
|
|
self.current = 1
|
2018-11-09 11:19:32 +01:00
|
|
|
self.processed = 0
|
|
|
|
self.title = ''
|
2018-10-21 12:03:21 +02:00
|
|
|
self.section_name = None
|
2018-10-24 07:08:32 +02:00
|
|
|
self.dialog = None
|
2018-10-20 14:49:04 +02:00
|
|
|
super(ProcessMetadata, self).__init__()
|
2017-04-02 17:02:41 +02:00
|
|
|
|
2018-11-03 17:18:25 +01:00
|
|
|
def update_progressbar(self):
|
2018-10-24 07:08:32 +02:00
|
|
|
if self.show_dialog:
|
|
|
|
try:
|
|
|
|
progress = int(float(self.current) / float(self.total) * 100.0)
|
|
|
|
except ZeroDivisionError:
|
|
|
|
progress = 0
|
|
|
|
self.dialog.update(progress,
|
2018-11-11 17:48:11 +01:00
|
|
|
'%s (%s)' % (self.section_name, self.section_type_text),
|
2018-11-12 16:37:56 +01:00
|
|
|
'%s/%s %s'
|
2018-11-09 09:05:14 +01:00
|
|
|
% (self.current, self.total, self.title))
|
2017-04-02 17:02:41 +02:00
|
|
|
|
|
|
|
def run(self):
|
2018-04-17 20:18:25 +02:00
|
|
|
LOG.debug('Processing thread started')
|
2018-11-25 17:21:32 +01:00
|
|
|
if self.show_dialog:
|
|
|
|
self.dialog = xbmcgui.DialogProgressBG()
|
|
|
|
self.dialog.create(utils.lang(39714))
|
2018-10-21 12:03:21 +02:00
|
|
|
try:
|
2018-11-25 17:21:32 +01:00
|
|
|
self._run()
|
|
|
|
except:
|
|
|
|
utils.ERROR(notify=True, cancel_sync=True)
|
2018-10-21 12:03:21 +02:00
|
|
|
finally:
|
2018-10-24 07:08:32 +02:00
|
|
|
if self.dialog:
|
|
|
|
self.dialog.close()
|
2018-11-09 08:56:57 +01:00
|
|
|
while not self.queue.empty():
|
|
|
|
# We need to empty the queue to let full_sync finish join()
|
|
|
|
self.queue.get()
|
|
|
|
self.queue.task_done()
|
2018-10-21 12:03:21 +02:00
|
|
|
LOG.debug('Processing thread terminated')
|
2018-11-25 17:21:32 +01:00
|
|
|
|
|
|
|
def _run(self):
|
|
|
|
"""
|
|
|
|
Do the work
|
|
|
|
"""
|
|
|
|
# Init with the very first library section. This will block!
|
|
|
|
section = self.queue.get()
|
|
|
|
self.queue.task_done()
|
|
|
|
if section is None:
|
|
|
|
return
|
|
|
|
while not self.isCanceled():
|
|
|
|
if section is None:
|
|
|
|
break
|
|
|
|
LOG.debug('Start processing section %s (%ss)',
|
|
|
|
section.name, section.plex_type)
|
|
|
|
self.current = 1
|
|
|
|
self.processed = 0
|
|
|
|
self.total = section.total
|
|
|
|
self.section_name = section.name
|
|
|
|
self.section_type_text = utils.lang(
|
|
|
|
v.TRANSLATION_FROM_PLEXTYPE[section.plex_type])
|
|
|
|
profile = Profile()
|
|
|
|
profile.enable()
|
|
|
|
with section.context(self.last_sync) as context:
|
|
|
|
while not self.isCanceled():
|
|
|
|
# grabs item from queue. This will block!
|
|
|
|
item = self.queue.get()
|
|
|
|
if isinstance(item, dict):
|
|
|
|
context.add_update(item['xml'][0],
|
|
|
|
section_name=section.name,
|
|
|
|
section_id=section.id,
|
|
|
|
children=item['children'])
|
|
|
|
self.title = item['xml'][0].get('title')
|
|
|
|
self.processed += 1
|
|
|
|
elif isinstance(item, UpdateLastSync):
|
|
|
|
context.plexdb.update_last_sync(item.plex_id,
|
|
|
|
section.plex_type,
|
|
|
|
self.last_sync)
|
|
|
|
elif isinstance(item, InitNewSection) or item is None:
|
|
|
|
section = item
|
|
|
|
self.queue.task_done()
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
context.remove(item.plex_id, plex_type=section.plex_type)
|
|
|
|
self.update_progressbar()
|
|
|
|
self.current += 1
|
|
|
|
if self.processed == 500:
|
|
|
|
self.processed = 0
|
|
|
|
context.commit()
|
|
|
|
self.queue.task_done()
|
|
|
|
profile.disable()
|
|
|
|
string_io = StringIO()
|
|
|
|
stats = Stats(profile, stream=string_io).sort_stats('cumulative')
|
|
|
|
stats.print_stats()
|
|
|
|
LOG.info('cProfile result: ')
|
|
|
|
LOG.info(string_io.getvalue())
|