update subliminal_patch; improve scoring; improve bsplayer; possibly fix bazarr#821

This commit is contained in:
panni 2020-02-16 05:53:32 +01:00
parent 9455847a0e
commit 584f4a7c04
3 changed files with 34 additions and 47 deletions

View file

@ -360,15 +360,16 @@ class SZProviderPool(ProviderPool):
orig_matches = matches.copy() orig_matches = matches.copy()
logger.debug('%r: Found matches %r', s, matches) logger.debug('%r: Found matches %r', s, matches)
score, score_without_hash = compute_score(matches, s, video, hearing_impaired=use_hearing_impaired)
unsorted_subtitles.append( unsorted_subtitles.append(
(s, compute_score(matches, s, video, hearing_impaired=use_hearing_impaired), matches, orig_matches)) (s, score, score_without_hash, matches, orig_matches))
# sort subtitles by score # sort subtitles by score
scored_subtitles = sorted(unsorted_subtitles, key=operator.itemgetter(1), reverse=True) scored_subtitles = sorted(unsorted_subtitles, key=operator.itemgetter(1, 2), reverse=True)
# download best subtitles, falling back on the next on error # download best subtitles, falling back on the next on error
downloaded_subtitles = [] downloaded_subtitles = []
for subtitle, score, matches, orig_matches in scored_subtitles: for subtitle, score, score_without_hash, matches, orig_matches in scored_subtitles:
# check score # check score
if score < min_score: if score < min_score:
logger.info('%r: Score %d is below min_score (%d)', subtitle, score, min_score) logger.info('%r: Score %d is below min_score (%d)', subtitle, score, min_score)
@ -552,7 +553,7 @@ def scan_video(path, dont_use_actual_file=False, hints=None, providers=None, ski
video.hashes['bsplayer'] = osub_hash = hash_opensubtitles(hash_path) video.hashes['bsplayer'] = osub_hash = hash_opensubtitles(hash_path)
if "opensubtitles" in providers: if "opensubtitles" in providers:
video.hashes['opensubtitles'] = osub_hash = hash_opensubtitles(hash_path) video.hashes['opensubtitles'] = osub_hash = osub_hash or hash_opensubtitles(hash_path)
if "shooter" in providers: if "shooter" in providers:
video.hashes['shooter'] = hash_shooter(hash_path) video.hashes['shooter'] = hash_shooter(hash_path)

View file

@ -19,9 +19,11 @@ from xml.etree import ElementTree
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
class BSPlayerSubtitle(Subtitle): class BSPlayerSubtitle(Subtitle):
"""BSPlayer Subtitle.""" """BSPlayer Subtitle."""
provider_name = 'bsplayer' provider_name = 'bsplayer'
hash_verifiable = True
def __init__(self, language, filename, subtype, video, link): def __init__(self, language, filename, subtype, video, link):
super(BSPlayerSubtitle, self).__init__(language) super(BSPlayerSubtitle, self).__init__(language)
@ -41,27 +43,12 @@ class BSPlayerSubtitle(Subtitle):
def get_matches(self, video): def get_matches(self, video):
matches = set() matches = set()
video_filename = video.name
video_filename = os.path.basename(video_filename)
video_filename, _ = os.path.splitext(video_filename)
video_filename = sanitize_release_group(video_filename)
subtitle_filename = self.filename
subtitle_filename = os.path.basename(subtitle_filename)
subtitle_filename, _ = os.path.splitext(subtitle_filename)
subtitle_filename = sanitize_release_group(subtitle_filename)
matches |= guess_matches(video, guessit(self.filename)) matches |= guess_matches(video, guessit(self.filename))
matches.add(id(self))
matches.add('hash') matches.add('hash')
return matches return matches
class BSPlayerProvider(Provider): class BSPlayerProvider(Provider):
"""BSPlayer Provider.""" """BSPlayer Provider."""
languages = {Language('por', 'BR')} | {Language(l) for l in [ languages = {Language('por', 'BR')} | {Language(l) for l in [
@ -69,6 +56,7 @@ class BSPlayerProvider(Provider):
'ron', 'rus', 'spa', 'swe', 'tur', 'ukr', 'zho' 'ron', 'rus', 'spa', 'swe', 'tur', 'ukr', 'zho'
]} ]}
SEARCH_THROTTLE = 8 SEARCH_THROTTLE = 8
hash_verifiable = True
# batantly based on kodi's bsplayer plugin # batantly based on kodi's bsplayer plugin
# also took from BSPlayer-Subtitles-Downloader # also took from BSPlayer-Subtitles-Downloader
@ -108,18 +96,11 @@ class BSPlayerProvider(Provider):
res = self.session.post(self.search_url, data) res = self.session.post(self.search_url, data)
return ElementTree.fromstring(res.text) return ElementTree.fromstring(res.text)
### with requests
# res = requests.post(
# url=self.search_url,
# data=data,
# headers=headers
# )
# return ElementTree.fromstring(res.text)
except Exception as ex: except Exception as ex:
logger.info("ERROR: %s." % ex) logger.info("ERROR: %s." % ex)
if func_name == 'logIn': if func_name == 'logIn':
self.search_url = self.get_sub_domain() self.search_url = self.get_sub_domain()
sleep(1) sleep(1)
logger.info('ERROR: Too many tries (%d)...' % tries) logger.info('ERROR: Too many tries (%d)...' % tries)
raise Exception('Too many tries...') raise Exception('Too many tries...')
@ -167,7 +148,6 @@ class BSPlayerProvider(Provider):
# language_ids = 'spa' # language_ids = 'spa'
language_ids = ','.join(sorted(l.opensubtitles for l in language)) language_ids = ','.join(sorted(l.opensubtitles for l in language))
if video.imdb_id is None: if video.imdb_id is None:
imdbId = '*' imdbId = '*'
else: else:
@ -193,13 +173,13 @@ class BSPlayerProvider(Provider):
if items: if items:
logger.info("Subtitles Found.") logger.info("Subtitles Found.")
for item in items: for item in items:
subID=item.find('subID').text subID = item.find('subID').text
subDownloadLink=item.find('subDownloadLink').text subDownloadLink = item.find('subDownloadLink').text
subLang= Language.fromopensubtitles(item.find('subLang').text) subLang = Language.fromopensubtitles(item.find('subLang').text)
subName=item.find('subName').text subName = item.find('subName').text
subFormat=item.find('subFormat').text subFormat = item.find('subFormat').text
subtitles.append( subtitles.append(
BSPlayerSubtitle(subLang,subName, subFormat, video, subDownloadLink) BSPlayerSubtitle(subLang, subName, subFormat, video, subDownloadLink)
) )
return subtitles return subtitles
@ -207,9 +187,9 @@ class BSPlayerProvider(Provider):
return self.query(video, video.hashes['bsplayer'], languages) return self.query(video, video.hashes['bsplayer'], languages)
def get_sub_domain(self): def get_sub_domain(self):
# s1-9, s101-109 # s1-9, s101-109
SUB_DOMAINS = ['s1', 's2', 's3', 's4', 's5', 's6', 's7', 's8', 's9', SUB_DOMAINS = ['s1', 's2', 's3', 's4', 's5', 's6', 's7', 's8', 's9',
's101', 's102', 's103', 's104', 's105', 's106', 's107', 's108', 's109'] 's101', 's102', 's103', 's104', 's105', 's106', 's107', 's108', 's109']
API_URL_TEMPLATE = "http://{sub_domain}.api.bsplayer-subtitles.com/v1.php" API_URL_TEMPLATE = "http://{sub_domain}.api.bsplayer-subtitles.com/v1.php"
sub_domains_end = len(SUB_DOMAINS) - 1 sub_domains_end = len(SUB_DOMAINS) - 1
return API_URL_TEMPLATE.format(sub_domain=SUB_DOMAINS[random.randint(0, sub_domains_end)]) return API_URL_TEMPLATE.format(sub_domain=SUB_DOMAINS[random.randint(0, sub_domains_end)])
@ -226,10 +206,8 @@ class BSPlayerProvider(Provider):
raise ValueError('Error 500 on server') raise ValueError('Error 500 on server')
with gzip.GzipFile(fileobj=io.BytesIO(res.content)) as gf: with gzip.GzipFile(fileobj=io.BytesIO(res.content)) as gf:
subtitle.content = gf.read() subtitle.content = gf.read()
subtitle.normalize() subtitle.normalize()
return subtitle return subtitle
raise ValueError('Problems conecting to the server') raise ValueError('Problems conecting to the server')

View file

@ -61,6 +61,8 @@ def compute_score(matches, subtitle, video, hearing_impaired=None):
episode_hash_valid_if = {"series", "season", "episode", "format"} episode_hash_valid_if = {"series", "season", "episode", "format"}
movie_hash_valid_if = {"video_codec", "format"} movie_hash_valid_if = {"video_codec", "format"}
orig_matches = matches.copy()
# on hash match, discard everything else # on hash match, discard everything else
if subtitle.hash_verifiable: if subtitle.hash_verifiable:
if 'hash' in matches: if 'hash' in matches:
@ -84,41 +86,47 @@ def compute_score(matches, subtitle, video, hearing_impaired=None):
matches &= {'hash'} matches &= {'hash'}
# handle equivalent matches # handle equivalent matches
eq_matches = set()
if is_episode: if is_episode:
if 'title' in matches: if 'title' in matches:
logger.debug('Adding title match equivalent') logger.debug('Adding title match equivalent')
matches.add('episode') eq_matches.add('episode')
if 'series_imdb_id' in matches: if 'series_imdb_id' in matches:
logger.debug('Adding series_imdb_id match equivalent') logger.debug('Adding series_imdb_id match equivalent')
matches |= {'series', 'year'} eq_matches |= {'series', 'year'}
if 'imdb_id' in matches: if 'imdb_id' in matches:
logger.debug('Adding imdb_id match equivalents') logger.debug('Adding imdb_id match equivalents')
matches |= {'series', 'year', 'season', 'episode'} eq_matches |= {'series', 'year', 'season', 'episode'}
if 'tvdb_id' in matches: if 'tvdb_id' in matches:
logger.debug('Adding tvdb_id match equivalents') logger.debug('Adding tvdb_id match equivalents')
matches |= {'series', 'year', 'season', 'episode', 'title'} eq_matches |= {'series', 'year', 'season', 'episode', 'title'}
if 'series_tvdb_id' in matches: if 'series_tvdb_id' in matches:
logger.debug('Adding series_tvdb_id match equivalents') logger.debug('Adding series_tvdb_id match equivalents')
matches |= {'series', 'year'} eq_matches |= {'series', 'year'}
# specials # specials
if video.is_special and 'title' in matches and 'series' in matches \ if video.is_special and 'title' in matches and 'series' in matches \
and 'year' in matches: and 'year' in matches:
logger.debug('Adding special title match equivalent') logger.debug('Adding special title match equivalent')
matches |= {'season', 'episode'} eq_matches |= {'season', 'episode'}
elif is_movie: elif is_movie:
if 'imdb_id' in matches: if 'imdb_id' in matches:
logger.debug('Adding imdb_id match equivalents') logger.debug('Adding imdb_id match equivalents')
matches |= {'title', 'year'} eq_matches |= {'title', 'year'}
matches |= eq_matches
# handle hearing impaired # handle hearing impaired
if hearing_impaired is not None and subtitle.hearing_impaired == hearing_impaired: if hearing_impaired is not None and subtitle.hearing_impaired == hearing_impaired:
logger.debug('Matched hearing_impaired') logger.debug('Matched hearing_impaired')
matches.add('hearing_impaired') matches.add('hearing_impaired')
orig_matches.add('hearing_impaired')
# compute the score # compute the score
score = sum((scores.get(match, 0) for match in matches)) score = sum((scores.get(match, 0) for match in matches))
logger.info('%r: Computed score %r with final matches %r', subtitle, score, matches) logger.info('%r: Computed score %r with final matches %r', subtitle, score, matches)
return score score_without_hash = sum((scores.get(match, 0) for match in orig_matches | eq_matches if match != "hash"))
return score, score_without_hash