jellyfin-kodi/resources/lib/database/__init__.py

397 lines
12 KiB
Python
Raw Normal View History

2018-09-06 08:36:32 +00:00
# -*- coding: utf-8 -*-
#################################################################################################
import datetime
import logging
import json
import os
import sqlite3
import xbmc
import xbmcvfs
import emby_db
from helper.utils import delete_folder
2018-09-06 08:36:32 +00:00
from helper import _, settings, window, dialog
2018-10-15 11:26:25 +00:00
from objects import obj
2018-09-06 08:36:32 +00:00
#################################################################################################
LOG = logging.getLogger("EMBY."+__name__)
#################################################################################################
class Database(object):
''' This should be called like a context.
i.e. with Database('emby') as db:
db.cursor
db.conn.commit()
'''
timeout = 120
2019-01-10 23:38:57 +00:00
discovered = False
2018-09-06 08:36:32 +00:00
def __init__(self, file=None, commit_close=True):
''' file: emby, texture, music, video, :memory: or path to file
'''
self.db_file = file or "video"
self.commit_close = commit_close
def __enter__(self):
''' Open the connection and return the Database class.
2019-01-10 23:38:57 +00:00
This is to allow for the cursor, conn and others to be accessible.
2018-09-06 08:36:32 +00:00
'''
2019-01-09 01:51:18 +00:00
self.path = self._sql(self.db_file)
self.conn = sqlite3.connect(self.path, timeout=self.timeout)
2018-09-06 08:36:32 +00:00
self.cursor = self.conn.cursor()
2018-10-08 06:21:30 +00:00
if self.db_file in ('video', 'music', 'texture', 'emby'):
2019-01-10 23:38:57 +00:00
self.conn.execute("PRAGMA journal_mode=WAL") # to avoid writing conflict with kodi
2018-09-06 08:36:32 +00:00
2019-01-10 23:38:57 +00:00
LOG.debug("--->[ database: %s ] %s", self.db_file, id(self.conn))
2018-09-06 08:36:32 +00:00
2018-09-13 00:43:51 +00:00
if not window('emby_db_check.bool') and self.db_file == 'emby':
2018-09-06 08:36:32 +00:00
2018-09-13 00:43:51 +00:00
window('emby_db_check.bool', True)
2018-09-06 08:36:32 +00:00
emby_tables(self.cursor)
self.conn.commit()
return self
2019-01-11 06:39:23 +00:00
def _get_database(self, path, silent=False):
2019-01-09 01:51:18 +00:00
path = xbmc.translatePath(path).decode('utf-8')
if not silent:
if not xbmcvfs.exists(path):
raise Exception("Database: %s missing" % path)
conn = sqlite3.connect(path)
cursor = conn.cursor()
cursor.execute("SELECT name FROM sqlite_master WHERE type='table';")
tables = cursor.fetchall()
conn.close()
if not len(tables):
raise Exception("Database: %s malformed?" % path)
2019-01-09 01:51:18 +00:00
return path
2019-01-10 23:38:57 +00:00
def _discover_database(self, database):
''' Grab the first database encountered, by most recent.
Will likely not work, but heck.
'''
databases = xbmc.translatePath("special://database/").decode('utf-8')
2019-01-10 23:38:57 +00:00
types = {
'video': "MyVideos",
'music': "MyMusic",
'texture': "Textures"
}
database = types[database]
dirs, files = xbmcvfs.listdir(databases)
modified = {'file': None, 'time': 0}
2019-01-10 23:38:57 +00:00
for file in reversed(files):
if file.startswith(database) and not file.endswith('-wal') and not file.endswith('-shm'):
st = xbmcvfs.Stat(databases + file.decode('utf-8'))
modified_int = st.st_mtime()
LOG.debug("Database detected: %s time: %s", file.decode('utf-8'), modified_int)
if modified_int > modified['time']:
modified['time'] = modified_int
modified['file'] = file.decode('utf-8')
LOG.info("Discovered database: %s", modified)
return xbmc.translatePath("special://database/%s" % modified['file']).decode('utf-8')
2019-01-10 23:38:57 +00:00
2018-09-06 08:36:32 +00:00
def _sql(self, file):
2019-01-09 01:51:18 +00:00
''' Get the database path based on the file objects/obj_map.json
Compatible check, in the event multiple db version are supported with the same Kodi version.
2019-01-10 23:38:57 +00:00
Discover by file as a last resort.
2019-01-09 01:51:18 +00:00
'''
2018-09-06 08:36:32 +00:00
databases = obj.Objects().objects
if file not in ('video', 'music', 'texture') or databases.get('database_set%s' % file):
return self._get_database(databases[file], True)
discovered = self._discover_database(file) if not databases.get('database_set%s' % file) else None
2019-01-09 01:51:18 +00:00
try:
loaded = self._get_database(databases[file]) if file in databases else file
2019-01-09 01:51:18 +00:00
except Exception as error:
LOG.error(error)
for i in range(1, 10):
alt_file = "%s-%s" % (file, i)
try:
loaded = self._get_database(databases[alt_file])
2019-01-10 23:38:57 +00:00
except KeyError: # No other db options
loaded = None
break
2019-01-09 01:51:18 +00:00
except Exception:
pass
2019-01-12 04:46:14 +00:00
if discovered and discovered != loaded:
databases[file] = discovered
self.discovered = True
else:
databases[file] = loaded
databases['database_set%s' % file] = True
LOG.info("Database locked in: %s", databases[file])
return databases[file]
2018-09-06 08:36:32 +00:00
def __exit__(self, exc_type, exc_val, exc_tb):
''' Close the connection and cursor.
'''
changes = self.conn.total_changes
if exc_type is not None: # errors raised
LOG.error("type: %s value: %s", exc_type, exc_val)
if self.commit_close and changes:
LOG.info("[%s] %s rows updated.", self.db_file, changes)
self.conn.commit()
LOG.debug("---<[ database: %s ] %s", self.db_file, id(self.conn))
self.cursor.close()
self.conn.close()
def emby_tables(cursor):
''' Create the tables for the emby database.
emby, view, version
'''
cursor.execute(
"""CREATE TABLE IF NOT EXISTS emby(
emby_id TEXT UNIQUE, media_folder TEXT, emby_type TEXT, media_type TEXT,
kodi_id INTEGER, kodi_fileid INTEGER, kodi_pathid INTEGER, parent_id INTEGER,
2018-09-13 00:43:51 +00:00
checksum INTEGER, emby_parent_id TEXT)""")
2018-09-06 08:36:32 +00:00
cursor.execute(
"""CREATE TABLE IF NOT EXISTS view(
view_id TEXT UNIQUE, view_name TEXT, media_type TEXT)""")
cursor.execute("CREATE TABLE IF NOT EXISTS version(idVersion TEXT)")
2018-09-13 00:43:51 +00:00
columns = cursor.execute("SELECT * FROM emby")
if 'emby_parent_id' not in [description[0] for description in columns.description]:
LOG.info("Add missing column emby_parent_id")
cursor.execute("ALTER TABLE emby ADD COLUMN emby_parent_id 'TEXT'")
2018-09-06 08:36:32 +00:00
def reset():
''' Reset both the emby database and the kodi database.
'''
from views import Views
views = Views()
if not dialog("yesno", heading="{emby}", line1=_(33074)):
return
window('emby_should_stop.bool', True)
count = 10
while window('emby_sync.bool'):
LOG.info("Sync is running...")
count -= 1
if not count:
dialog("ok", heading="{emby}", line1=_(33085))
return
if xbmc.Monitor().waitForAbort(1):
return
reset_kodi()
reset_emby()
views.delete_playlists()
views.delete_nodes()
if dialog("yesno", heading="{emby}", line1=_(33086)):
reset_artwork()
addon_data = xbmc.translatePath("special://profile/addon_data/plugin.video.emby/").decode('utf-8')
if dialog("yesno", heading="{emby}", line1=_(33087)):
xbmcvfs.delete(os.path.join(addon_data, "settings.xml"))
xbmcvfs.delete(os.path.join(addon_data, "data.json"))
LOG.info("[ reset settings ]")
if xbmcvfs.exists(os.path.join(addon_data, "sync.json")):
xbmcvfs.delete(os.path.join(addon_data, "sync.json"))
2018-09-18 07:49:08 +00:00
settings('enableMusic.bool', False)
2018-12-13 10:41:34 +00:00
settings('MinimumSetup', "")
2018-09-08 19:40:19 +00:00
settings('MusicRescan.bool', False)
2018-09-06 08:36:32 +00:00
settings('SyncInstallRunDone.bool', False)
dialog("ok", heading="{emby}", line1=_(33088))
xbmc.executebuiltin('RestartApp')
def reset_kodi():
with Database() as videodb:
videodb.cursor.execute("SELECT tbl_name FROM sqlite_master WHERE type='table'")
for table in videodb.cursor.fetchall():
name = table[0]
if name != 'version':
videodb.cursor.execute("DELETE FROM " + name)
if settings('enableMusic.bool') or dialog("yesno", heading="{emby}", line1=_(33162)):
2018-09-06 08:36:32 +00:00
with Database('music') as musicdb:
musicdb.cursor.execute("SELECT tbl_name FROM sqlite_master WHERE type='table'")
for table in musicdb.cursor.fetchall():
name = table[0]
if name != 'version':
musicdb.cursor.execute("DELETE FROM " + name)
LOG.warn("[ reset kodi ]")
def reset_emby():
with Database('emby') as embydb:
embydb.cursor.execute("SELECT tbl_name FROM sqlite_master WHERE type='table'")
for table in embydb.cursor.fetchall():
name = table[0]
if name not in ('version', 'view'):
embydb.cursor.execute("DELETE FROM " + name)
embydb.cursor.execute("DROP table IF EXISTS emby")
embydb.cursor.execute("DROP table IF EXISTS view")
embydb.cursor.execute("DROP table IF EXISTS version")
LOG.warn("[ reset emby ]")
def reset_artwork():
''' Remove all existing texture.
'''
thumbnails = xbmc.translatePath('special://thumbnails/').decode('utf-8')
if xbmcvfs.exists(thumbnails):
dirs, ignore = xbmcvfs.listdir(thumbnails)
for directory in dirs:
ignore, thumbs = xbmcvfs.listdir(os.path.join(thumbnails, directory.decode('utf-8')))
for thumb in thumbs:
LOG.debug("DELETE thumbnail %s", thumb)
xbmcvfs.delete(os.path.join(thumbnails, directory.decode('utf-8'), thumb.decode('utf-8')))
with Database('texture') as texdb:
texdb.cursor.execute("SELECT tbl_name FROM sqlite_master WHERE type='table'")
for table in texdb.cursor.fetchall():
name = table[0]
if name != 'version':
texdb.cursor.execute("DELETE FROM " + name)
LOG.warn("[ reset artwork ]")
def get_sync():
path = xbmc.translatePath("special://profile/addon_data/plugin.video.emby/").decode('utf-8')
if not xbmcvfs.exists(path):
xbmcvfs.mkdirs(path)
try:
with open(os.path.join(path, 'sync.json')) as infile:
sync = json.load(infile)
except Exception:
sync = {}
2018-09-13 06:57:57 +00:00
sync['Libraries'] = sync.get('Libraries', [])
sync['RestorePoint'] = sync.get('RestorePoint', {})
2018-09-13 06:44:39 +00:00
sync['Whitelist'] = list(set(sync.get('Whitelist', [])))
2018-09-13 06:57:57 +00:00
sync['SortedViews'] = sync.get('SortedViews', [])
2018-09-06 08:36:32 +00:00
return sync
def save_sync(sync):
path = xbmc.translatePath("special://profile/addon_data/plugin.video.emby/").decode('utf-8')
if not xbmcvfs.exists(path):
xbmcvfs.mkdirs(path)
sync['Date'] = datetime.datetime.utcnow().strftime('%Y-%m-%dT%H:%M:%SZ')
with open(os.path.join(path, 'sync.json'), 'w') as outfile:
json.dump(sync, outfile, sort_keys=True, indent=4, ensure_ascii=False)
def get_credentials():
path = xbmc.translatePath("special://profile/addon_data/plugin.video.emby/").decode('utf-8')
if not xbmcvfs.exists(path):
xbmcvfs.mkdirs(path)
try:
with open(os.path.join(path, 'data.json')) as infile:
credentials = json.load(infile)
except Exception:
try:
with open(os.path.join(path, 'data.txt')) as infile:
credentials = json.load(infile)
save_credentials(credentials)
xbmcvfs.delete(os.path.join(path, 'data.txt'))
except Exception:
credentials = {}
credentials['Servers'] = credentials.get('Servers', [])
2018-09-06 08:36:32 +00:00
return credentials
def save_credentials(credentials):
credentials = credentials or {}
path = xbmc.translatePath("special://profile/addon_data/plugin.video.emby/").decode('utf-8')
if not xbmcvfs.exists(path):
xbmcvfs.mkdirs(path)
with open(os.path.join(path, 'data.json'), 'w') as outfile:
json.dump(credentials, outfile, sort_keys=True, indent=4, ensure_ascii=False)
def get_item(kodi_id, media):
''' Get emby item based on kodi id and media.
'''
with Database('emby') as embydb:
item = emby_db.EmbyDatabase(embydb.cursor).get_full_item_by_kodi_id(kodi_id, media)
if not item:
LOG.debug("Not an emby item")
return
return item