mirror of
https://github.com/moparisthebest/SickRage
synced 2025-01-09 04:48:41 -05:00
199 lines
7.6 KiB
Python
199 lines
7.6 KiB
Python
# -*- coding: utf-8 -*-
|
|
from __future__ import unicode_literals
|
|
import base64
|
|
import logging
|
|
import os
|
|
import re
|
|
import zlib
|
|
import babelfish
|
|
import guessit
|
|
from . import Provider
|
|
from .. import __version__
|
|
from ..compat import ServerProxy, TimeoutTransport
|
|
from ..exceptions import ProviderError, AuthenticationError, DownloadLimitExceeded
|
|
from ..subtitle import Subtitle, fix_line_endings, compute_guess_matches
|
|
from ..video import Episode, Movie
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class OpenSubtitlesSubtitle(Subtitle):
|
|
provider_name = 'opensubtitles'
|
|
series_re = re.compile('^"(?P<series_name>.*)" (?P<series_title>.*)$')
|
|
|
|
def __init__(self, language, hearing_impaired, id, matched_by, movie_kind, hash, movie_name, movie_release_name, # @ReservedAssignment
|
|
movie_year, movie_imdb_id, series_season, series_episode, page_link):
|
|
super(OpenSubtitlesSubtitle, self).__init__(language, hearing_impaired, page_link)
|
|
self.id = id
|
|
self.matched_by = matched_by
|
|
self.movie_kind = movie_kind
|
|
self.hash = hash
|
|
self.movie_name = movie_name
|
|
self.movie_release_name = movie_release_name
|
|
self.movie_year = movie_year
|
|
self.movie_imdb_id = movie_imdb_id
|
|
self.series_season = series_season
|
|
self.series_episode = series_episode
|
|
|
|
@property
|
|
def series_name(self):
|
|
return self.series_re.match(self.movie_name).group('series_name')
|
|
|
|
@property
|
|
def series_title(self):
|
|
return self.series_re.match(self.movie_name).group('series_title')
|
|
|
|
def compute_matches(self, video):
|
|
matches = set()
|
|
# episode
|
|
if isinstance(video, Episode) and self.movie_kind == 'episode':
|
|
# series
|
|
if video.series and self.series_name.lower() == video.series.lower():
|
|
matches.add('series')
|
|
# season
|
|
if video.season and self.series_season == video.season:
|
|
matches.add('season')
|
|
# episode
|
|
if video.episode and self.series_episode == video.episode:
|
|
matches.add('episode')
|
|
# guess
|
|
matches |= compute_guess_matches(video, guessit.guess_episode_info(self.movie_release_name + '.mkv'))
|
|
# movie
|
|
elif isinstance(video, Movie) and self.movie_kind == 'movie':
|
|
# year
|
|
if video.year and self.movie_year == video.year:
|
|
matches.add('year')
|
|
# guess
|
|
matches |= compute_guess_matches(video, guessit.guess_movie_info(self.movie_release_name + '.mkv'))
|
|
else:
|
|
logger.info('%r is not a valid movie_kind for %r', self.movie_kind, video)
|
|
return matches
|
|
# hash
|
|
if 'opensubtitles' in video.hashes and self.hash == video.hashes['opensubtitles']:
|
|
matches.add('hash')
|
|
# imdb_id
|
|
if video.imdb_id and self.movie_imdb_id == video.imdb_id:
|
|
matches.add('imdb_id')
|
|
# title
|
|
if video.title and self.movie_name.lower() == video.title.lower():
|
|
matches.add('title')
|
|
return matches
|
|
|
|
|
|
class OpenSubtitlesProvider(Provider):
|
|
languages = {babelfish.Language.fromopensubtitles(l) for l in babelfish.language_converters['opensubtitles'].codes}
|
|
|
|
def __init__(self):
|
|
self.server = ServerProxy('http://api.opensubtitles.org/xml-rpc', transport=TimeoutTransport(10))
|
|
self.token = None
|
|
|
|
def initialize(self):
|
|
response = checked(self.server.LogIn('', '', 'eng', 'subliminal v%s' % __version__.split('-')[0]))
|
|
self.token = response['token']
|
|
|
|
def terminate(self):
|
|
checked(self.server.LogOut(self.token))
|
|
self.server.close()
|
|
|
|
def no_operation(self):
|
|
checked(self.server.NoOperation(self.token))
|
|
|
|
def query(self, languages, hash=None, size=None, imdb_id=None, query=None): # @ReservedAssignment
|
|
searches = []
|
|
if hash and size:
|
|
searches.append({'moviehash': hash, 'moviebytesize': str(size)})
|
|
if imdb_id:
|
|
searches.append({'imdbid': imdb_id})
|
|
if query:
|
|
searches.append({'query': query})
|
|
if not searches:
|
|
raise ValueError('One or more parameter missing')
|
|
for search in searches:
|
|
search['sublanguageid'] = ','.join(l.opensubtitles for l in languages)
|
|
logger.debug('Searching subtitles %r', searches)
|
|
response = checked(self.server.SearchSubtitles(self.token, searches))
|
|
if not response['data']:
|
|
logger.debug('No subtitle found')
|
|
return []
|
|
return [OpenSubtitlesSubtitle(babelfish.Language.fromopensubtitles(r['SubLanguageID']),
|
|
bool(int(r['SubHearingImpaired'])), r['IDSubtitleFile'], r['MatchedBy'],
|
|
r['MovieKind'], r['MovieHash'], r['MovieName'], r['MovieReleaseName'],
|
|
int(r['MovieYear']) if r['MovieYear'] else None, int(r['IDMovieImdb']),
|
|
int(r['SeriesSeason']) if r['SeriesSeason'] else None,
|
|
int(r['SeriesEpisode']) if r['SeriesEpisode'] else None, r['SubtitlesLink'])
|
|
for r in response['data']]
|
|
|
|
def list_subtitles(self, video, languages):
|
|
query = None
|
|
if ('opensubtitles' not in video.hashes or not video.size) and not video.imdb_id:
|
|
query = video.name.split(os.sep)[-1]
|
|
return self.query(languages, hash=video.hashes.get('opensubtitles'), size=video.size, imdb_id=video.imdb_id,
|
|
query=query)
|
|
|
|
def download_subtitle(self, subtitle):
|
|
response = checked(self.server.DownloadSubtitles(self.token, [subtitle.id]))
|
|
if not response['data']:
|
|
raise ProviderError('Nothing to download')
|
|
subtitle.content = fix_line_endings(zlib.decompress(base64.b64decode(response['data'][0]['data']), 47))
|
|
|
|
|
|
class OpenSubtitlesError(ProviderError):
|
|
"""Base class for non-generic :class:`OpenSubtitlesProvider` exceptions"""
|
|
|
|
|
|
class Unauthorized(OpenSubtitlesError, AuthenticationError):
|
|
"""Exception raised when status is '401 Unauthorized'"""
|
|
|
|
|
|
class NoSession(OpenSubtitlesError, AuthenticationError):
|
|
"""Exception raised when status is '406 No session'"""
|
|
|
|
|
|
class DownloadLimitReached(OpenSubtitlesError, DownloadLimitExceeded):
|
|
"""Exception raised when status is '407 Download limit reached'"""
|
|
|
|
|
|
class InvalidImdbid(OpenSubtitlesError):
|
|
"""Exception raised when status is '413 Invalid ImdbID'"""
|
|
|
|
|
|
class UnknownUserAgent(OpenSubtitlesError, AuthenticationError):
|
|
"""Exception raised when status is '414 Unknown User Agent'"""
|
|
|
|
|
|
class DisabledUserAgent(OpenSubtitlesError, AuthenticationError):
|
|
"""Exception raised when status is '415 Disabled user agent'"""
|
|
|
|
|
|
class ServiceUnavailable(OpenSubtitlesError):
|
|
"""Exception raised when status is '503 Service Unavailable'"""
|
|
|
|
|
|
def checked(response):
|
|
"""Check a response status before returning it
|
|
|
|
:param response: a response from a XMLRPC call to OpenSubtitles
|
|
:return: the response
|
|
:raise: :class:`OpenSubtitlesError`
|
|
|
|
"""
|
|
status_code = int(response['status'][:3])
|
|
if status_code == 401:
|
|
raise Unauthorized
|
|
if status_code == 406:
|
|
raise NoSession
|
|
if status_code == 407:
|
|
raise DownloadLimitReached
|
|
if status_code == 413:
|
|
raise InvalidImdbid
|
|
if status_code == 414:
|
|
raise UnknownUserAgent
|
|
if status_code == 415:
|
|
raise DisabledUserAgent
|
|
if status_code == 503:
|
|
raise ServiceUnavailable
|
|
if status_code != 200:
|
|
raise OpenSubtitlesError(response['status'])
|
|
return response
|