2015-03-14 08:24:59 +11:00
|
|
|
#################################################################################################
|
|
|
|
# LibrarySync
|
|
|
|
#################################################################################################
|
|
|
|
|
|
|
|
import xbmc
|
|
|
|
import xbmcgui
|
|
|
|
import xbmcaddon
|
|
|
|
import xbmcvfs
|
|
|
|
import json
|
2015-03-14 11:46:54 +11:00
|
|
|
import sqlite3
|
2015-03-17 04:51:49 +11:00
|
|
|
import inspect
|
2015-03-14 08:24:59 +11:00
|
|
|
import threading
|
|
|
|
import urllib
|
|
|
|
from datetime import datetime, timedelta, time
|
2015-05-01 21:30:21 +10:00
|
|
|
from itertools import chain
|
2015-03-14 08:24:59 +11:00
|
|
|
import urllib2
|
|
|
|
import os
|
2015-03-14 09:39:35 +11:00
|
|
|
|
2015-06-16 15:53:01 +10:00
|
|
|
import KodiMonitor
|
2015-03-14 08:24:59 +11:00
|
|
|
from API import API
|
|
|
|
import Utils as utils
|
2015-06-16 15:53:01 +10:00
|
|
|
from ClientInformation import ClientInformation
|
2015-03-14 08:24:59 +11:00
|
|
|
from DownloadUtils import DownloadUtils
|
2015-03-18 04:51:45 +11:00
|
|
|
from ReadEmbyDB import ReadEmbyDB
|
2015-03-18 05:41:26 +11:00
|
|
|
from ReadKodiDB import ReadKodiDB
|
2015-05-07 19:45:24 +10:00
|
|
|
from WriteKodiVideoDB import WriteKodiVideoDB
|
2015-05-08 08:04:40 +10:00
|
|
|
from WriteKodiMusicDB import WriteKodiMusicDB
|
2015-05-07 07:24:13 +10:00
|
|
|
from VideoNodes import VideoNodes
|
2015-03-14 08:24:59 +11:00
|
|
|
|
2015-03-26 04:37:21 +11:00
|
|
|
addondir = xbmc.translatePath(xbmcaddon.Addon(id='plugin.video.emby').getAddonInfo('profile'))
|
2015-03-14 08:24:59 +11:00
|
|
|
dataPath = os.path.join(addondir,"library")
|
2015-03-15 09:33:16 +11:00
|
|
|
movieLibrary = os.path.join(dataPath,'movies')
|
|
|
|
tvLibrary = os.path.join(dataPath,'tvshows')
|
2015-03-14 08:24:59 +11:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
WINDOW = xbmcgui.Window( 10000 )
|
|
|
|
|
2015-06-16 15:53:01 +10:00
|
|
|
class LibrarySync(threading.Thread):
|
|
|
|
|
2015-06-28 22:08:06 +10:00
|
|
|
_shared_state = {}
|
|
|
|
|
2015-06-16 15:53:01 +10:00
|
|
|
KodiMonitor = KodiMonitor.Kodi_Monitor()
|
|
|
|
clientInfo = ClientInformation()
|
|
|
|
|
|
|
|
addonName = clientInfo.getAddonName()
|
|
|
|
|
2015-06-28 22:08:06 +10:00
|
|
|
doIncrementalSync = False
|
|
|
|
updateItems = []
|
|
|
|
removeItems = []
|
|
|
|
|
2015-06-16 15:53:01 +10:00
|
|
|
def __init__(self, *args):
|
|
|
|
|
2015-06-28 22:08:06 +10:00
|
|
|
self.__dict__ = self._shared_state
|
2015-06-16 15:53:01 +10:00
|
|
|
threading.Thread.__init__(self, *args)
|
|
|
|
|
|
|
|
def logMsg(self, msg, lvl=1):
|
|
|
|
|
|
|
|
className = self.__class__.__name__
|
|
|
|
utils.logMsg("%s %s" % (self.addonName, className), msg, int(lvl))
|
2015-03-14 08:24:59 +11:00
|
|
|
|
2015-05-07 19:36:34 +10:00
|
|
|
def FullLibrarySync(self,manualRun=False):
|
2015-05-02 19:56:31 +10:00
|
|
|
|
2015-03-27 11:16:45 +11:00
|
|
|
#set some variable to check if this is the first run
|
2015-03-26 04:37:21 +11:00
|
|
|
addon = xbmcaddon.Addon(id='plugin.video.emby')
|
2015-05-02 19:56:31 +10:00
|
|
|
|
2015-03-27 11:16:45 +11:00
|
|
|
startupDone = WINDOW.getProperty("startup") == "done"
|
|
|
|
syncInstallRunDone = addon.getSetting("SyncInstallRunDone") == "true"
|
2015-05-31 09:02:45 +10:00
|
|
|
performMusicSync = addon.getSetting("enableMusicSync") == "true"
|
2015-05-03 03:49:39 +10:00
|
|
|
dbSyncIndication = addon.getSetting("dbSyncIndication") == "true"
|
2015-04-03 19:39:16 +11:00
|
|
|
WINDOW.setProperty("SyncDatabaseRunning", "true")
|
2015-03-27 11:16:45 +11:00
|
|
|
|
2015-05-03 03:49:39 +10:00
|
|
|
|
|
|
|
#show the progress dialog
|
|
|
|
pDialog = None
|
2015-05-07 19:36:34 +10:00
|
|
|
if (syncInstallRunDone == False or dbSyncIndication or manualRun):
|
2015-05-03 03:49:39 +10:00
|
|
|
pDialog = xbmcgui.DialogProgressBG()
|
|
|
|
pDialog.create('Emby for Kodi', 'Performing full sync')
|
|
|
|
|
2015-04-03 19:39:16 +11:00
|
|
|
if(WINDOW.getProperty("SyncDatabaseShouldStop") == "true"):
|
|
|
|
utils.logMsg("Sync Database", "Can not start SyncDatabaseShouldStop=True", 0)
|
|
|
|
return True
|
|
|
|
|
|
|
|
try:
|
|
|
|
completed = True
|
2015-05-08 08:04:40 +10:00
|
|
|
|
|
|
|
|
|
|
|
### BUILD VIDEO NODES LISTING ###
|
|
|
|
VideoNodes().buildVideoNodesListing()
|
|
|
|
|
|
|
|
### PROCESS VIDEO LIBRARY ###
|
|
|
|
|
|
|
|
#create the sql connection to video db
|
|
|
|
connection = utils.KodiSQL("video")
|
2015-04-05 03:20:48 +10:00
|
|
|
cursor = connection.cursor()
|
2015-05-01 21:30:21 +10:00
|
|
|
|
2015-05-02 11:47:05 +10:00
|
|
|
#Add the special emby table
|
2015-05-13 05:27:27 +10:00
|
|
|
cursor.execute("CREATE TABLE IF NOT EXISTS emby(emby_id TEXT, kodi_id INTEGER, media_type TEXT, checksum TEXT, parent_id INTEGER)")
|
|
|
|
connection.commit()
|
2015-05-01 21:30:21 +10:00
|
|
|
|
2015-04-03 19:39:16 +11:00
|
|
|
# sync movies
|
2015-05-02 19:56:31 +10:00
|
|
|
self.MoviesFullSync(connection,cursor,pDialog)
|
2015-05-03 03:49:39 +10:00
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
2015-05-03 06:02:06 +10:00
|
|
|
return False
|
2015-05-03 03:49:39 +10:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
#sync Tvshows and episodes
|
2015-05-02 19:56:31 +10:00
|
|
|
self.TvShowsFullSync(connection,cursor,pDialog)
|
2015-05-03 06:02:06 +10:00
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
|
|
|
# sync musicvideos
|
|
|
|
self.MusicVideosFullSync(connection,cursor,pDialog)
|
|
|
|
|
2015-05-08 08:04:40 +10:00
|
|
|
#close sql connection
|
|
|
|
cursor.close()
|
|
|
|
|
|
|
|
### PROCESS MUSIC LIBRARY ###
|
|
|
|
if performMusicSync:
|
|
|
|
#create the sql connection to music db
|
|
|
|
connection = utils.KodiSQL("music")
|
|
|
|
cursor = connection.cursor()
|
|
|
|
|
|
|
|
#Add the special emby table
|
2015-05-13 05:27:27 +10:00
|
|
|
cursor.execute("CREATE TABLE IF NOT EXISTS emby(emby_id TEXT, kodi_id INTEGER, media_type TEXT, checksum TEXT, parent_id INTEGER)")
|
|
|
|
connection.commit()
|
2015-05-08 08:04:40 +10:00
|
|
|
|
|
|
|
self.MusicFullSync(connection,cursor,pDialog)
|
|
|
|
cursor.close()
|
|
|
|
|
2015-04-03 19:39:16 +11:00
|
|
|
# set the install done setting
|
|
|
|
if(syncInstallRunDone == False and completed):
|
|
|
|
addon = xbmcaddon.Addon(id='plugin.video.emby') #force a new instance of the addon
|
|
|
|
addon.setSetting("SyncInstallRunDone", "true")
|
|
|
|
|
2015-05-03 06:46:12 +10:00
|
|
|
# Commit all DB changes at once and Force refresh the library
|
2015-05-02 19:56:31 +10:00
|
|
|
xbmc.executebuiltin("UpdateLibrary(video)")
|
|
|
|
|
2015-04-03 19:39:16 +11:00
|
|
|
# set prop to show we have run for the first time
|
|
|
|
WINDOW.setProperty("startup", "done")
|
|
|
|
|
2015-06-24 17:03:49 +10:00
|
|
|
# tell any widgets to refresh because the content has changed
|
|
|
|
WINDOW.setProperty("widgetreload", datetime.now().strftime('%Y-%m-%d %H:%M:%S'))
|
|
|
|
|
2015-04-03 19:39:16 +11:00
|
|
|
finally:
|
|
|
|
WINDOW.setProperty("SyncDatabaseRunning", "false")
|
2015-04-05 09:59:15 +10:00
|
|
|
utils.logMsg("Sync DB", "syncDatabase Exiting", 0)
|
2015-05-03 06:46:12 +10:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
|
|
|
|
if(pDialog != None):
|
|
|
|
pDialog.close()
|
|
|
|
|
2015-03-19 07:34:52 +11:00
|
|
|
return True
|
2015-05-01 21:30:21 +10:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
def MoviesFullSync(self,connection,cursor, pDialog):
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
views = ReadEmbyDB().getCollections("movies")
|
|
|
|
|
|
|
|
allKodiMovieIds = list()
|
|
|
|
allEmbyMovieIds = list()
|
|
|
|
|
|
|
|
for view in views:
|
2015-03-19 02:47:55 +11:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
allEmbyMovies = ReadEmbyDB().getMovies(view.get('id'))
|
2015-05-01 21:30:21 +10:00
|
|
|
allKodiMovies = ReadKodiDB().getKodiMovies(connection, cursor)
|
2015-05-02 11:47:05 +10:00
|
|
|
|
|
|
|
for kodimovie in allKodiMovies:
|
|
|
|
allKodiMovieIds.append(kodimovie[1])
|
2015-05-02 19:56:31 +10:00
|
|
|
|
|
|
|
total = len(allEmbyMovies) + 1
|
|
|
|
count = 1
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
#### PROCESS ADDS AND UPDATES ###
|
2015-05-02 19:56:31 +10:00
|
|
|
for item in allEmbyMovies:
|
2015-05-03 03:49:39 +10:00
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
if not item.get('IsFolder'):
|
|
|
|
allEmbyMovieIds.append(item["Id"])
|
2015-03-19 02:47:55 +11:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
if(pDialog != None):
|
|
|
|
progressTitle = "Processing " + view.get('title') + " (" + str(count) + " of " + str(total) + ")"
|
2015-05-08 20:05:15 +10:00
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-02 20:51:46 +10:00
|
|
|
count += 1
|
2015-05-02 19:56:31 +10:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
kodiMovie = None
|
|
|
|
for kodimovie in allKodiMovies:
|
|
|
|
if kodimovie[1] == item["Id"]:
|
|
|
|
kodiMovie = kodimovie
|
|
|
|
|
|
|
|
if kodiMovie == None:
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateMovieToKodiLibrary(item["Id"],connection, cursor, view.get('title'))
|
2015-05-01 21:30:21 +10:00
|
|
|
else:
|
2015-05-02 19:56:31 +10:00
|
|
|
if kodiMovie[2] != API().getChecksum(item):
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateMovieToKodiLibrary(item["Id"],connection, cursor, view.get('title'))
|
2015-05-03 22:44:23 +10:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#### PROCESS BOX SETS #####
|
|
|
|
if(pDialog != None):
|
|
|
|
utils.logMsg("Sync Movies", "BoxSet Sync Started", 1)
|
|
|
|
boxsets = ReadEmbyDB().getBoxSets()
|
|
|
|
|
|
|
|
total = len(boxsets) + 1
|
|
|
|
count = 1
|
|
|
|
for boxset in boxsets:
|
2015-05-08 20:05:15 +10:00
|
|
|
progressTitle = "Processing BoxSets"+ " (" + str(count) + " of " + str(total) + ")"
|
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-03 22:44:23 +10:00
|
|
|
count += 1
|
|
|
|
if(self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
boxsetMovies = ReadEmbyDB().getMoviesInBoxSet(boxset["Id"])
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addBoxsetToKodiLibrary(boxset,connection, cursor)
|
2015-05-03 22:44:23 +10:00
|
|
|
|
|
|
|
for boxsetMovie in boxsetMovies:
|
|
|
|
if(self.ShouldStop()):
|
|
|
|
return False
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().updateBoxsetToKodiLibrary(boxsetMovie,boxset, connection, cursor)
|
2015-05-03 22:44:23 +10:00
|
|
|
|
|
|
|
utils.logMsg("Sync Movies", "BoxSet Sync Finished", 1)
|
2015-05-01 21:30:21 +10:00
|
|
|
|
2015-05-02 20:51:46 +10:00
|
|
|
#### PROCESS DELETES #####
|
|
|
|
allEmbyMovieIds = set(allEmbyMovieIds)
|
|
|
|
for kodiId in allKodiMovieIds:
|
|
|
|
if not kodiId in allEmbyMovieIds:
|
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
2015-05-03 06:46:12 +10:00
|
|
|
|
|
|
|
### commit all changes to database ###
|
|
|
|
connection.commit()
|
2015-05-01 21:30:21 +10:00
|
|
|
|
2015-05-03 06:02:06 +10:00
|
|
|
def MusicVideosFullSync(self,connection,cursor, pDialog):
|
|
|
|
|
|
|
|
allKodiMusicvideoIds = list()
|
|
|
|
allEmbyMusicvideoIds = list()
|
|
|
|
|
|
|
|
allEmbyMusicvideos = ReadEmbyDB().getMusicVideos()
|
|
|
|
allKodiMusicvideos = ReadKodiDB().getKodiMusicVideos(connection, cursor)
|
|
|
|
|
|
|
|
for kodivideo in allKodiMusicvideos:
|
|
|
|
allKodiMusicvideoIds.append(kodivideo[1])
|
|
|
|
|
|
|
|
total = len(allEmbyMusicvideos) + 1
|
|
|
|
count = 1
|
|
|
|
|
|
|
|
#### PROCESS ADDS AND UPDATES ###
|
|
|
|
for item in allEmbyMusicvideos:
|
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
|
|
|
if not item.get('IsFolder'):
|
|
|
|
allEmbyMusicvideoIds.append(item["Id"])
|
|
|
|
|
|
|
|
if(pDialog != None):
|
|
|
|
progressTitle = "Processing MusicVideos (" + str(count) + " of " + str(total) + ")"
|
2015-05-08 20:05:15 +10:00
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-03 06:02:06 +10:00
|
|
|
count += 1
|
|
|
|
|
|
|
|
kodiVideo = None
|
|
|
|
for kodivideo in allKodiMusicvideos:
|
|
|
|
if kodivideo[1] == item["Id"]:
|
|
|
|
kodiVideo = kodivideo
|
|
|
|
|
|
|
|
if kodiVideo == None:
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateMusicVideoToKodiLibrary(item["Id"],connection, cursor)
|
2015-05-03 06:02:06 +10:00
|
|
|
else:
|
|
|
|
if kodiVideo[2] != API().getChecksum(item):
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateMusicVideoToKodiLibrary(item["Id"],connection, cursor)
|
2015-05-03 06:02:06 +10:00
|
|
|
|
|
|
|
#### PROCESS DELETES #####
|
|
|
|
allEmbyMusicvideoIds = set(allEmbyMusicvideoIds)
|
|
|
|
for kodiId in allKodiMusicvideoIds:
|
|
|
|
if not kodiId in allEmbyMusicvideoIds:
|
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
2015-05-03 06:46:12 +10:00
|
|
|
|
|
|
|
### commit all changes to database ###
|
|
|
|
connection.commit()
|
2015-05-03 06:02:06 +10:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
def TvShowsFullSync(self,connection,cursor,pDialog):
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
views = ReadEmbyDB().getCollections("tvshows")
|
2015-03-19 04:00:38 +11:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
allKodiTvShowIds = list()
|
|
|
|
allEmbyTvShowIds = list()
|
2015-05-02 11:47:05 +10:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
for view in views:
|
2015-03-20 19:15:06 +11:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
allEmbyTvShows = ReadEmbyDB().getTvShows(view.get('id'))
|
|
|
|
allKodiTvShows = ReadKodiDB().getKodiTvShows(connection, cursor)
|
2015-05-02 11:47:05 +10:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
total = len(allEmbyTvShows) + 1
|
|
|
|
count = 1
|
|
|
|
|
2015-05-02 11:47:05 +10:00
|
|
|
for kodishow in allKodiTvShows:
|
|
|
|
allKodiTvShowIds.append(kodishow[1])
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
#### TVSHOW: PROCESS ADDS AND UPDATES ###
|
|
|
|
for item in allEmbyTvShows:
|
2015-05-03 03:49:39 +10:00
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
2015-05-03 06:34:55 +10:00
|
|
|
if(pDialog != None):
|
|
|
|
progressTitle = "Processing " + view.get('title') + " (" + str(count) + " of " + str(total) + ")"
|
2015-05-08 20:05:15 +10:00
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-03 06:34:55 +10:00
|
|
|
count += 1
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
if item.get('IsFolder') and item.get('RecursiveItemCount') != 0:
|
|
|
|
allEmbyTvShowIds.append(item["Id"])
|
2015-03-19 08:38:02 +11:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
#build a list with all Id's and get the existing entry (if exists) in Kodi DB
|
|
|
|
kodiShow = None
|
|
|
|
for kodishow in allKodiTvShows:
|
|
|
|
if kodishow[1] == item["Id"]:
|
|
|
|
kodiShow = kodishow
|
|
|
|
|
|
|
|
if kodiShow == None:
|
|
|
|
# Tv show doesn't exist in Kodi yet so proceed and add it
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateTvShowToKodiLibrary(item["Id"],connection, cursor, view.get('title'))
|
2015-05-01 21:30:21 +10:00
|
|
|
else:
|
|
|
|
# If there are changes to the item, perform a full sync of the item
|
2015-05-02 19:56:31 +10:00
|
|
|
if kodiShow[2] != API().getChecksum(item):
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateTvShowToKodiLibrary(item["Id"],connection, cursor, view.get('title'))
|
2015-04-05 07:48:02 +10:00
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
#### PROCESS EPISODES ######
|
2015-05-02 22:57:43 +10:00
|
|
|
self.EpisodesFullSync(connection,cursor,item["Id"])
|
2015-05-01 21:30:21 +10:00
|
|
|
|
2015-05-02 20:51:46 +10:00
|
|
|
#### TVSHOW: PROCESS DELETES #####
|
|
|
|
allEmbyTvShowIds = set(allEmbyTvShowIds)
|
|
|
|
for kodiId in allKodiTvShowIds:
|
|
|
|
if not kodiId in allEmbyTvShowIds:
|
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
2015-05-03 06:46:12 +10:00
|
|
|
|
|
|
|
### commit all changes to database ###
|
|
|
|
connection.commit()
|
2015-05-02 19:56:31 +10:00
|
|
|
|
2015-05-02 22:57:43 +10:00
|
|
|
def EpisodesFullSync(self,connection,cursor,showId):
|
2015-05-01 21:30:21 +10:00
|
|
|
|
|
|
|
WINDOW = xbmcgui.Window( 10000 )
|
|
|
|
|
|
|
|
allKodiEpisodeIds = list()
|
|
|
|
allEmbyEpisodeIds = list()
|
2015-03-19 04:00:38 +11:00
|
|
|
|
2015-05-02 22:57:43 +10:00
|
|
|
#get the kodi parent id
|
|
|
|
cursor.execute("SELECT kodi_id FROM emby WHERE emby_id=?",(showId,))
|
|
|
|
kodiShowId = cursor.fetchone()[0]
|
|
|
|
|
|
|
|
allEmbyEpisodes = ReadEmbyDB().getEpisodes(showId)
|
2015-05-01 21:30:21 +10:00
|
|
|
allKodiEpisodes = ReadKodiDB().getKodiEpisodes(connection, cursor, kodiShowId)
|
2015-05-02 11:47:05 +10:00
|
|
|
|
|
|
|
for kodiepisode in allKodiEpisodes:
|
|
|
|
allKodiEpisodeIds.append(kodiepisode[1])
|
2015-05-01 21:30:21 +10:00
|
|
|
|
|
|
|
#### EPISODES: PROCESS ADDS AND UPDATES ###
|
|
|
|
for item in allEmbyEpisodes:
|
2015-05-03 03:49:39 +10:00
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
2015-05-01 21:30:21 +10:00
|
|
|
allEmbyEpisodeIds.append(item["Id"])
|
|
|
|
|
2015-05-02 11:47:05 +10:00
|
|
|
#get the existing entry (if exists) in Kodi DB
|
2015-05-01 21:30:21 +10:00
|
|
|
kodiEpisode = None
|
|
|
|
for kodiepisode in allKodiEpisodes:
|
|
|
|
if kodiepisode[1] == item["Id"]:
|
|
|
|
kodiEpisode = kodiepisode
|
|
|
|
|
|
|
|
if kodiEpisode == None:
|
|
|
|
# Episode doesn't exist in Kodi yet so proceed and add it
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateEpisodeToKodiLibrary(item["Id"], kodiShowId, connection, cursor)
|
2015-05-01 21:30:21 +10:00
|
|
|
else:
|
|
|
|
# If there are changes to the item, perform a full sync of the item
|
2015-05-02 19:56:31 +10:00
|
|
|
if kodiEpisode[2] != API().getChecksum(item):
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().addOrUpdateEpisodeToKodiLibrary(item["Id"], kodiShowId, connection, cursor)
|
2015-05-01 21:30:21 +10:00
|
|
|
|
|
|
|
#### EPISODES: PROCESS DELETES #####
|
|
|
|
allEmbyEpisodeIds = set(allEmbyEpisodeIds)
|
|
|
|
for kodiId in allKodiEpisodeIds:
|
2015-05-02 11:47:05 +10:00
|
|
|
if (not kodiId in allEmbyEpisodeIds):
|
2015-05-01 21:30:21 +10:00
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
2015-05-07 19:45:24 +10:00
|
|
|
WriteKodiVideoDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
2015-05-03 06:46:12 +10:00
|
|
|
|
2015-05-08 08:04:40 +10:00
|
|
|
def MusicFullSync(self, connection,cursor, pDialog):
|
|
|
|
|
|
|
|
self.ProcessMusicArtists(connection,cursor,pDialog)
|
|
|
|
self.ProcessMusicAlbums(connection,cursor,pDialog)
|
|
|
|
self.ProcessMusicSongs(connection,cursor,pDialog)
|
|
|
|
|
|
|
|
### commit all changes to database ###
|
|
|
|
connection.commit()
|
|
|
|
|
|
|
|
def ProcessMusicSongs(self,connection,cursor,pDialog):
|
|
|
|
|
|
|
|
allKodiSongIds = list()
|
|
|
|
allEmbySongIds = list()
|
|
|
|
|
|
|
|
allEmbySongs = ReadEmbyDB().getMusicSongs()
|
|
|
|
allKodiSongs = ReadKodiDB().getKodiMusicSongs(connection, cursor)
|
|
|
|
|
|
|
|
for kodisong in allKodiSongs:
|
|
|
|
allKodiSongIds.append(kodisong[1])
|
|
|
|
|
|
|
|
total = len(allEmbySongs) + 1
|
|
|
|
count = 1
|
|
|
|
|
|
|
|
#### PROCESS SONGS ADDS AND UPDATES ###
|
|
|
|
for item in allEmbySongs:
|
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
|
|
|
allEmbySongIds.append(item["Id"])
|
|
|
|
|
|
|
|
if(pDialog != None):
|
|
|
|
progressTitle = "Processing Music Songs (" + str(count) + " of " + str(total) + ")"
|
2015-05-08 20:05:15 +10:00
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-08 08:04:40 +10:00
|
|
|
count += 1
|
|
|
|
|
|
|
|
kodiSong = None
|
|
|
|
for kodisong in allKodiSongs:
|
|
|
|
if kodisong[1] == item["Id"]:
|
|
|
|
kodiSong = kodisong
|
|
|
|
|
|
|
|
if kodiSong == None:
|
|
|
|
WriteKodiMusicDB().addOrUpdateSongToKodiLibrary(item["Id"],connection, cursor)
|
|
|
|
else:
|
|
|
|
if kodiSong[2] != API().getChecksum(item):
|
|
|
|
WriteKodiMusicDB().addOrUpdateSongToKodiLibrary(item["Id"],connection, cursor)
|
|
|
|
|
|
|
|
#### PROCESS DELETES #####
|
|
|
|
allEmbySongIds = set(allEmbySongIds)
|
|
|
|
for kodiId in allKodiSongIds:
|
|
|
|
if not kodiId in allEmbySongIds:
|
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
|
|
|
WriteKodiMusicDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
|
|
|
|
|
|
|
def ProcessMusicArtists(self,connection,cursor,pDialog):
|
|
|
|
|
|
|
|
allKodiArtistIds = list()
|
|
|
|
allEmbyArtistIds = list()
|
|
|
|
|
|
|
|
allEmbyArtists = ReadEmbyDB().getMusicArtists()
|
|
|
|
allKodiArtists = ReadKodiDB().getKodiMusicArtists(connection, cursor)
|
|
|
|
|
|
|
|
for kodiartist in allKodiArtists:
|
|
|
|
allKodiArtistIds.append(kodiartist[1])
|
|
|
|
|
|
|
|
total = len(allEmbyArtists) + 1
|
|
|
|
count = 1
|
|
|
|
|
|
|
|
#### PROCESS SONGS ADDS AND UPDATES ###
|
|
|
|
for item in allEmbyArtists:
|
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
|
|
|
allEmbyArtistIds.append(item["Id"])
|
|
|
|
|
|
|
|
if(pDialog != None):
|
|
|
|
progressTitle = "Processing Music Artists (" + str(count) + " of " + str(total) + ")"
|
2015-05-08 20:05:15 +10:00
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-08 08:04:40 +10:00
|
|
|
count += 1
|
|
|
|
|
|
|
|
kodiArtist = None
|
|
|
|
for kodiartist in allKodiArtists:
|
|
|
|
if kodiartist[1] == item["Id"]:
|
|
|
|
kodiArtist = kodiartist
|
|
|
|
|
|
|
|
if kodiArtist == None:
|
|
|
|
WriteKodiMusicDB().addOrUpdateArtistToKodiLibrary(item["Id"],connection, cursor)
|
|
|
|
else:
|
|
|
|
if kodiArtist[2] != API().getChecksum(item):
|
|
|
|
WriteKodiMusicDB().addOrUpdateArtistToKodiLibrary(item["Id"],connection, cursor)
|
|
|
|
|
|
|
|
#### PROCESS DELETES #####
|
|
|
|
allEmbyArtistIds = set(allEmbyArtistIds)
|
|
|
|
for kodiId in allKodiArtistIds:
|
|
|
|
if not kodiId in allEmbyArtistIds:
|
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
|
|
|
WriteKodiMusicDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
|
|
|
|
|
|
|
def ProcessMusicAlbums(self,connection,cursor,pDialog):
|
|
|
|
|
|
|
|
allKodiAlbumIds = list()
|
|
|
|
allEmbyAlbumIds = list()
|
|
|
|
|
|
|
|
allEmbyAlbums = ReadEmbyDB().getMusicAlbums()
|
|
|
|
allKodiAlbums = ReadKodiDB().getKodiMusicAlbums(connection, cursor)
|
|
|
|
|
|
|
|
for kodialbum in allKodiAlbums:
|
|
|
|
allKodiAlbumIds.append(kodialbum[1])
|
|
|
|
|
|
|
|
total = len(allEmbyAlbums) + 1
|
|
|
|
count = 1
|
|
|
|
|
|
|
|
#### PROCESS SONGS ADDS AND UPDATES ###
|
|
|
|
for item in allEmbyAlbums:
|
|
|
|
|
|
|
|
if (self.ShouldStop()):
|
|
|
|
return False
|
|
|
|
|
|
|
|
allEmbyAlbumIds.append(item["Id"])
|
|
|
|
|
|
|
|
if(pDialog != None):
|
|
|
|
progressTitle = "Processing Music Albums (" + str(count) + " of " + str(total) + ")"
|
2015-05-08 20:05:15 +10:00
|
|
|
percentage = int(((float(count) / float(total)) * 100))
|
|
|
|
pDialog.update(percentage, "Emby for Kodi - Running Sync", progressTitle)
|
2015-05-08 08:04:40 +10:00
|
|
|
count += 1
|
|
|
|
|
|
|
|
kodiAlbum = None
|
|
|
|
for kodialbum in allKodiAlbums:
|
|
|
|
if kodialbum[1] == item["Id"]:
|
|
|
|
kodiAlbum = kodialbum
|
|
|
|
|
|
|
|
if kodiAlbum == None:
|
|
|
|
WriteKodiMusicDB().addOrUpdateAlbumToKodiLibrary(item["Id"],connection, cursor)
|
|
|
|
else:
|
|
|
|
if kodiAlbum[2] != API().getChecksum(item):
|
|
|
|
WriteKodiMusicDB().addOrUpdateAlbumToKodiLibrary(item["Id"],connection, cursor)
|
|
|
|
|
|
|
|
#### PROCESS DELETES #####
|
|
|
|
allEmbyAlbumIds = set(allEmbyAlbumIds)
|
|
|
|
for kodiId in allKodiAlbumIds:
|
|
|
|
if not kodiId in allEmbyAlbumIds:
|
|
|
|
WINDOW.setProperty(kodiId,"deleted")
|
|
|
|
WriteKodiMusicDB().deleteItemFromKodiLibrary(kodiId, connection, cursor)
|
2015-03-19 04:00:38 +11:00
|
|
|
|
2015-05-02 19:56:31 +10:00
|
|
|
def IncrementalSync(self, itemList):
|
2015-03-22 00:31:30 +11:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
startupDone = WINDOW.getProperty("startup") == "done"
|
2015-03-22 00:31:30 +11:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
#only perform incremental scan when full scan is completed
|
|
|
|
if startupDone:
|
2015-03-22 00:31:30 +11:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
#this will only perform sync for items received by the websocket
|
|
|
|
addon = xbmcaddon.Addon(id='plugin.video.emby')
|
|
|
|
dbSyncIndication = addon.getSetting("dbSyncIndication") == "true"
|
2015-05-31 09:02:45 +10:00
|
|
|
performMusicSync = addon.getSetting("enableMusicSync") == "true"
|
2015-05-08 08:06:49 +10:00
|
|
|
WINDOW.setProperty("SyncDatabaseRunning", "true")
|
|
|
|
|
|
|
|
#show the progress dialog
|
2015-05-31 10:47:44 +10:00
|
|
|
if (dbSyncIndication and xbmc.Player().isPlaying() == False):
|
2015-05-08 23:46:07 +10:00
|
|
|
xbmcgui.Dialog().notification('Emby for Kodi', 'Performing incremental sync...', "special://home/addons/plugin.video.emby/icon.png")
|
2015-05-08 08:06:49 +10:00
|
|
|
|
|
|
|
connection = utils.KodiSQL("video")
|
|
|
|
cursor = connection.cursor()
|
|
|
|
|
|
|
|
try:
|
|
|
|
#### PROCESS MOVIES ####
|
|
|
|
views = ReadEmbyDB().getCollections("movies")
|
|
|
|
for view in views:
|
|
|
|
allEmbyMovies = ReadEmbyDB().getMovies(view.get('id'), itemList)
|
|
|
|
for item in allEmbyMovies:
|
|
|
|
|
|
|
|
if not item.get('IsFolder'):
|
|
|
|
WriteKodiVideoDB().addOrUpdateMovieToKodiLibrary(item["Id"],connection, cursor, view.get('title'))
|
|
|
|
|
2015-05-08 20:05:15 +10:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
#### PROCESS BOX SETS #####
|
|
|
|
boxsets = ReadEmbyDB().getBoxSets()
|
|
|
|
|
|
|
|
for boxset in boxsets:
|
|
|
|
boxsetMovies = ReadEmbyDB().getMoviesInBoxSet(boxset["Id"])
|
|
|
|
WriteKodiVideoDB().addBoxsetToKodiLibrary(boxset,connection, cursor)
|
2015-05-02 20:51:46 +10:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
for boxsetMovie in boxsetMovies:
|
2015-05-08 20:05:15 +10:00
|
|
|
WriteKodiVideoDB().updateBoxsetToKodiLibrary(boxsetMovie,boxset, connection, cursor)
|
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
#### PROCESS TV SHOWS ####
|
|
|
|
views = ReadEmbyDB().getCollections("tvshows")
|
|
|
|
for view in views:
|
|
|
|
allEmbyTvShows = ReadEmbyDB().getTvShows(view.get('id'),itemList)
|
|
|
|
for item in allEmbyTvShows:
|
|
|
|
if item.get('IsFolder') and item.get('RecursiveItemCount') != 0:
|
|
|
|
kodiId = WriteKodiVideoDB().addOrUpdateTvShowToKodiLibrary(item["Id"],connection, cursor, view.get('title'))
|
2015-05-08 20:05:15 +10:00
|
|
|
|
2015-05-13 05:40:58 +10:00
|
|
|
|
|
|
|
#### PROCESS OTHERS BY THE ITEMLIST ######
|
2015-05-08 08:06:49 +10:00
|
|
|
for item in itemList:
|
2015-05-02 20:51:46 +10:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
MBitem = ReadEmbyDB().getItem(item)
|
2015-05-13 05:40:58 +10:00
|
|
|
|
|
|
|
#### PROCESS EPISODES ######
|
2015-05-08 08:06:49 +10:00
|
|
|
if MBitem["Type"] == "Episode":
|
2015-05-02 21:47:04 +10:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
#get the tv show
|
|
|
|
cursor.execute("SELECT kodi_id FROM emby WHERE media_type='tvshow' AND emby_id=?", (MBitem["SeriesId"],))
|
|
|
|
result = cursor.fetchone()
|
|
|
|
if result:
|
|
|
|
kodi_show_id = result[0]
|
|
|
|
else:
|
|
|
|
kodi_show_id = None
|
2015-05-02 21:47:04 +10:00
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
if kodi_show_id:
|
|
|
|
WriteKodiVideoDB().addOrUpdateEpisodeToKodiLibrary(MBitem["Id"], kodi_show_id, connection, cursor)
|
2015-05-13 05:27:27 +10:00
|
|
|
else:
|
|
|
|
#tv show doesn't exist
|
|
|
|
#perform full tvshow sync instead so both the show and episodes get added
|
|
|
|
self.TvShowsFullSync(connection,cursor,None)
|
2015-06-18 18:03:44 +10:00
|
|
|
|
|
|
|
elif u"Season" in MBitem['Type']:
|
|
|
|
|
|
|
|
#get the tv show
|
|
|
|
cursor.execute("SELECT kodi_id FROM emby WHERE media_type='tvshow' AND emby_id=?", (MBitem["SeriesId"],))
|
|
|
|
result = cursor.fetchone()
|
|
|
|
if result:
|
|
|
|
kodi_show_id = result[0]
|
|
|
|
# update season
|
|
|
|
WriteKodiVideoDB().updateSeasons(MBitem["SeriesId"], kodi_show_id, connection, cursor)
|
2015-05-13 05:40:58 +10:00
|
|
|
|
|
|
|
#### PROCESS BOXSETS ######
|
|
|
|
elif MBitem["Type"] == "BoxSet":
|
|
|
|
boxsetMovies = ReadEmbyDB().getMoviesInBoxSet(boxset["Id"])
|
|
|
|
WriteKodiVideoDB().addBoxsetToKodiLibrary(boxset,connection, cursor)
|
|
|
|
|
|
|
|
for boxsetMovie in boxsetMovies:
|
|
|
|
WriteKodiVideoDB().updateBoxsetToKodiLibrary(boxsetMovie,boxset, connection, cursor)
|
|
|
|
|
|
|
|
#### PROCESS MUSICVIDEOS ####
|
|
|
|
elif MBitem["Type"] == "MusicVideo":
|
|
|
|
if not MBitem.get('IsFolder'):
|
|
|
|
WriteKodiVideoDB().addOrUpdateMusicVideoToKodiLibrary(MBitem["Id"],connection, cursor)
|
2015-05-08 08:06:49 +10:00
|
|
|
|
|
|
|
### commit all changes to database ###
|
2015-05-08 08:04:40 +10:00
|
|
|
connection.commit()
|
|
|
|
cursor.close()
|
|
|
|
|
2015-05-08 08:06:49 +10:00
|
|
|
### PROCESS MUSIC LIBRARY ###
|
|
|
|
if performMusicSync:
|
|
|
|
connection = utils.KodiSQL("music")
|
|
|
|
cursor = connection.cursor()
|
|
|
|
for item in itemList:
|
|
|
|
MBitem = ReadEmbyDB().getItem(item)
|
|
|
|
if MBitem["Type"] == "MusicArtist":
|
|
|
|
WriteKodiMusicDB().addOrUpdateArtistToKodiLibrary(MBitem["Id"],connection, cursor)
|
|
|
|
if MBitem["Type"] == "MusicAlbum":
|
2015-05-08 08:15:31 +10:00
|
|
|
WriteKodiMusicDB().addOrUpdateAlbumToKodiLibrary(MBitem["Id"],connection, cursor)
|
2015-05-08 08:06:49 +10:00
|
|
|
if MBitem["Type"] == "Audio":
|
2015-05-08 08:15:31 +10:00
|
|
|
WriteKodiMusicDB().addOrUpdateSongToKodiLibrary(MBitem["Id"],connection, cursor)
|
2015-05-08 08:06:49 +10:00
|
|
|
connection.commit()
|
|
|
|
cursor.close()
|
|
|
|
|
|
|
|
finally:
|
|
|
|
xbmc.executebuiltin("UpdateLibrary(video)")
|
|
|
|
WINDOW.setProperty("SyncDatabaseRunning", "false")
|
2015-06-24 17:03:49 +10:00
|
|
|
# tell any widgets to refresh because the content has changed
|
|
|
|
WINDOW.setProperty("widgetreload", datetime.now().strftime('%Y-%m-%d %H:%M:%S'))
|
2015-05-08 23:46:07 +10:00
|
|
|
|
2015-06-28 22:08:06 +10:00
|
|
|
def removefromDB(self, itemList, deleteEmbyItem = False):
|
|
|
|
# Delete from Kodi before Emby
|
|
|
|
# To be able to get mediaType
|
|
|
|
doUtils = DownloadUtils()
|
|
|
|
|
|
|
|
video = []
|
|
|
|
music = []
|
|
|
|
|
|
|
|
itemIds = ','.join(itemList)
|
|
|
|
url = "{server}/mediabrowser/Users/{UserId}/Items?Ids=%s&format=json" % itemIds
|
|
|
|
result = doUtils.downloadUrl(url)
|
|
|
|
|
|
|
|
if result is "":
|
|
|
|
# Websocket feedback
|
|
|
|
self.logMsg("Item %s is removed." % itemIds)
|
|
|
|
return
|
|
|
|
|
|
|
|
for item in result[u'Items']:
|
|
|
|
# Sort by type for database deletion
|
|
|
|
itemId = item["Id"]
|
|
|
|
mediaType = item["MediaType"]
|
|
|
|
|
|
|
|
if "Video" in mediaType:
|
|
|
|
video.append(itemId)
|
|
|
|
elif "Audio" in mediaType:
|
|
|
|
music.append(itemId)
|
|
|
|
|
|
|
|
if len(video) > 0:
|
|
|
|
#Process video library
|
|
|
|
connection = utils.KodiSQL("video")
|
|
|
|
cursor = connection.cursor()
|
|
|
|
|
|
|
|
for item in video:
|
|
|
|
type = ReadKodiDB().getTypeByEmbyId(item, connection, cursor)
|
|
|
|
self.logMsg("Type: %s" % type)
|
|
|
|
self.logMsg("Message: Doing LibraryChanged: Items Removed: Calling deleteItemFromKodiLibrary: %s" % item, 0)
|
|
|
|
if "episode" in type:
|
|
|
|
# Get the TV Show Id for reference later
|
|
|
|
showId = ReadKodiDB().getShowIdByEmbyId(item, connection, cursor)
|
|
|
|
self.logMsg("ShowId: %s" % showId, 0)
|
|
|
|
WriteKodiVideoDB().deleteItemFromKodiLibrary(item, connection, cursor)
|
|
|
|
# Verification
|
|
|
|
if "episode" in type:
|
|
|
|
showTotalCount = ReadKodiDB().getShowTotalCount(showId, connection, cursor)
|
|
|
|
self.logMsg("ShowTotalCount: %s" % showTotalCount, 0)
|
|
|
|
# If there are no episodes left
|
2015-06-28 23:36:44 +10:00
|
|
|
if showTotalCount == 0 or showTotalCount == None:
|
2015-06-28 22:08:06 +10:00
|
|
|
# Delete show
|
|
|
|
embyId = ReadKodiDB().getEmbyIdByKodiId(showId, "tvshow", connection, cursor)
|
|
|
|
self.logMsg("Message: Doing LibraryChanged: Deleting show: %s" % embyId, 0)
|
|
|
|
WriteKodiVideoDB().deleteItemFromKodiLibrary(embyId, connection, cursor)
|
|
|
|
|
|
|
|
connection.commit()
|
|
|
|
cursor.close()
|
|
|
|
|
|
|
|
if len(music) > 0:
|
|
|
|
#Process music library
|
|
|
|
addon = xbmcaddon.Addon(id='plugin.video.emby')
|
|
|
|
if addon.getSetting("enableMusicSync") is "true":
|
|
|
|
connection = utils.KodiSQL("music")
|
|
|
|
cursor = connection.cursor()
|
|
|
|
|
|
|
|
for item in music:
|
|
|
|
self.logMsg("Message : Doing LibraryChanged : Items Removed : Calling deleteItemFromKodiLibrary (musiclibrary): " + item, 0)
|
|
|
|
WriteKodiMusicDB().deleteItemFromKodiLibrary(item, connection, cursor)
|
|
|
|
|
|
|
|
connection.commit()
|
|
|
|
cursor.close()
|
|
|
|
|
|
|
|
if deleteEmbyItem:
|
|
|
|
for item in itemList:
|
|
|
|
url = "{server}/mediabrowser/Items/%s" % item
|
|
|
|
self.logMsg('Deleting via URL: %s' % url)
|
|
|
|
doUtils.downloadUrl(url, type="DELETE")
|
|
|
|
xbmc.executebuiltin("Container.Refresh")
|
|
|
|
|
|
|
|
def remove_items(self, itemsRemoved):
|
|
|
|
|
|
|
|
self.removeItems.extend(itemsRemoved)
|
|
|
|
|
|
|
|
def update_items(self, itemsToUpdate):
|
|
|
|
# doing adds and updates
|
|
|
|
if(len(itemsToUpdate) > 0):
|
|
|
|
self.logMsg("Message : Doing LibraryChanged : Processing Added and Updated : " + str(itemsToUpdate), 0)
|
|
|
|
self.updateItems.extend(itemsToUpdate)
|
|
|
|
self.doIncrementalSync = True
|
|
|
|
|
|
|
|
def user_data_update(self, userDataList):
|
2015-06-28 23:33:58 +10:00
|
|
|
# do full playcount update for now
|
2015-06-28 22:08:06 +10:00
|
|
|
for userData in userDataList:
|
|
|
|
itemId = userData.get("ItemId")
|
|
|
|
if(itemId != None):
|
|
|
|
self.updateItems.append(itemId)
|
|
|
|
if(len(self.updateItems) > 0):
|
|
|
|
self.logMsg("Message : Doing UserDataChanged : Processing Updated : " + str(self.updateItems), 0)
|
|
|
|
self.doIncrementalSync = True
|
|
|
|
|
2015-05-03 03:49:39 +10:00
|
|
|
def ShouldStop(self):
|
|
|
|
|
|
|
|
if(xbmc.abortRequested):
|
2015-03-16 14:10:41 +11:00
|
|
|
return True
|
2015-04-03 19:39:16 +11:00
|
|
|
|
|
|
|
if(WINDOW.getProperty("SyncDatabaseShouldStop") == "true"):
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|
2015-03-17 04:51:49 +11:00
|
|
|
|
2015-06-16 15:53:01 +10:00
|
|
|
def run(self):
|
|
|
|
|
|
|
|
self.logMsg("--- Starting Library Sync Thread ---", 0)
|
|
|
|
WINDOW = xbmcgui.Window(10000)
|
|
|
|
startupComplete = False
|
|
|
|
|
|
|
|
while not self.KodiMonitor.abortRequested():
|
|
|
|
|
|
|
|
# Library sync
|
|
|
|
if not startupComplete:
|
|
|
|
# Run full sync
|
|
|
|
self.logMsg("Doing_Db_Sync: syncDatabase (Started)", 1)
|
|
|
|
libSync = self.FullLibrarySync()
|
|
|
|
self.logMsg("Doing_Db_Sync: syncDatabase (Finished) %s" % libSync, 1)
|
|
|
|
|
|
|
|
if libSync:
|
|
|
|
startupComplete = True
|
|
|
|
|
|
|
|
if WINDOW.getProperty("OnWakeSync") == "true":
|
|
|
|
WINDOW.clearProperty("OnWakeSync")
|
|
|
|
if WINDOW.getProperty("SyncDatabaseRunning") != "true":
|
|
|
|
utils.logMsg("Doing_Db_Sync Post Resume: syncDatabase (Started)",0)
|
|
|
|
libSync = self.FullLibrarySync()
|
|
|
|
utils.logMsg("Doing_Db_Sync Post Resume: syncDatabase (Finished) " + str(libSync),0)
|
|
|
|
|
2015-06-28 22:08:06 +10:00
|
|
|
if self.doIncrementalSync:
|
|
|
|
# Add or update item to Kodi library
|
|
|
|
listItems = self.updateItems
|
|
|
|
self.updateItems = []
|
|
|
|
self.doIncrementalSync = False
|
|
|
|
self.IncrementalSync(listItems)
|
|
|
|
|
|
|
|
if len(self.removeItems) > 0:
|
|
|
|
# Remove item from Kodi library
|
|
|
|
listItems = self.removeItems
|
|
|
|
self.removeItems = []
|
|
|
|
self.removefromDB(listItems)
|
|
|
|
|
2015-06-16 15:53:01 +10:00
|
|
|
if self.KodiMonitor.waitForAbort(1):
|
|
|
|
# Abort was requested while waiting. We should exit
|
|
|
|
break
|
|
|
|
|
2015-06-28 14:37:40 +10:00
|
|
|
self.logMsg("--- Library Sync Thread stopped ---", 0)
|