2015-03-13 22:24:59 +01:00
|
|
|
#################################################################################################
|
|
|
|
# utils
|
|
|
|
#################################################################################################
|
|
|
|
|
|
|
|
import xbmc
|
|
|
|
import xbmcgui
|
|
|
|
import xbmcaddon
|
|
|
|
import xbmcvfs
|
|
|
|
import json
|
|
|
|
import os
|
2015-03-20 10:48:59 +11:00
|
|
|
import cProfile
|
|
|
|
import pstats
|
|
|
|
import time
|
2015-03-13 22:24:59 +01:00
|
|
|
import inspect
|
2015-03-27 12:20:40 +01:00
|
|
|
import sqlite3
|
2015-03-30 00:43:53 +02:00
|
|
|
import string
|
|
|
|
import unicodedata
|
2015-05-06 23:24:13 +02:00
|
|
|
|
2015-03-13 22:24:59 +01:00
|
|
|
|
|
|
|
from API import API
|
|
|
|
from PlayUtils import PlayUtils
|
|
|
|
from DownloadUtils import DownloadUtils
|
|
|
|
downloadUtils = DownloadUtils()
|
2015-07-22 08:16:08 -05:00
|
|
|
addonSettings = xbmcaddon.Addon()
|
2015-03-23 11:30:03 +00:00
|
|
|
language = addonSettings.getLocalizedString
|
2015-03-26 22:35:11 +01:00
|
|
|
|
2015-03-13 22:24:59 +01:00
|
|
|
|
|
|
|
def logMsg(title, msg, level = 1):
|
2015-06-05 05:12:09 -05:00
|
|
|
|
2015-04-28 17:23:26 -05:00
|
|
|
WINDOW = xbmcgui.Window(10000)
|
2015-06-05 05:12:09 -05:00
|
|
|
# Get the logLevel set in UserClient
|
|
|
|
logLevel = int(WINDOW.getProperty('getLogLevel'))
|
|
|
|
|
2015-03-13 22:24:59 +01:00
|
|
|
if(logLevel >= level):
|
2015-03-20 10:48:59 +11:00
|
|
|
if(logLevel == 2): # inspect.stack() is expensive
|
2015-03-13 22:24:59 +01:00
|
|
|
try:
|
|
|
|
xbmc.log(title + " -> " + inspect.stack()[1][3] + " : " + str(msg))
|
|
|
|
except UnicodeEncodeError:
|
|
|
|
xbmc.log(title + " -> " + inspect.stack()[1][3] + " : " + str(msg.encode('utf-8')))
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
xbmc.log(title + " -> " + str(msg))
|
|
|
|
except UnicodeEncodeError:
|
|
|
|
xbmc.log(title + " -> " + str(msg.encode('utf-8')))
|
2015-03-16 22:31:32 +01:00
|
|
|
|
|
|
|
def convertEncoding(data):
|
|
|
|
#nasty hack to make sure we have a unicode string
|
|
|
|
try:
|
|
|
|
return data.decode('utf-8')
|
|
|
|
except:
|
|
|
|
return data
|
|
|
|
|
2015-05-08 00:04:40 +02:00
|
|
|
def KodiSQL(type="video"):
|
|
|
|
|
|
|
|
if type == "music":
|
|
|
|
dbPath = getKodiMusicDBPath()
|
2015-06-19 05:53:22 -05:00
|
|
|
elif type == "texture":
|
|
|
|
dbPath = xbmc.translatePath("special://database/Textures13.db")
|
2015-05-08 00:04:40 +02:00
|
|
|
else:
|
|
|
|
dbPath = getKodiVideoDBPath()
|
|
|
|
|
|
|
|
connection = sqlite3.connect(dbPath)
|
2015-05-01 13:30:21 +02:00
|
|
|
|
2015-04-03 10:58:21 +01:00
|
|
|
return connection
|
|
|
|
|
2015-05-08 00:04:40 +02:00
|
|
|
def getKodiVideoDBPath():
|
2015-06-15 19:41:40 -05:00
|
|
|
|
2015-06-18 05:42:57 -05:00
|
|
|
kodibuild = xbmc.getInfoLabel("System.BuildVersion")
|
2015-06-17 21:34:45 -05:00
|
|
|
|
2015-06-18 05:42:57 -05:00
|
|
|
if kodibuild.startswith("13"):
|
|
|
|
# Gotham
|
|
|
|
dbVersion = "78"
|
|
|
|
elif kodibuild.startswith("14"):
|
|
|
|
# Helix
|
|
|
|
dbVersion = "90"
|
|
|
|
elif kodibuild.startswith("15"):
|
|
|
|
# Isengard
|
|
|
|
dbVersion = "93"
|
2015-07-24 12:29:41 +01:00
|
|
|
elif kodibuild.startswith("16"):
|
|
|
|
# Jarvis
|
|
|
|
dbVersion = "94"
|
2015-06-18 05:42:57 -05:00
|
|
|
else:
|
|
|
|
# Not a compatible build
|
|
|
|
xbmc.log("This Kodi version is incompatible. Current version: %s" % kodibuild)
|
2015-06-15 19:41:40 -05:00
|
|
|
|
2015-06-18 05:42:57 -05:00
|
|
|
dbPath = xbmc.translatePath("special://profile/Database/MyVideos" + dbVersion + ".db")
|
2015-04-03 10:58:21 +01:00
|
|
|
|
2015-06-18 05:42:57 -05:00
|
|
|
return dbPath
|
2015-03-27 11:24:49 +11:00
|
|
|
|
2015-05-08 00:04:40 +02:00
|
|
|
def getKodiMusicDBPath():
|
2015-06-18 05:42:57 -05:00
|
|
|
if xbmc.getInfoLabel("System.BuildVersion").startswith("13"):
|
|
|
|
#gotham
|
|
|
|
dbVersion = "46"
|
2015-07-22 14:24:24 +01:00
|
|
|
elif xbmc.getInfoLabel("System.BuildVersion").startswith("14"):
|
|
|
|
#helix
|
|
|
|
dbVersion = "48"
|
2015-06-18 05:42:57 -05:00
|
|
|
elif xbmc.getInfoLabel("System.BuildVersion").startswith("15"):
|
|
|
|
#isengard
|
|
|
|
dbVersion = "52"
|
2015-07-24 12:29:41 +01:00
|
|
|
elif xbmc.getInfoLabel("System.BuildVersion").startswith("16"):
|
|
|
|
#jarvis
|
|
|
|
dbVersion = "53"
|
2015-06-18 05:42:57 -05:00
|
|
|
else:
|
2015-07-22 14:24:24 +01:00
|
|
|
# Not a compatible build
|
|
|
|
xbmc.log("This Kodi version is incompatible. Current version: %s" % kodibuild)
|
|
|
|
|
2015-06-18 05:42:57 -05:00
|
|
|
|
|
|
|
dbPath = xbmc.translatePath("special://profile/Database/MyMusic" + dbVersion + ".db")
|
|
|
|
|
|
|
|
return dbPath
|
2015-03-13 22:24:59 +01:00
|
|
|
|
|
|
|
def prettifyXml(elem):
|
|
|
|
rough_string = etree.tostring(elem, "utf-8")
|
|
|
|
reparsed = minidom.parseString(rough_string)
|
2015-07-22 08:16:08 -05:00
|
|
|
return reparsed.toprettyxml(indent="\t")
|
2015-03-13 22:24:59 +01:00
|
|
|
|
2015-03-20 10:48:59 +11:00
|
|
|
def startProfiling():
|
|
|
|
pr = cProfile.Profile()
|
|
|
|
pr.enable()
|
|
|
|
return pr
|
|
|
|
|
|
|
|
def stopProfiling(pr, profileName):
|
|
|
|
pr.disable()
|
|
|
|
ps = pstats.Stats(pr)
|
|
|
|
|
2015-03-25 18:37:21 +01:00
|
|
|
addondir = xbmc.translatePath(xbmcaddon.Addon(id='plugin.video.emby').getAddonInfo('profile'))
|
2015-03-20 10:48:59 +11:00
|
|
|
|
|
|
|
fileTimeStamp = time.strftime("%Y-%m-%d %H-%M-%S")
|
|
|
|
tabFileNamepath = os.path.join(addondir, "profiles")
|
|
|
|
tabFileName = os.path.join(addondir, "profiles" , profileName + "_profile_(" + fileTimeStamp + ").tab")
|
|
|
|
|
|
|
|
if not xbmcvfs.exists(tabFileNamepath):
|
|
|
|
xbmcvfs.mkdir(tabFileNamepath)
|
|
|
|
|
|
|
|
f = open(tabFileName, 'wb')
|
|
|
|
f.write("NumbCalls\tTotalTime\tCumulativeTime\tFunctionName\tFileName\r\n")
|
|
|
|
for (key, value) in ps.stats.items():
|
|
|
|
(filename, count, func_name) = key
|
|
|
|
(ccalls, ncalls, total_time, cumulative_time, callers) = value
|
|
|
|
try:
|
|
|
|
f.write(str(ncalls) + "\t" + "{:10.4f}".format(total_time) + "\t" + "{:10.4f}".format(cumulative_time) + "\t" + func_name + "\t" + filename + "\r\n")
|
|
|
|
except ValueError:
|
|
|
|
f.write(str(ncalls) + "\t" + "{0}".format(total_time) + "\t" + "{0}".format(cumulative_time) + "\t" + func_name + "\t" + filename + "\r\n")
|
|
|
|
f.close()
|
|
|
|
|
2015-07-19 20:35:14 -05:00
|
|
|
def createSources():
|
|
|
|
# To make Master lock compatible
|
|
|
|
path = xbmc.translatePath("special://profile/").decode("utf-8")
|
|
|
|
xmlpath = "%ssources.xml" % path
|
|
|
|
|
2015-07-26 07:09:09 -05:00
|
|
|
if xbmcvfs.exists(xmlpath):
|
|
|
|
# add some way to writing dummy path to existing sources.xml
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
sources = open(xmlpath, 'w')
|
|
|
|
sources.write(
|
|
|
|
|
|
|
|
'<sources>\n\t'
|
|
|
|
'<programs>\n\t\t'
|
|
|
|
'<default pathversion="1"></default>\n\t'
|
|
|
|
'</programs>\n\t'
|
|
|
|
'<video>\n\t\t'
|
|
|
|
'<default pathversion="1"></default>\n\t\t'
|
|
|
|
'<source>\n\t\t\t'
|
2015-07-27 04:24:00 -05:00
|
|
|
'<name>Emby</name>\n\t\t\t'
|
2015-07-26 07:09:09 -05:00
|
|
|
'<path pathversion="1">smb://embydummy/dummypath1/</path>\n\t\t\t'
|
|
|
|
'<allowsharing>true</allowsharing>\n\t\t'
|
|
|
|
'</source>\n\t\t'
|
|
|
|
'<source>\n\t\t\t'
|
2015-07-27 04:24:00 -05:00
|
|
|
'<name>Emby</name>\n\t\t\t'
|
2015-07-26 07:09:09 -05:00
|
|
|
'<path pathversion="1">smb://embydummy/dummypath2/</path>\n\t\t\t'
|
|
|
|
'<allowsharing>true</allowsharing>\n\t\t'
|
|
|
|
'</source>\n\t'
|
|
|
|
'</video>\n\t'
|
|
|
|
'<music>\n\t\t'
|
|
|
|
'<default pathversion="1"></default>\n\t'
|
|
|
|
'</music>\n\t'
|
|
|
|
'<pictures>\n\t\t'
|
|
|
|
'<default pathversion="1"></default>\n\t'
|
|
|
|
'</pictures>\n\t'
|
|
|
|
'<files>\n\t\t'
|
|
|
|
'<default pathversion="1"></default>\n\t'
|
|
|
|
'</files>\n'
|
|
|
|
'</sources>'
|
|
|
|
)
|
2015-07-19 20:35:14 -05:00
|
|
|
|
2015-03-30 00:43:53 +02:00
|
|
|
def CleanName(filename):
|
|
|
|
validFilenameChars = "-_.() %s%s" % (string.ascii_letters, string.digits)
|
|
|
|
cleanedFilename = unicodedata.normalize('NFKD', filename).encode('ASCII', 'ignore')
|
|
|
|
return ''.join(c for c in cleanedFilename if c in validFilenameChars)
|
2015-06-19 03:10:41 -05:00
|
|
|
|
|
|
|
def normalize_string(text):
|
|
|
|
try:
|
|
|
|
text = text.replace(":", "")
|
|
|
|
text = text.replace("/", "-")
|
|
|
|
text = text.replace("\\", "-")
|
2015-06-20 18:05:54 +01:00
|
|
|
text = text.replace("<", "")
|
|
|
|
text = text.replace(">", "")
|
|
|
|
text = text.replace("*", "")
|
|
|
|
text = text.replace("?", "")
|
|
|
|
text = text.replace('|', "")
|
2015-06-19 03:10:41 -05:00
|
|
|
text = text.strip()
|
|
|
|
# Remove dots from the last character as windows can not have directories
|
|
|
|
# with dots at the end
|
|
|
|
text = text.rstrip('.')
|
|
|
|
text = unicodedata.normalize('NFKD', unicode(text, 'utf-8')).encode('ascii', 'ignore')
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
return text
|
2015-03-17 13:41:26 -05:00
|
|
|
|
2015-04-02 20:47:06 +01:00
|
|
|
|
|
|
|
def reset():
|
2015-04-03 19:39:16 +11:00
|
|
|
|
2015-07-22 08:16:08 -05:00
|
|
|
WINDOW = xbmcgui.Window( 10000 )
|
2015-04-15 12:20:08 +10:00
|
|
|
return_value = xbmcgui.Dialog().yesno("Warning", "Are you sure you want to reset your local Kodi database?")
|
2015-04-12 08:49:36 -05:00
|
|
|
|
2015-04-03 19:39:16 +11:00
|
|
|
if return_value == 0:
|
|
|
|
return
|
2015-04-12 08:49:36 -05:00
|
|
|
|
2015-05-05 01:43:46 +02:00
|
|
|
#cleanup video nodes
|
|
|
|
import shutil
|
2015-05-05 21:45:29 -05:00
|
|
|
path = "special://profile/library/video/"
|
2015-05-05 01:43:46 +02:00
|
|
|
if xbmcvfs.exists(path):
|
|
|
|
allDirs, allFiles = xbmcvfs.listdir(path)
|
|
|
|
for dir in allDirs:
|
|
|
|
if dir.startswith("Emby "):
|
2015-05-05 21:45:29 -05:00
|
|
|
shutil.rmtree(xbmc.translatePath("special://profile/library/video/" + dir))
|
2015-05-05 16:16:34 +02:00
|
|
|
for file in allFiles:
|
|
|
|
if file.startswith("emby"):
|
|
|
|
xbmcvfs.delete(path + file)
|
2015-05-05 01:43:46 +02:00
|
|
|
|
2015-04-12 08:49:36 -05:00
|
|
|
# Ask if user information should be deleted too.
|
2015-04-15 12:20:08 +10:00
|
|
|
return_user = xbmcgui.Dialog().yesno("Warning", "Reset all Emby Addon settings?")
|
2015-04-12 08:49:36 -05:00
|
|
|
if return_user == 1:
|
2015-07-22 08:16:08 -05:00
|
|
|
WINDOW.setProperty('deletesettings', "true")
|
2015-04-03 19:39:16 +11:00
|
|
|
|
|
|
|
# first stop any db sync
|
|
|
|
WINDOW.setProperty("SyncDatabaseShouldStop", "true")
|
|
|
|
|
|
|
|
count = 0
|
|
|
|
while(WINDOW.getProperty("SyncDatabaseRunning") == "true"):
|
2015-04-15 12:20:08 +10:00
|
|
|
xbmc.log("Sync Running, will wait : " + str(count))
|
2015-04-03 19:39:16 +11:00
|
|
|
count += 1
|
|
|
|
if(count > 10):
|
2015-04-03 11:49:39 +01:00
|
|
|
dialog = xbmcgui.Dialog()
|
2015-04-03 19:39:16 +11:00
|
|
|
dialog.ok('Warning', 'Could not stop DB sync, you should try again.')
|
|
|
|
return
|
|
|
|
xbmc.sleep(1000)
|
2015-04-12 18:34:00 +10:00
|
|
|
|
2015-05-08 00:46:41 +02:00
|
|
|
# delete video db table data
|
|
|
|
print "Doing Video DB Reset"
|
|
|
|
connection = KodiSQL("video")
|
2015-04-12 18:34:00 +10:00
|
|
|
cursor = connection.cursor( )
|
|
|
|
cursor.execute('SELECT tbl_name FROM sqlite_master WHERE type="table"')
|
|
|
|
rows = cursor.fetchall()
|
|
|
|
for row in rows:
|
|
|
|
tableName = row[0]
|
|
|
|
if(tableName != "version"):
|
|
|
|
cursor.execute("DELETE FROM " + tableName)
|
|
|
|
connection.commit()
|
|
|
|
cursor.close()
|
|
|
|
|
2015-05-30 18:02:45 -05:00
|
|
|
if addonSettings.getSetting("enableMusicSync") == "true":
|
2015-05-08 00:46:41 +02:00
|
|
|
# delete video db table data
|
|
|
|
print "Doing Music DB Reset"
|
|
|
|
connection = KodiSQL("music")
|
|
|
|
cursor = connection.cursor( )
|
|
|
|
cursor.execute('SELECT tbl_name FROM sqlite_master WHERE type="table"')
|
|
|
|
rows = cursor.fetchall()
|
|
|
|
for row in rows:
|
|
|
|
tableName = row[0]
|
|
|
|
if(tableName != "version"):
|
|
|
|
cursor.execute("DELETE FROM " + tableName)
|
|
|
|
connection.commit()
|
|
|
|
cursor.close()
|
|
|
|
|
|
|
|
|
2015-04-12 18:34:00 +10:00
|
|
|
# reset the install run flag
|
2015-04-12 04:49:02 -05:00
|
|
|
WINDOW.setProperty("SyncInstallRunDone", "false")
|
2015-04-12 18:34:00 +10:00
|
|
|
|
2015-04-02 20:47:06 +01:00
|
|
|
dialog = xbmcgui.Dialog()
|
2015-04-12 08:49:36 -05:00
|
|
|
dialog.ok('Emby Reset', 'Database reset has completed, Kodi will now restart to apply the changes.')
|
2015-07-22 08:16:08 -05:00
|
|
|
xbmc.executebuiltin("RestartApp")
|