2018-10-20 14:49:04 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
from __future__ import absolute_import, division, unicode_literals
|
|
|
|
from logging import getLogger
|
|
|
|
|
2018-11-07 10:37:32 +01:00
|
|
|
from .get_metadata import GetMetadataTask, reset_collections
|
2018-11-25 17:21:32 +01:00
|
|
|
from .process_metadata import InitNewSection, UpdateLastSync, ProcessMetadata, \
|
|
|
|
DeleteItem
|
2018-11-09 11:19:32 +01:00
|
|
|
from . import common, sections
|
2018-11-18 14:59:17 +01:00
|
|
|
from .. import utils, timing, backgroundthread, variables as v, app
|
2018-10-20 14:49:04 +02:00
|
|
|
from .. import plex_functions as PF, itemtypes
|
2018-10-23 13:54:09 +02:00
|
|
|
from ..plex_db import PlexDB
|
2018-10-20 14:49:04 +02:00
|
|
|
|
2018-10-23 13:54:09 +02:00
|
|
|
if (v.PLATFORM != 'Microsoft UWP' and
|
|
|
|
utils.settings('enablePlaylistSync') == 'true'):
|
|
|
|
# Xbox cannot use watchdog, a dependency for PKC playlist features
|
|
|
|
from .. import playlists
|
|
|
|
PLAYLIST_SYNC_ENABLED = True
|
|
|
|
else:
|
|
|
|
PLAYLIST_SYNC_ENABLED = False
|
2018-10-20 14:49:04 +02:00
|
|
|
|
2018-11-01 15:43:43 +01:00
|
|
|
LOG = getLogger('PLEX.sync.full_sync')
|
2018-10-20 14:49:04 +02:00
|
|
|
|
|
|
|
|
2018-11-05 13:52:31 +01:00
|
|
|
class FullSync(common.libsync_mixin):
|
2018-10-24 07:08:32 +02:00
|
|
|
def __init__(self, repair, callback, show_dialog):
|
2018-10-20 14:49:04 +02:00
|
|
|
"""
|
|
|
|
repair=True: force sync EVERY item
|
|
|
|
"""
|
2018-10-24 18:08:00 +02:00
|
|
|
self._canceled = False
|
2018-10-20 14:49:04 +02:00
|
|
|
self.repair = repair
|
|
|
|
self.callback = callback
|
2018-10-24 07:08:32 +02:00
|
|
|
self.show_dialog = show_dialog
|
2018-10-21 12:03:21 +02:00
|
|
|
self.queue = None
|
|
|
|
self.process_thread = None
|
2018-11-04 16:53:42 +01:00
|
|
|
self.current_sync = None
|
2018-10-28 16:14:37 +01:00
|
|
|
self.plexdb = None
|
2018-10-21 16:56:13 +02:00
|
|
|
self.plex_type = None
|
2018-11-04 16:53:42 +01:00
|
|
|
self.section_type = None
|
2018-10-21 18:32:11 +02:00
|
|
|
self.processing_thread = None
|
2018-11-25 20:31:40 +01:00
|
|
|
self.section_initiated = False
|
2018-10-25 13:27:12 +02:00
|
|
|
self.install_sync_done = utils.settings('SyncInstallRunDone') == 'true'
|
2018-11-06 11:17:21 +01:00
|
|
|
self.threader = backgroundthread.ThreaderManager(
|
|
|
|
worker=backgroundthread.NonstoppingBackgroundWorker)
|
2018-10-20 14:49:04 +02:00
|
|
|
super(FullSync, self).__init__()
|
|
|
|
|
2018-10-21 16:56:13 +02:00
|
|
|
def process_item(self, xml_item):
|
2018-10-20 14:49:04 +02:00
|
|
|
"""
|
|
|
|
Processes a single library item
|
|
|
|
"""
|
2018-10-25 15:57:12 +02:00
|
|
|
plex_id = int(xml_item.get('ratingKey'))
|
2018-11-04 16:53:42 +01:00
|
|
|
if not self.repair and self.plexdb.checksum(plex_id, self.plex_type) == \
|
|
|
|
int('%s%s' % (plex_id,
|
2018-11-07 07:45:19 +01:00
|
|
|
xml_item.get('updatedAt',
|
|
|
|
xml_item.get('addedAt', 1541572987)))):
|
2018-11-09 11:19:32 +01:00
|
|
|
# Already got EXACTLY this item in our DB. BUT need to collect all
|
|
|
|
# DB updates within the same thread
|
|
|
|
self.queue.put(UpdateLastSync(plex_id))
|
2018-11-04 16:53:42 +01:00
|
|
|
return
|
2018-10-25 17:50:59 +02:00
|
|
|
task = GetMetadataTask()
|
2018-11-07 10:37:32 +01:00
|
|
|
task.setup(self.queue, plex_id, self.plex_type, self.get_children)
|
2018-11-06 11:17:21 +01:00
|
|
|
self.threader.addTask(task)
|
2018-10-20 14:49:04 +02:00
|
|
|
|
2018-10-21 16:56:13 +02:00
|
|
|
def process_delete(self):
|
|
|
|
"""
|
2018-11-04 16:53:42 +01:00
|
|
|
Removes all the items that have NOT been updated (last_sync timestamp
|
|
|
|
is different)
|
2018-10-21 16:56:13 +02:00
|
|
|
"""
|
2018-11-25 17:21:32 +01:00
|
|
|
for plex_id in self.plexdb.plex_id_by_last_sync(self.plex_type,
|
|
|
|
self.current_sync):
|
|
|
|
if self.isCanceled():
|
|
|
|
return
|
|
|
|
self.queue.put(DeleteItem(plex_id))
|
2018-10-21 16:56:13 +02:00
|
|
|
|
2018-11-04 16:53:42 +01:00
|
|
|
@utils.log_time
|
|
|
|
def process_playstate(self, iterator):
|
|
|
|
"""
|
|
|
|
Updates the playstate (resume point, number of views, userrating, last
|
|
|
|
played date, etc.) for all elements in the (xml-)iterator
|
|
|
|
"""
|
|
|
|
with self.context(self.current_sync) as c:
|
|
|
|
for xml_item in iterator:
|
|
|
|
if self.isCanceled():
|
|
|
|
return False
|
|
|
|
c.update_userdata(xml_item, self.plex_type)
|
|
|
|
|
2018-10-21 12:03:21 +02:00
|
|
|
@utils.log_time
|
2018-10-21 16:56:13 +02:00
|
|
|
def process_kind(self):
|
2018-10-20 14:49:04 +02:00
|
|
|
"""
|
|
|
|
"""
|
2018-11-04 16:53:42 +01:00
|
|
|
successful = True
|
|
|
|
LOG.debug('Start processing %ss', self.section_type)
|
2018-10-25 13:25:25 +02:00
|
|
|
sects = (x for x in sections.SECTIONS
|
2018-11-04 16:53:42 +01:00
|
|
|
if x['plex_type'] == self.section_type)
|
2018-10-25 13:25:25 +02:00
|
|
|
for section in sects:
|
2018-10-20 14:49:04 +02:00
|
|
|
LOG.debug('Processing library section %s', section)
|
|
|
|
if self.isCanceled():
|
|
|
|
return False
|
|
|
|
if not self.install_sync_done:
|
2018-11-18 14:59:17 +01:00
|
|
|
app.SYNC.path_verified = False
|
2018-10-20 14:49:04 +02:00
|
|
|
try:
|
2018-11-04 16:53:42 +01:00
|
|
|
# Sync new, updated and deleted items
|
2018-11-25 20:31:40 +01:00
|
|
|
self.section_initiated = True
|
2018-10-25 15:54:22 +02:00
|
|
|
iterator = PF.SectionItems(section['section_id'],
|
2018-11-04 16:53:42 +01:00
|
|
|
plex_type=self.plex_type)
|
2018-10-21 12:03:21 +02:00
|
|
|
# Tell the processing thread about this new section
|
2018-11-09 11:19:32 +01:00
|
|
|
queue_info = InitNewSection(self.context,
|
2018-11-11 17:48:11 +01:00
|
|
|
iterator.total,
|
2018-11-09 11:19:32 +01:00
|
|
|
iterator.get('librarySectionTitle'),
|
|
|
|
section['section_id'],
|
|
|
|
self.plex_type)
|
2018-10-21 12:03:21 +02:00
|
|
|
self.queue.put(queue_info)
|
2018-11-20 19:23:42 +01:00
|
|
|
for xml_item in iterator:
|
|
|
|
if self.isCanceled():
|
|
|
|
return False
|
|
|
|
self.process_item(xml_item)
|
2018-10-20 14:49:04 +02:00
|
|
|
except RuntimeError:
|
|
|
|
LOG.error('Could not entirely process section %s', section)
|
2018-11-04 16:53:42 +01:00
|
|
|
successful = False
|
2018-10-21 12:03:21 +02:00
|
|
|
continue
|
2018-11-09 08:44:05 +01:00
|
|
|
LOG.debug('Waiting for download threads to finish')
|
|
|
|
while self.threader.threader.working():
|
2018-11-20 16:58:25 +01:00
|
|
|
app.APP.monitor.waitForAbort(0.1)
|
2018-11-04 16:53:42 +01:00
|
|
|
LOG.debug('Waiting for processing thread to finish section')
|
2018-10-23 13:54:09 +02:00
|
|
|
self.queue.join()
|
2018-11-07 10:37:32 +01:00
|
|
|
reset_collections()
|
2018-11-04 16:53:42 +01:00
|
|
|
try:
|
|
|
|
# Sync playstate of every item
|
|
|
|
iterator = PF.SectionItems(section['section_id'],
|
|
|
|
plex_type=self.plex_type)
|
|
|
|
# Tell the processing thread that we're syncing playstate
|
2018-11-09 11:19:32 +01:00
|
|
|
queue_info = InitNewSection(self.context,
|
2018-11-11 17:48:11 +01:00
|
|
|
iterator.total,
|
2018-11-09 11:19:32 +01:00
|
|
|
iterator.get('librarySectionTitle'),
|
|
|
|
section['section_id'],
|
|
|
|
self.plex_type)
|
2018-11-04 16:53:42 +01:00
|
|
|
self.queue.put(queue_info)
|
2018-11-05 12:19:08 +01:00
|
|
|
# Ensure that the DB connection is closed to commit the
|
|
|
|
# changes above - avoids "Item not yet synced" error
|
|
|
|
self.queue.join()
|
2018-11-21 20:20:06 +01:00
|
|
|
if self.plex_type != v.PLEX_TYPE_ARTIST:
|
|
|
|
self.process_playstate(iterator)
|
2018-11-04 16:53:42 +01:00
|
|
|
except RuntimeError:
|
|
|
|
LOG.error('Could not process playstate for section %s', section)
|
|
|
|
successful = False
|
|
|
|
continue
|
|
|
|
LOG.debug('Done processing playstate for section')
|
2018-10-21 16:56:13 +02:00
|
|
|
|
|
|
|
LOG.debug('Finished processing %ss', self.plex_type)
|
2018-11-04 16:53:42 +01:00
|
|
|
return successful
|
2018-10-20 14:49:04 +02:00
|
|
|
|
2018-10-21 16:56:13 +02:00
|
|
|
def full_library_sync(self):
|
2018-10-20 14:49:04 +02:00
|
|
|
"""
|
|
|
|
"""
|
2018-10-21 12:03:21 +02:00
|
|
|
kinds = [
|
2018-11-04 16:53:42 +01:00
|
|
|
(v.PLEX_TYPE_MOVIE, v.PLEX_TYPE_MOVIE, itemtypes.Movie, False),
|
|
|
|
(v.PLEX_TYPE_SHOW, v.PLEX_TYPE_SHOW, itemtypes.Show, False),
|
|
|
|
(v.PLEX_TYPE_SEASON, v.PLEX_TYPE_SHOW, itemtypes.Season, False),
|
|
|
|
(v.PLEX_TYPE_EPISODE, v.PLEX_TYPE_SHOW, itemtypes.Episode, False)
|
2018-10-21 12:03:21 +02:00
|
|
|
]
|
2018-11-18 14:59:17 +01:00
|
|
|
if app.SYNC.enable_music:
|
2018-10-25 13:22:34 +02:00
|
|
|
kinds.extend([
|
2018-11-04 16:53:42 +01:00
|
|
|
(v.PLEX_TYPE_ARTIST, v.PLEX_TYPE_ARTIST, itemtypes.Artist, False),
|
|
|
|
(v.PLEX_TYPE_ALBUM, v.PLEX_TYPE_ARTIST, itemtypes.Album, True),
|
2018-11-03 18:47:51 +01:00
|
|
|
])
|
2018-10-28 16:14:37 +01:00
|
|
|
with PlexDB() as self.plexdb:
|
|
|
|
for kind in kinds:
|
2018-11-25 20:31:40 +01:00
|
|
|
self.section_initiated = False
|
2018-10-28 16:14:37 +01:00
|
|
|
# Setup our variables
|
|
|
|
self.plex_type = kind[0]
|
2018-11-04 16:53:42 +01:00
|
|
|
self.section_type = kind[1]
|
|
|
|
self.context = kind[2]
|
|
|
|
self.get_children = kind[3]
|
2018-10-28 16:14:37 +01:00
|
|
|
# Now do the heavy lifting
|
2018-10-23 13:54:09 +02:00
|
|
|
if self.isCanceled() or not self.process_kind():
|
|
|
|
return False
|
2018-11-04 16:53:42 +01:00
|
|
|
# Delete movies that are not on Plex anymore
|
2018-11-25 20:31:40 +01:00
|
|
|
if not self.section_initiated:
|
|
|
|
# Need to make sure that we're telling about this section
|
|
|
|
queue_info = InitNewSection(self.context,
|
|
|
|
0,
|
|
|
|
'',
|
|
|
|
'',
|
|
|
|
self.plex_type)
|
|
|
|
self.queue.put(queue_info)
|
2018-11-04 16:53:42 +01:00
|
|
|
self.process_delete()
|
2018-10-20 14:49:04 +02:00
|
|
|
return True
|
|
|
|
|
|
|
|
@utils.log_time
|
|
|
|
def run(self):
|
2018-10-21 18:32:11 +02:00
|
|
|
if self.isCanceled():
|
|
|
|
return
|
2018-10-23 13:54:09 +02:00
|
|
|
successful = False
|
2018-11-18 14:59:17 +01:00
|
|
|
self.current_sync = timing.unix_timestamp()
|
2018-10-24 07:08:32 +02:00
|
|
|
# Delete playlist and video node files from Kodi
|
|
|
|
utils.delete_playlists()
|
|
|
|
utils.delete_nodes()
|
|
|
|
# Get latest Plex libraries and build playlist and video node files
|
2018-10-21 18:32:11 +02:00
|
|
|
if not sections.sync_from_pms():
|
|
|
|
return
|
2018-10-20 14:49:04 +02:00
|
|
|
try:
|
2018-10-21 18:32:11 +02:00
|
|
|
# Fire up our single processing thread
|
2018-11-09 11:19:32 +01:00
|
|
|
self.queue = backgroundthread.Queue.Queue(maxsize=600)
|
|
|
|
self.processing_thread = ProcessMetadata(self.queue,
|
|
|
|
self.current_sync,
|
|
|
|
self.show_dialog)
|
2018-10-21 18:32:11 +02:00
|
|
|
self.processing_thread.start()
|
2018-10-23 13:54:09 +02:00
|
|
|
|
|
|
|
# Actual syncing - do only new items first
|
2018-11-04 16:53:42 +01:00
|
|
|
LOG.info('Running full_library_sync with repair=%s',
|
2018-10-23 13:54:09 +02:00
|
|
|
self.repair)
|
2018-10-20 14:49:04 +02:00
|
|
|
if not self.full_library_sync():
|
|
|
|
return
|
2018-11-04 16:53:42 +01:00
|
|
|
# Tell the processing thread to exit with one last element None
|
|
|
|
self.queue.put(None)
|
2018-10-20 14:49:04 +02:00
|
|
|
if self.isCanceled():
|
|
|
|
return
|
|
|
|
if PLAYLIST_SYNC_ENABLED and not playlists.full_sync():
|
|
|
|
return
|
|
|
|
successful = True
|
|
|
|
except:
|
|
|
|
utils.ERROR(txt='full_sync.py crashed', notify=True)
|
|
|
|
finally:
|
2018-11-04 16:53:42 +01:00
|
|
|
# This will block until the processing thread really exits
|
2018-10-21 18:32:11 +02:00
|
|
|
LOG.debug('Waiting for processing thread to exit')
|
|
|
|
self.processing_thread.join()
|
2018-11-05 13:53:57 +01:00
|
|
|
common.update_kodi_library(video=True, music=True)
|
2018-11-06 11:17:21 +01:00
|
|
|
self.threader.shutdown()
|
2018-10-24 07:08:32 +02:00
|
|
|
if self.callback:
|
|
|
|
self.callback(successful)
|
2018-10-21 18:32:11 +02:00
|
|
|
LOG.info('Done full_sync')
|
2018-10-20 14:49:04 +02:00
|
|
|
|
|
|
|
|
2018-10-24 07:08:32 +02:00
|
|
|
def start(show_dialog, repair=False, callback=None):
|
2018-10-20 14:49:04 +02:00
|
|
|
"""
|
|
|
|
"""
|
2018-11-05 13:52:31 +01:00
|
|
|
# FullSync(repair, callback, show_dialog).start()
|
|
|
|
FullSync(repair, callback, show_dialog).run()
|