From 6f5100c5410815b2a3fedd76c0da82624cfd2064 Mon Sep 17 00:00:00 2001 From: Prinz23 Date: Mon, 17 Jul 2017 20:29:32 +0100 Subject: [PATCH] Change showupdate add integrity safeguards and improve image processing. Add an episode view filter, improve WEB PROPER selection. Change show update, don't delete any ep in DB if eps are not returned from indexer. Change prevent unneeded error message during show update. Change improve performance, don't fetch episode list when retrieving a show image. Change don't remove episodes from DB with status: SNATCHED, SNATCHED_PROPER, SNATCHED_BEST, DOWNLOADED, ARCHIVED, IGNORED. Change add additional episode removal protections for TVDb_api v2. Change filter SKIPPED items from episode view. Change improve clarity of various error message by including relevant show name. Change extend WEB PROPER release group check to ignore SD releases. --- CHANGES.md | 8 ++++++ sickbeard/helpers.py | 10 ++++++- sickbeard/metadata/generic.py | 10 +++---- sickbeard/metadata/helpers.py | 4 +-- sickbeard/properFinder.py | 6 +++-- sickbeard/show_queue.py | 31 +++++++++++++-------- sickbeard/tv.py | 51 ++++++++++++++++++----------------- sickbeard/webserve.py | 4 +-- 8 files changed, 76 insertions(+), 48 deletions(-) diff --git a/CHANGES.md b/CHANGES.md index 5ae290a5..7ef63324 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -65,6 +65,14 @@ * Change editshow saving empty scene exceptions * Change improve TVDB data handling * Change improve post processing by using more snatch history data +* Change show update, don't delete any ep in DB if eps are not returned from indexer +* Change prevent unneeded error message during show update +* Change improve performance, don't fetch episode list when retrieving a show image +* Change don't remove episodes from DB with status: SNATCHED, SNATCHED_PROPER, SNATCHED_BEST, DOWNLOADED, ARCHIVED, IGNORED +* Change add additional episode removal protections for TVDb_api v2 +* Change filter SKIPPED items from episode view +* Change improve clarity of various error message by including relevant show name +* Change extend WEB PROPER release group check to ignore SD releases [develop changelog] diff --git a/sickbeard/helpers.py b/sickbeard/helpers.py index df6e44fe..cf0195ac 100644 --- a/sickbeard/helpers.py +++ b/sickbeard/helpers.py @@ -54,7 +54,7 @@ except ImportError: from sickbeard.exceptions import MultipleShowObjectsException, ex from sickbeard import logger, db, notifiers, clients -from sickbeard.common import USER_AGENT, mediaExtensions, subtitleExtensions, cpu_presets +from sickbeard.common import USER_AGENT, mediaExtensions, subtitleExtensions, cpu_presets, statusStrings, SNATCHED, SNATCHED_PROPER, SNATCHED_BEST, DOWNLOADED, ARCHIVED, IGNORED, Quality from sickbeard import encodingKludge as ek from lib.cachecontrol import CacheControl, caches @@ -1540,3 +1540,11 @@ def set_file_timestamp(filename, min_age=3, new_time=None): ek.ek(os.utime, filename, new_time) except (StandardError, Exception): pass + + +def should_delete_episode(status): + s = Quality.splitCompositeStatus(status) + if s not in (SNATCHED, SNATCHED_PROPER, SNATCHED_BEST, DOWNLOADED, ARCHIVED, IGNORED): + return True + logger.log('not safe to delete episode from db because of status: %s' % statusStrings[s], logger.DEBUG) + return False \ No newline at end of file diff --git a/sickbeard/metadata/generic.py b/sickbeard/metadata/generic.py index 1c6833d4..6257ec52 100644 --- a/sickbeard/metadata/generic.py +++ b/sickbeard/metadata/generic.py @@ -517,7 +517,7 @@ class GenericMetadata(): logger.log(u"No thumb is available for this episode, not creating a thumb", logger.DEBUG) return False - thumb_data = metadata_helpers.getShowImage(thumb_url) + thumb_data = metadata_helpers.getShowImage(thumb_url, showName=ep_obj.show.name) result = self._write_image(thumb_data, file_path) @@ -620,7 +620,7 @@ class GenericMetadata(): logger.DEBUG) continue - seasonData = metadata_helpers.getShowImage(season_url) + seasonData = metadata_helpers.getShowImage(season_url, showName=show_obj.name) if not seasonData: logger.log(u"No season poster data available, skipping this season", logger.DEBUG) @@ -668,7 +668,7 @@ class GenericMetadata(): logger.DEBUG) continue - seasonData = metadata_helpers.getShowImage(season_url) + seasonData = metadata_helpers.getShowImage(season_url, showName=show_obj.name) if not seasonData: logger.log(u"No season banner data available, skipping this season", logger.DEBUG) @@ -771,7 +771,7 @@ class GenericMetadata(): lINDEXER_API_PARMS['language'] = indexer_lang t = sickbeard.indexerApi(show_obj.indexer).indexer(**lINDEXER_API_PARMS) - indexer_show_obj = t[show_obj.indexerid] + indexer_show_obj = t[show_obj.indexerid, False] except (sickbeard.indexer_error, IOError) as e: logger.log(u"Unable to look up show on " + sickbeard.indexerApi( show_obj.indexer).name + ", not downloading images: " + ex(e), logger.ERROR) @@ -827,7 +827,7 @@ class GenericMetadata(): if return_links: return image_urls else: - image_data = metadata_helpers.getShowImage((init_url, image_urls[0])[None is init_url], which) + image_data = metadata_helpers.getShowImage((init_url, image_urls[0])[None is init_url], which, show_obj.name) if None is not image_data: return image_data diff --git a/sickbeard/metadata/helpers.py b/sickbeard/metadata/helpers.py index 604feebd..70cff329 100644 --- a/sickbeard/metadata/helpers.py +++ b/sickbeard/metadata/helpers.py @@ -20,7 +20,7 @@ from sickbeard import helpers from sickbeard import logger -def getShowImage(url, imgNum=None): +def getShowImage(url, imgNum=None, showName=None): if None is url: return None @@ -32,7 +32,7 @@ def getShowImage(url, imgNum=None): image_data = helpers.getURL(temp_url) if None is image_data: - logger.log(u'There was an error trying to retrieve the image, aborting', logger.ERROR) + logger.log('There was an error trying to retrieve the image%s, aborting' % ('', ' for show: %s' % showName)[None is not showName], logger.ERROR) return return image_data diff --git a/sickbeard/properFinder.py b/sickbeard/properFinder.py index 767808c8..a09729e8 100644 --- a/sickbeard/properFinder.py +++ b/sickbeard/properFinder.py @@ -21,6 +21,7 @@ import operator import os import threading import traceback +import re import sickbeard @@ -165,7 +166,7 @@ def _get_proper_list(aired_since_shows, recent_shows, recent_anime): # check if we actually want this proper (if it's the right quality) my_db = db.DBConnection() sql_results = my_db.select( - 'SELECT release_group, status, version FROM tv_episodes WHERE showid = ? AND season = ? AND episode = ?', + 'SELECT release_group, status, version, release_name FROM tv_episodes WHERE showid = ? AND season = ? AND episode = ?', [cur_proper.indexerid, cur_proper.season, cur_proper.episode]) if not sql_results: continue @@ -182,7 +183,8 @@ def _get_proper_list(aired_since_shows, recent_shows, recent_anime): # for webldls, prevent propers from different groups if sickbeard.PROPERS_WEBDL_ONEGRP and \ - old_quality in (Quality.HDWEBDL, Quality.FULLHDWEBDL, Quality.UHD4KWEB) and \ + (old_quality in (Quality.HDWEBDL, Quality.FULLHDWEBDL, Quality.UHD4KWEB) or + (old_quality == Quality.SDTV and re.search(r'\Wweb.?(dl|rip|.[hx]26[45])\W', str(sql_results[0]['release_name']), re.I))) and \ cur_proper.release_group != old_release_group: logger.log(log_same_grp, logger.DEBUG) continue diff --git a/sickbeard/show_queue.py b/sickbeard/show_queue.py index 99e7bd26..dfa0f941 100644 --- a/sickbeard/show_queue.py +++ b/sickbeard/show_queue.py @@ -22,12 +22,13 @@ import traceback import sickbeard -from sickbeard.common import SKIPPED, WANTED, UNAIRED +from sickbeard.common import SKIPPED, WANTED, UNAIRED, statusStrings from sickbeard.tv import TVShow from sickbeard import exceptions, logger, ui, db from sickbeard import generic_queue from sickbeard import name_cache from sickbeard.exceptions import ex +from sickbeard.helpers import should_delete_episode from sickbeard.blackandwhitelist import BlackAndWhiteList @@ -615,7 +616,7 @@ class QueueItemUpdate(ShowQueueItem): try: self.show.saveToDB() except Exception as e: - logger.log('Error saving the episode to the database: %s' % ex(e), logger.ERROR) + logger.log('Error saving the show to the database: %s' % ex(e), logger.ERROR) logger.log(traceback.format_exc(), logger.ERROR) # get episode list from DB @@ -631,9 +632,12 @@ class QueueItemUpdate(ShowQueueItem): (sickbeard.indexerApi(self.show.indexer).name, ex(e)), logger.ERROR) IndexerEpList = None - if IndexerEpList == None: - logger.log('No data returned from %s, unable to update this show' % - sickbeard.indexerApi(self.show.indexer).name, logger.ERROR) + if None is IndexerEpList: + logger.log('No data returned from %s, unable to update episodes for show: %s' % + (sickbeard.indexerApi(self.show.indexer).name, self.show.name), logger.ERROR) + elif not IndexerEpList or 0 == len(IndexerEpList): + logger.log('No episodes returned from %s for show: %s' % + (sickbeard.indexerApi(self.show.indexer).name, self.show.name), logger.WARNING) else: # for each ep we found on TVDB delete it from the DB list for curSeason in IndexerEpList: @@ -645,13 +649,18 @@ class QueueItemUpdate(ShowQueueItem): # for the remaining episodes in the DB list just delete them from the DB for curSeason in DBEpList: for curEpisode in DBEpList[curSeason]: - logger.log('Permanently deleting episode %sx%s from the database' % - (curSeason, curEpisode), logger.MESSAGE) curEp = self.show.getEpisode(curSeason, curEpisode) - try: - curEp.deleteEpisode() - except exceptions.EpisodeDeletedException: - pass + status = sickbeard.common.Quality.splitCompositeStatus(curEp.status)[0] + if should_delete_episode(status): + logger.log('Permanently deleting episode %sx%s from the database' % + (curSeason, curEpisode), logger.MESSAGE) + try: + curEp.deleteEpisode() + except exceptions.EpisodeDeletedException: + pass + else: + logger.log('Not deleting episode %sx%s from the database because status is: %s' % + (curSeason, curEpisode, statusStrings[status]), logger.MESSAGE) if self.priority != generic_queue.QueuePriorities.NORMAL: self.kwargs['priority'] = self.priority diff --git a/sickbeard/tv.py b/sickbeard/tv.py index 9fc6211b..a3e2d479 100644 --- a/sickbeard/tv.py +++ b/sickbeard/tv.py @@ -484,7 +484,7 @@ class TVShow(object): def loadEpisodesFromDB(self, update=False): - logger.log('Loading all episodes from the DB') + logger.log('Loading all episodes for [%s] from the DB' % self.name) myDB = db.DBConnection() sql = 'SELECT * FROM tv_episodes WHERE showid = ? AND indexer = ?' @@ -518,27 +518,27 @@ class TVShow(object): try: cachedSeasons[curSeason] = cachedShow[curSeason] except sickbeard.indexer_seasonnotfound as e: - logger.log('Error when trying to load the episode from %s: %s' % - (sickbeard.indexerApi(self.indexer).name, e.message), logger.WARNING) + logger.log('Error when trying to load the episode for [%s] from %s: %s' % + (self.name, sickbeard.indexerApi(self.indexer).name, e.message), logger.WARNING) deleteEp = True if not curSeason in scannedEps: scannedEps[curSeason] = {} - logger.log('Loading episode %sx%s from the DB' % (curSeason, curEpisode), logger.DEBUG) + logger.log('Loading episode %sx%s for [%s] from the DB' % (curSeason, curEpisode, self.name), logger.DEBUG) try: curEp = self.getEpisode(curSeason, curEpisode) # if we found out that the ep is no longer on TVDB then delete it from our database too - if deleteEp: + if deleteEp and helpers.should_delete_episode(curEp.status): curEp.deleteEpisode() curEp.loadFromDB(curSeason, curEpisode) curEp.loadFromIndexer(tvapi=t, cachedSeason=cachedSeasons[curSeason], update=update) scannedEps[curSeason][curEpisode] = True except exceptions.EpisodeDeletedException: - logger.log('Tried loading an episode from the DB that should have been deleted, skipping it', + logger.log('Tried loading an episode from [%s] from the DB that should have been deleted, skipping it' % self.name, logger.DEBUG) continue @@ -557,14 +557,14 @@ class TVShow(object): if self.dvdorder != 0: lINDEXER_API_PARMS['dvdorder'] = True - logger.log('%s: Loading all episodes from %s..' % (self.indexerid, sickbeard.indexerApi(self.indexer).name)) + logger.log('%s: Loading all episodes for [%s] from %s..' % (self.indexerid, self.name, sickbeard.indexerApi(self.indexer).name)) try: t = sickbeard.indexerApi(self.indexer).indexer(**lINDEXER_API_PARMS) showObj = t[self.indexerid] except sickbeard.indexer_error: - logger.log('%s timed out, unable to update episodes from %s' % - (sickbeard.indexerApi(self.indexer).name, sickbeard.indexerApi(self.indexer).name), logger.ERROR) + logger.log('%s timed out, unable to update episodes for [%s] from %s' % + (sickbeard.indexerApi(self.indexer).name, self.name, sickbeard.indexerApi(self.indexer).name), logger.ERROR) return None scannedEps = {} @@ -579,19 +579,19 @@ class TVShow(object): try: ep = self.getEpisode(season, episode) except exceptions.EpisodeNotFoundException: - logger.log('%s: %s object for %sx%s is incomplete, skipping this episode' % - (self.indexerid, sickbeard.indexerApi(self.indexer).name, season, episode)) + logger.log('%s: %s object for %sx%s from [%s] is incomplete, skipping this episode' % + (self.indexerid, sickbeard.indexerApi(self.indexer).name, season, episode, self.name)) continue else: try: ep.loadFromIndexer(tvapi=t, update=update) except exceptions.EpisodeDeletedException: - logger.log('The episode was deleted, skipping the rest of the load') + logger.log('The episode from [%s] was deleted, skipping the rest of the load' % self.name) continue with ep.lock: - logger.log('%s: Loading info from %s for episode %sx%s' % - (self.indexerid, sickbeard.indexerApi(self.indexer).name, season, episode), logger.DEBUG) + logger.log('%s: Loading info from %s for episode %sx%s from [%s]' % + (self.indexerid, sickbeard.indexerApi(self.indexer).name, season, episode, self.name), logger.DEBUG) ep.loadFromIndexer(season, episode, tvapi=t, update=update) result = ep.get_sql() @@ -779,10 +779,10 @@ class TVShow(object): sqlResults = myDB.select('SELECT * FROM tv_shows WHERE indexer_id = ?', [self.indexerid]) if len(sqlResults) > 1: - logger.log('%s: Loading show info from database' % self.indexerid) + logger.log('%s: Loading show info [%s] from database' % (self.indexerid, self.name)) raise exceptions.MultipleDBShowsException() elif len(sqlResults) == 0: - logger.log('%s: Unable to find the show in the database' % self.indexerid) + logger.log('%s: Unable to find the show [%s] in the database' % (self.indexerid, self.name)) return else: if not self.indexer: @@ -889,7 +889,7 @@ class TVShow(object): def loadFromIndexer(self, cache=True, tvapi=None, cachedSeason=None): - logger.log('%s: Loading show info from %s' % (self.indexerid, sickbeard.indexerApi(self.indexer).name)) + logger.log('%s: Loading show info [%s] from %s' % (self.indexerid, self.name, sickbeard.indexerApi(self.indexer).name)) # There's gotta be a better way of doing this but we don't wanna # change the cache value elsewhere @@ -912,7 +912,7 @@ class TVShow(object): myEp = t[self.indexerid, False] if None is myEp: - logger.log('Show not found (maybe even removed?)', logger.WARNING) + logger.log('Show [%s] not found (maybe even removed?)' % self.name, logger.WARNING) return False try: @@ -944,7 +944,7 @@ class TVShow(object): from lib.imdb import _exceptions as imdb_exceptions - logger.log('Retrieving show info from IMDb', logger.DEBUG) + logger.log('Retrieving show info [%s] from IMDb' % self.name, logger.DEBUG) try: self._get_imdb_info() except imdb_exceptions.IMDbDataAccessError as e: @@ -1029,7 +1029,7 @@ class TVShow(object): logger.log('%s: Parsed latest IMDb show info for [%s]' % (self.indexerid, self.name)) def nextEpisode(self): - logger.log('%s: Finding the episode which airs next' % self.indexerid, logger.DEBUG) + logger.log('%s: Finding the episode which airs next for: %s' % (self.indexerid, self.name), logger.DEBUG) curDate = datetime.date.today().toordinal() if not self.nextaired or self.nextaired and curDate > self.nextaired: @@ -1135,7 +1135,7 @@ class TVShow(object): self.loadEpisodesFromDir() # run through all locations from DB, check that they exist - logger.log('%s: Loading all episodes with a location from the database' % self.indexerid) + logger.log('%s: Loading all episodes for [%s] with a location from the database' % (self.indexerid, self.name)) myDB = db.DBConnection() sqlResults = myDB.select("SELECT * FROM tv_episodes WHERE showid = ? AND location != ''", [self.indexerid]) @@ -1149,7 +1149,7 @@ class TVShow(object): try: curEp = self.getEpisode(season, episode) except exceptions.EpisodeDeletedException: - logger.log('The episode was deleted while we were refreshing it, moving on to the next one', + logger.log('The episode from [%s] was deleted while we were refreshing it, moving on to the next one' % self.name, logger.DEBUG) continue @@ -1783,7 +1783,7 @@ class TVEpisode(object): logger.log('Unable to find the episode on %s... has it been removed? Should I delete from db?' % sickbeard.indexerApi(self.indexer).name, logger.DEBUG) # if I'm no longer on the Indexers but I once was then delete myself from the DB - if -1 != self.indexerid: + if -1 != self.indexerid and helpers.should_delete_episode(self.status): self.deleteEpisode() return @@ -1827,7 +1827,7 @@ class TVEpisode(object): logger.log('Malformed air date retrieved from %s (%s - %sx%s)' % (sickbeard.indexerApi(self.indexer).name, self.show.name, season, episode), logger.ERROR) # if I'm incomplete on TVDB but I once was complete then just delete myself from the DB for now - if -1 != self.indexerid: + if -1 != self.indexerid and helpers.should_delete_episode(self.status): self.deleteEpisode() return False @@ -1835,7 +1835,8 @@ class TVEpisode(object): self.indexerid = getattr(myEp, 'id', None) if None is self.indexerid: logger.log('Failed to retrieve ID from %s' % sickbeard.indexerApi(self.indexer).name, logger.ERROR) - self.deleteEpisode() + if helpers.should_delete_episode(self.status): + self.deleteEpisode() return False # don't update show status if show dir is missing, unless it's missing on purpose diff --git a/sickbeard/webserve.py b/sickbeard/webserve.py index be8414bc..465d4029 100644 --- a/sickbeard/webserve.py +++ b/sickbeard/webserve.py @@ -447,7 +447,7 @@ class MainHandler(WebHandler): recently = (yesterday_dt - datetime.timedelta(days=sickbeard.EPISODE_VIEW_MISSED_RANGE)).toordinal() done_show_list = [] - qualities = Quality.DOWNLOADED + Quality.SNATCHED + [ARCHIVED, IGNORED] + qualities = Quality.DOWNLOADED + Quality.SNATCHED + [ARCHIVED, IGNORED, SKIPPED] myDB = db.DBConnection() sql_results = myDB.select( @@ -475,7 +475,7 @@ class MainHandler(WebHandler): # make a dict out of the sql results sql_results = [dict(row) for row in sql_results if Quality.splitCompositeStatus(helpers.tryInt(row['status']))[0] not in - [DOWNLOADED, SNATCHED, SNATCHED_PROPER, SNATCHED_BEST, ARCHIVED, IGNORED]] + [DOWNLOADED, SNATCHED, SNATCHED_PROPER, SNATCHED_BEST, ARCHIVED, IGNORED, SKIPPED]] # multi dimension sort sorts = {