| #!/usr/bin/env python3 |
| |
| import argparse |
| import logging |
| import shlex |
| import subprocess |
| |
| HELP = ''' |
| Normalize audio input. |
| |
| The command uses ffprobe to analyze an input file with the ebur128 |
| filter, and finally run ffmpeg to normalize the input depending on the |
| computed adjustment. |
| |
| ffmpeg encoding arguments can be passed through the extra arguments |
| after options, for example as in: |
| normalize.py --input input.mp3 --output output.mp3 -- -loglevel debug -y |
| ''' |
| |
| logging.basicConfig(format='normalize|%(levelname)s> %(message)s', level=logging.INFO) |
| log = logging.getLogger() |
| |
| |
| class Formatter( |
| argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter |
| ): |
| pass |
| |
| |
| def normalize(): |
| parser = argparse.ArgumentParser(description=HELP, formatter_class=Formatter) |
| parser.add_argument('--input', '-i', required=True, help='specify input file') |
| parser.add_argument('--output', '-o', required=True, help='specify output file') |
| parser.add_argument('--dry-run', '-n', help='simulate commands', action='store_true') |
| parser.add_argument('encode_arguments', nargs='*', help='specify encode options used for the actual encoding') |
| |
| args = parser.parse_args() |
| |
| analysis_cmd = [ |
| 'ffprobe', '-v', 'error', '-of', 'compact=p=0:nk=1', |
| '-show_entries', 'frame_tags=lavfi.r128.I', '-f', 'lavfi', |
| f"amovie='{args.input}',ebur128=metadata=1" |
| ] |
| |
| def _run_command(cmd, dry_run=False): |
| log.info(f"Running command:\n$ {shlex.join(cmd)}") |
| if not dry_run: |
| result = subprocess.run(cmd, check=True, stdout=subprocess.PIPE) |
| return result |
| |
| result = _run_command(analysis_cmd) |
| |
| loudness = ref = -23 |
| for line in result.stdout.splitlines(): |
| sline = line.rstrip() |
| if sline: |
| loudness = sline |
| |
| adjust = ref - float(loudness) |
| if abs(adjust) < 0.0001: |
| logging.info(f"No normalization needed for '{args.input}'") |
| return |
| |
| logging.info(f"Adjusting '{args.input}' by {adjust:.2f}dB...") |
| normalize_cmd = [ |
| 'ffmpeg', '-i', args.input, '-af', f'volume={adjust:.2f}dB' |
| ] + args.encode_arguments + [args.output] |
| |
| _run_command(normalize_cmd, args.dry_run) |
| |
| |
| if __name__ == '__main__': |
| normalize() |