Plex-Meta-Manager/modules/cache.py

263 lines
14 KiB
Python
Raw Normal View History

2021-01-20 21:37:59 +00:00
import logging, os, random, sqlite3
from contextlib import closing
from datetime import datetime, timedelta
logger = logging.getLogger("Plex Meta Manager")
class Cache:
def __init__(self, config_path, expiration):
2021-02-24 06:44:06 +00:00
cache = f"{os.path.splitext(config_path)[0]}.cache"
2021-01-20 21:37:59 +00:00
with sqlite3.connect(cache) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
2021-05-08 23:49:55 +00:00
cursor.execute("SELECT count(name) FROM sqlite_master WHERE type='table' AND name='guid_map'")
2021-01-20 21:37:59 +00:00
if cursor.fetchone()[0] == 0:
2021-02-24 06:44:06 +00:00
logger.info(f"Initializing cache database at {cache}")
2021-01-20 21:37:59 +00:00
else:
2021-02-24 06:44:06 +00:00
logger.info(f"Using cache database at {cache}")
2021-05-08 23:49:55 +00:00
cursor.execute("DROP TABLE IF EXISTS guids")
2021-05-27 17:41:04 +00:00
cursor.execute("DROP TABLE IF EXISTS imdb_to_tvdb_map")
cursor.execute("DROP TABLE IF EXISTS tmdb_to_tvdb_map")
2021-05-09 05:38:41 +00:00
cursor.execute("DROP TABLE IF EXISTS imdb_map")
2021-03-05 16:04:28 +00:00
cursor.execute(
2021-05-08 23:49:55 +00:00
"""CREATE TABLE IF NOT EXISTS guid_map (
2021-03-05 16:04:28 +00:00
INTEGER PRIMARY KEY,
plex_guid TEXT UNIQUE,
2021-05-08 23:49:55 +00:00
t_id TEXT,
media_type TEXT,
expiration_date TEXT)"""
2021-03-05 16:04:28 +00:00
)
cursor.execute(
2021-05-09 05:38:41 +00:00
"""CREATE TABLE IF NOT EXISTS imdb_to_tmdb_map (
2021-03-05 16:04:28 +00:00
INTEGER PRIMARY KEY,
imdb_id TEXT UNIQUE,
2021-05-09 05:38:41 +00:00
tmdb_id TEXT,
2021-05-08 23:49:55 +00:00
media_type TEXT,
expiration_date TEXT)"""
2021-03-05 16:04:28 +00:00
)
2021-05-09 05:38:41 +00:00
cursor.execute(
2021-05-27 17:41:04 +00:00
"""CREATE TABLE IF NOT EXISTS imdb_to_tvdb_map2 (
2021-05-09 05:38:41 +00:00
INTEGER PRIMARY KEY,
imdb_id TEXT UNIQUE,
2021-05-27 17:41:04 +00:00
tvdb_id TEXT,
2021-05-09 05:38:41 +00:00
expiration_date TEXT)"""
)
cursor.execute(
2021-05-27 17:41:04 +00:00
"""CREATE TABLE IF NOT EXISTS tmdb_to_tvdb_map2 (
2021-05-09 05:38:41 +00:00
INTEGER PRIMARY KEY,
tmdb_id TEXT UNIQUE,
2021-05-27 17:41:04 +00:00
tvdb_id TEXT,
2021-05-09 05:38:41 +00:00
expiration_date TEXT)"""
)
2021-03-12 19:49:23 +00:00
cursor.execute(
"""CREATE TABLE IF NOT EXISTS letterboxd_map (
INTEGER PRIMARY KEY,
letterboxd_id TEXT UNIQUE,
tmdb_id TEXT,
expiration_date TEXT)"""
)
2021-03-05 16:04:28 +00:00
cursor.execute(
"""CREATE TABLE IF NOT EXISTS omdb_data (
INTEGER PRIMARY KEY,
imdb_id TEXT UNIQUE,
title TEXT,
year INTEGER,
content_rating TEXT,
genres TEXT,
imdb_rating REAL,
imdb_votes INTEGER,
metacritic_rating INTEGER,
type TEXT,
expiration_date TEXT)"""
)
2021-04-22 18:03:27 +00:00
cursor.execute(
"""CREATE TABLE IF NOT EXISTS anime_map (
INTEGER PRIMARY KEY,
anidb TEXT UNIQUE,
anilist TEXT,
myanimelist TEXT,
kitsu TEXT,
expiration_date TEXT)"""
)
cursor.execute(
"""CREATE TABLE IF NOT EXISTS image_map (
INTEGER PRIMARY KEY,
rating_key TEXT,
library TEXT,
type TEXT,
2021-06-11 14:26:11 +00:00
overlay TEXT,
compare TEXT,
location TEXT)"""
)
2021-01-20 21:37:59 +00:00
self.expiration = expiration
self.cache_path = cache
2021-05-08 23:49:55 +00:00
def query_guid_map(self, plex_guid):
2021-01-20 21:37:59 +00:00
id_to_return = None
2021-05-08 23:49:55 +00:00
media_type = None
2021-01-20 21:37:59 +00:00
expired = None
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
2021-05-08 23:49:55 +00:00
cursor.execute(f"SELECT * FROM guid_map WHERE plex_guid = ?", (plex_guid,))
2021-01-20 21:37:59 +00:00
row = cursor.fetchone()
2021-05-08 23:49:55 +00:00
if row:
time_between_insertion = datetime.now() - datetime.strptime(row["expiration_date"], "%Y-%m-%d")
id_to_return = row["t_id"]
media_type = row["media_type"]
2021-01-20 21:37:59 +00:00
expired = time_between_insertion.days > self.expiration
2021-05-08 23:49:55 +00:00
return id_to_return, media_type, expired
2021-01-20 21:37:59 +00:00
2021-05-09 05:38:41 +00:00
def update_guid_map(self, media_type, plex_guid, t_id, expired):
self._update_map("guid_map", "plex_guid", plex_guid, "t_id", t_id, expired, media_type=media_type)
2021-03-05 16:04:28 +00:00
2021-05-09 05:38:41 +00:00
def query_imdb_to_tmdb_map(self, media_type, _id, imdb=True):
from_id = "imdb_id" if imdb else "tmdb_id"
to_id = "tmdb_id" if imdb else "imdb_id"
return self._query_map("imdb_to_tmdb_map", _id, from_id, to_id, media_type=media_type)
2021-01-20 21:37:59 +00:00
2021-05-09 05:38:41 +00:00
def update_imdb_to_tmdb_map(self, media_type, expired, imdb_id, tmdb_id):
self._update_map("imdb_to_tmdb_map", "imdb_id", imdb_id, "tmdb_id", tmdb_id, expired, media_type=media_type)
2021-01-20 21:37:59 +00:00
2021-05-09 05:38:41 +00:00
def query_imdb_to_tvdb_map(self, _id, imdb=True):
from_id = "imdb_id" if imdb else "tvdb_id"
to_id = "tvdb_id" if imdb else "imdb_id"
2021-05-27 17:41:04 +00:00
return self._query_map("imdb_to_tvdb_map2", _id, from_id, to_id)
2021-05-09 05:38:41 +00:00
def update_imdb_to_tvdb_map(self, expired, imdb_id, tvdb_id):
2021-05-27 17:41:04 +00:00
self._update_map("imdb_to_tvdb_map2", "imdb_id", imdb_id, "tvdb_id", tvdb_id, expired)
2021-05-09 05:38:41 +00:00
def query_tmdb_to_tvdb_map(self, _id, tmdb=True):
from_id = "tmdb_id" if tmdb else "tvdb_id"
to_id = "tvdb_id" if tmdb else "tmdb_id"
2021-05-27 17:41:04 +00:00
return self._query_map("tmdb_to_tvdb_map2", _id, from_id, to_id)
2021-05-09 05:38:41 +00:00
def update_tmdb_to_tvdb_map(self, expired, tmdb_id, tvdb_id):
2021-05-27 17:41:04 +00:00
self._update_map("tmdb_to_tvdb_map2", "tmdb_id", tmdb_id, "tvdb_id", tvdb_id, expired)
2021-03-05 16:04:28 +00:00
2021-03-12 19:49:23 +00:00
def query_letterboxd_map(self, letterboxd_id):
2021-05-09 05:38:41 +00:00
return self._query_map("letterboxd_map", letterboxd_id, "letterboxd_id", "tmdb_id")
def update_letterboxd_map(self, expired, letterboxd_id, tmdb_id):
self._update_map("letterboxd_map", "letterboxd_id", letterboxd_id, "tmdb_id", tmdb_id, expired)
def _query_map(self, map_name, _id, from_id, to_id, media_type=None):
id_to_return = None
2021-03-12 19:49:23 +00:00
expired = None
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
2021-05-09 05:38:41 +00:00
if media_type is None:
cursor.execute(f"SELECT * FROM {map_name} WHERE {from_id} = ?", (_id,))
else:
cursor.execute(f"SELECT * FROM {map_name} WHERE {from_id} = ? AND media_type = ?", (_id, media_type))
2021-03-12 19:49:23 +00:00
row = cursor.fetchone()
2021-05-09 05:38:41 +00:00
if row and row[to_id]:
2021-03-12 19:49:23 +00:00
datetime_object = datetime.strptime(row["expiration_date"], "%Y-%m-%d")
time_between_insertion = datetime.now() - datetime_object
2021-06-20 22:31:33 +00:00
id_to_return = row[to_id] if to_id == "imdb_id" else int(row[to_id])
2021-03-12 19:49:23 +00:00
expired = time_between_insertion.days > self.expiration
2021-05-09 05:38:41 +00:00
return id_to_return, expired
2021-03-12 19:49:23 +00:00
2021-05-09 05:49:13 +00:00
def _update_map(self, map_name, val1_name, val1, val2_name, val2, expired, media_type=None):
2021-03-12 19:49:23 +00:00
expiration_date = datetime.now() if expired is True else (datetime.now() - timedelta(days=random.randint(1, self.expiration)))
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
2021-05-09 05:38:41 +00:00
cursor.execute(f"INSERT OR IGNORE INTO {map_name}({val1_name}) VALUES(?)", (val1,))
if media_type is None:
sql = f"UPDATE {map_name} SET {val2_name} = ?, expiration_date = ? WHERE {val1_name} = ?"
cursor.execute(sql, (val2, expiration_date.strftime("%Y-%m-%d"), val1))
else:
sql = f"UPDATE {map_name} SET {val2_name} = ?, expiration_date = ?{'' if media_type is None else ', media_type = ?'} WHERE {val1_name} = ?"
cursor.execute(sql, (val2, expiration_date.strftime("%Y-%m-%d"), media_type, val1))
2021-03-12 19:49:23 +00:00
2021-03-05 16:04:28 +00:00
def query_omdb(self, imdb_id):
omdb_dict = {}
expired = None
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
cursor.execute("SELECT * FROM omdb_data WHERE imdb_id = ?", (imdb_id,))
row = cursor.fetchone()
if row:
omdb_dict["imdbID"] = row["imdb_id"] if row["imdb_id"] else None
omdb_dict["Title"] = row["title"] if row["title"] else None
omdb_dict["Year"] = row["year"] if row["year"] else None
omdb_dict["Rated"] = row["content_rating"] if row["content_rating"] else None
omdb_dict["Genre"] = row["genres"] if row["genres"] else None
omdb_dict["imdbRating"] = row["imdb_rating"] if row["imdb_rating"] else None
omdb_dict["imdbVotes"] = row["imdb_votes"] if row["imdb_votes"] else None
omdb_dict["Metascore"] = row["metacritic_rating"] if row["metacritic_rating"] else None
omdb_dict["Type"] = row["type"] if row["type"] else None
2021-06-24 21:42:15 +00:00
omdb_dict["Response"] = "True"
2021-03-05 16:04:28 +00:00
datetime_object = datetime.strptime(row["expiration_date"], "%Y-%m-%d")
time_between_insertion = datetime.now() - datetime_object
2021-03-05 20:33:24 +00:00
expired = time_between_insertion.days > self.expiration
2021-03-05 16:04:28 +00:00
return omdb_dict, expired
def update_omdb(self, expired, omdb):
2021-03-05 20:33:24 +00:00
expiration_date = datetime.now() if expired is True else (datetime.now() - timedelta(days=random.randint(1, self.expiration)))
2021-03-05 16:04:28 +00:00
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
cursor.execute("INSERT OR IGNORE INTO omdb_data(imdb_id) VALUES(?)", (omdb.imdb_id,))
update_sql = "UPDATE omdb_data SET title = ?, year = ?, content_rating = ?, genres = ?, imdb_rating = ?, imdb_votes = ?, metacritic_rating = ?, type = ?, expiration_date = ? WHERE imdb_id = ?"
cursor.execute(update_sql, (omdb.title, omdb.year, omdb.content_rating, omdb.genres_str, omdb.imdb_rating, omdb.imdb_votes, omdb.metacritic_rating, omdb.type, expiration_date.strftime("%Y-%m-%d"), omdb.imdb_id))
2021-04-22 18:03:27 +00:00
def query_anime_map(self, anime_id, id_type):
ids = None
expired = None
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
cursor.execute(f"SELECT * FROM anime_map WHERE {id_type} = ?", (anime_id, ))
row = cursor.fetchone()
if row and row["anidb"]:
datetime_object = datetime.strptime(row["expiration_date"], "%Y-%m-%d")
time_between_insertion = datetime.now() - datetime_object
ids = {
"anilist": int(row["anilist"]) if row["anilist"] else None,
"anidb": int(row["anidb"]) if row["anidb"] else None,
"myanimelist": int(row["myanimelist"]) if row["myanimelist"] else None,
"kitsu": int(row["kitsu"]) if row["kitsu"] else None
}
expired = time_between_insertion.days > self.expiration
return ids, expired
2021-05-09 05:38:41 +00:00
def update_anime_map(self, expired, anime_ids):
2021-04-22 18:03:27 +00:00
expiration_date = datetime.now() if expired is True else (datetime.now() - timedelta(days=random.randint(1, self.expiration)))
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
cursor.execute("INSERT OR IGNORE INTO anime_map(anidb) VALUES(?)", (anime_ids["anidb"],))
cursor.execute("UPDATE anime_map SET anilist = ?, myanimelist = ?, kitsu = ?, expiration_date = ? WHERE anidb = ?", (anime_ids["anidb"], anime_ids["myanimelist"], anime_ids["kitsu"], expiration_date.strftime("%Y-%m-%d"), anime_ids["anidb"]))
2021-06-30 03:08:38 +00:00
def query_image_map_overlay(self, library, image_type, overlay):
rks = []
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
cursor.execute(f"SELECT * FROM image_map WHERE overlay = ? AND library = ? AND type = ?", (overlay, library, image_type))
rows = cursor.fetchall()
for row in rows:
rks.append(int(row["rating_key"]))
return rks
def query_image_map(self, rating_key, library, image_type):
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
cursor.execute(f"SELECT * FROM image_map WHERE rating_key = ? AND library = ? AND type = ?", (rating_key, library, image_type))
row = cursor.fetchone()
if row and row["location"]:
2021-06-30 03:08:38 +00:00
return row["location"], row["compare"], row["overlay"]
return None, None, None
2021-06-30 03:08:38 +00:00
def update_image_map(self, rating_key, library, image_type, location, compare, overlay):
with sqlite3.connect(self.cache_path) as connection:
connection.row_factory = sqlite3.Row
with closing(connection.cursor()) as cursor:
2021-06-12 15:29:17 +00:00
cursor.execute("INSERT OR IGNORE INTO image_map(rating_key, library, type) VALUES(?, ?, ?)", (rating_key, library, image_type))
2021-06-30 03:08:38 +00:00
cursor.execute("UPDATE image_map SET location = ?, compare = ?, overlay = ? WHERE rating_key = ? AND library = ? AND type = ?", (location, compare, overlay, rating_key, library, image_type))