Private GIT

Skip to content
Snippets Groups Projects
Select Git revision
  • 4d9663c13fa520287aa351fad1a508a14fbd1f26
  • master default protected
  • fix_nzb_cat
  • develop
  • guessit2-minimal
  • ssl_warning
  • UHD-qualities
  • fix_providers8
  • !
  • tvvault
  • provider_alpharatio
  • v5.1.1
  • v5.1
  • v5.0.3
  • v5.0.2
  • v5.0.1
  • v5.0
  • v4.2.1.07
  • v4.2.1.06
  • v4.2.1.05
  • v4.2.1.04
  • v4.2.1.03
  • v4.2.1.02
  • v4.2.1.01
  • v4.2.1.0
  • v4.2.0.6
  • v4.2.0.5
  • v4.2.0.4
  • v4.2.0.3
  • v4.2.0.2
  • v4.2.0.1
31 results

SickBeard.py

Blame
  • scene_exceptions.py 13.13 KiB
    # coding=utf-8
    # Author: Nic Wolfe <nic@wolfeden.ca>
    # URL: https://sickrage.github.io
    # Git: https://github.com/SickRage/SickRage.git
    #
    # This file is part of SickRage.
    #
    # SickRage is free software: you can redistribute it and/or modify
    # it under the terms of the GNU General Public License as published by
    # the Free Software Foundation, either version 3 of the License, or
    # (at your option) any later version.
    #
    # SickRage is distributed in the hope that it will be useful,
    # but WITHOUT ANY WARRANTY; without even the implied warranty of
    # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
    # GNU General Public License for more details.
    #
    # You should have received a copy of the GNU General Public License
    # along with SickRage. If not, see <http://www.gnu.org/licenses/>.
    
    from __future__ import print_function, unicode_literals
    
    import datetime
    import threading
    import time
    
    import adba
    import six
    
    import sickbeard
    from sickbeard import db, helpers, logger
    from sickbeard.indexers.indexer_config import INDEXER_TVDB
    
    exception_dict = {}
    anidb_exception_dict = {}
    xem_exception_dict = {}
    
    exceptionsCache = {}
    exceptionsSeasonCache = {}
    
    exceptionLock = threading.Lock()
    
    
    def shouldRefresh(exList):
        """
        Check if we should refresh cache for items in exList
    
        :param exList: exception list to check if needs a refresh
        :return: True if refresh is needed
        """
        MAX_REFRESH_AGE_SECS = 86400  # 1 day
    
        cache_db_con = db.DBConnection('cache.db')
        rows = cache_db_con.select("SELECT last_refreshed FROM scene_exceptions_refresh WHERE list = ?", [exList])
        if rows:
            lastRefresh = int(rows[0][b'last_refreshed'])
            return int(time.mktime(datetime.datetime.today().timetuple())) > lastRefresh + MAX_REFRESH_AGE_SECS
        else:
            return True
    
    
    def setLastRefresh(exList):
        """
        Update last cache update time for shows in list
    
        :param exList: exception list to set refresh time
        """
        cache_db_con = db.DBConnection('cache.db')
        cache_db_con.upsert(
            "scene_exceptions_refresh",
            {'last_refreshed': int(time.mktime(datetime.datetime.today().timetuple()))},
            {'list': exList}
        )
    
    
    def get_scene_exceptions(indexer_id, season=-1):
        """
        Given a indexer_id, return a list of all the scene exceptions.
        """
    
        exceptionsList = []
    
        if indexer_id not in exceptionsCache or season not in exceptionsCache[indexer_id]:
            cache_db_con = db.DBConnection('cache.db')
            exceptions = cache_db_con.select("SELECT show_name FROM scene_exceptions WHERE indexer_id = ? and season = ?",
                                             [indexer_id, season])
            if exceptions:
                exceptionsList = list({cur_exception[b"show_name"] for cur_exception in exceptions})
    
                if indexer_id not in exceptionsCache:
                    exceptionsCache[indexer_id] = {}
                exceptionsCache[indexer_id][season] = exceptionsList
        else:
            exceptionsList = exceptionsCache[indexer_id][season]
    
        # Add generic exceptions regardless of the season if there is no exception for season
        if season != -1 and not exceptionsList:
            exceptionsList += get_scene_exceptions(indexer_id, season=-1)
    
        return list({exception for exception in exceptionsList})
    
    
    def get_all_scene_exceptions(indexer_id):
        """
        Get all scene exceptions for a show ID
    
        :param indexer_id: ID to check
        :return: dict of exceptions
        """
        exceptionsDict = {}
    
        cache_db_con = db.DBConnection('cache.db')
        exceptions = cache_db_con.select("SELECT show_name,season,custom FROM scene_exceptions WHERE indexer_id = ?", [indexer_id])
    
        if exceptions:
            for cur_exception in exceptions:
                if not cur_exception[b"season"] in exceptionsDict:
                    exceptionsDict[cur_exception[b"season"]] = []
                exceptionsDict[cur_exception[b"season"]].append({
                    "show_name": cur_exception[b"show_name"],
                    "custom": bool(cur_exception[b"custom"])
                })
    
        return exceptionsDict
    
    
    def get_scene_seasons(indexer_id):
        """
        return a list of season numbers that have scene exceptions
        """
        exceptionsSeasonList = []
    
        if indexer_id not in exceptionsSeasonCache:
            cache_db_con = db.DBConnection('cache.db')
            sql_results = cache_db_con.select("SELECT DISTINCT(season) as season FROM scene_exceptions WHERE indexer_id = ?",
                                              [indexer_id])
            if sql_results:
                exceptionsSeasonList = list({int(x[b"season"]) for x in sql_results})
    
                if indexer_id not in exceptionsSeasonCache:
                    exceptionsSeasonCache[indexer_id] = {}
    
                exceptionsSeasonCache[indexer_id] = exceptionsSeasonList
        else:
            exceptionsSeasonList = exceptionsSeasonCache[indexer_id]
    
        return exceptionsSeasonList
    
    
    def get_scene_exception_by_name(show_name):
        return get_scene_exception_by_name_multiple(show_name)[0]
    
    
    def get_scene_exception_by_name_multiple(show_name):
        """
        Given a show name, return the indexerid of the exception, None if no exception
        is present.
        """
    
        # try the obvious case first
        cache_db_con = db.DBConnection('cache.db')
        exception_result = cache_db_con.select(
            "SELECT indexer_id, season FROM scene_exceptions WHERE LOWER(show_name) = ? ORDER BY season ASC",
            [show_name.lower()])
        if exception_result:
            return [(int(x[b"indexer_id"]), int(x[b"season"])) for x in exception_result]
    
        out = []
        all_exception_results = cache_db_con.select("SELECT show_name, indexer_id, season FROM scene_exceptions")
    
        for cur_exception in all_exception_results:
    
            cur_exception_name = cur_exception[b"show_name"]
            cur_indexer_id = int(cur_exception[b"indexer_id"])
    
            if show_name.lower() in (
                    cur_exception_name.lower(),
                    sickbeard.helpers.sanitizeSceneName(cur_exception_name).lower().replace('.', ' ')):
    
                logger.log("Scene exception lookup got indexer id {0}, using that".format
                           (cur_indexer_id), logger.DEBUG)
    
                out.append((cur_indexer_id, int(cur_exception[b"season"])))
    
        if out:
            return out
    
        return [(None, None)]
    
    
    def retrieve_exceptions():  # pylint:disable=too-many-locals, too-many-branches
        """
        Looks up the exceptions on github, parses them into a dict, and inserts them into the
        scene_exceptions table in cache.db. Also clears the scene name cache.
        """
    
        do_refresh = False
        for indexer in sickbeard.indexerApi().indexers:
            if shouldRefresh(sickbeard.indexerApi(indexer).name):
                do_refresh = True
    
        if do_refresh:
            loc = sickbeard.indexerApi(INDEXER_TVDB).config['scene_loc']
            logger.log("Checking for scene exception updates from {0}".format(loc))
    
            session = sickbeard.indexerApi(INDEXER_TVDB).session
            proxy = sickbeard.PROXY_SETTING
            if proxy and sickbeard.PROXY_INDEXERS:
                session.proxies = {
                    "http": proxy,
                    "https": proxy,
                }
    
            try:
                jdata = helpers.getURL(loc, session=session, returns='json')
            except Exception:
                jdata = None
    
            if not jdata:
                # When jdata is None, trouble connecting to github, or reading file failed
                logger.log("Check scene exceptions update failed. Unable to update from {0}".format(loc), logger.DEBUG)
            else:
                for indexer in sickbeard.indexerApi().indexers:
                    try:
                        setLastRefresh(sickbeard.indexerApi(indexer).name)
                        for indexer_id in jdata[sickbeard.indexerApi(indexer).config['xem_origin']]:
                            alias_list = [
                                {scene_exception: int(scene_season)}
                                for scene_season in jdata[sickbeard.indexerApi(indexer).config['xem_origin']][indexer_id]
                                for scene_exception in jdata[sickbeard.indexerApi(indexer).config['xem_origin']][indexer_id][scene_season]
                            ]
                            exception_dict[indexer_id] = alias_list
                    except Exception:
                        continue
    
        # XEM scene exceptions
        _xem_exceptions_fetcher()
        for xem_ex in xem_exception_dict:
            if xem_ex in exception_dict:
                exception_dict[xem_ex] += exception_dict[xem_ex]
            else:
                exception_dict[xem_ex] = xem_exception_dict[xem_ex]
    
        # AniDB scene exceptions
        _anidb_exceptions_fetcher()
        for anidb_ex in anidb_exception_dict:
            if anidb_ex in exception_dict:
                exception_dict[anidb_ex] += anidb_exception_dict[anidb_ex]
            else:
                exception_dict[anidb_ex] = anidb_exception_dict[anidb_ex]
    
        queries = []
        cache_db_con = db.DBConnection('cache.db')
        for cur_indexer_id in exception_dict:
            sql_ex = cache_db_con.select("SELECT show_name FROM scene_exceptions WHERE indexer_id = ?;", [cur_indexer_id])
            existing_exceptions = [x[b"show_name"] for x in sql_ex]
            if cur_indexer_id not in exception_dict:
                continue
    
            for cur_exception_dict in exception_dict[cur_indexer_id]:
                for ex in six.iteritems(cur_exception_dict):
                    cur_exception, curSeason = ex
                    if cur_exception not in existing_exceptions:
                        queries.append(
                            ["INSERT OR IGNORE INTO scene_exceptions (indexer_id, show_name, season) VALUES (?,?,?);",
                             [cur_indexer_id, cur_exception, curSeason]])
        if queries:
            cache_db_con.mass_action(queries)
            logger.log("Updated scene exceptions", logger.DEBUG)
    
        # cleanup
        exception_dict.clear()
        anidb_exception_dict.clear()
        xem_exception_dict.clear()
    
    
    def update_scene_exceptions(indexer_id, scene_exceptions):
        """
        Given a indexer_id, and a list of all show scene exceptions, update the db.
        """
        cache_db_con = db.DBConnection('cache.db')
        cache_db_con.action('DELETE FROM scene_exceptions WHERE indexer_id=? and custom=1', [indexer_id])
    
        logger.log("Updating scene exceptions", logger.INFO)
    
        for season in scene_exceptions:
            for cur_exception in scene_exceptions[season]:
                cache_db_con.action("INSERT INTO scene_exceptions (indexer_id, show_name, season, custom) VALUES (?,?,?,?)",
                                    [indexer_id, cur_exception["show_name"], season, cur_exception["custom"]])
    
        rebuild_exception_cache(indexer_id)
    
    
    def _anidb_exceptions_fetcher():
        if shouldRefresh('anidb'):
            logger.log("Checking for scene exception updates for AniDB")
            for show in sickbeard.showList:
                if show.is_anime and show.indexer == 1:
                    try:
                        anime = adba.Anime(None, name=show.name, tvdbid=show.indexerid, autoCorrectName=True)
                    except Exception:
                        continue
                    else:
                        if anime.name and anime.name != show.name:
                            anidb_exception_dict[show.indexerid] = [{anime.name: -1}]
    
            setLastRefresh('anidb')
        return anidb_exception_dict
    
    xem_session = helpers.make_session()
    
    def _xem_exceptions_fetcher():
        if shouldRefresh('xem'):
            for indexer in sickbeard.indexerApi().indexers:
                logger.log("Checking for XEM scene exception updates for {0}".format
                           (sickbeard.indexerApi(indexer).name))
    
                url = "http://thexem.de/map/allNames?origin={0}&seasonNumbers=1".format(sickbeard.indexerApi(indexer).config['xem_origin'])
    
                parsed_json = helpers.getURL(url, session=xem_session, timeout=90, returns='json')
                if not parsed_json:
                    logger.log("Check scene exceptions update failed for {0}, Unable to get URL: {1}".format
                               (sickbeard.indexerApi(indexer).name, url), logger.DEBUG)
                    continue
    
                if parsed_json['result'] == 'failure':
                    continue
    
                for indexerid, names in six.iteritems(parsed_json['data']):
                    try:
                        xem_exception_dict[int(indexerid)] = names
                    except Exception as error:
                        logger.log("XEM: Rejected entry: indexerid:{0}; names:{1}".format(indexerid, names), logger.WARNING)
                        logger.log("XEM: Rejected entry error message:{0}".format(error), logger.DEBUG)
    
            setLastRefresh('xem')
    
        return xem_exception_dict
    
    
    def getSceneSeasons(indexer_id):
        """get a list of season numbers that have scene exceptions"""
        cache_db_con = db.DBConnection('cache.db')
        seasons = cache_db_con.select("SELECT DISTINCT season FROM scene_exceptions WHERE indexer_id = ?", [indexer_id])
        return [cur_exception[b"season"] for cur_exception in seasons]
    
    
    def rebuild_exception_cache(indexer_id):
        if indexer_id not in exceptionsCache:
            exceptionsCache[indexer_id] = {}
    
        cache_db_con = db.DBConnection('cache.db')
        results = cache_db_con.action('SELECT show_name, season FROM scene_exceptions WHERE indexer_id=?', [indexer_id])
    
        for result in results:
            if result[b"season"] not in exceptionsCache[indexer_id]:
                exceptionsCache[indexer_id][result[b"season"]] = []
    
            exceptionsCache[indexer_id][result[b"season"]].append(result[b"show_name"])