You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
210 lines
7.2 KiB
210 lines
7.2 KiB
#!/usr/bin/env python
|
|
# -*- coding: utf-8 -*-
|
|
"""
|
|
audio_codec, audio_profile and audio_channels property
|
|
"""
|
|
from rebulk.remodule import re
|
|
|
|
from rebulk import Rebulk, Rule, RemoveMatch
|
|
|
|
from ..common import dash
|
|
from ..common.pattern import is_disabled
|
|
from ..common.validators import seps_before, seps_after
|
|
|
|
audio_properties = ['audio_codec', 'audio_profile', 'audio_channels']
|
|
|
|
|
|
def audio_codec(config): # pylint:disable=unused-argument
|
|
"""
|
|
Builder for rebulk object.
|
|
|
|
:param config: rule configuration
|
|
:type config: dict
|
|
:return: Created Rebulk object
|
|
:rtype: Rebulk
|
|
"""
|
|
rebulk = Rebulk().regex_defaults(flags=re.IGNORECASE, abbreviations=[dash]).string_defaults(ignore_case=True)
|
|
|
|
def audio_codec_priority(match1, match2):
|
|
"""
|
|
Gives priority to audio_codec
|
|
:param match1:
|
|
:type match1:
|
|
:param match2:
|
|
:type match2:
|
|
:return:
|
|
:rtype:
|
|
"""
|
|
if match1.name == 'audio_codec' and match2.name in ['audio_profile', 'audio_channels']:
|
|
return match2
|
|
if match1.name in ['audio_profile', 'audio_channels'] and match2.name == 'audio_codec':
|
|
return match1
|
|
return '__default__'
|
|
|
|
rebulk.defaults(name='audio_codec',
|
|
conflict_solver=audio_codec_priority,
|
|
disabled=lambda context: is_disabled(context, 'audio_codec'))
|
|
|
|
rebulk.regex("MP3", "LAME", r"LAME(?:\d)+-?(?:\d)+", value="MP3")
|
|
rebulk.regex('Dolby', 'DolbyDigital', 'Dolby-Digital', 'DD', 'AC3D?', value='Dolby Digital')
|
|
rebulk.regex('Dolby-?Atmos', 'Atmos', value='Dolby Atmos')
|
|
rebulk.string("AAC", value="AAC")
|
|
rebulk.string('EAC3', 'DDP', 'DD+', value='Dolby Digital Plus')
|
|
rebulk.string("Flac", value="FLAC")
|
|
rebulk.string("DTS", value="DTS")
|
|
rebulk.regex('DTS-?HD', 'DTS(?=-?MA)', value='DTS-HD',
|
|
conflict_solver=lambda match, other: other if other.name == 'audio_codec' else '__default__')
|
|
rebulk.regex('True-?HD', value='Dolby TrueHD')
|
|
rebulk.string('Opus', value='Opus')
|
|
rebulk.string('Vorbis', value='Vorbis')
|
|
rebulk.string('PCM', value='PCM')
|
|
rebulk.string('LPCM', value='LPCM')
|
|
|
|
rebulk.defaults(name='audio_profile', disabled=lambda context: is_disabled(context, 'audio_profile'))
|
|
rebulk.string('MA', value='Master Audio', tags='DTS-HD')
|
|
rebulk.string('HR', 'HRA', value='High Resolution Audio', tags='DTS-HD')
|
|
rebulk.string('ES', value='Extended Surround', tags='DTS')
|
|
rebulk.string('HE', value='High Efficiency', tags='AAC')
|
|
rebulk.string('LC', value='Low Complexity', tags='AAC')
|
|
rebulk.string('HQ', value='High Quality', tags='Dolby Digital')
|
|
rebulk.string('EX', value='EX', tags='Dolby Digital')
|
|
|
|
rebulk.defaults(name="audio_channels", disabled=lambda context: is_disabled(context, 'audio_channels'))
|
|
rebulk.regex(r'(7[\W_][01](?:ch)?)(?=[^\d]|$)', value='7.1', children=True)
|
|
rebulk.regex(r'(5[\W_][01](?:ch)?)(?=[^\d]|$)', value='5.1', children=True)
|
|
rebulk.regex(r'(2[\W_]0(?:ch)?)(?=[^\d]|$)', value='2.0', children=True)
|
|
rebulk.regex('7[01]', value='7.1', validator=seps_after, tags='weak-audio_channels')
|
|
rebulk.regex('5[01]', value='5.1', validator=seps_after, tags='weak-audio_channels')
|
|
rebulk.string('20', value='2.0', validator=seps_after, tags='weak-audio_channels')
|
|
rebulk.string('7ch', '8ch', value='7.1')
|
|
rebulk.string('5ch', '6ch', value='5.1')
|
|
rebulk.string('2ch', 'stereo', value='2.0')
|
|
rebulk.string('1ch', 'mono', value='1.0')
|
|
|
|
rebulk.rules(DtsHDRule, AacRule, DolbyDigitalRule, AudioValidatorRule, HqConflictRule, AudioChannelsValidatorRule)
|
|
|
|
return rebulk
|
|
|
|
|
|
class AudioValidatorRule(Rule):
|
|
"""
|
|
Remove audio properties if not surrounded by separators and not next each others
|
|
"""
|
|
priority = 64
|
|
consequence = RemoveMatch
|
|
|
|
def when(self, matches, context):
|
|
ret = []
|
|
|
|
audio_list = matches.range(predicate=lambda match: match.name in audio_properties)
|
|
for audio in audio_list:
|
|
if not seps_before(audio):
|
|
valid_before = matches.range(audio.start - 1, audio.start,
|
|
lambda match: match.name in audio_properties)
|
|
if not valid_before:
|
|
ret.append(audio)
|
|
continue
|
|
if not seps_after(audio):
|
|
valid_after = matches.range(audio.end, audio.end + 1,
|
|
lambda match: match.name in audio_properties)
|
|
if not valid_after:
|
|
ret.append(audio)
|
|
continue
|
|
|
|
return ret
|
|
|
|
|
|
class AudioProfileRule(Rule):
|
|
"""
|
|
Abstract rule to validate audio profiles
|
|
"""
|
|
priority = 64
|
|
dependency = AudioValidatorRule
|
|
consequence = RemoveMatch
|
|
|
|
def __init__(self, codec):
|
|
super(AudioProfileRule, self).__init__()
|
|
self.codec = codec
|
|
|
|
def enabled(self, context):
|
|
return not is_disabled(context, 'audio_profile')
|
|
|
|
def when(self, matches, context):
|
|
profile_list = matches.named('audio_profile', lambda match: self.codec in match.tags)
|
|
ret = []
|
|
for profile in profile_list:
|
|
codec = matches.previous(profile, lambda match: match.name == 'audio_codec' and match.value == self.codec)
|
|
if not codec:
|
|
codec = matches.next(profile, lambda match: match.name == 'audio_codec' and match.value == self.codec)
|
|
if not codec:
|
|
ret.append(profile)
|
|
if codec:
|
|
ret.extend(matches.conflicting(profile))
|
|
return ret
|
|
|
|
|
|
class DtsHDRule(AudioProfileRule):
|
|
"""
|
|
Rule to validate DTS-HD profile
|
|
"""
|
|
|
|
def __init__(self):
|
|
super(DtsHDRule, self).__init__('DTS-HD')
|
|
|
|
|
|
class AacRule(AudioProfileRule):
|
|
"""
|
|
Rule to validate AAC profile
|
|
"""
|
|
|
|
def __init__(self):
|
|
super(AacRule, self).__init__("AAC")
|
|
|
|
|
|
class DolbyDigitalRule(AudioProfileRule):
|
|
"""
|
|
Rule to validate Dolby Digital profile
|
|
"""
|
|
|
|
def __init__(self):
|
|
super(DolbyDigitalRule, self).__init__('Dolby Digital')
|
|
|
|
|
|
class HqConflictRule(Rule):
|
|
"""
|
|
Solve conflict between HQ from other property and from audio_profile.
|
|
"""
|
|
|
|
dependency = [DtsHDRule, AacRule, DolbyDigitalRule]
|
|
consequence = RemoveMatch
|
|
|
|
def enabled(self, context):
|
|
return not is_disabled(context, 'audio_profile')
|
|
|
|
def when(self, matches, context):
|
|
hq_audio = matches.named('audio_profile', lambda m: m.value == 'High Quality')
|
|
hq_audio_spans = [match.span for match in hq_audio]
|
|
return matches.named('other', lambda m: m.span in hq_audio_spans)
|
|
|
|
|
|
class AudioChannelsValidatorRule(Rule):
|
|
"""
|
|
Remove audio_channel if no audio codec as previous match.
|
|
"""
|
|
priority = 128
|
|
consequence = RemoveMatch
|
|
|
|
def enabled(self, context):
|
|
return not is_disabled(context, 'audio_channels')
|
|
|
|
def when(self, matches, context):
|
|
ret = []
|
|
|
|
for audio_channel in matches.tagged('weak-audio_channels'):
|
|
valid_before = matches.range(audio_channel.start - 1, audio_channel.start,
|
|
lambda match: match.name == 'audio_codec')
|
|
if not valid_before:
|
|
ret.append(audio_channel)
|
|
|
|
return ret
|