jellyfin-kodi/resources/lib/library.py

866 lines
28 KiB
Python
Raw Normal View History

2018-09-06 08:36:32 +00:00
# -*- coding: utf-8 -*-
##################################################################################################
import logging
import Queue
import threading
import sys
from datetime import datetime, timedelta
import xbmc
2018-10-04 03:36:57 +00:00
import xbmcgui
2018-09-06 08:36:32 +00:00
from objects import Movies, TVShows, MusicVideos, Music
from database import Database, emby_db, get_sync, save_sync
from full_sync import FullSync
from views import Views
from downloader import GetItemWorker
2018-10-04 05:41:31 +00:00
from helper import _, api, stop, settings, window, dialog, event, progress, LibraryException
from helper.utils import split_list, set_screensaver, get_screensaver
2018-09-06 08:36:32 +00:00
from emby import Emby
##################################################################################################
LOG = logging.getLogger("EMBY."+__name__)
2018-10-01 00:35:31 +00:00
LIMIT = min(int(settings('limitIndex') or 50), 50)
DTHREADS = int(settings('limitThreads') or 3)
2018-09-06 08:36:32 +00:00
MEDIA = {
'Movie': Movies,
'BoxSet': Movies,
'MusicVideo': MusicVideos,
'Series': TVShows,
'Season': TVShows,
'Episode': TVShows,
'MusicAlbum': Music,
'MusicArtist': Music,
'AlbumArtist': Music,
'Audio': Music
}
##################################################################################################
class Library(threading.Thread):
started = False
stop_thread = False
suspend = False
pending_refresh = False
2018-10-04 03:36:57 +00:00
screensaver = None
progress_updates = None
total_updates = 0
2018-09-06 08:36:32 +00:00
def __init__(self, monitor):
self.direct_path = settings('useDirectPaths') == "1"
2018-10-04 03:36:57 +00:00
self.progress_display = int(settings('syncProgress') or 50)
2018-09-06 08:36:32 +00:00
self.monitor = monitor
2018-10-04 05:41:31 +00:00
self.player = monitor.monitor.player
2018-09-06 08:36:32 +00:00
self.server = Emby()
self.updated_queue = Queue.Queue()
self.userdata_queue = Queue.Queue()
self.removed_queue = Queue.Queue()
self.updated_output = self.__new_queues__()
self.userdata_output = self.__new_queues__()
self.removed_output = self.__new_queues__()
2018-10-04 05:41:31 +00:00
self.notify_output = Queue.Queue()
2018-09-06 08:36:32 +00:00
self.emby_threads = []
self.download_threads = []
2018-10-04 05:41:31 +00:00
self.notify_threads = []
2018-09-06 08:36:32 +00:00
self.writer_threads = {'updated': [], 'userdata': [], 'removed': []}
self.database_lock = threading.Lock()
self.music_database_lock = threading.Lock()
threading.Thread.__init__(self)
def __new_queues__(self):
return {
'Movie': Queue.Queue(),
'BoxSet': Queue.Queue(),
'MusicVideo': Queue.Queue(),
'Series': Queue.Queue(),
'Season': Queue.Queue(),
'Episode': Queue.Queue(),
'MusicAlbum': Queue.Queue(),
'MusicArtist': Queue.Queue(),
'AlbumArtist': Queue.Queue(),
'Audio': Queue.Queue()
}
def run(self):
LOG.warn("--->[ library ]")
if not self.startup():
self.stop_client()
window('emby_startup.bool', True)
2018-09-06 08:36:32 +00:00
while not self.stop_thread:
try:
self.service()
except LibraryException as error:
break
except Exception as error:
LOG.exception(error)
break
if self.monitor.waitForAbort(2):
break
LOG.warn("---<[ library ]")
2019-01-10 23:38:57 +00:00
def test_databases(self):
''' Open the databases to test if the file exists.
'''
with Database('video') as kodidb:
with Database('music') as musicdb:
if kodidb.discovered or musicdb.discovered:
if kodidb.path != settings('DiscoveredDatabase'):
LOG.info("Newly discovered database: %s", kodidb.path)
settings('DiscoveredDatabase', kodidb.path)
self.monitor.settings['enable_db_discovery'] = False
settings('AskDiscoverDatabase.bool', False)
return False
else:
settings('DiscoveredDatabase', "")
return True
2018-09-06 08:36:32 +00:00
@stop()
def service(self):
''' If error is encountered, it will rerun this function.
Start new "daemon threads" to process library updates.
(actual daemon thread is not supported in Kodi)
'''
for threads in (self.download_threads, self.writer_threads['updated'],
self.writer_threads['userdata'], self.writer_threads['removed']):
for thread in threads:
if thread.is_done:
threads.remove(thread)
2018-11-19 06:12:46 +00:00
if not self.player.isPlayingVideo() or settings('syncDuringPlay.bool') or xbmc.getCondVisibility('VideoPlayer.Content(livetv)'):
2018-10-04 03:36:57 +00:00
2018-11-17 09:46:22 +00:00
self.worker_downloads()
self.worker_sort()
self.worker_updates()
self.worker_userdata()
self.worker_remove()
self.worker_notify()
2018-10-04 03:36:57 +00:00
if self.pending_refresh:
2018-11-17 09:46:22 +00:00
if self.total_updates > self.progress_display:
queue_size = self.worker_queue_size()
2018-10-04 03:36:57 +00:00
if self.progress_updates is None:
2018-10-04 03:36:57 +00:00
self.progress_updates = xbmcgui.DialogProgressBG()
self.progress_updates.create(_('addon_name'), _(33178))
self.progress_updates.update(int((float(self.total_updates - queue_size) / float(self.total_updates))*100), message="%s: %s" % (_(33178), queue_size))
elif queue_size:
self.progress_updates.update(int((float(self.total_updates - queue_size) / float(self.total_updates))*100), message="%s: %s" % (_(33178), queue_size))
2018-10-04 03:36:57 +00:00
else:
self.progress_updates.update(int((float(self.total_updates - queue_size) / float(self.total_updates))*100), message=_(33178))
2018-10-04 03:36:57 +00:00
if not settings('dbSyncScreensaver.bool') and self.screensaver is None:
xbmc.executebuiltin('InhibitIdleShutdown(true)')
self.screensaver = get_screensaver()
set_screensaver(value="")
if (self.pending_refresh and not self.download_threads and not self.writer_threads['updated'] and
not self.writer_threads['userdata'] and not self.writer_threads['removed']):
self.pending_refresh = False
self.save_last_sync()
self.total_updates = 0
if self.progress_updates:
self.progress_updates.close()
self.progress_updates = None
if not settings('dbSyncScreensaver.bool') and self.screensaver is not None:
xbmc.executebuiltin('InhibitIdleShutdown(false)')
set_screensaver(value=self.screensaver)
self.screensaver = None
if xbmc.getCondVisibility('Container.Content(musicvideos)') or xbmc.getCondVisibility('Window.IsMedia'): # Prevent cursor from moving
xbmc.executebuiltin('Container.Refresh')
else: # Update widgets
xbmc.executebuiltin('UpdateLibrary(video)')
def stop_client(self):
self.stop_thread = True
def worker_queue_size(self):
''' Get how many items are queued up for worker threads.
2018-10-04 03:36:57 +00:00
'''
total = 0
for queues in self.updated_output:
total += self.updated_output[queues].qsize()
for queues in self.userdata_output:
total += self.userdata_output[queues].qsize()
for queues in self.removed_output:
total += self.removed_output[queues].qsize()
return total
def worker_downloads(self):
''' Get items from emby and place them in the appropriate queues.
'''
2018-09-06 08:36:32 +00:00
for queue in ((self.updated_queue, self.updated_output), (self.userdata_queue, self.userdata_output)):
2018-10-01 00:35:31 +00:00
if queue[0].qsize() and len(self.download_threads) < DTHREADS:
2018-09-06 08:36:32 +00:00
new_thread = GetItemWorker(self.server, queue[0], queue[1])
new_thread.start()
LOG.info("-->[ q:download/%s ]", id(new_thread))
def worker_sort(self):
2018-10-04 03:36:57 +00:00
''' Get items based on the local emby database and place item in appropriate queues.
'''
2018-09-06 08:36:32 +00:00
if self.removed_queue.qsize() and len(self.emby_threads) < 2:
new_thread = SortWorker(self.removed_queue, self.removed_output)
new_thread.start()
LOG.info("-->[ q:sort/%s ]", id(new_thread))
2018-10-04 03:36:57 +00:00
def worker_updates(self):
''' Update items in the Kodi database.
'''
2018-09-06 08:36:32 +00:00
for queues in self.updated_output:
queue = self.updated_output[queues]
2018-10-07 23:19:06 +00:00
if queue.qsize() and not len(self.writer_threads['updated']):
2018-09-06 08:36:32 +00:00
if queues in ('Audio', 'MusicArtist', 'AlbumArtist', 'MusicAlbum'):
2018-10-04 05:41:31 +00:00
new_thread = UpdatedWorker(queue, self.notify_output, self.music_database_lock, "music", self.server, self.direct_path)
2018-09-06 08:36:32 +00:00
else:
2018-10-04 05:41:31 +00:00
new_thread = UpdatedWorker(queue, self.notify_output, self.database_lock, "video", self.server, self.direct_path)
2018-09-06 08:36:32 +00:00
new_thread.start()
LOG.info("-->[ q:updated/%s/%s ]", queues, id(new_thread))
self.writer_threads['updated'].append(new_thread)
self.pending_refresh = True
2018-10-04 03:36:57 +00:00
def worker_userdata(self):
''' Update userdata in the Kodi database.
'''
2018-09-06 08:36:32 +00:00
for queues in self.userdata_output:
queue = self.userdata_output[queues]
2018-10-07 23:19:06 +00:00
if queue.qsize() and not len(self.writer_threads['userdata']):
2018-09-06 08:36:32 +00:00
if queues in ('Audio', 'MusicArtist', 'AlbumArtist', 'MusicAlbum'):
new_thread = UserDataWorker(queue, self.music_database_lock, "music", self.server, self.direct_path)
else:
new_thread = UserDataWorker(queue, self.database_lock, "video", self.server, self.direct_path)
new_thread.start()
LOG.info("-->[ q:userdata/%s/%s ]", queues, id(new_thread))
self.writer_threads['userdata'].append(new_thread)
self.pending_refresh = True
2018-10-04 03:36:57 +00:00
def worker_remove(self):
''' Remove items from the Kodi database.
'''
2018-09-06 08:36:32 +00:00
for queues in self.removed_output:
queue = self.removed_output[queues]
2018-10-07 23:19:06 +00:00
if queue.qsize() and not len(self.writer_threads['removed']):
2018-09-06 08:36:32 +00:00
if queues in ('Audio', 'MusicArtist', 'AlbumArtist', 'MusicAlbum'):
new_thread = RemovedWorker(queue, self.music_database_lock, "music", self.server, self.direct_path)
else:
new_thread = RemovedWorker(queue, self.database_lock, "video", self.server, self.direct_path)
new_thread.start()
LOG.info("-->[ q:removed/%s/%s ]", queues, id(new_thread))
self.writer_threads['removed'].append(new_thread)
self.pending_refresh = True
2018-10-04 05:41:31 +00:00
def worker_notify(self):
''' Notify the user of new additions.
'''
2018-10-07 23:19:06 +00:00
if self.notify_output.qsize() and not len(self.notify_threads):
2018-10-04 05:41:31 +00:00
new_thread = NotifyWorker(self.notify_output, self.player)
new_thread.start()
LOG.info("-->[ q:notify/%s ]", id(new_thread))
self.notify_threads.append(new_thread)
2018-09-06 08:36:32 +00:00
def startup(self):
2019-01-10 23:38:57 +00:00
''' Run at startup.
Check databases.
Check for the server plugin.
2018-09-06 08:36:32 +00:00
'''
2019-01-10 23:38:57 +00:00
if not self.test_databases():
if settings('AskDiscoverDatabase.bool'):
self.monitor.settings['enable_db_discovery'] = False
settings('AskDiscoverDatabase.bool', False)
result = dialog("yesno", heading="{emby}", line1=_(33189))
settings('DiscoverDatabase.bool', result == 1)
if not result:
LOG.info("Do not discover database again.")
return False
elif not settings('DiscoverDatabase.bool'):
LOG.info("Do not re-discover database again.")
return False
2018-09-06 08:36:32 +00:00
Views().get_views()
Views().get_nodes()
try:
if get_sync()['Libraries']:
try:
FullSync(self)
Views().get_nodes()
except Exception as error:
LOG.error(error)
elif not settings('SyncInstallRunDone.bool'):
FullSync(self)
Views().get_nodes()
2018-10-12 05:26:44 +00:00
return True
2018-09-06 08:36:32 +00:00
if settings('SyncInstallRunDone.bool'):
if settings('kodiCompanion.bool'):
for plugin in self.server['api'].get_plugins():
if plugin['Name'] in ("Emby.Kodi Sync Queue", "Kodi companion"):
if not self.fast_sync():
dialog("ok", heading="{emby}", line1=_(33128))
2018-09-06 08:36:32 +00:00
raise Exception("Failed to retrieve latest updates")
LOG.info("--<[ retrieve changes ]")
2018-09-06 08:36:32 +00:00
break
else:
raise LibraryException('CompanionMissing')
2018-09-06 08:36:32 +00:00
return True
2018-09-06 08:36:32 +00:00
except LibraryException as error:
LOG.error(error.status)
if error.status in 'SyncLibraryLater':
2018-09-06 08:36:32 +00:00
dialog("ok", heading="{emby}", line1=_(33129))
settings('SyncInstallRunDone.bool', True)
sync = get_sync()
sync['Libraries'] = []
save_sync(sync)
2018-09-06 08:36:32 +00:00
return True
elif error.status == 'CompanionMissing':
dialog("ok", heading="{emby}", line1=_(33099))
2018-09-10 04:16:44 +00:00
settings('kodiCompanion.bool', False)
2018-09-06 08:36:32 +00:00
return True
except Exception as error:
LOG.exception(error)
return False
def fast_sync(self):
''' Movie and userdata not provided by server yet.
'''
last_sync = settings('LastIncrementalSync')
filters = ["tvshows", "boxsets", "musicvideos", "music", "movies"]
sync = get_sync()
LOG.info("--[ retrieve changes ] %s", last_sync)
"""
for library in sync['Whitelist']:
data = self.server['api'].get_date_modified(last_sync, library.replace('Mixed:', ""), "Series,Episode,BoxSet,Movie,MusicVideo,MusicArtist,MusicAlbum,Audio")
[self.updated_output[query['Type']].put(query) for query in data['Items']]
"""
try:
updated = []
userdata = []
removed = []
2018-09-06 08:36:32 +00:00
for media in filters:
result = self.server['api'].get_sync_queue(last_sync, ",".join([x for x in filters if x != media]))
updated.extend(result['ItemsAdded'])
updated.extend(result['ItemsUpdated'])
userdata.extend(result['UserDataChanged'])
removed.extend(result['ItemsRemoved'])
total = len(updated) + len(userdata)
if total > int(settings('syncIndicator') or 99):
''' Inverse yes no, in case the dialog is forced closed by Kodi.
'''
if dialog("yesno", heading="{emby}", line1=_(33172).replace('{number}', str(total)), nolabel=_(107), yeslabel=_(106)):
LOG.warn("Large updates skipped.")
return True
self.updated(updated)
self.userdata(userdata)
self.removed(removed)
2018-09-06 08:36:32 +00:00
"""
result = self.server['api'].get_sync_queue(last_sync)
self.userdata(result['UserDataChanged'])
self.removed(result['ItemsRemoved'])
filters.extend(["tvshows", "boxsets", "musicvideos", "music"])
# Get only movies.
result = self.server['api'].get_sync_queue(last_sync, ",".join(filters))
self.updated(result['ItemsAdded'])
self.updated(result['ItemsUpdated'])
self.userdata(result['UserDataChanged'])
self.removed(result['ItemsRemoved'])
"""
except Exception as error:
LOG.exception(error)
return False
return True
def save_last_sync(self):
try:
time_now = datetime.strptime(self.server['config/server-time'].split(', ', 1)[1], '%d %b %Y %H:%M:%S GMT') - timedelta(minutes=2)
except Exception as error:
LOG.error(error)
time_now = datetime.utcnow() - timedelta(minutes=2)
2018-09-06 08:36:32 +00:00
last_sync = time_now.strftime('%Y-%m-%dT%H:%M:%Sz')
settings('LastIncrementalSync', value=last_sync)
LOG.info("--[ sync/%s ]", last_sync)
def select_libraries(self, mode=None):
''' Select from libraries synced. Either update or repair libraries.
Send event back to service.py
'''
modes = {
'SyncLibrarySelection': 'SyncLibrary',
'RepairLibrarySelection': 'RepairLibrary',
'AddLibrarySelection': 'SyncLibrary',
'RemoveLibrarySelection': 'RemoveLibrary'
}
2018-09-06 08:36:32 +00:00
sync = get_sync()
whitelist = [x.replace('Mixed:', "") for x in sync['Whitelist']]
2018-09-06 08:36:32 +00:00
libraries = []
with Database('emby') as embydb:
db = emby_db.EmbyDatabase(embydb.cursor)
if mode in ('SyncLibrarySelection', 'RepairLibrarySelection', 'RemoveLibrarySelection'):
for library in sync['Whitelist']:
name = db.get_view_name(library.replace('Mixed:', ""))
libraries.append({'Id': library, 'Name': name})
else:
available = [x for x in sync['SortedViews'] if x not in whitelist]
for library in available:
name, media = db.get_view(library)
2018-09-06 08:36:32 +00:00
if media in ('movies', 'tvshows', 'musicvideos', 'mixed', 'music'):
libraries.append({'Id': library, 'Name': name})
2018-09-06 08:36:32 +00:00
choices = [x['Name'] for x in libraries]
choices.insert(0, _(33121))
selection = dialog("multi", _(33120), choices)
if selection is None:
return
if 0 in selection:
selection = list(range(1, len(libraries) + 1))
selected_libraries = []
for x in selection:
library = libraries[x - 1]
selected_libraries.append(library['Id'])
event(modes[mode], {'Id': ','.join([libraries[x - 1]['Id'] for x in selection]), 'Update': mode == 'SyncLibrarySelection'})
2018-09-06 08:36:32 +00:00
def add_library(self, library_id, update=False):
2018-09-06 08:36:32 +00:00
try:
FullSync(self, library_id, update=update)
2018-09-06 08:36:32 +00:00
except Exception as error:
LOG.exception(error)
return False
Views().get_nodes()
return True
@progress(_(33144))
def remove_library(self, library_id, dialog):
try:
with Database('emby') as embydb:
db = emby_db.EmbyDatabase(embydb.cursor)
library = db.get_view(library_id.replace('Mixed:', ""))
items = db.get_item_by_media_folder(library_id.replace('Mixed:', ""))
2018-09-06 08:36:32 +00:00
media = 'music' if library[1] == 'music' else 'video'
2018-09-14 08:22:48 +00:00
if media == 'music':
settings('MusicRescan.bool', False)
2018-09-06 08:36:32 +00:00
if items:
count = 0
2018-09-06 08:36:32 +00:00
with self.music_database_lock if media == 'music' else self.database_lock:
with Database(media) as kodidb:
if library[1] == 'mixed':
movies = [x for x in items if x[1] == 'Movie']
tvshows = [x for x in items if x[1] == 'Series']
obj = MEDIA['Movie'](self.server, embydb, kodidb, self.direct_path)['Remove']
for item in movies:
obj(item[0])
dialog.update(int((float(count) / float(len(items))*100)), heading="%s: %s" % (_('addon_name'), library[0]))
count += 1
obj = MEDIA['Series'](self.server, embydb, kodidb, self.direct_path)['Remove']
for item in tvshows:
obj(item[0])
dialog.update(int((float(count) / float(len(items))*100)), heading="%s: %s" % (_('addon_name'), library[0]))
count += 1
else:
obj = MEDIA[items[0][1]](self.server, embydb, kodidb, self.direct_path)['Remove']
for item in items:
obj(item[0])
dialog.update(int((float(count) / float(len(items))*100)), heading="%s: %s" % (_('addon_name'), library[0]))
count += 1
2018-09-06 08:36:32 +00:00
sync = get_sync()
if library_id in sync['Whitelist']:
sync['Whitelist'].remove(library_id)
elif 'Mixed:%s' % library_id in sync['Whitelist']:
sync['Whitelist'].remove('Mixed:%s' % library_id)
save_sync(sync)
Views().remove_library(library_id)
except Exception as error:
LOG.exception(error)
dialog.close()
return False
Views().get_views()
2018-09-06 08:36:32 +00:00
Views().get_nodes()
return True
def userdata(self, data):
''' Add item_id to userdata queue.
'''
if not data:
return
2018-10-01 00:35:31 +00:00
items = [x['ItemId'] for x in data]
2018-09-06 08:36:32 +00:00
2018-10-01 00:35:31 +00:00
for item in split_list(items, LIMIT):
self.userdata_queue.put(item)
2018-09-06 08:36:32 +00:00
2018-10-04 03:36:57 +00:00
self.total_updates += len(items)
2018-10-01 00:35:31 +00:00
LOG.info("---[ userdata:%s ]", len(items))
2018-09-06 08:36:32 +00:00
def updated(self, data):
''' Add item_id to updated queue.
'''
if not data:
return
2018-10-01 00:35:31 +00:00
for item in split_list(data, LIMIT):
2018-09-06 08:36:32 +00:00
self.updated_queue.put(item)
2018-10-04 03:36:57 +00:00
self.total_updates += len(data)
2018-10-01 00:35:31 +00:00
LOG.info("---[ updated:%s ]", len(data))
2018-09-06 08:36:32 +00:00
def removed(self, data):
''' Add item_id to removed queue.
'''
if not data:
return
for item in data:
if item in list(self.removed_queue.queue):
continue
self.removed_queue.put(item)
2018-10-04 03:36:57 +00:00
self.total_updates += len(data)
2018-10-01 00:35:31 +00:00
LOG.info("---[ removed:%s ]", len(data))
2018-09-06 08:36:32 +00:00
class UpdatedWorker(threading.Thread):
is_done = False
2018-10-04 05:41:31 +00:00
def __init__(self, queue, notify, lock, database, *args):
2018-09-06 08:36:32 +00:00
self.queue = queue
2018-10-04 05:41:31 +00:00
self.notify_output = notify
self.notify = settings('newContent.bool')
2018-09-06 08:36:32 +00:00
self.lock = lock
self.database = Database(database)
self.args = args
threading.Thread.__init__(self)
def run(self):
2018-10-03 22:18:05 +00:00
with self.lock:
with self.database as kodidb:
with Database('emby') as embydb:
2018-09-06 08:36:32 +00:00
2018-10-03 22:18:05 +00:00
while True:
2018-09-06 08:36:32 +00:00
2018-10-03 22:18:05 +00:00
try:
item = self.queue.get(timeout=1)
2018-10-03 22:18:05 +00:00
except Queue.Empty:
break
2018-09-06 08:36:32 +00:00
obj = MEDIA[item['Type']](self.args[0], embydb, kodidb, self.args[1])[item['Type']]
2018-09-06 08:36:32 +00:00
try:
2018-10-04 05:41:31 +00:00
if obj(item) and self.notify:
self.notify_output.put((item['Type'], api.API(item).get_naming()))
2018-09-06 08:36:32 +00:00
except LibraryException as error:
if error.status == 'StopCalled':
break
except Exception as error:
LOG.exception(error)
2018-10-03 22:18:05 +00:00
self.queue.task_done()
2018-10-03 22:18:05 +00:00
if window('emby_should_stop.bool'):
break
2018-09-06 08:36:32 +00:00
LOG.info("--<[ q:updated/%s ]", id(self))
self.is_done = True
2018-09-06 08:36:32 +00:00
class UserDataWorker(threading.Thread):
is_done = False
def __init__(self, queue, lock, database, *args):
self.queue = queue
self.lock = lock
self.database = Database(database)
self.args = args
threading.Thread.__init__(self)
def run(self):
2018-10-03 22:18:05 +00:00
with self.lock:
with self.database as kodidb:
with Database('emby') as embydb:
2018-09-06 08:36:32 +00:00
2018-10-03 22:18:05 +00:00
while True:
2018-09-06 08:36:32 +00:00
2018-10-03 22:18:05 +00:00
try:
item = self.queue.get(timeout=1)
2018-10-03 22:18:05 +00:00
except Queue.Empty:
break
2018-09-06 08:36:32 +00:00
obj = MEDIA[item['Type']](self.args[0], embydb, kodidb, self.args[1])['UserData']
try:
obj(item)
except LibraryException as error:
if error.status == 'StopCalled':
break
except Exception as error:
LOG.exception(error)
2018-10-03 22:18:05 +00:00
self.queue.task_done()
2018-10-03 22:18:05 +00:00
if window('emby_should_stop.bool'):
break
2018-09-06 08:36:32 +00:00
LOG.info("--<[ q:userdata/%s ]", id(self))
self.is_done = True
2018-09-06 08:36:32 +00:00
class SortWorker(threading.Thread):
is_done = False
def __init__(self, queue, output, *args):
self.queue = queue
self.output = output
self.args = args
threading.Thread.__init__(self)
def run(self):
with Database('emby') as embydb:
database = emby_db.EmbyDatabase(embydb.cursor)
while True:
try:
item_id = self.queue.get(timeout=1)
except Queue.Empty:
break
2018-09-06 08:36:32 +00:00
2018-09-13 00:43:51 +00:00
try:
media = database.get_media_by_id(item_id)
2018-09-06 08:36:32 +00:00
self.output[media].put({'Id': item_id, 'Type': media})
2018-09-13 00:43:51 +00:00
except Exception:
items = database.get_media_by_parent_id(item_id)
if not items:
LOG.info("Could not find media %s in the emby database.", item_id)
else:
for item in items:
self.output[item[1]].put({'Id': item[0], 'Type': item[1]})
2018-09-06 08:36:32 +00:00
self.queue.task_done()
if window('emby_should_stop.bool'):
2018-09-06 08:36:32 +00:00
break
LOG.info("--<[ q:sort/%s ]", id(self))
self.is_done = True
2018-09-06 08:36:32 +00:00
class RemovedWorker(threading.Thread):
is_done = False
def __init__(self, queue, lock, database, *args):
self.queue = queue
self.lock = lock
self.database = Database(database)
self.args = args
threading.Thread.__init__(self)
def run(self):
2018-10-03 22:18:05 +00:00
with self.lock:
with self.database as kodidb:
with Database('emby') as embydb:
2018-09-06 08:36:32 +00:00
2018-10-03 22:18:05 +00:00
while True:
2018-09-06 08:36:32 +00:00
2018-10-03 22:18:05 +00:00
try:
item = self.queue.get(timeout=1)
2018-10-03 22:18:05 +00:00
except Queue.Empty:
break
2018-09-06 08:36:32 +00:00
obj = MEDIA[item['Type']](self.args[0], embydb, kodidb, self.args[1])['Remove']
try:
obj(item['Id'])
except LibraryException as error:
if error.status == 'StopCalled':
break
except Exception as error:
LOG.exception(error)
2018-10-03 22:18:05 +00:00
self.queue.task_done()
2018-10-03 22:18:05 +00:00
if window('emby_should_stop.bool'):
break
2018-09-06 08:36:32 +00:00
LOG.info("--<[ q:removed/%s ]", id(self))
self.is_done = True
2018-09-06 08:36:32 +00:00
class NotifyWorker(threading.Thread):
is_done = False
2018-10-04 05:41:31 +00:00
def __init__(self, queue, player):
2018-09-06 08:36:32 +00:00
self.queue = queue
2018-10-04 05:41:31 +00:00
self.video_time = int(settings('newvideotime')) * 1000
self.music_time = int(settings('newmusictime')) * 1000
self.player = player
threading.Thread.__init__(self)
2018-09-06 08:36:32 +00:00
def run(self):
while True:
try:
item = self.queue.get(timeout=3)
except Queue.Empty:
break
2018-10-04 05:41:31 +00:00
time = self.music_time if item[0] == 'Audio' else self.video_time
if time and (not self.player.isPlayingVideo() or xbmc.getCondVisibility('VideoPlayer.Content(livetv)')):
dialog("notification", heading="%s %s" % (_(33049), item[0]), message=item[1],
icon="{emby}", time=time, sound=False)
2018-09-06 08:36:32 +00:00
self.queue.task_done()
2018-10-04 05:41:31 +00:00
if window('emby_should_stop.bool'):
2018-09-06 08:36:32 +00:00
break
LOG.info("--<[ q:notify/%s ]", id(self))
self.is_done = True