mirror of
https://github.com/SickGear/SickGear.git
synced 2025-01-07 10:33:38 +00:00
Add Rarbg torrent provider.
Note: The merging of all qualities into one category in API v2 mean more results likely to be cropped. The vendor switched to this method from their (imho, better) original of listing category numbers. It was suggested that clients add and maintain all possible search strings that signify as quality. However, this would still trigger misses, introduce other issues, plus add needless non-standard API complexity. Conclusion; as designed, SG will utilise other configured providers when this one fails to hit targets, this provider should not be configured as a sole provider.
This commit is contained in:
parent
1d72320757
commit
93c219d82e
6 changed files with 246 additions and 3 deletions
|
@ -17,6 +17,7 @@
|
|||
* Change provider SCC remove 1 of 3 requests per search to save 30% time
|
||||
* Change provider Womble's use SSL
|
||||
* Change refactor and code simplification for torrent providers
|
||||
* Add Rarbg torrent provider
|
||||
* Fix getManualSearchStatus: object has no attribute 'segment'
|
||||
* Change handling of general HTTP error response codes to prevent issues
|
||||
* Add handling for CloudFlare custom HTTP response codes
|
||||
|
|
BIN
gui/slick/images/providers/rarbg.png
Normal file
BIN
gui/slick/images/providers/rarbg.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 635 B |
|
@ -37,7 +37,7 @@ from sickbeard import providers, metadata, config, webserveInit
|
|||
from sickbeard.providers.generic import GenericProvider
|
||||
from providers import btn, newznab, womble, thepiratebay, torrentleech, kat, iptorrents, \
|
||||
omgwtfnzbs, scc, hdtorrents, torrentday, hdbits, nextgen, speedcd, nyaatorrents, torrentbytes, \
|
||||
freshontv, bitsoup, tokyotoshokan, animenzb, totv
|
||||
freshontv, bitsoup, tokyotoshokan, animenzb, totv, rarbg
|
||||
from sickbeard.config import CheckSection, check_setting_int, check_setting_str, check_setting_float, ConfigMigrator, \
|
||||
naming_ep_type, minimax
|
||||
from sickbeard import searchBacklog, showUpdater, versionChecker, autoPostProcesser, \
|
||||
|
@ -1209,7 +1209,7 @@ def initialize(consoleLogging=True):
|
|||
# processors
|
||||
autoPostProcesserScheduler = scheduler.Scheduler(autoPostProcesser.PostProcesser(),
|
||||
cycleTime=datetime.timedelta(
|
||||
minutes=AUTOPOSTPROCESSER_FREQUENCY),
|
||||
minutes=AUTOPOSTPROCESSER_FREQUENCY),
|
||||
threadName='POSTPROCESSER',
|
||||
silent=not PROCESS_AUTOMATICALLY)
|
||||
|
||||
|
|
|
@ -35,6 +35,7 @@ __all__ = ['womble',
|
|||
'bitsoup',
|
||||
'tokyotoshokan',
|
||||
'animenzb',
|
||||
'rarbg',
|
||||
'totv',
|
||||
]
|
||||
|
||||
|
|
|
@ -493,6 +493,8 @@ class TorrentProvider(GenericProvider):
|
|||
def getQuality(self, item, anime=False):
|
||||
if isinstance(item, tuple):
|
||||
name = item[0]
|
||||
elif isinstance(item, dict):
|
||||
name, url = self._get_title_and_url(item)
|
||||
else:
|
||||
name = item.title
|
||||
return Quality.sceneQuality(name, anime)
|
||||
|
@ -535,9 +537,10 @@ class TorrentProvider(GenericProvider):
|
|||
for proper_string in method:
|
||||
search_string = self._get_episode_search_strings(cur_ep, add_string=proper_string)
|
||||
|
||||
proper_exp = re.sub(r'(?i)[^a-z\|\.]+', '', proper_string)
|
||||
for item in self._doSearch(search_string[0]):
|
||||
title, url = self._get_title_and_url(item)
|
||||
if not re.search('(?i)(?:%s)' % '|'.join(method), title):
|
||||
if not re.search('(?i)(?:%s)' % proper_exp, title):
|
||||
continue
|
||||
results.append(classes.Proper(title, url, datetime.datetime.today(), self.show))
|
||||
|
||||
|
|
238
sickbeard/providers/rarbg.py
Normal file
238
sickbeard/providers/rarbg.py
Normal file
|
@ -0,0 +1,238 @@
|
|||
# Author: SickGear
|
||||
#
|
||||
# This file is part of SickGear.
|
||||
#
|
||||
# SickGear 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.
|
||||
#
|
||||
# SickGear 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 SickGear. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
import re
|
||||
import datetime
|
||||
import time
|
||||
|
||||
import sickbeard
|
||||
import generic
|
||||
from sickbeard import logger
|
||||
from sickbeard import tvcache
|
||||
from sickbeard import show_name_helpers
|
||||
from sickbeard import helpers
|
||||
from sickbeard.indexers.indexer_config import INDEXER_TVDB
|
||||
|
||||
|
||||
class RarbgProvider(generic.TorrentProvider):
|
||||
|
||||
def __init__(self):
|
||||
generic.TorrentProvider.__init__(self, 'Rarbg')
|
||||
self.session = None
|
||||
|
||||
self.request_throttle = datetime.datetime.now()
|
||||
self.token = None
|
||||
self.token_expiry = None
|
||||
|
||||
self.confirmed = False
|
||||
self.ratio = None
|
||||
self.minseed = None
|
||||
self.minleech = None
|
||||
|
||||
# api_spec: https://rarbg.com/pubapi/apidocs_v2.txt
|
||||
self.url_api = 'https://torrentapi.org/pubapi_v2.php?'
|
||||
self.urls = {'config_provider_home_uri': 'https://rarbg.com',
|
||||
'api_token': self.url_api + 'get_token=get_token',
|
||||
'api_list': self.url_api + 'mode=list',
|
||||
'api_search': self.url_api + 'mode=search'}
|
||||
|
||||
self.params = {'defaults': '&category=%(cat)s&ranked=%(ranked)s&limit=100&sort=last'
|
||||
% {'cat': 'tv', 'ranked': int(self.confirmed)} + '&token=%(token)s',
|
||||
'param_iid': '&search_imdb=%(sid)s',
|
||||
'param_tid': '&search_tvdb=%(sid)s',
|
||||
'param_rid': '&search_tvrage=%(sid)s',
|
||||
'param_str': '&search_string=%(str)s',
|
||||
'param_seed': '&min_seeders=%(min_seeds)s',
|
||||
'param_peer': '&min_leechers=%(min_peers)s'}
|
||||
|
||||
self.url = self.urls['config_provider_home_uri']
|
||||
self.cache = RarbgCache(self)
|
||||
|
||||
def _doLogin(self, reset=False):
|
||||
|
||||
if not reset and self.token and self.token_expiry and datetime.datetime.now() < self.token_expiry:
|
||||
return True
|
||||
|
||||
try:
|
||||
data = helpers.getURL(self.urls['api_token'], headers=self.headers, json=True)
|
||||
self.token = data['token']
|
||||
self.token_expiry = datetime.datetime.now() + datetime.timedelta(minutes=14)
|
||||
return True
|
||||
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
logger.log(u'No usable API token returned from: %s' % self.urls['api_token'], logger.ERROR)
|
||||
return False
|
||||
|
||||
def _get_season_search_strings(self, ep_obj):
|
||||
|
||||
if ep_obj.show.air_by_date or ep_obj.show.sports:
|
||||
ep_detail = str(ep_obj.airdate).split('-')[0]
|
||||
elif ep_obj.show.anime:
|
||||
ep_detail = '%d' % ep_obj.scene_absolute_number
|
||||
else:
|
||||
ep_detail = 'S%02d' % int(ep_obj.scene_season)
|
||||
|
||||
search_string = {'Season': []}
|
||||
for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
|
||||
search_string['Season'].append('%s %s' % (show_name, ep_detail))
|
||||
|
||||
if not self.show.sports and not self.show.anime:
|
||||
search_string.update({'Season_only': [ep_detail]})
|
||||
|
||||
return [search_string]
|
||||
|
||||
def _get_episode_search_strings(self, ep_obj, add_string=''):
|
||||
|
||||
if not ep_obj:
|
||||
return []
|
||||
|
||||
airdate = str(ep_obj.airdate).replace('-', ' ')
|
||||
if self.show.air_by_date:
|
||||
ep_detail = airdate
|
||||
elif self.show.sports:
|
||||
ep_detail = '%s|%s' % (airdate, ep_obj.airdate.strftime('%b'))
|
||||
elif self.show.anime:
|
||||
ep_detail = ep_obj.scene_absolute_number
|
||||
else:
|
||||
ep_detail = sickbeard.config.naming_ep_type[2] % {'seasonnumber': ep_obj.scene_season,
|
||||
'episodenumber': ep_obj.scene_episode}
|
||||
if add_string and not self.show.anime:
|
||||
ep_detail += ' ' + add_string
|
||||
|
||||
search_string = {'Episode': []}
|
||||
for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
|
||||
search_string['Episode'].append(re.sub('\s+', ' ', '%s %s' % (show_name, ep_detail)))
|
||||
|
||||
if not self.show.sports and not self.show.anime:
|
||||
search_string.update({'Episode_only': [ep_detail]})
|
||||
|
||||
return [search_string]
|
||||
|
||||
def _doSearch(self, search_params, search_mode='eponly', epcount=0, age=0):
|
||||
|
||||
results = []
|
||||
items = {'Season': [], 'Episode': [], 'RSS': []}
|
||||
|
||||
if not self._doLogin(reset=True):
|
||||
return results
|
||||
|
||||
id_search = None
|
||||
if hasattr(self, 'show') and self.show and self.show.indexer and self.show.indexerid:
|
||||
if 0 < len(self.show.imdb_info):
|
||||
sid = self.show.imdb_info['imdb_id']
|
||||
search_with = 'param_iid'
|
||||
else:
|
||||
sid = self.show.indexerid
|
||||
if INDEXER_TVDB == self.show.indexer:
|
||||
search_with = 'param_tid'
|
||||
else: # INDEXER_TVRAGE == self.show.indexer:
|
||||
search_with = 'param_rid'
|
||||
id_search = self.params[search_with] % {'sid': sid}
|
||||
|
||||
dedupe = []
|
||||
search_types = sorted([x for x in search_params.items()], key=lambda tup: tup[1], reverse=True) # sort type "_only" as first to process
|
||||
for mode_params in search_types:
|
||||
mode_search = mode_params[0]
|
||||
mode_base = mode_search.replace('_only', '')
|
||||
for search_string in mode_params[1]:
|
||||
search_url = ''
|
||||
if 'RSS' == mode_search:
|
||||
url = 'api_list'
|
||||
else:
|
||||
url = 'api_search'
|
||||
|
||||
if '_only' in mode_search and id_search:
|
||||
search_url = id_search
|
||||
|
||||
if None is not search_string:
|
||||
search_url += self.params['param_str'] % {'str': search_string}
|
||||
|
||||
search_url = self.urls[url] + self.params['defaults'] + search_url
|
||||
|
||||
if self.minseed:
|
||||
search_url += self.params['param_seed'] % {'min_seeds': self.minseed}
|
||||
|
||||
if self.minleech:
|
||||
search_url += self.params['param_peer'] % {'min_peers': self.minleech}
|
||||
|
||||
logger.log(u'Base search URL: %s' % search_url, logger.DEBUG)
|
||||
|
||||
for r in xrange(0, 3):
|
||||
time_out = 0
|
||||
while(self.request_throttle > datetime.datetime.now()) and 15 >= time_out:
|
||||
time_out += 1
|
||||
time.sleep(1)
|
||||
|
||||
try:
|
||||
data = self.getURL(search_url % {'token': self.token}, json=True)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
self.request_throttle = datetime.datetime.now() + datetime.timedelta(seconds=3)
|
||||
|
||||
if 'error' in data:
|
||||
if 5 == data['error_code']: # Too many requests per second.
|
||||
continue
|
||||
|
||||
elif 2 == data['error_code']: # Invalid token set!
|
||||
if self._doLogin(reset=True):
|
||||
continue
|
||||
return results
|
||||
break
|
||||
|
||||
if 'error' not in data:
|
||||
for item in data['torrent_results']:
|
||||
try:
|
||||
title = item['filename']
|
||||
get = item['download']
|
||||
if not (title and get) or get in dedupe:
|
||||
continue
|
||||
dedupe += [get]
|
||||
items[mode_base].append((title, get))
|
||||
logger.log(u'Found result: %s' % title, logger.DEBUG)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
if 0 < len(items[mode_base]):
|
||||
results += items[mode_base]
|
||||
items[mode_base] = []
|
||||
else:
|
||||
logger.log(u'No results found for: %s' % search_string, logger.DEBUG)
|
||||
|
||||
if '_only' in mode_search and 0 < len(results):
|
||||
break
|
||||
|
||||
return results
|
||||
|
||||
def findPropers(self, search_date=datetime.datetime.today()):
|
||||
return self._find_propers(search_date, '{{.proper.|.repack.}}')
|
||||
|
||||
|
||||
class RarbgCache(tvcache.TVCache):
|
||||
def __init__(self, this_provider):
|
||||
|
||||
tvcache.TVCache.__init__(self, this_provider)
|
||||
|
||||
self.minTime = 10 # cache update frequency
|
||||
|
||||
def _getRSSData(self):
|
||||
return self.provider.get_cache_data()
|
||||
|
||||
|
||||
provider = RarbgProvider()
|
Loading…
Reference in a new issue