2018-03-23 23:59:59 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
import io
|
|
|
|
import json
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
|
|
|
|
from babelfish import Language, language_converters
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
from dogpile.cache.api import NO_VALUE
|
|
|
|
from guessit import guessit
|
|
|
|
import pytz
|
|
|
|
import rarfile
|
|
|
|
from rarfile import RarFile, is_rarfile
|
|
|
|
from requests import Session
|
|
|
|
from zipfile import ZipFile, is_zipfile
|
|
|
|
|
|
|
|
from . import ParserBeautifulSoup, Provider
|
|
|
|
from .. import __short_version__
|
|
|
|
from ..cache import SHOW_EXPIRATION_TIME, region
|
2019-09-13 19:12:26 +00:00
|
|
|
from ..exceptions import AuthenticationError, ConfigurationError, ProviderError
|
2018-03-23 23:59:59 +00:00
|
|
|
from ..subtitle import SUBTITLE_EXTENSIONS, Subtitle, fix_line_ending, guess_matches, sanitize
|
|
|
|
from ..video import Episode, Movie
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
language_converters.register('legendastv = subliminal.converters.legendastv:LegendasTVConverter')
|
|
|
|
|
|
|
|
# Configure :mod:`rarfile` to use the same path separator as :mod:`zipfile`
|
|
|
|
rarfile.PATH_SEP = '/'
|
|
|
|
|
|
|
|
#: Conversion map for types
|
|
|
|
type_map = {'M': 'movie', 'S': 'episode', 'C': 'episode'}
|
|
|
|
|
|
|
|
#: BR title season parsing regex
|
|
|
|
season_re = re.compile(r' - (?P<season>\d+)(\xaa|a|st|nd|rd|th) (temporada|season)', re.IGNORECASE)
|
|
|
|
|
|
|
|
#: Downloads parsing regex
|
|
|
|
downloads_re = re.compile(r'(?P<downloads>\d+) downloads')
|
|
|
|
|
|
|
|
#: Rating parsing regex
|
|
|
|
rating_re = re.compile(r'nota (?P<rating>\d+)')
|
|
|
|
|
|
|
|
#: Timestamp parsing regex
|
|
|
|
timestamp_re = re.compile(r'(?P<day>\d+)/(?P<month>\d+)/(?P<year>\d+) - (?P<hour>\d+):(?P<minute>\d+)')
|
|
|
|
|
|
|
|
#: Cache key for releases
|
2019-09-13 19:12:26 +00:00
|
|
|
releases_key = __name__ + ':releases|{archive_id}'
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LegendasTVArchive(object):
|
|
|
|
"""LegendasTV Archive.
|
|
|
|
|
|
|
|
:param str id: identifier.
|
|
|
|
:param str name: name.
|
|
|
|
:param bool pack: contains subtitles for multiple episodes.
|
|
|
|
:param bool pack: featured.
|
|
|
|
:param str link: link.
|
|
|
|
:param int downloads: download count.
|
|
|
|
:param int rating: rating (0-10).
|
|
|
|
:param timestamp: timestamp.
|
|
|
|
:type timestamp: datetime.datetime
|
2018-04-23 12:36:17 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
"""
|
2018-03-23 23:59:59 +00:00
|
|
|
def __init__(self, id, name, pack, featured, link, downloads=0, rating=0, timestamp=None):
|
|
|
|
#: Identifier
|
|
|
|
self.id = id
|
|
|
|
|
|
|
|
#: Name
|
|
|
|
self.name = name
|
|
|
|
|
|
|
|
#: Pack
|
|
|
|
self.pack = pack
|
|
|
|
|
|
|
|
#: Featured
|
|
|
|
self.featured = featured
|
|
|
|
|
|
|
|
#: Link
|
|
|
|
self.link = link
|
|
|
|
|
|
|
|
#: Download count
|
|
|
|
self.downloads = downloads
|
|
|
|
|
|
|
|
#: Rating (0-10)
|
|
|
|
self.rating = rating
|
|
|
|
|
|
|
|
#: Timestamp
|
|
|
|
self.timestamp = timestamp
|
|
|
|
|
|
|
|
#: Compressed content as :class:`rarfile.RarFile` or :class:`zipfile.ZipFile`
|
|
|
|
self.content = None
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return '<%s [%s] %r>' % (self.__class__.__name__, self.id, self.name)
|
|
|
|
|
|
|
|
|
|
|
|
class LegendasTVSubtitle(Subtitle):
|
|
|
|
"""LegendasTV Subtitle."""
|
|
|
|
provider_name = 'legendastv'
|
|
|
|
|
|
|
|
def __init__(self, language, type, title, year, imdb_id, season, archive, name):
|
2019-09-13 19:12:26 +00:00
|
|
|
super(LegendasTVSubtitle, self).__init__(language, archive.link)
|
2018-03-23 23:59:59 +00:00
|
|
|
self.type = type
|
|
|
|
self.title = title
|
|
|
|
self.year = year
|
|
|
|
self.imdb_id = imdb_id
|
|
|
|
self.season = season
|
|
|
|
self.archive = archive
|
|
|
|
self.name = name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def id(self):
|
|
|
|
return '%s-%s' % (self.archive.id, self.name.lower())
|
|
|
|
|
|
|
|
def get_matches(self, video, hearing_impaired=False):
|
|
|
|
matches = set()
|
|
|
|
|
|
|
|
# episode
|
|
|
|
if isinstance(video, Episode) and self.type == 'episode':
|
|
|
|
# series
|
2019-09-13 19:12:26 +00:00
|
|
|
if video.series and sanitize(self.title) == sanitize(video.series):
|
2018-03-23 23:59:59 +00:00
|
|
|
matches.add('series')
|
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
# year (year is based on season air date hence the adjustment)
|
|
|
|
if video.original_series and self.year is None or video.year and video.year == self.year - self.season + 1:
|
2018-03-23 23:59:59 +00:00
|
|
|
matches.add('year')
|
|
|
|
|
|
|
|
# imdb_id
|
|
|
|
if video.series_imdb_id and self.imdb_id == video.series_imdb_id:
|
|
|
|
matches.add('series_imdb_id')
|
|
|
|
|
|
|
|
# movie
|
|
|
|
elif isinstance(video, Movie) and self.type == 'movie':
|
|
|
|
# title
|
2019-09-13 19:12:26 +00:00
|
|
|
if video.title and sanitize(self.title) == sanitize(video.title):
|
2018-03-23 23:59:59 +00:00
|
|
|
matches.add('title')
|
|
|
|
|
|
|
|
# year
|
|
|
|
if video.year and self.year == video.year:
|
|
|
|
matches.add('year')
|
|
|
|
|
|
|
|
# imdb_id
|
|
|
|
if video.imdb_id and self.imdb_id == video.imdb_id:
|
|
|
|
matches.add('imdb_id')
|
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
# archive name
|
|
|
|
matches |= guess_matches(video, guessit(self.archive.name, {'type': self.type}))
|
|
|
|
|
2018-03-23 23:59:59 +00:00
|
|
|
# name
|
|
|
|
matches |= guess_matches(video, guessit(self.name, {'type': self.type}))
|
|
|
|
|
|
|
|
return matches
|
|
|
|
|
|
|
|
|
|
|
|
class LegendasTVProvider(Provider):
|
|
|
|
"""LegendasTV Provider.
|
|
|
|
|
|
|
|
:param str username: username.
|
|
|
|
:param str password: password.
|
2018-04-23 12:36:17 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
"""
|
2018-03-23 23:59:59 +00:00
|
|
|
languages = {Language.fromlegendastv(l) for l in language_converters['legendastv'].codes}
|
|
|
|
server_url = 'http://legendas.tv/'
|
|
|
|
|
|
|
|
def __init__(self, username=None, password=None):
|
2019-09-13 19:12:26 +00:00
|
|
|
if username and not password or not username and password:
|
2018-03-23 23:59:59 +00:00
|
|
|
raise ConfigurationError('Username and password must be specified')
|
|
|
|
|
|
|
|
self.username = username
|
|
|
|
self.password = password
|
|
|
|
self.logged_in = False
|
|
|
|
|
|
|
|
def initialize(self):
|
|
|
|
self.session = Session()
|
|
|
|
self.session.headers['User-Agent'] = 'Subliminal/%s' % __short_version__
|
|
|
|
|
|
|
|
# login
|
2019-09-13 19:12:26 +00:00
|
|
|
if self.username is not None and self.password is not None:
|
2018-03-23 23:59:59 +00:00
|
|
|
logger.info('Logging in')
|
|
|
|
data = {'_method': 'POST', 'data[User][username]': self.username, 'data[User][password]': self.password}
|
|
|
|
r = self.session.post(self.server_url + 'login', data, allow_redirects=False, timeout=10)
|
2019-09-13 19:12:26 +00:00
|
|
|
r.raise_for_status()
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
soup = ParserBeautifulSoup(r.content, ['html.parser'])
|
|
|
|
if soup.find('div', {'class': 'alert-error'}, string=re.compile(u'Usuário ou senha inválidos')):
|
|
|
|
raise AuthenticationError(self.username)
|
|
|
|
|
|
|
|
logger.debug('Logged in')
|
|
|
|
self.logged_in = True
|
|
|
|
|
|
|
|
def terminate(self):
|
|
|
|
# logout
|
|
|
|
if self.logged_in:
|
|
|
|
logger.info('Logging out')
|
|
|
|
r = self.session.get(self.server_url + 'users/logout', allow_redirects=False, timeout=10)
|
2019-09-13 19:12:26 +00:00
|
|
|
r.raise_for_status()
|
2018-03-23 23:59:59 +00:00
|
|
|
logger.debug('Logged out')
|
|
|
|
self.logged_in = False
|
|
|
|
|
|
|
|
self.session.close()
|
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
@region.cache_on_arguments(expiration_time=SHOW_EXPIRATION_TIME)
|
|
|
|
def search_titles(self, title):
|
2018-03-23 23:59:59 +00:00
|
|
|
"""Search for titles matching the `title`.
|
|
|
|
|
|
|
|
:param str title: the title to search for.
|
|
|
|
:return: found titles.
|
|
|
|
:rtype: dict
|
2018-04-23 12:36:17 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
"""
|
|
|
|
# make the query
|
|
|
|
logger.info('Searching title %r', title)
|
|
|
|
r = self.session.get(self.server_url + 'legenda/sugestao/{}'.format(title), timeout=10)
|
|
|
|
r.raise_for_status()
|
|
|
|
results = json.loads(r.text)
|
2018-04-23 12:36:17 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
# loop over results
|
|
|
|
titles = {}
|
|
|
|
for result in results:
|
|
|
|
source = result['_source']
|
|
|
|
|
|
|
|
# extract id
|
|
|
|
title_id = int(source['id_filme'])
|
|
|
|
|
|
|
|
# extract type and title
|
|
|
|
title = {'type': type_map[source['tipo']], 'title': source['dsc_nome']}
|
|
|
|
|
|
|
|
# extract year
|
|
|
|
if source['dsc_data_lancamento'] and source['dsc_data_lancamento'].isdigit():
|
|
|
|
title['year'] = int(source['dsc_data_lancamento'])
|
|
|
|
|
|
|
|
# extract imdb_id
|
|
|
|
if source['id_imdb'] != '0':
|
|
|
|
if not source['id_imdb'].startswith('tt'):
|
|
|
|
title['imdb_id'] = 'tt' + source['id_imdb'].zfill(7)
|
|
|
|
else:
|
|
|
|
title['imdb_id'] = source['id_imdb']
|
|
|
|
|
|
|
|
# extract season
|
|
|
|
if title['type'] == 'episode':
|
|
|
|
if source['temporada'] and source['temporada'].isdigit():
|
|
|
|
title['season'] = int(source['temporada'])
|
|
|
|
else:
|
|
|
|
match = season_re.search(source['dsc_nome_br'])
|
|
|
|
if match:
|
|
|
|
title['season'] = int(match.group('season'))
|
2018-04-23 12:36:17 +00:00
|
|
|
else:
|
2019-09-13 19:12:26 +00:00
|
|
|
logger.warning('No season detected for title %d', title_id)
|
2018-04-23 12:36:17 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
# add title
|
|
|
|
titles[title_id] = title
|
|
|
|
|
|
|
|
logger.debug('Found %d titles', len(titles))
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
return titles
|
|
|
|
|
|
|
|
@region.cache_on_arguments(expiration_time=timedelta(minutes=15).total_seconds())
|
2019-09-13 19:12:26 +00:00
|
|
|
def get_archives(self, title_id, language_code):
|
|
|
|
"""Get the archive list from a given `title_id` and `language_code`.
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
:param int title_id: title id.
|
|
|
|
:param int language_code: language code.
|
|
|
|
:return: the archives.
|
|
|
|
:rtype: list of :class:`LegendasTVArchive`
|
|
|
|
|
|
|
|
"""
|
2019-09-13 19:12:26 +00:00
|
|
|
logger.info('Getting archives for title %d and language %d', title_id, language_code)
|
2018-03-23 23:59:59 +00:00
|
|
|
archives = []
|
2019-09-13 19:12:26 +00:00
|
|
|
page = 1
|
2018-03-23 23:59:59 +00:00
|
|
|
while True:
|
|
|
|
# get the archive page
|
2019-09-13 19:12:26 +00:00
|
|
|
url = self.server_url + 'util/carrega_legendas_busca_filme/{title}/{language}/-/{page}'.format(
|
|
|
|
title=title_id, language=language_code, page=page)
|
2018-03-23 23:59:59 +00:00
|
|
|
r = self.session.get(url)
|
2019-09-13 19:12:26 +00:00
|
|
|
r.raise_for_status()
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
# parse the results
|
|
|
|
soup = ParserBeautifulSoup(r.content, ['lxml', 'html.parser'])
|
2019-09-13 19:12:26 +00:00
|
|
|
for archive_soup in soup.select('div.list_element > article > div'):
|
2018-03-23 23:59:59 +00:00
|
|
|
# create archive
|
2019-09-13 19:12:26 +00:00
|
|
|
archive = LegendasTVArchive(archive_soup.a['href'].split('/')[2], archive_soup.a.text,
|
|
|
|
'pack' in archive_soup['class'], 'destaque' in archive_soup['class'],
|
2018-03-23 23:59:59 +00:00
|
|
|
self.server_url + archive_soup.a['href'][1:])
|
|
|
|
|
|
|
|
# extract text containing downloads, rating and timestamp
|
|
|
|
data_text = archive_soup.find('p', class_='data').text
|
|
|
|
|
|
|
|
# match downloads
|
|
|
|
archive.downloads = int(downloads_re.search(data_text).group('downloads'))
|
|
|
|
|
|
|
|
# match rating
|
|
|
|
match = rating_re.search(data_text)
|
|
|
|
if match:
|
|
|
|
archive.rating = int(match.group('rating'))
|
|
|
|
|
|
|
|
# match timestamp and validate it
|
|
|
|
time_data = {k: int(v) for k, v in timestamp_re.search(data_text).groupdict().items()}
|
|
|
|
archive.timestamp = pytz.timezone('America/Sao_Paulo').localize(datetime(**time_data))
|
|
|
|
if archive.timestamp > datetime.utcnow().replace(tzinfo=pytz.utc):
|
|
|
|
raise ProviderError('Archive timestamp is in the future')
|
|
|
|
|
|
|
|
# add archive
|
|
|
|
archives.append(archive)
|
|
|
|
|
|
|
|
# stop on last page
|
|
|
|
if soup.find('a', attrs={'class': 'load_more'}, string='carregar mais') is None:
|
|
|
|
break
|
|
|
|
|
|
|
|
# increment page count
|
|
|
|
page += 1
|
|
|
|
|
|
|
|
logger.debug('Found %d archives', len(archives))
|
|
|
|
|
|
|
|
return archives
|
|
|
|
|
|
|
|
def download_archive(self, archive):
|
|
|
|
"""Download an archive's :attr:`~LegendasTVArchive.content`.
|
|
|
|
|
|
|
|
:param archive: the archive to download :attr:`~LegendasTVArchive.content` of.
|
|
|
|
:type archive: :class:`LegendasTVArchive`
|
|
|
|
|
|
|
|
"""
|
|
|
|
logger.info('Downloading archive %s', archive.id)
|
|
|
|
r = self.session.get(self.server_url + 'downloadarquivo/{}'.format(archive.id))
|
2019-09-13 19:12:26 +00:00
|
|
|
r.raise_for_status()
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
# open the archive
|
|
|
|
archive_stream = io.BytesIO(r.content)
|
|
|
|
if is_rarfile(archive_stream):
|
|
|
|
logger.debug('Identified rar archive')
|
|
|
|
archive.content = RarFile(archive_stream)
|
|
|
|
elif is_zipfile(archive_stream):
|
|
|
|
logger.debug('Identified zip archive')
|
|
|
|
archive.content = ZipFile(archive_stream)
|
|
|
|
else:
|
|
|
|
raise ValueError('Not a valid archive')
|
|
|
|
|
|
|
|
def query(self, language, title, season=None, episode=None, year=None):
|
|
|
|
# search for titles
|
2019-09-13 19:12:26 +00:00
|
|
|
titles = self.search_titles(sanitize(title))
|
|
|
|
|
|
|
|
# search for titles with the quote or dot character
|
|
|
|
ignore_characters = {'\'', '.'}
|
|
|
|
if any(c in title for c in ignore_characters):
|
|
|
|
titles.update(self.search_titles(sanitize(title, ignore_characters=ignore_characters)))
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
subtitles = []
|
|
|
|
# iterate over titles
|
|
|
|
for title_id, t in titles.items():
|
2019-09-13 19:12:26 +00:00
|
|
|
# discard mismatches on title
|
|
|
|
if sanitize(t['title']) != sanitize(title):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# episode
|
|
|
|
if season and episode:
|
|
|
|
# discard mismatches on type
|
|
|
|
if t['type'] != 'episode':
|
|
|
|
continue
|
|
|
|
|
|
|
|
# discard mismatches on season
|
|
|
|
if 'season' not in t or t['season'] != season:
|
|
|
|
continue
|
|
|
|
# movie
|
|
|
|
else:
|
|
|
|
# discard mismatches on type
|
|
|
|
if t['type'] != 'movie':
|
|
|
|
continue
|
2018-03-23 23:59:59 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
# discard mismatches on year
|
|
|
|
if year is not None and 'year' in t and t['year'] != year:
|
|
|
|
continue
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
# iterate over title's archives
|
2019-09-13 19:12:26 +00:00
|
|
|
for a in self.get_archives(title_id, language.legendastv):
|
|
|
|
# clean name of path separators and pack flags
|
|
|
|
clean_name = a.name.replace('/', '-')
|
|
|
|
if a.pack and clean_name.startswith('(p)'):
|
|
|
|
clean_name = clean_name[3:]
|
|
|
|
|
|
|
|
# guess from name
|
|
|
|
guess = guessit(clean_name, {'type': t['type']})
|
|
|
|
|
|
|
|
# episode
|
|
|
|
if season and episode:
|
|
|
|
# discard mismatches on episode in non-pack archives
|
|
|
|
if not a.pack and 'episode' in guess and guess['episode'] != episode:
|
|
|
|
continue
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
# compute an expiration time based on the archive timestamp
|
|
|
|
expiration_time = (datetime.utcnow().replace(tzinfo=pytz.utc) - a.timestamp).total_seconds()
|
|
|
|
|
|
|
|
# attempt to get the releases from the cache
|
2019-09-13 19:12:26 +00:00
|
|
|
releases = region.get(releases_key.format(archive_id=a.id), expiration_time=expiration_time)
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
# the releases are not in cache or cache is expired
|
|
|
|
if releases == NO_VALUE:
|
|
|
|
logger.info('Releases not found in cache')
|
|
|
|
|
|
|
|
# download archive
|
|
|
|
self.download_archive(a)
|
|
|
|
|
|
|
|
# extract the releases
|
|
|
|
releases = []
|
|
|
|
for name in a.content.namelist():
|
|
|
|
# discard the legendastv file
|
|
|
|
if name.startswith('Legendas.tv'):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# discard hidden files
|
|
|
|
if os.path.split(name)[-1].startswith('.'):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# discard non-subtitle files
|
|
|
|
if not name.lower().endswith(SUBTITLE_EXTENSIONS):
|
|
|
|
continue
|
|
|
|
|
|
|
|
releases.append(name)
|
|
|
|
|
|
|
|
# cache the releases
|
2019-09-13 19:12:26 +00:00
|
|
|
region.set(releases_key.format(archive_id=a.id), releases)
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
# iterate over releases
|
|
|
|
for r in releases:
|
2019-09-13 19:12:26 +00:00
|
|
|
subtitle = LegendasTVSubtitle(language, t['type'], t['title'], t.get('year'), t.get('imdb_id'),
|
|
|
|
t.get('season'), a, r)
|
2018-03-23 23:59:59 +00:00
|
|
|
logger.debug('Found subtitle %r', subtitle)
|
|
|
|
subtitles.append(subtitle)
|
|
|
|
|
|
|
|
return subtitles
|
|
|
|
|
|
|
|
def list_subtitles(self, video, languages):
|
|
|
|
season = episode = None
|
|
|
|
if isinstance(video, Episode):
|
2019-09-13 19:12:26 +00:00
|
|
|
title = video.series
|
2018-03-23 23:59:59 +00:00
|
|
|
season = video.season
|
|
|
|
episode = video.episode
|
|
|
|
else:
|
2019-09-13 19:12:26 +00:00
|
|
|
title = video.title
|
2018-03-23 23:59:59 +00:00
|
|
|
|
2019-09-13 19:12:26 +00:00
|
|
|
return [s for l in languages for s in self.query(l, title, season=season, episode=episode, year=video.year)]
|
2018-03-23 23:59:59 +00:00
|
|
|
|
|
|
|
def download_subtitle(self, subtitle):
|
|
|
|
# download archive in case we previously hit the releases cache and didn't download it
|
|
|
|
if subtitle.archive.content is None:
|
|
|
|
self.download_archive(subtitle.archive)
|
|
|
|
|
|
|
|
# extract subtitle's content
|
|
|
|
subtitle.content = fix_line_ending(subtitle.archive.content.read(subtitle.name))
|