2014-03-10 05:18:05 +00:00
|
|
|
#
|
2014-11-12 16:43:14 +00:00
|
|
|
# This file is part of SickGear.
|
2014-03-10 05:18:05 +00:00
|
|
|
#
|
2014-11-12 16:43:14 +00:00
|
|
|
# SickGear is free software: you can redistribute it and/or modify
|
2014-03-10 05:18:05 +00: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-11-12 16:43:14 +00:00
|
|
|
# SickGear is distributed in the hope that it will be useful,
|
2014-03-10 05:18:05 +00:00
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2014-07-31 09:45:51 +00:00
|
|
|
# GNU General Public License for more details.
|
2014-03-10 05:18:05 +00:00
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
2014-11-12 16:43:14 +00:00
|
|
|
# along with SickGear. If not, see <http://www.gnu.org/licenses/>.
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
import re
|
2014-03-10 05:18:05 +00:00
|
|
|
import urllib
|
2014-05-07 07:50:49 +00:00
|
|
|
|
2015-07-13 09:39:20 +00:00
|
|
|
from . import generic
|
2015-09-18 00:06:34 +00:00
|
|
|
from sickbeard import logger, show_name_helpers, tvcache
|
2016-08-26 23:36:01 +00:00
|
|
|
from sickbeard.helpers import tryInt
|
2014-03-10 05:18:05 +00:00
|
|
|
|
|
|
|
|
2014-03-25 05:57:24 +00:00
|
|
|
class NyaaProvider(generic.TorrentProvider):
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2015-07-13 09:39:20 +00:00
|
|
|
def __init__(self):
|
|
|
|
generic.TorrentProvider.__init__(self, 'NyaaTorrents', anime_only=True)
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
self.url_base = self.url = 'https://www.nyaa.se/'
|
|
|
|
|
|
|
|
self.minseed, self.minleech = 2 * [None]
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2015-07-13 09:39:20 +00:00
|
|
|
self.cache = NyaaCache(self)
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
def _search_provider(self, search_string, search_mode='eponly', **kwargs):
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2014-05-26 06:29:22 +00:00
|
|
|
if self.show and not self.show.is_anime:
|
2016-08-26 23:36:01 +00:00
|
|
|
return []
|
|
|
|
|
|
|
|
params = urllib.urlencode({'term': search_string.encode('utf-8'),
|
|
|
|
'cats': '1_37', # Limit to English-translated Anime (for now)
|
|
|
|
# 'sort': '2', # Sort Descending By Seeders
|
|
|
|
})
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
return self.get_data(getrss_func=self.cache.getRSSFeed,
|
|
|
|
search_url='%s?page=rss&%s' % (self.url, params),
|
|
|
|
mode=('Episode', 'Season')['sponly' == search_mode])
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
def get_data(self, getrss_func, search_url, mode='cache'):
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
data = getrss_func(search_url)
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
results = []
|
2015-07-13 09:39:20 +00:00
|
|
|
if data and 'entries' in data:
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
rc = dict((k, re.compile('(?i)' + v)) for (k, v) in {
|
|
|
|
'stats': '(\d+)\W+seed[^\d]+(\d+)\W+leech[^\d]+\d+\W+down[^\d]+([\d.,]+\s\w+)'}.iteritems())
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
for cur_item in data.get('entries', []):
|
|
|
|
try:
|
|
|
|
seeders, leechers, size = 0, 0, 0
|
|
|
|
stats = rc['stats'].findall(cur_item.get('summary_detail', {'value': ''}).get('value', ''))
|
|
|
|
if len(stats):
|
|
|
|
seeders, leechers, size = (tryInt(n, n) for n in stats[0])
|
|
|
|
if self._peers_fail(mode, seeders, leechers):
|
|
|
|
continue
|
|
|
|
title, download_url = self._title_and_url(cur_item)
|
|
|
|
download_url = self._link(download_url)
|
|
|
|
except (AttributeError, TypeError, ValueError, IndexError):
|
|
|
|
continue
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
if title and download_url:
|
|
|
|
results.append((title, download_url, seeders, self._bytesizer(size)))
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
self._log_search(mode, len(results), search_url)
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
return self._sort_seeding(mode, results)
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2015-09-18 00:06:34 +00:00
|
|
|
def _season_strings(self, ep_obj, **kwargs):
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2015-07-13 09:39:20 +00:00
|
|
|
return show_name_helpers.makeSceneShowSearchStrings(self.show)
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2015-09-18 00:06:34 +00:00
|
|
|
def _episode_strings(self, ep_obj, **kwargs):
|
2015-07-13 09:39:20 +00:00
|
|
|
|
2015-09-18 00:06:34 +00:00
|
|
|
return self._season_strings(ep_obj)
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2014-07-31 09:45:51 +00:00
|
|
|
|
2014-03-25 05:57:24 +00:00
|
|
|
class NyaaCache(tvcache.TVCache):
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2015-07-13 09:39:20 +00:00
|
|
|
def __init__(self, this_provider):
|
|
|
|
tvcache.TVCache.__init__(self, this_provider)
|
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
self.update_freq = 15
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2015-09-18 00:06:34 +00:00
|
|
|
def _cache_data(self):
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
params = urllib.urlencode({'page': 'rss', # Use RSS page
|
|
|
|
'order': '1', # Sort Descending By Date
|
|
|
|
'cats': '1_37' # Limit to English-translated Anime (for now)
|
|
|
|
})
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2016-08-26 23:36:01 +00:00
|
|
|
return self.provider.get_data(getrss_func=self.getRSSFeed,
|
|
|
|
search_url='%s?%s' % (self.provider.url, params))
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2014-03-25 05:57:24 +00:00
|
|
|
|
2014-06-01 22:05:07 +00:00
|
|
|
provider = NyaaProvider()
|