2014-03-10 01:18:05 -04:00
|
|
|
# Author: Nic Wolfe <nic@wolfeden.ca>
|
|
|
|
# URL: http://code.google.com/p/sickbeard/
|
|
|
|
#
|
2014-05-23 08:37:22 -04:00
|
|
|
# This file is part of SickRage.
|
2014-03-10 01:18:05 -04:00
|
|
|
#
|
2014-05-23 08:37:22 -04:00
|
|
|
# SickRage is free software: you can redistribute it and/or modify
|
2014-03-10 01:18:05 -04:00
|
|
|
# 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.
|
|
|
|
#
|
2014-05-23 08:37:22 -04:00
|
|
|
# SickRage is distributed in the hope that it will be useful,
|
2014-03-10 01:18:05 -04:00
|
|
|
# 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
|
2014-05-23 08:37:22 -04:00
|
|
|
# along with SickRage. If not, see <http://www.gnu.org/licenses/>.
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
import datetime
|
|
|
|
import operator
|
2014-05-19 13:40:25 -04:00
|
|
|
import threading
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
import sickbeard
|
|
|
|
|
|
|
|
from sickbeard import db
|
|
|
|
from sickbeard import exceptions
|
|
|
|
from sickbeard.exceptions import ex
|
|
|
|
from sickbeard import helpers, logger, show_name_helpers
|
|
|
|
from sickbeard import search
|
|
|
|
from sickbeard import history
|
|
|
|
|
|
|
|
from sickbeard.common import DOWNLOADED, SNATCHED, SNATCHED_PROPER, Quality
|
|
|
|
|
2014-07-06 09:11:04 -04:00
|
|
|
from name_parser.parser import NameParser, InvalidNameException, InvalidShowException
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
|
|
|
|
class ProperFinder():
|
|
|
|
def __init__(self):
|
2014-05-19 13:40:25 -04:00
|
|
|
self.amActive = False
|
2014-05-14 18:23:59 -04:00
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
def run(self, force=False):
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
if not sickbeard.DOWNLOAD_PROPERS:
|
|
|
|
return
|
|
|
|
|
2014-05-14 18:23:59 -04:00
|
|
|
logger.log(u"Beginning the search for new propers")
|
2014-03-10 01:18:05 -04:00
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
self.amActive = True
|
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
propers = self._getProperList()
|
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
if propers:
|
|
|
|
self._downloadPropers(propers)
|
2014-03-25 01:57:24 -04:00
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
self._set_lastProperSearch(datetime.datetime.today().toordinal())
|
|
|
|
|
2014-06-19 03:12:30 -04:00
|
|
|
run_at = ""
|
|
|
|
if None is sickbeard.properFinderScheduler.start_time:
|
|
|
|
run_in = sickbeard.properFinderScheduler.lastRun + sickbeard.properFinderScheduler.cycleTime - datetime.datetime.now()
|
|
|
|
hours, remainder = divmod(run_in.seconds, 3600)
|
|
|
|
minutes, seconds = divmod(remainder, 60)
|
|
|
|
run_at = u", next check in approx. " + ("%dh, %dm" % (hours, minutes) if 0 < hours else "%dm, %ds" % (minutes, seconds))
|
|
|
|
|
|
|
|
logger.log(u"Completed the search for new propers%s" % run_at)
|
2014-05-14 18:23:59 -04:00
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
self.amActive = False
|
2014-03-10 01:18:05 -04:00
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
def _getProperList(self):
|
2014-03-10 01:18:05 -04:00
|
|
|
propers = {}
|
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
# for each provider get a list of the
|
|
|
|
origThreadName = threading.currentThread().name
|
|
|
|
providers = [x for x in sickbeard.providers.sortedProviderList() if x.isActive()]
|
|
|
|
for curProvider in providers:
|
|
|
|
threading.currentThread().name = origThreadName + " :: [" + curProvider.name + "]"
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
search_date = datetime.datetime.today() - datetime.timedelta(days=2)
|
|
|
|
|
|
|
|
logger.log(u"Searching for any new PROPER releases from " + curProvider.name)
|
|
|
|
try:
|
|
|
|
curPropers = curProvider.findPropers(search_date)
|
|
|
|
except exceptions.AuthException, e:
|
|
|
|
logger.log(u"Authentication error: " + ex(e), logger.ERROR)
|
|
|
|
continue
|
|
|
|
|
|
|
|
# if they haven't been added by a different provider than add the proper to the list
|
|
|
|
for x in curPropers:
|
|
|
|
name = self._genericName(x.name)
|
|
|
|
if not name in propers:
|
|
|
|
logger.log(u"Found new proper: " + x.name, logger.DEBUG)
|
|
|
|
x.provider = curProvider
|
|
|
|
propers[name] = x
|
|
|
|
|
2014-05-19 13:40:25 -04:00
|
|
|
# reset thread name back to original
|
|
|
|
threading.currentThread().name = origThreadName
|
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
# take the list of unique propers and get it sorted by
|
|
|
|
sortedPropers = sorted(propers.values(), key=operator.attrgetter('date'), reverse=True)
|
|
|
|
finalPropers = []
|
|
|
|
for curProper in sortedPropers:
|
|
|
|
|
|
|
|
try:
|
2014-07-14 22:00:53 -04:00
|
|
|
myParser = NameParser(False, showObj=curProper.show)
|
2014-05-31 06:35:11 -04:00
|
|
|
parse_result = myParser.parse(curProper.name)
|
2014-03-10 01:18:05 -04:00
|
|
|
except InvalidNameException:
|
|
|
|
logger.log(u"Unable to parse the filename " + curProper.name + " into a valid episode", logger.DEBUG)
|
|
|
|
continue
|
2014-07-06 09:11:04 -04:00
|
|
|
except InvalidShowException:
|
2014-07-11 02:44:36 -04:00
|
|
|
logger.log(u"Unable to parse the filename " + curProper.name + " into a valid show", logger.DEBUG)
|
2014-05-07 03:50:49 -04:00
|
|
|
continue
|
|
|
|
|
2014-07-06 09:11:04 -04:00
|
|
|
if not parse_result.series_name:
|
2014-05-07 03:50:49 -04:00
|
|
|
continue
|
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
if not parse_result.episode_numbers:
|
2014-03-25 01:57:24 -04:00
|
|
|
logger.log(
|
|
|
|
u"Ignoring " + curProper.name + " because it's for a full season rather than specific episode",
|
|
|
|
logger.DEBUG)
|
2014-03-10 01:18:05 -04:00
|
|
|
continue
|
|
|
|
|
2014-05-30 03:36:47 -04:00
|
|
|
logger.log(
|
2014-07-14 22:00:53 -04:00
|
|
|
u"Successful match! Result " + parse_result.original_name + " matched to show " + parse_result.show.name,
|
2014-05-30 03:36:47 -04:00
|
|
|
logger.DEBUG)
|
|
|
|
|
|
|
|
# set the indexerid in the db to the show's indexerid
|
2014-07-14 22:00:53 -04:00
|
|
|
curProper.indexerid = parse_result.show.indexerid
|
2014-05-30 03:36:47 -04:00
|
|
|
|
|
|
|
# set the indexer in the db to the show's indexer
|
2014-07-14 22:00:53 -04:00
|
|
|
curProper.indexer = parse_result.show.indexer
|
2014-05-30 03:36:47 -04:00
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
# populate our Proper instance
|
2014-07-14 22:00:53 -04:00
|
|
|
if parse_result.is_air_by_date or parse_result.is_sports:
|
2014-03-10 01:18:05 -04:00
|
|
|
curProper.season = -1
|
2014-07-21 05:20:54 -04:00
|
|
|
curProper.episode = parse_result.air_date or parse_result.sports_air_date
|
2014-03-10 01:18:05 -04:00
|
|
|
else:
|
2014-05-26 02:29:22 -04:00
|
|
|
if parse_result.is_anime:
|
|
|
|
logger.log(u"I am sorry '"+curProper.name+"' seams to be an anime proper seach is not yet suported", logger.DEBUG)
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
curProper.season = parse_result.season_number if parse_result.season_number != None else 1
|
|
|
|
curProper.episode = parse_result.episode_numbers[0]
|
|
|
|
|
|
|
|
curProper.quality = Quality.nameQuality(curProper.name, parse_result.is_anime)
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
if not show_name_helpers.filterBadReleases(curProper.name):
|
2014-03-25 01:57:24 -04:00
|
|
|
logger.log(u"Proper " + curProper.name + " isn't a valid scene release that we want, igoring it",
|
|
|
|
logger.DEBUG)
|
2014-03-10 01:18:05 -04:00
|
|
|
continue
|
|
|
|
|
2014-07-14 22:00:53 -04:00
|
|
|
if parse_result.show.rls_ignore_words and search.filter_release_name(curProper.name, parse_result.show.rls_ignore_words):
|
|
|
|
logger.log(u"Ignoring " + curProper.name + " based on ignored words filter: " + parse_result.show.rls_ignore_words,
|
2014-04-24 01:18:16 -04:00
|
|
|
logger.MESSAGE)
|
|
|
|
continue
|
|
|
|
|
2014-07-14 22:00:53 -04:00
|
|
|
if parse_result.show.rls_require_words and not search.filter_release_name(curProper.name, parse_result.show.rls_require_words):
|
|
|
|
logger.log(u"Ignoring " + curProper.name + " based on required words filter: " + parse_result.show.rls_require_words,
|
2014-04-24 01:18:16 -04:00
|
|
|
logger.MESSAGE)
|
|
|
|
continue
|
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
# if we have an air-by-date show then get the real season/episode numbers
|
2014-07-14 22:00:53 -04:00
|
|
|
if (parse_result.is_air_by_date or parse_result.is_sports_air_date) and curProper.indexerid:
|
2014-05-02 06:09:17 -04:00
|
|
|
logger.log(
|
|
|
|
u"Looks like this is an air-by-date or sports show, attempting to convert the date to season/episode",
|
|
|
|
logger.DEBUG)
|
2014-05-04 23:04:46 -04:00
|
|
|
airdate = curProper.episode.toordinal()
|
2014-06-21 18:46:59 -04:00
|
|
|
myDB = db.DBConnection()
|
|
|
|
sql_result = myDB.select(
|
|
|
|
"SELECT season, episode FROM tv_episodes WHERE showid = ? and indexer = ? and airdate = ?",
|
|
|
|
[curProper.indexerid, curProper.indexer, airdate])
|
2014-05-02 06:09:17 -04:00
|
|
|
|
|
|
|
if sql_result:
|
|
|
|
curProper.season = int(sql_result[0][0])
|
|
|
|
curProper.episodes = [int(sql_result[0][1])]
|
|
|
|
else:
|
2014-03-25 01:57:24 -04:00
|
|
|
logger.log(u"Unable to find episode with date " + str(
|
|
|
|
curProper.episode) + " for show " + parse_result.series_name + ", skipping", logger.WARNING)
|
2014-03-10 01:18:05 -04:00
|
|
|
continue
|
|
|
|
|
|
|
|
# check if we actually want this proper (if it's the right quality)
|
2014-06-21 18:46:59 -04:00
|
|
|
myDB = db.DBConnection()
|
|
|
|
sqlResults = myDB.select(
|
|
|
|
"SELECT status FROM tv_episodes WHERE showid = ? AND season = ? AND episode = ?",
|
|
|
|
[curProper.indexerid, curProper.season, curProper.episode])
|
2014-06-07 17:32:38 -04:00
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
if not sqlResults:
|
|
|
|
continue
|
2014-06-07 17:32:38 -04:00
|
|
|
|
2014-03-10 01:18:05 -04:00
|
|
|
oldStatus, oldQuality = Quality.splitCompositeStatus(int(sqlResults[0]["status"]))
|
|
|
|
|
|
|
|
# only keep the proper if we have already retrieved the same quality ep (don't get better/worse ones)
|
|
|
|
if oldStatus not in (DOWNLOADED, SNATCHED) or oldQuality != curProper.quality:
|
|
|
|
continue
|
|
|
|
|
|
|
|
# if the show is in our list and there hasn't been a proper already added for that particular episode then add it to our list of propers
|
2014-03-25 01:57:24 -04:00
|
|
|
if curProper.indexerid != -1 and (curProper.indexerid, curProper.season, curProper.episode) not in map(
|
|
|
|
operator.attrgetter('indexerid', 'season', 'episode'), finalPropers):
|
2014-03-10 01:18:05 -04:00
|
|
|
logger.log(u"Found a proper that we need: " + str(curProper.name))
|
|
|
|
finalPropers.append(curProper)
|
|
|
|
|
|
|
|
return finalPropers
|
|
|
|
|
|
|
|
def _downloadPropers(self, properList):
|
|
|
|
|
|
|
|
for curProper in properList:
|
|
|
|
|
|
|
|
historyLimit = datetime.datetime.today() - datetime.timedelta(days=30)
|
|
|
|
|
|
|
|
# make sure the episode has been downloaded before
|
2014-06-21 18:46:59 -04:00
|
|
|
myDB = db.DBConnection()
|
|
|
|
historyResults = myDB.select(
|
|
|
|
"SELECT resource FROM history "
|
|
|
|
"WHERE showid = ? AND season = ? AND episode = ? AND quality = ? AND date >= ? "
|
|
|
|
"AND action IN (" + ",".join([str(x) for x in Quality.SNATCHED]) + ")",
|
|
|
|
[curProper.indexerid, curProper.season, curProper.episode, curProper.quality,
|
|
|
|
historyLimit.strftime(history.dateFormat)])
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
# if we didn't download this episode in the first place we don't know what quality to use for the proper so we can't do it
|
|
|
|
if len(historyResults) == 0:
|
2014-03-25 01:57:24 -04:00
|
|
|
logger.log(
|
|
|
|
u"Unable to find an original history entry for proper " + curProper.name + " so I'm not downloading it.")
|
2014-03-10 01:18:05 -04:00
|
|
|
continue
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
# make sure that none of the existing history downloads are the same proper we're trying to download
|
2014-07-19 18:23:01 -04:00
|
|
|
clean_proper_name = self._genericName(helpers.remove_non_release_groups(curProper.name))
|
2014-03-10 01:18:05 -04:00
|
|
|
isSame = False
|
|
|
|
for curResult in historyResults:
|
|
|
|
# if the result exists in history already we need to skip it
|
2014-07-19 18:23:01 -04:00
|
|
|
if self._genericName(helpers.remove_non_release_groups(curResult["resource"])) == clean_proper_name:
|
2014-03-10 01:18:05 -04:00
|
|
|
isSame = True
|
|
|
|
break
|
|
|
|
if isSame:
|
|
|
|
logger.log(u"This proper is already in history, skipping it", logger.DEBUG)
|
|
|
|
continue
|
|
|
|
|
|
|
|
# get the episode object
|
|
|
|
showObj = helpers.findCertainShow(sickbeard.showList, curProper.indexerid)
|
2014-03-20 14:03:22 -04:00
|
|
|
if showObj == None:
|
2014-03-25 01:57:24 -04:00
|
|
|
logger.log(u"Unable to find the show with indexerid " + str(
|
2014-05-04 23:04:46 -04:00
|
|
|
curProper .indexerid) + " so unable to download the proper", logger.ERROR)
|
2014-03-10 01:18:05 -04:00
|
|
|
continue
|
|
|
|
epObj = showObj.getEpisode(curProper.season, curProper.episode)
|
|
|
|
|
|
|
|
# make the result object
|
|
|
|
result = curProper.provider.getResult([epObj])
|
|
|
|
result.url = curProper.url
|
|
|
|
result.name = curProper.name
|
|
|
|
result.quality = curProper.quality
|
|
|
|
|
|
|
|
# snatch it
|
2014-04-02 02:29:46 -04:00
|
|
|
search.snatchEpisode(result, SNATCHED_PROPER)
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
def _genericName(self, name):
|
|
|
|
return name.replace(".", " ").replace("-", " ").replace("_", " ").lower()
|
|
|
|
|
|
|
|
def _set_lastProperSearch(self, when):
|
|
|
|
|
|
|
|
logger.log(u"Setting the last Proper search in the DB to " + str(when), logger.DEBUG)
|
|
|
|
|
2014-06-21 18:46:59 -04:00
|
|
|
myDB = db.DBConnection()
|
|
|
|
sqlResults = myDB.select("SELECT * FROM info")
|
2014-03-10 01:18:05 -04:00
|
|
|
|
2014-06-21 18:46:59 -04:00
|
|
|
if len(sqlResults) == 0:
|
|
|
|
myDB.action("INSERT INTO info (last_backlog, last_indexer, last_proper_search) VALUES (?,?,?)",
|
|
|
|
[0, 0, str(when)])
|
|
|
|
else:
|
|
|
|
myDB.action("UPDATE info SET last_proper_search=" + str(when))
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
def _get_lastProperSearch(self):
|
|
|
|
|
2014-06-21 18:46:59 -04:00
|
|
|
myDB = db.DBConnection()
|
|
|
|
sqlResults = myDB.select("SELECT * FROM info")
|
2014-03-10 01:18:05 -04:00
|
|
|
|
|
|
|
try:
|
|
|
|
last_proper_search = datetime.date.fromordinal(int(sqlResults[0]["last_proper_search"]))
|
|
|
|
except:
|
|
|
|
return datetime.date.fromordinal(1)
|
|
|
|
|
|
|
|
return last_proper_search
|