ffmpeg-normalize


Nameffmpeg-normalize JSON
Version 1.31.0 PyPI version JSON
download
home_pagehttps://github.com/slhck/ffmpeg-normalize
SummaryNormalize audio via ffmpeg
upload_time2024-12-15 19:51:43
maintainerNone
docs_urlNone
authorWerner Robitza
requires_python>=3.9
licenseMIT
keywords ffmpeg normalize audio
VCS
bugtrack_url
requirements tqdm colorama ffmpeg-progress-yield colorlog
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # ffmpeg-normalize

[![PyPI version](https://img.shields.io/pypi/v/ffmpeg-normalize.svg)](https://pypi.org/project/ffmpeg-normalize)
![Docker Image Version](https://img.shields.io/docker/v/slhck/ffmpeg-normalize?sort=semver&label=Docker%20image)
![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/slhck/ffmpeg-normalize/python-package.yml)

<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-20-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END -->

A utility for batch-normalizing audio using ffmpeg.

This program normalizes media files to a certain loudness level using the EBU R128 loudness normalization procedure. It can also perform RMS-based normalization (where the mean is lifted or attenuated), or peak normalization to a certain target level.

Batch processing of several input files is possible, including video files.

**A very quick how-to:**

1. Install a recent version of [ffmpeg](https://ffmpeg.org/download.html)
2. Run `pip3 install ffmpeg-normalize`
3. Run `ffmpeg-normalize /path/to/your/file.mp4`
4. Done! 🎧 (the file will be in a folder called `normalized`)

Read on for more info.

**Contents:**

- [Requirements](#requirements)
  - [ffmpeg](#ffmpeg)
- [Installation](#installation)
  - [Shell Completions](#shell-completions)
- [Usage with Docker](#usage-with-docker)
- [High LeveL Introduction](#high-level-introduction)
- [Basic Usage](#basic-usage)
- [Examples](#examples)
- [Detailed Options](#detailed-options)
  - [File Input/Output](#file-inputoutput)
  - [General](#general)
  - [Normalization](#normalization)
  - [EBU R128 Normalization](#ebu-r128-normalization)
  - [Audio Encoding](#audio-encoding)
  - [Other Encoding Options](#other-encoding-options)
  - [Input/Output Format](#inputoutput-format)
  - [Environment Variables](#environment-variables)
- [API](#api)
- [FAQ](#faq)
  - [My output file is too large?](#my-output-file-is-too-large)
  - [What options should I choose for the EBU R128 filter? What is linear and dynamic mode?](#what-options-should-i-choose-for-the-ebu-r128-filter-what-is-linear-and-dynamic-mode)
  - [The program doesn't work because the "loudnorm" filter can't be found](#the-program-doesnt-work-because-the-loudnorm-filter-cant-be-found)
  - [Should I use this to normalize my music collection?](#should-i-use-this-to-normalize-my-music-collection)
  - [Why are my output files MKV?](#why-are-my-output-files-mkv)
  - [I get a "Could not write header for output file" error](#i-get-a-could-not-write-header-for-output-file-error)
  - [The conversion does not work and I get a cryptic ffmpeg error!](#the-conversion-does-not-work-and-i-get-a-cryptic-ffmpeg-error)
  - [What are the different normalization algorithms?](#what-are-the-different-normalization-algorithms)
  - [Couldn't I just run `loudnorm` with ffmpeg?](#couldnt-i-just-run-loudnorm-with-ffmpeg)
  - [What about speech?](#what-about-speech)
  - [After updating, this program does not work as expected anymore!](#after-updating-this-program-does-not-work-as-expected-anymore)
  - [Can I buy you a beer / coffee / random drink?](#can-i-buy-you-a-beer--coffee--random-drink)
- [Related Tools and Articles](#related-tools-and-articles)
- [Contributors](#contributors)
- [License](#license)

-------------

## Requirements

You need Python 3.9 or higher, and ffmpeg.

### ffmpeg

- ffmpeg 5.x is required, ffmpeg 6.x is recommended (it fixes [a bug for short files](https://github.com/slhck/ffmpeg-normalize/issues/87))
- Download a [static build](https://ffmpeg.org/download.html) for your system
- Place the `ffmpeg` executable in your `$PATH`, or specify the path to the binary with the `FFMPEG_PATH` environment variable in `ffmpeg-normalize`

For instance, under Linux:

```bash
wget https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-amd64-static.tar.xz
mkdir -p ffmpeg
tar -xf ffmpeg-release-amd64-static.tar.xz -C ffmpeg --strip-components=1
sudo cp ffmpeg/ffmpeg /usr/local/bin
sudo cp ffmpeg/ffprobe /usr/local/bin
sudo chmod +x /usr/local/bin/ffmpeg /usr/local/bin/ffprobe
```

For Windows, follow [this guide](https://www.wikihow.com/Install-FFmpeg-on-Windows).

For macOS and Linux, you can also use [Homebrew](https://brew.sh):

```bash
brew install ffmpeg
```

Note that using distribution packages (e.g., `apt install ffmpeg`) is not recommended, as these are often outdated.

## Installation

For Python 3 and pip:

```bash
pip3 install ffmpeg-normalize
```

Or download this repository, then run `pip3 install .`.

To later upgrade to the latest version, run `pip3 install --upgrade ffmpeg-normalize`.

### Shell Completions

This tool provides shell completions for bash and zsh. To install them:

<!--
Note to self: Generate the shtab ones with:

  shtab --shell=bash -u ffmpeg_normalize.__main__.create_parser > completions/ffmpeg-normalize-shtab.bash
  shtab --shell=zsh -u ffmpeg_normalize.__main__.create_parser > completions/ffmpeg-normalize-shtab.zsh

but these are not properly working yet.
-->

#### Bash

If you have [`bash-completion`](https://github.com/scop/bash-completion) installed, you can just copy your new completion script to the `/usr/local/etc/bash_completion.d` directory.

```bash
curl -L https://raw.githubusercontent.com/slhck/ffmpeg-normalize/master/completions/ffmpeg-normalize-completion.bash \
  -o /usr/local/etc/bash_completion.d/ffmpeg-normalize
```

Without bash-completion, you can manually install the completion script:

```bash
# create completions directory if it doesn't exist
mkdir -p ~/.bash_completions.d

# download and install completion script
curl -L https://raw.githubusercontent.com/slhck/ffmpeg-normalize/master/completions/ffmpeg-normalize-completion.bash \
  -o ~/.bash_completions.d/ffmpeg-normalize

# source it in your ~/.bashrc
echo 'source ~/.bash_completions.d/ffmpeg-normalize' >> ~/.bashrc
```

#### Zsh

Download the completion script and place it in the default `site-functions` directory:

```bash
curl -L https://raw.githubusercontent.com/slhck/ffmpeg-normalize/master/completions/ffmpeg-normalize.zsh \
  -o /usr/local/share/zsh/site-functions/
```

You may choose any other directory that is in your `$FPATH` variable.
Make sure your `.zshrc` file contains `autoload -Uz compinit && compinit`.

## Usage with Docker

You can use the pre-built image from Docker Hub:

```bash
docker run -v "$(pwd):/tmp" -it slhck/ffmpeg-normalize
```

Alternatively, download this repository and run

```bash
docker build -t ffmpeg-normalize .
```

Then run the container with:

```bash
docker run  -v "$(pwd):/tmp" -it ffmpeg-normalize
```

This will mount your current directory to the `/tmp` directory inside the container. Everything else works the same way as if you had installed the program locally. For example, to normalize a file:

```bash
docker run  -v "$(pwd):/tmp" -it ffmpeg-normalize /tmp/yourfile.mp4 -o /tmp/yourfile-normalized.wav
```

You will then find the normalized file in your current directory.

## High LeveL Introduction

Please read this section for a high level introduction.

**What does the program do?**

The program takes one or more input files and, by default, writes them to a folder called `normalized`, using an `.mkv` container. All audio streams will be normalized so that they have the same (perceived) volume according to the EBU R128 standard. This is done by analyzing the audio streams and applying a filter to bring them to a target level. Under the hood, the program uses ffmpeg's `loudnorm` filter to do this.

**How do I specify the input?**

Just give the program one or more input files as arguments. It works with most media files, including video files.

**How do I specify the output?**

You don't have to specify an output file name (the default is `normalized/<input>.mkv`), but if you want to override it, you can specify one output file name for each input file with the `-o` option. In this case, the container format (e.g. `.wav`) will be inferred from the file name extension that you've given.

Example:

```bash
ffmpeg-normalize 1.wav 2.wav -o 1-normalized.wav 2-normalized.wav
```

Note that if you don't specify the output file name for an input file, the container format will be MKV, and the output will be written to `normalized/<input>.mkv`. The reason for choosing the MKV container is that it can handle almost any codec combination.

Using the `-ext` option, you can supply a different output extension common to all output files, e.g. `-ext m4a`. However, you need to make sure that the container supports the codecs used for the output (see below).

**What will get normalized?**

By default, all streams from the input file will be written to the output file. For example, if your input is a video with two language tracks and a subtitle track, both audio tracks will be normalized independently. The video and subtitle tracks will be copied over to the output file.

**How will the normalization be done?**

The normalization will be performed according to the EBU R128 algorithm with the [`loudnorm` filter](https://ffmpeg.org/ffmpeg-filters.html#loudnorm) from FFmpeg, which was [originally written by Kyle Swanson](https://k.ylo.ph/2016/04/04/loudnorm.html). It will bring the audio to a specified target level. This ensures that multiple files normalized with this filter will have the same perceived loudness.

**What codec is chosen?**

The default audio encoding method is uncompressed PCM (`pcm_s16le`) to avoid introducing compression artifacts. This will result in a much higher bitrate than you might want, for example if your input files are MP3s.

Some containers (like MP4) also cannot handle PCM audio. If you want to use such containers and/or keep the file size down, use `-c:a` and specify an audio codec (e.g., `-c:a aac` for ffmpeg's built-in AAC encoder).

## Basic Usage

Supply one or more input files, and optionally, output file names:

```bash
ffmpeg-normalize input [input ...][-h][-o OUTPUT [OUTPUT ...]] [options]
```

Example:

```bash
ffmpeg-normalize 1.wav 2.wav -o 1-normalized.m4a 2-normalized.m4a -c:a aac -b:a 192k
```

For more information on the options (`[options]`) available, run `ffmpeg-normalize -h`, or read on.

## Examples

[Read the examples on the wiki.](https://github.com/slhck/ffmpeg-normalize/wiki/examples)

## Detailed Options

### File Input/Output

- `input`: Input media file(s)

- `-o OUTPUT [OUTPUT ...], --output OUTPUT [OUTPUT ...]`: Output file names.

    Will be applied per input file.

    If no output file name is specified for an input file, the output files
    will be written to the default output folder with the name `<input>.<ext>`, where `<ext>` is the output extension (see `-ext` option).

    Example: `ffmpeg-normalize 1.wav 2.wav -o 1n.wav 2n.wav`

- `-of OUTPUT_FOLDER, --output-folder OUTPUT_FOLDER`: Output folder (default: `normalized`)

    This folder will be used for input files that have no explicit output name specified.

### General

- `-f, --force`: Force overwrite existing files

- `-d, --debug`: Print debugging output

- `-v, --verbose`: Print verbose output

- `-q, --quiet`: Only print errors

- `-n, --dry-run`: Do not run normalization, only print what would be done

- `-pr`, `--progress`: Show progress bar for files and streams

- `--version`: Print version and exit

### Normalization

- `-nt {ebu,rms,peak}, --normalization-type {ebu,rms,peak}`: Normalization type (default: `ebu`).

    EBU normalization performs two passes and normalizes according to EBU R128.

    RMS-based normalization brings the input file to the specified RMS level.

    Peak normalization brings the signal to the specified peak level.

- `-t TARGET_LEVEL, --target-level TARGET_LEVEL`: Normalization target level in dB/LUFS (default: -23).

    For EBU normalization, it corresponds to Integrated Loudness Target in LUFS. The range is -70.0 - -5.0.

    Otherwise, the range is -99 to 0.

- `-p, --print-stats`: Print loudness statistics for both passes formatted as JSON to stdout.

### EBU R128 Normalization

- `-lrt LOUDNESS_RANGE_TARGET, --loudness-range-target LOUDNESS_RANGE_TARGET`: EBU Loudness Range Target in LUFS (default: 7.0).

    Range is 1.0 - 50.0.

- `--keep-loudness-range-target`: Keep the input loudness range target to allow for linear normalization.

- `--keep-lra-above-loudness-range-target`: Keep input loudness range above loudness range target.

    Can be used as an alternative to `--keep-loudness-range-target` to allow for linear normalization.

- `-tp TRUE_PEAK, --true-peak TRUE_PEAK`: EBU Maximum True Peak in dBTP (default: -2.0).

    Range is -9.0 - +0.0.

- `--offset OFFSET`: EBU Offset Gain (default: 0.0).

    The gain is applied before the true-peak limiter in the first pass only. The offset for the second pass will be automatically determined based on the first pass statistics.

    Range is -99.0 - +99.0.

- `--lower-only`: Whether the audio should not increase in loudness.

    If the measured loudness from the first pass is lower than the target loudness then normalization pass will be skipped for the measured audio source.

- `--auto-lower-loudness-target`: Automatically lower EBU Integrated Loudness Target.

    Automatically lower EBU Integrated Loudness Target to prevent falling back to dynamic filtering.

    Makes sure target loudness is lower than measured loudness minus peak loudness (input_i - input_tp) by a small amount.

- `--dual-mono`: Treat mono input files as "dual-mono".

    If a mono file is intended for playback on a stereo system, its EBU R128 measurement will be perceptually incorrect. If set, this option will compensate for this effect. Multi-channel input files are not affected by this option.

- `--dynamic`: Force dynamic normalization mode.

    Instead of applying linear EBU R128 normalization, choose a dynamic normalization. This is not usually recommended.

    Dynamic mode will automatically change the sample rate to 192 kHz. Use `-ar`/`--sample-rate` to specify a different output sample rate.

### Audio Encoding

- `-c:a AUDIO_CODEC, --audio-codec AUDIO_CODEC`: Audio codec to use for output files.

    See `ffmpeg -encoders` for a list.

    Will use PCM audio with input stream bit depth by default.

- `-b:a AUDIO_BITRATE, --audio-bitrate AUDIO_BITRATE`: Audio bitrate in bits/s, or with K suffix.

    If not specified, will use codec default.

- `-ar SAMPLE_RATE, --sample-rate SAMPLE_RATE`: Audio sample rate to use for output files in Hz.

    Will use input sample rate by default, except for EBU normalization, which will change the input sample rate to 192 kHz.

- `-ac`, `--audio-channels`: Set the number of audio channels. If not specified, the input channel layout will be used. This is equivalent to `-ac` in ffmpeg.

- `-koa, --keep-original-audio`: Copy original, non-normalized audio streams to output file

- `-prf PRE_FILTER, --pre-filter PRE_FILTER`: Add an audio filter chain before applying normalization.

    Multiple filters can be specified by comma-separating them.

- `-pof POST_FILTER, --post-filter POST_FILTER`: Add an audio filter chain after applying normalization.

    Multiple filters can be specified by comma-separating them.

    For EBU, the filter will be applied during the second pass.

### Other Encoding Options

- `-vn, --video-disable`: Do not write video streams to output

- `-c:v VIDEO_CODEC, --video-codec VIDEO_CODEC`: Video codec to use for output files (default: 'copy').

    See `ffmpeg -encoders` for a list.

    Will attempt to copy video codec by default.

- `-sn, --subtitle-disable`: Do not write subtitle streams to output

- `-mn, --metadata-disable`: Do not write metadata to output

- `-cn, --chapters-disable`: Do not write chapters to output


### Input/Output Format

- `-ei EXTRA_INPUT_OPTIONS, --extra-input-options EXTRA_INPUT_OPTIONS`: Extra input options list.

    A list of extra ffmpeg command line arguments valid for the input, applied before ffmpeg's `-i`.

    You can either use a JSON-formatted list (i.e., a list of comma-separated, quoted elements within square brackets), or a simple string of space-separated arguments.

    If JSON is used, you need to wrap the whole argument in quotes to prevent shell expansion and to preserve literal quotes inside the string. If a simple string is used, you need to specify the argument with `-e=`.

    Examples: `-ei '[ "-f", "mpegts", "-r", "24" ]'` or `-ei="-f mpegts -r 24"`

- `-e EXTRA_OUTPUT_OPTIONS, --extra-output-options EXTRA_OUTPUT_OPTIONS`: Extra output options list.

    A list of extra ffmpeg command line arguments valid for the output.

    You can either use a JSON-formatted list (i.e., a list of comma-separated, quoted elements within square brackets), or a simple string of space-separated arguments.

    If JSON is used, you need to wrap the whole argument in quotes to prevent shell expansion and to preserve literal quotes inside the string. If a simple string is used, you need to specify the argument with `-e=`.

    Examples: `-e '[ "-vbr", "3", "-preset:v", "ultrafast" ]'` or `-e="-vbr 3 -preset:v ultrafast"`

- `-ofmt OUTPUT_FORMAT, --output-format OUTPUT_FORMAT`: Media format to use for output file(s).

    See `ffmpeg -formats` for a list.

    If not specified, the format will be inferred by ffmpeg from the output file name. If the output file name is not explicitly specified, the extension will govern the format (see '--extension' option).

- `-ext EXTENSION, --extension EXTENSION`: Output file extension to use for output files that were not explicitly specified. (Default: `mkv`)

### Environment Variables

The program additionally respects environment variables:

- `TMP` / `TEMP` / `TMPDIR`

    Sets the path to the temporary directory in which files are
    stored before being moved to the final output directory.
    Note: You need to use full paths.

- `FFMPEG_PATH`

    Sets the full path to an `ffmpeg` executable other than
    the system default or you can provide a file name available on $PATH

## API

This program has a simple API that can be used to integrate it into other Python programs.

For more information see the [API documentation](https://htmlpreview.github.io/?https://github.com/slhck/ffmpeg-normalize/blob/master/docs/ffmpeg_normalize.html).

## FAQ

### My output file is too large?

This is because the default output codec is PCM, which is uncompressed. If you want to reduce the file size, you can specify an audio codec with `-c:a` (e.g., `-c:a aac` for ffmpeg's built-in AAC encoder), and optionally a bitrate with `-b:a`.

For example:

```bash
ffmpeg-normalize input.wav -o output.m4a -c:a aac -b:a 192k
```

### What options should I choose for the EBU R128 filter? What is linear and dynamic mode?

EBU R128 is a method for normalizing audio loudness across different tracks or programs. It works by analyzing the audio content and adjusting it to meet specific loudness targets. The main components are:

* Integrated Loudness (I): The overall loudness of the entire audio.
* Loudness Range (LRA): The variation in loudness over time.
* True Peak (TP): The maximum level of the audio signal.

The normalization process involves measuring these values (input) and then applying gain adjustments to meet target levels (output), typically -23 LUFS for integrated loudness. You can also specify a target loudness range (LRA) and true peak level (TP).

**Linear mode** applies a constant gain adjustment across the entire audio file. This is generally preferred because:

* It preserves the original dynamic range of the audio.
* It maintains the relative loudness between different parts of the audio.
* It avoids potential artifacts or pumping effects that can occur with dynamic processing.

**Dynamic mode**, on the other hand, can change the volume dynamically throughout the file. While this can achieve more consistent loudness, it may alter the original artistic intent and potentially introduce audible artifacts (possibly due to some bugs in the ffmpeg filter).

For most cases, linear mode is recommended. Dynamic mode should only be used when linear mode is not suitable or when a specific effect is desired. In some cases, `loudnorm` will still fall back to dynamic mode, and a warning will be printed to the console. Here's when this can happen:

* When the input loudness range (LRA) is larger than the target loudness range: If the input file has a loudness range that exceeds the specified loudness range target, the loudnorm filter will automatically switch to dynamic mode. This is because linear normalization alone cannot reduce the loudness range without dynamic processing (limiting). The `--keep-loudness-range-target` option can be used to keep the input loudness range target above the specified target.

* When the required gain adjustment to meet the integrated loudness target would result in the true peak exceeding the specified true peak limit. This is because linear processing alone cannot reduce peaks without affecting the entire signal. For example, if a file needs to be amplified by 6 dB to reach the target integrated loudness, but doing so would push the true peak above the specified limit, the filter might switch to dynamic mode to handle this situation. If your content allows for it, you can increase the true peak target to give more headroom for linear processing. If you're consistently running into true peak issues, you might also consider lowering your target integrated loudness level.

At this time, the `loudnorm` filter in ffmpeg does not provide a way to force linear mode when the input loudness range exceeds the target or when the true peak would be exceeded. There are some options to mitigate this:

- The `--keep-lra-above-loudness-range-target` option can be used to keep the input loudness range above the specified target, but it will not force linear mode in all cases.
- Similarly, the `--keep-loudness-range-target` option can be used to keep the input loudness range target.
- The `--lower-only` option can be used to skip the normalization pass completely if the measured loudness is lower than the target loudness.

### The program doesn't work because the "loudnorm" filter can't be found

Make sure you run a recent ffmpeg version and that `loudnorm` is part of the output when you run `ffmpeg -filters`. Many distributions package outdated ffmpeg versions, or (even worse), Libav's `ffmpeg` disguising as a real `ffmpeg` from the FFmpeg project.

Some ffmpeg builds also do not have the `loudnorm` filter enabled.

You can always download a static build from [their website](http://ffmpeg.org/download.html) and use that.

If you have to use an outdated ffmpeg version, you can only use `rms` or `peak` as normalization types, but I can't promise that the program will work correctly.

### Should I use this to normalize my music collection?

Generally, no.

When you run `ffmpeg-normalize` and re-encode files with MP3 or AAC, you will inevitably introduce [generation loss](https://en.wikipedia.org/wiki/Generation_loss). Therefore, I do not recommend running this on your precious music collection, unless you have a backup of the originals or accept potential quality reduction. If you just want to normalize the subjective volume of the files without changing the actual content, consider using [MP3Gain](http://mp3gain.sourceforge.net/) and [aacgain](http://aacgain.altosdesign.com/).

### Why are my output files MKV?

I chose MKV as a default output container since it handles almost every possible combination of audio, video, and subtitle codecs. If you know which audio/video codec you want, and which container is supported, use the output options to specify the encoder and output file name manually.

### I get a "Could not write header for output file" error

See the [next section](#the-conversion-does-not-work-and-i-get-a-cryptic-ffmpeg-error).

### The conversion does not work and I get a cryptic ffmpeg error!

Maybe ffmpeg says something like:

> Could not write header for output file #0 (incorrect codec parameters ?): Invalid argument

Or the program says:

> … Please choose a suitable audio codec with the `-c:a` option.

One possible reason is that the input file contains some streams that cannot be mapped to the output file, or that you are using a codec that does not work for the output file. Examples:

- You are trying to normalize a movie file, writing to a `.wav` or `.mp3` file. WAV/MP3 files only support audio, not video. Disable video and subtitles with `-vn` and `-sn`, or choose a container that supports video (e.g. `.mkv`).

- You are trying to normalize a file, writing to an `.mp4` container. This program defaults to PCM audio, but MP4 does not support PCM audio. Make sure that your audio codec is set to something MP4 containers support (e.g. `-c:a aac`).

The default output container is `.mkv` as it will support most input stream types. If you want a different output container, [make sure that it supports](https://en.wikipedia.org/wiki/Comparison_of_container_file_formats) your input file's video, audio, and subtitle streams (if any).

Also, if there is some other broken metadata, you can try to disable copying over of metadata with `-mn`.

Finally, make sure you use a recent version of ffmpeg. The [static builds](https://ffmpeg.org/download.html) are usually the best option.

### What are the different normalization algorithms?

- **EBU R128** is an EBU standard that is commonly used in the broadcasting world. The normalization is performed using a psychoacoustic model that targets a subjective loudness level measured in LUFS (Loudness Unit Full Scale). R128 is subjectively more accurate than any peak or RMS-based normalization. More info on R128 can be found in the [official document](https://tech.ebu.ch/docs/r/r128.pdf) and [the `loudnorm` filter description](http://k.ylo.ph/2016/04/04/loudnorm.html) by its original author.

- **Peak Normalization** analyzes the peak signal level in dBFS and increases the volume of the input signal such that the maximum in the output is 0 dB (or any other chosen threshold). Since spikes in the signal can cause high volume peaks, peak normalization might still result in files that are subjectively quieter than other, non-peak-normalized files.

- **RMS-based Normalization** analyzes the [RMS power](https://en.wikipedia.org/wiki/Root_mean_square#Average_power) of the signal and changes the volume such that a new RMS target is reached. Otherwise it works similar to peak normalization.

### Couldn't I just run `loudnorm` with ffmpeg?

You absolutely can. However, you can get better accuracy and linear normalization with two passes of the filter. Since ffmpeg does not allow you to automatically run these two passes, you have to do it yourself and parse the output values from the first run.

If ffmpeg-normalize is too over-engineered for you, you could also use an approach such as featured [in this Ruby script](https://gist.github.com/kylophone/84ba07f6205895e65c9634a956bf6d54) that performs the two `loudnorm` passes.

If you want dynamic normalization (the loudnorm default), simply use ffmpeg with one pass, e.g.:

```bash
ffmpeg -i input.mp3 -af loudnorm -c:a aac -b:a 192k output.m4a
```

### What about speech?

You should check out the `speechnorm` filter that is part of ffmpeg. It is a designed to be used in one pass, so you don't need this script at all.

See [the documentation](https://ffmpeg.org/ffmpeg-all.html#speechnorm) for more information.

### After updating, this program does not work as expected anymore!

You are probably using a 0.x version of this program. There are significant changes to the command line arguments and inner workings of this program, so please  adapt your scripts to the new one. Those changes were necessary to address a few issues that kept piling up; leaving the program as-is would have made it hard to extend it. You can continue using the old version (find it under *Releases* on GitHub or request the specific version from PyPi), but it will not be supported anymore.

### Can I buy you a beer / coffee / random drink?

If you found this program useful and feel like giving back, feel free to send a donation [via PayPal](https://paypal.me/WernerRobitza).

## Related Tools and Articles

- [Create an AppleScript application to drop or open a folder of files in ffmpeg-normalize](https://prehensileblog.wordpress.com/2022/04/15/create-an-applescript-application-to-drop-or-open-a-folder-of-files-in-ffmpeg-normalize/)

*(Have a link? Please propose an edit to this section via a pull request!)*

## Contributors

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://overtag.dk/"><img src="https://avatars.githubusercontent.com/u/374612?v=4?s=100" width="100px;" alt="Benjamin Balder Bach"/><br /><sub><b>Benjamin Balder Bach</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=benjaoming" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://chaos.social/@eleni"><img src="https://avatars.githubusercontent.com/u/511547?v=4?s=100" width="100px;" alt="Eleni Lixourioti"/><br /><sub><b>Eleni Lixourioti</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=Geekfish" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/thenewguy"><img src="https://avatars.githubusercontent.com/u/77731?v=4?s=100" width="100px;" alt="thenewguy"/><br /><sub><b>thenewguy</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=thenewguy" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/aviolo"><img src="https://avatars.githubusercontent.com/u/560229?v=4?s=100" width="100px;" alt="Anthony Violo"/><br /><sub><b>Anthony Violo</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=aviolo" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://jacobs.af/"><img src="https://avatars.githubusercontent.com/u/952830?v=4?s=100" width="100px;" alt="Eric Jacobs"/><br /><sub><b>Eric Jacobs</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=jetpks" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kostalski"><img src="https://avatars.githubusercontent.com/u/34033008?v=4?s=100" width="100px;" alt="kostalski"/><br /><sub><b>kostalski</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=kostalski" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://justinppearson.com/"><img src="https://avatars.githubusercontent.com/u/8844823?v=4?s=100" width="100px;" alt="Justin Pearson"/><br /><sub><b>Justin Pearson</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=justinpearson" title="Code">πŸ’»</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Nottt"><img src="https://avatars.githubusercontent.com/u/13532436?v=4?s=100" width="100px;" alt="ad90xa0-aa"/><br /><sub><b>ad90xa0-aa</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=Nottt" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Mathijsz"><img src="https://avatars.githubusercontent.com/u/1891187?v=4?s=100" width="100px;" alt="Mathijs"/><br /><sub><b>Mathijs</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=Mathijsz" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/mpuels"><img src="https://avatars.githubusercontent.com/u/2924816?v=4?s=100" width="100px;" alt="Marc PΓΌls"/><br /><sub><b>Marc PΓΌls</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=mpuels" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://www.mvbattista.com/"><img src="https://avatars.githubusercontent.com/u/158287?v=4?s=100" width="100px;" alt="Michael V. Battista"/><br /><sub><b>Michael V. Battista</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=mvbattista" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://auto-editor.com"><img src="https://avatars.githubusercontent.com/u/57511737?v=4?s=100" width="100px;" alt="WyattBlue"/><br /><sub><b>WyattBlue</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=WyattBlue" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/g3n35i5"><img src="https://avatars.githubusercontent.com/u/17593457?v=4?s=100" width="100px;" alt="Jan-Frederik Schmidt"/><br /><sub><b>Jan-Frederik Schmidt</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=g3n35i5" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/mjhalwa"><img src="https://avatars.githubusercontent.com/u/8994014?v=4?s=100" width="100px;" alt="mjhalwa"/><br /><sub><b>mjhalwa</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=mjhalwa" title="Code">πŸ’»</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/07416"><img src="https://avatars.githubusercontent.com/u/14923168?v=4?s=100" width="100px;" alt="07416"/><br /><sub><b>07416</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=07416" title="Documentation">πŸ“–</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/sian1468"><img src="https://avatars.githubusercontent.com/u/58017832?v=4?s=100" width="100px;" alt="sian1468"/><br /><sub><b>sian1468</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=sian1468" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/psavva"><img src="https://avatars.githubusercontent.com/u/1454758?v=4?s=100" width="100px;" alt="Panayiotis Savva"/><br /><sub><b>Panayiotis Savva</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=psavva" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/HighMans"><img src="https://avatars.githubusercontent.com/u/42877729?v=4?s=100" width="100px;" alt="HighMans"/><br /><sub><b>HighMans</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=HighMans" title="Code">πŸ’»</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kanjieater"><img src="https://avatars.githubusercontent.com/u/32607317?v=4?s=100" width="100px;" alt="kanjieater"/><br /><sub><b>kanjieater</b></sub></a><br /><a href="#ideas-kanjieater" title="Ideas, Planning, & Feedback">πŸ€”</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://ahmetsait.com/"><img src="https://avatars.githubusercontent.com/u/8372246?v=4?s=100" width="100px;" alt="Ahmet Sait"/><br /><sub><b>Ahmet Sait</b></sub></a><br /><a href="https://github.com/slhck/ffmpeg-normalize/commits?author=ahmetsait" title="Code">πŸ’»</a></td>
    </tr>
  </tbody>
  <tfoot>
    <tr>
      <td align="center" size="13px" colspan="7">
        <img src="https://raw.githubusercontent.com/all-contributors/all-contributors-cli/1b8533af435da9854653492b1327a23a4dbd0a10/assets/logo-small.svg">
          <a href="https://all-contributors.js.org/docs/en/bot/usage">Add your contributions</a>
        </img>
      </td>
    </tr>
  </tfoot>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

## License

The MIT License (MIT)

Copyright (c) 2015-2022 Werner Robitza

Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to
the following conditions:

The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.


# Changelog


## v1.31.0 (2024-12-15)

* Update docs and completions.

* Implement `--auto-lower-loudness-target`

* Fix deprecations and mypy --strict errors.

* Feat: add completions.

* Docs: update explainer.

* Docs: update docs to include lower-only.


## v1.30.0 (2024-11-22)

* Change lower-only message to warning.

* Make setup name PEP 625 compliant.

* Docs: add @ahmetsait as a contributor.

* Implement `--lower-only`

* Fix: `--print-stats` only outputs the last stream.

* More robust `loudnorm` output parsing.

* Remove unnecessary conversions.

* Update .editorconfig.

* Remove python 3.8, add python 3.12, 3.13.

* Add README on file size.


## v1.29.2 (2024-11-18)

* Fix: show percentage with two decimal digits in progress.

* Chore: add python 12.


## v1.29.1 (2024-10-22)

* Fix: override argparse usage.


## v1.29.0 (2024-10-14)

* Add option to set audio channels directly.


## v1.28.3 (2024-08-16)

* Make colorama dependency windows-only.

* Update.

* Update badges.

* Docs: add @kanjieater as a contributor.


## v1.28.2 (2024-06-22)

* Provide an entrypoint for the Docker image.

* Add docker push (#261)


## v1.28.1 (2024-05-15)

* Fix assignment of audio statistics, fixes #257.


## v1.28.0 (2024-05-13)

* Warn if dynamic mode is used but linear specified (#256)

* Print debug commands in shell-escaped form.


## v1.27.7 (2023-09-26)

* Allow cover art in MP3.


## v1.27.6 (2023-07-22)

* Remove warning for short files (#87)


## v1.27.5 (2023-07-12)

* Building on Windows failed due to character conversion (#244)


## v1.27.4 (2023-06-29)

* Docs: add HighMans as a contributor for code (#242)

  * docs: update README.md

  * docs: update .all-contributorsrc

  ---------

* Properly handle non-existing input files and skipping.

* Fix: Dockerfile used wrong path.

* Use `https` rather than `http` (#238)

* Docs: add psavva as a contributor for code (#236)

  * docs: update README.md

  * docs: update .all-contributorsrc

  ---------

* Add Docker support (#235)

* Docs: add @sian1468 as a contributor.

* Round percentage in test (#234)


## v1.27.3 (2023-05-06)

* Bump ffmpeg-progress-yield, round percentage output.

* Fix test for progress.

  seems flaky, worked locally?


## v1.27.2 (2023-05-05)

* Bump ffmpeg-progress-yield, output progress in percent.

* Docs: add @07416 as a contributor.

* A typo in README (#231)


## v1.27.1 (2023-04-25)

* Fix capping to [1, 50] instead of [1, 7] (#230)

* Clarify usage, output/input order.

* Add note on ffmpeg 6.0.

* Docs: add @mjhalwa as a contributor.


## v1.27.0 (2023-04-24)

* Constrain input LRA for second pass, addresses #227.

* Feat: add linear loudnorm option to set lra up to target, then keep input lra.

* Update question template.


## v1.26.6 (2023-03-16)

* Production status stable.

* Make install_requires more abstract.


## v1.26.5 (2023-03-15)

* Add "-hide_banner" remove "-nostdin" (#222)

  The `-nostdin` option is unnessary because of the `-y` option.
  Adding `-hide_banner` makes DEBUG statements shorter.

* Homebrew works on linux too.

* Explain ffmpeg installation steps.


## v1.26.4 (2023-02-08)

* Re-add requirements.txt to (maybe) fix conda-forge builds.


## v1.26.3 (2023-02-08)

* Fix requirements (#218)

* Improve types.

* Docs: add @g3n35i5 as a contributor.


## v1.26.2 (2023-02-06)

* Add ignore-revs file.

* Formatting and import sorting.

* Refactor: Improved logging behavior (#216)

* Add "apt update" (#215)

* Remove stalebot.

* Update README.

* Move to_ms and make CommandRunner more ergonomic (#212)

* Upgrade workflow, get ffmpeg from apt (#213)

* Turn FFmpegNormalizeError into a normal Exception (#211)

* Remove manifest.in (#210)

* Simplify logging (#209)

* Use pep585 type hints (#207)

* Don't use tempfile's private module function (#206)

* Fix smaller type errors.

* Reduce mypy errors 12 -> 4 (#204)

* Make input validation more efficient.

  Make input validation more efficient

  Re-separate formats and exts


## v1.26.1 (2022-12-18)

* Bump requirements.

* Add py.typed support.

* General refactoring + type hints (#202)

* Re-write to f-strings when possible (#201)

* Remove unnecessary utf-8 declarations (#200)

  "-*- coding: utf-8 -*-" is a Python 2 construct and can be safely
  removed. Other utf-8 declarations are also unnecessary.


## v1.26.0 (2022-12-14)

* Add .editorconfig.

* Link to API docs.

* Add docs.

* Add type hints, document everything, refactor some code.

* Add more audio formats (#199)

* Add python 3.11 to CI.

* Docs: add WyattBlue as a contributor for code (#198)

  * docs: update README.md

  * docs: update .all-contributorsrc

* Upgrade to Python 3.8 syntax (#197)

* Fix python version in github tests.

* Bump requirements to latest versions.

* Add python 3.11 to list of languages.

* Bump required python version to 3.8.

* Various minor code cleanups and type hints.

* Harmonize logger code.

* Update python version in tests.

* Docs: add @mvbattista as a contributor.

* Docs: add @mpuels as a contributor.

* Docs: add @Mathijsz as a contributor.

* Docs: add @Nottt as a contributor.

* Docs: add @justinpearson as a contributor.

* Docs: add @kostalski as a contributor.

* Docs: add @jetpks as a contributor.

* Docs: add @aviolo as a contributor.

* Docs: add @thenewguy as a contributor.

* Docs: add @Geekfish as a contributor.

* Docs: add @benjaoming as a contributor.

* Reference speechnorm.


## v1.25.3 (2022-11-09)

* Update README.

* Update list of pcm-incompatible extensions.


## v1.25.2 (2022-09-14)

* Constrain parsed ranges to avoid out of bounds, fixes #189.

* Fix readme for extra-input-options.

* Warn about dynamic mode only if not already set, fixes #187.


## v1.25.1 (2022-08-21)

* Add warning in case user specifies both --lrt and --keep-loudness-range-target.


## v1.25.0 (2022-08-20)

* Add option to keep loudness range target, fixes #181.

* Only show warning about disabling video if not yet disabled, addresses #184.


## v1.24.1 (2022-08-20)

* Code formatting.

* Extend warning for audio-only format to opus, fixes #184.


## v1.24.0 (2022-08-02)

* Update python requirements.

* Prevent race condition in output dir creation.


## v1.23.1 (2022-07-12)

* Increase possible loudness range target to 50.


## v1.23.0 (2022-05-01)

* Add way to force dynamic mode, clarify usage, fixes #176.


## v1.22.10 (2022-04-25)

* Add warning for cover art, addresses #174 and #175.

* Update README.


## v1.22.9 (2022-04-17)

* Improve issue templates.

* Do not print ffmpeg progress in debug logs.

* Remove unused import.

* Replace which() function with shlex version.

* Add python 3.10 in setup.py.

* Clarify minimum ffmpeg version.


## v1.22.8 (2022-03-07)

* Properly detect -inf dB input.


## v1.22.7 (2022-02-25)

* Debug command output for ffmpeg commands.

* Remove unneeded warning message.


## v1.22.6 (2022-02-20)

* Use astats instead of volumedetect filter, fixes #163.

  Allows floating point calculation.


## v1.22.5 (2022-01-25)

* Print warning for bit depths > 16, addresses #163.


## v1.22.4 (2021-10-18)

* Re-raise error on ffmpeg command failure.

  This prevents incorrectly telling the user that a normalized file was written when it wasn't.


## v1.22.3 (2021-08-31)

* Set tqdm lock for logging only when multiprocessing is available.

  Multiprocessing is not available in all environments, for example
  on AWS lambda python run time lacks /dev/shm, so trying to acquire
  a multiprocessing Lock throws an OSError. The module could also be
  missing in some cases (ex. Jython, although this library doesn't support
  Jython anyway).

  The solution to this is to only try to set the lock when multiprocessing
  is available. The tqdm library solves this in the same manner.

  For more details: https://github.com/slhck/ffmpeg-normalize/issues/156

* Add instructions on how to run tests.


## v1.22.2 (2021-08-14)

* Bump requirements, should fix #155.

* Move all examples to Wiki.

* Update badge link.


## v1.22.1 (2021-03-10)

* Add python_requires to setup.py.


## v1.22.0 (2021-03-09)

* Improve README.

* Add GitHub actions badge.

* Add GitHub actions tests.

* Properly convert EBU JSON values to float.

* Switch to f strings, remove Python 3.5 support.

* Format code with black.

* Fix flake8 errors.

* Factor out method.

* WIP: new tests.

* Log to stderr by default to enable JSON parsing.

* Remove release script.


## v1.21.2 (2021-03-06)

* Format setup.py.

* Switch progress to external lib.

* Remove support for older versions.


## v1.21.1 (2021-03-05)

* Adjusted handling of FFMPEG_PATH for binaries available via $PATH (#149)

  * adjusted handling of FFMPEG_PATH for binaries available via $PATH

  fixes #147

  * adjusted use of %s to {} to match style

  * documented the feature

  * condensed error message as other lines are longer


## v1.21.0 (2021-02-27)

* Fix JSON output for multiple files.

* Update badge URL.

* Update README.md (#142)

  * Update README.md

  Added example of verifying levels

  Fixes #141

  * shorten example, add link to wiki page

* Error if no ffmpeg exec exists.

* Add stalebot.


## v1.20.2 (2020-11-06)

* Fixing stdin corruption caused by new subprocess (#138)

* Update issue template.

* Create FUNDING.yml.

* Fix usage, addresses #132.


## v1.20.1 (2020-07-22)

* Manually specify usage string, fixes #132.

* Fix local import for tests.


## v1.20.0 (2020-07-04)

* Add extra input options.


## v1.19.1 (2020-06-25)

* Add colorama to requirements, fixes #131.

* Fix warning that is printed with default options.


## v1.19.0 (2020-05-02)

* Fix issue with output folder, fixes #126.

* Fix typo in README's table of contents link to "File Input/Output". (#124)

* Clarify readme, fixes #122.


## v1.18.2 (2020-04-19)

* Add warning for automatic sample rate conversion, addresses #122.

* Ignore vscode folder.

* Fix printing of errors in conversion.


## v1.18.1 (2020-04-16)

* Fix unit tests.

* Improve handling of output file folder and errors.

* Clarify usage of output options, add warning.

* Improve documentation, fixes #120.

* Do not include bump messages in changelog.


## v1.18.0 (2020-04-13)

* Use measured offset in second pass, fixes #119.

* Update release instructions.

* Remove author names from changelog.


## v1.17.0 (2020-04-10)

* Update release script and changelog template.

* Apply pre-filters in all first passes, fixes #118.

  This allows properly reading the level for any kind of normalization, even if
  filters affect the loudness in the first pass.


## v1.16.0 (2020-04-07)

* Add all commits to changelog.

* Remove python 2 support.

* Add quiet option, fixes #116.

  - Add a new quiet option
  - Promote some warnings to actual errors that need to be shown
  - Add a very basic test case


## v1.15.8 (2020-03-15)

* Improve release script.

* Python 3.8.


## v1.15.7 (2020-03-14)

* Only print length warning for non-EBU type normalization.


## v1.15.6 (2019-12-04)

* Remove build and dist folder on release.

* Do not exit on error in batch processing.

  Simply process the next file if one has errors, addresses #110.


## v1.15.5 (2019-11-19)

* Use minimal dependency for tqdm.

* Remove specific python version requirement.


## v1.15.4 (2019-11-19)

* Freeze tqdm version.

* Update python to 3.7.

* Improve release documentation.


## v1.15.3 (2019-10-15)

* Do not print stream warning when there is only one stream.

* Remove previous dist versions before release.


## v1.15.2 (2019-07-12)

* Warn when duration cannot be read, fixes #105.

* Update README.

  minor improvements in the description


## v1.15.1 (2019-06-17)

* Add output to unit test failures.

* Fix input label for audio stream.


## v1.15.0 (2019-06-17)

* Add pre-and post-filter hooks, fixes #67.

  This allows users to specify filters to be run before or after the actual
  normalization call, using regular ffmpeg syntax.
  Only applies to audio.

* Document audiostream class.

* Warn when file is too short, fixes #87.

* Update release method to twine.


## v1.14.1 (2019-06-14)

* Handle progress output from ffmpeg, fixes #10.

* Merge pull request #99 from Nottt/patch-1.

  fix -cn description

* Fix -cn description.

* Add nicer headers for options in README.


## v1.14.0 (2019-04-24)

* Add version file in release script before committing.

* Add option to keep original audio, fixes #83.

* Add pypi badge.

* Allow release script to add changelog for future version; upload to pypi.


## v1.13.11 (2019-04-16)

* Add release script.

* Add small developer guide on releasing.

* Move HISTORY.md to CHANGELOG.md.

* Fix ffmpeg static build download location.


## v1.3.10 (2019-02-22)

* Bump version.

* Cap measured input loudness, fixes #92.


## v1.3.9 (2019-01-10)

* Bump version.

* Fix handling of errors with tqdm.

* Improve readme.

* Delete issue template.

* Bump version.

* Clarify extra argument options, move to main entry point.

* Update issue templates.


## v1.3.8 (2018-11-28)

* Bump version.

* Clarify extra argument options, move to main entry point.


## v1.3.7 (2018-10-28)

* Bump version.

* Copy metadata from individual streams, fixes #86.

* Add python version for pyenv.


## v1.3.6 (2018-07-09)

* Bump version.

* Update README, fixes #79 and addresses #80.


## v1.3.5 (2018-06-12)

* Bump version.

* Minor README updates.

* Fix documentation of TMPDIR parameter.


## v1.3.4 (2018-05-05)

* Bump version.

* New way to specify extra options.


## v1.3.3 (2018-05-05)

* Update README.

* Decode strings in extra options.


## v1.3.2 (2018-04-25)

* Bump version.

* Merge pull request #69 from UbiCastTeam/master.

  Stderror decoding ignoring utf8 encoding errors

* Stderror decoding ignoring utf8 encoding errors.


## v1.3.1 (2018-04-24)

* Bump version.

* Do not require main module in setup.py, fixes #68.


## v1.3.0 (2018-04-15)

* Bump version.

* Remove dead code.

* Fix for python2 division.

* Update documentation.

* Progress bar.

* Remove imports from test file.

* Fix travis file.

* WIP: progress bar.

* Minor typo in option group.

* Add simple unit test for disabling chapters.


## v1.2.3 (2018-04-11)

* Fix unit test.

* Bump version.

* Add option to disable chapters, fixes #65, also fix issue with metadata.


## v1.2.2 (2018-04-10)

* Bump version.

* Set default loudness target to -23, fixes #48.


## v1.2.1 (2018-04-04)

* Bump version.

* Merge pull request #64 from UbiCastTeam/encoding-issue.

  Stdout and stderror decoding ignoring utf8 encoding errors

* Stdout and stderror decoding ignoring utf8 encoding errors.


## v1.2.0 (2018-03-22)

* Bump version.

* Add errors for impossible format combinations, fixes #60.

* Fix ordering of output maps, fixes #63.

* Improve documentation.


## v1.1.0 (2018-03-06)

* Add option to print first pass statistics.


## v1.0.10 (2018-03-04)

* Bump version.

* Restrict parsing to valid JSON part only, fixes #61.

* Add an example for MP3 encoding.

* Update paypal link.


## v1.0.9 (2018-02-08)

* Bump version.

* Add normalized folder to gitignore.

* Do not print escape sequences on Windows.

* Do not check for file existence, fixes #57.

* Add github issue template.


## v1.0.8 (2018-02-01)

* Bump version.

* Do not check for ffmpeg upon module import.


## v1.0.7 (2018-02-01)

* Bump version.

* Rename function test.

* Fix issue with wrong adjustment parameters, fixes #54.


## v1.0.6 (2018-01-30)

* Allow setting FFMPEG_PATH and document TMP.


## v1.0.5 (2018-01-26)

* Handle edge case for short input clips.


## v1.0.4 (2018-01-26)

* Bump version.

* Do not try to remove nonexisting file in case of error in command.


## v1.0.3 (2018-01-26)

* Bump version.

* Always streamcopy when detecting streams to avoid initializing encoder.

* Fix handling of temporary file.

* Add build status.

* Travis tests.


## v1.0.2 (2018-01-25)

* Fix bug with target level for peak/RMS.

* Update documentation formatting.

* Update history.


## v1.0.1 (2018-01-24)

* Bump version.

* Set default target to -23.


## v1.0.0 (2018-01-23)

* Add version info and test case for dry run.

* New feature detection, add documentation, contributors guide etc.

* WIP: v1.0 rewrite.


## v0.7.3 (2017-10-09)

* Use shutil.move instead of os.rename.


## v0.7.2 (2017-09-17)

* Allow setting threshold to 0.


## v0.7.1 (2017-09-14)

* Bump version.

* Update HISTORY.md.

* Merge pull request #37 from Mathijsz/fix-which-path-expansion.

  expand tilde and environment variables, fixes #36

* Expand tilde and environment variables, fixes #36.

* Update HISTORY.md.

* Update README w.r.t. loudnorm filter.

* Update README and indentation.


## v0.7.0 (2017-08-02)

* Bump version.

* Fix handling of extra options with spaces.

* Include test script.

* Logging and other improvements.

* Add test files.

* Autopep8 that thing.

* Logger improvements.

* Add example for overwriting.


## v0.6.0 (2017-07-31)

* Allow overwriting input file, fixes #22.

* Version bump.

* Better handle cmd arguments.

* Update README.md.

  add another example


## v0.5.1 (2017-04-04)

* Fix for problem introduced in 304e8df.


## v0.5 (2017-04-02)

* Fix pypi topics.

* Bump version and README.

* Fix issue where output was wrong format.

* Add EBU R128 filter.

* Use Markdown instead of RST for README/HISTORY.

* Define file encode for python3, fixes #24.

* Fix history.

* Fix option -np.

* Clarify merge option.

* Minor documentation improvements.

  - change README from CRLF to LF
  - add "attenuated" in description
  - extend LICENSE year
  - add license to main README


## v0.4.1 (2017-02-13)

* Update for release.

* Merge pull request #21 from mpuels/patch-1.

  Fix for #13

* Fix for #13.

* Mention Python 3.

  mention that Python 3 may work, just didn't have time to test

* Fix README's code blocks.


## v0.4 (2017-01-24)

* Code cleanup, add option to set format and audio codec.


## v0.3 (2017-01-19)

* Add option for no prefix, fixes #20.

* Handle multiple spaces in path; fixes issue #18.

* Handle spaces in path, fixes #12.

* Update README.rst.

* Change default level back  to -26.

* Typo in README example.

* Update documentation.

* Bump to v0.2.0.

  * Support for multiple files and output directories.
  * Support merging of audio with input file
  * Set audio codec and additional options
  * User-definable threshold
  * Better error handling and logging
  * Deprecates avconv

* Change default level back to -28.

* Merge pull request #15 from auricgoldfinger/master.

  Add extended normalisation options

* Add extended normalisation options.

  - add program option to write output in a separate directory in stead of
     prefixing it

  - add program option to merge the normalized audio in the original
     (video) file rather than creating a separate WAV file

  - change the maximum setting: will now normalize so that max
     volume is set to 0, adjusted with the given level.
     e.g. : -m -l -5 will increase the audio level to max = -5.0dB

  - improve verbose logging: number of files are written to the
     info log

  - improve performance: check first whether the output file
     exists before calculating the volume levels + not modifying
     the file if the adjustment < 0.5dB (level is never exactly 0)

* Update README, fixes #11.


## v0.1.3 (2015-12-15)

* Check for Windows .exe, fixes #10.

* Check path and fix #9.

* Merge pull request #8 from benjaoming/master.

  Add MANIFEST.in

* Bump version.

* Add manifest to include missing files in sdist.

* Merge pull request #6 from jetpks/master.

  Fixed ffmpeg v2.6.3 compatibility and docopt config

* Updated to work with ffmpeg v2.6.3, and fixed broken docopt config.

  ffmpeg update:

  ffmpeg v2.6.3 puts mean_volume on stderr instead of stdout, causing
  `output` in `ffmpeg_get_mean` to be completely empty, and no match for
  mean_volume or max_volume to be found.

  Fixed by adding `stderr=subprocess.PIPE` in both Popen calls in
  `run_command`, and combining stdout and stderr on return. We already
  exit with non-zero return, so combining stderr/stdout shouldn't cause
  any poor side-effects.

  docopt config:

  - args['--level'] was not recognizing its default because there was
    an errant comma between -l and --level, and it needed <level> after
    the arguments.
  - Fixed spacing for --max
  - Removed quotes around 'normalized' so single quote characters don't
    end up in the output file names.

* Removed Windows carraige returns from __main__.py.

* Merge pull request #5 from mvbattista/master.

  Installation update to ffmpeg

* Installation update to ffmpeg.

* Update to ffmpeg.

* Update HISTORY.rst.

* Update to ffmpeg.

* Merge pull request #4 from benjaoming/rename.

  Rename project

* Make at least one file mandatory.

* Rename project and remove pyc file.

* Merge pull request #2 from benjaoming/docopt-setuptools-avconv.

  Docopt, Setuptools, avconv compatibility

* Use docopt.

* Use normalize-audio when using avconv because it doesn't have a way to measure volume.

* Functional setup.py, communicate with avconv/ffmpeg about overwriting.

* Also detect avconv.

* Use a main function instead.

* Add a history for the project.

* Move to more unique module name.

* Update README.rst.

* Change the README to rst (PyPi)

* Delete .gitignore.

* Update README.md.

* Various improvements, fixes #1.

* License.

* Livense.

* Update README.md.

* Merge branch 'master' of https://github.com/slhck/audio-normalize.

* Initial commit.

* Initial commit.



            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/slhck/ffmpeg-normalize",
    "name": "ffmpeg-normalize",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.9",
    "maintainer_email": null,
    "keywords": "ffmpeg, normalize, audio",
    "author": "Werner Robitza",
    "author_email": "werner.robitza@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/f6/7f/ba9076fd5b1090631a6d8ee7e701ef5f480646c933bcfe045c64b7c21162/ffmpeg_normalize-1.31.0.tar.gz",
    "platform": null,
    "description": "# ffmpeg-normalize\n\n[![PyPI version](https://img.shields.io/pypi/v/ffmpeg-normalize.svg)](https://pypi.org/project/ffmpeg-normalize)\n![Docker Image Version](https://img.shields.io/docker/v/slhck/ffmpeg-normalize?sort=semver&label=Docker%20image)\n![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/slhck/ffmpeg-normalize/python-package.yml)\n\n<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->\n[![All Contributors](https://img.shields.io/badge/all_contributors-20-orange.svg?style=flat-square)](#contributors-)\n<!-- ALL-CONTRIBUTORS-BADGE:END -->\n\nA utility for batch-normalizing audio using ffmpeg.\n\nThis program normalizes media files to a certain loudness level using the EBU R128 loudness normalization procedure. It can also perform RMS-based normalization (where the mean is lifted or attenuated), or peak normalization to a certain target level.\n\nBatch processing of several input files is possible, including video files.\n\n**A very quick how-to:**\n\n1. Install a recent version of [ffmpeg](https://ffmpeg.org/download.html)\n2. Run `pip3 install ffmpeg-normalize`\n3. Run `ffmpeg-normalize /path/to/your/file.mp4`\n4. Done! \ud83c\udfa7 (the file will be in a folder called `normalized`)\n\nRead on for more info.\n\n**Contents:**\n\n- [Requirements](#requirements)\n  - [ffmpeg](#ffmpeg)\n- [Installation](#installation)\n  - [Shell Completions](#shell-completions)\n- [Usage with Docker](#usage-with-docker)\n- [High LeveL Introduction](#high-level-introduction)\n- [Basic Usage](#basic-usage)\n- [Examples](#examples)\n- [Detailed Options](#detailed-options)\n  - [File Input/Output](#file-inputoutput)\n  - [General](#general)\n  - [Normalization](#normalization)\n  - [EBU R128 Normalization](#ebu-r128-normalization)\n  - [Audio Encoding](#audio-encoding)\n  - [Other Encoding Options](#other-encoding-options)\n  - [Input/Output Format](#inputoutput-format)\n  - [Environment Variables](#environment-variables)\n- [API](#api)\n- [FAQ](#faq)\n  - [My output file is too large?](#my-output-file-is-too-large)\n  - [What options should I choose for the EBU R128 filter? What is linear and dynamic mode?](#what-options-should-i-choose-for-the-ebu-r128-filter-what-is-linear-and-dynamic-mode)\n  - [The program doesn't work because the \"loudnorm\" filter can't be found](#the-program-doesnt-work-because-the-loudnorm-filter-cant-be-found)\n  - [Should I use this to normalize my music collection?](#should-i-use-this-to-normalize-my-music-collection)\n  - [Why are my output files MKV?](#why-are-my-output-files-mkv)\n  - [I get a \"Could not write header for output file\" error](#i-get-a-could-not-write-header-for-output-file-error)\n  - [The conversion does not work and I get a cryptic ffmpeg error!](#the-conversion-does-not-work-and-i-get-a-cryptic-ffmpeg-error)\n  - [What are the different normalization algorithms?](#what-are-the-different-normalization-algorithms)\n  - [Couldn't I just run `loudnorm` with ffmpeg?](#couldnt-i-just-run-loudnorm-with-ffmpeg)\n  - [What about speech?](#what-about-speech)\n  - [After updating, this program does not work as expected anymore!](#after-updating-this-program-does-not-work-as-expected-anymore)\n  - [Can I buy you a beer / coffee / random drink?](#can-i-buy-you-a-beer--coffee--random-drink)\n- [Related Tools and Articles](#related-tools-and-articles)\n- [Contributors](#contributors)\n- [License](#license)\n\n-------------\n\n## Requirements\n\nYou need Python 3.9 or higher, and ffmpeg.\n\n### ffmpeg\n\n- ffmpeg 5.x is required, ffmpeg 6.x is recommended (it fixes [a bug for short files](https://github.com/slhck/ffmpeg-normalize/issues/87))\n- Download a [static build](https://ffmpeg.org/download.html) for your system\n- Place the `ffmpeg` executable in your `$PATH`, or specify the path to the binary with the `FFMPEG_PATH` environment variable in `ffmpeg-normalize`\n\nFor instance, under Linux:\n\n```bash\nwget https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-amd64-static.tar.xz\nmkdir -p ffmpeg\ntar -xf ffmpeg-release-amd64-static.tar.xz -C ffmpeg --strip-components=1\nsudo cp ffmpeg/ffmpeg /usr/local/bin\nsudo cp ffmpeg/ffprobe /usr/local/bin\nsudo chmod +x /usr/local/bin/ffmpeg /usr/local/bin/ffprobe\n```\n\nFor Windows, follow [this guide](https://www.wikihow.com/Install-FFmpeg-on-Windows).\n\nFor macOS and Linux, you can also use [Homebrew](https://brew.sh):\n\n```bash\nbrew install ffmpeg\n```\n\nNote that using distribution packages (e.g., `apt install ffmpeg`) is not recommended, as these are often outdated.\n\n## Installation\n\nFor Python 3 and pip:\n\n```bash\npip3 install ffmpeg-normalize\n```\n\nOr download this repository, then run `pip3 install .`.\n\nTo later upgrade to the latest version, run `pip3 install --upgrade ffmpeg-normalize`.\n\n### Shell Completions\n\nThis tool provides shell completions for bash and zsh. To install them:\n\n<!--\nNote to self: Generate the shtab ones with:\n\n  shtab --shell=bash -u ffmpeg_normalize.__main__.create_parser > completions/ffmpeg-normalize-shtab.bash\n  shtab --shell=zsh -u ffmpeg_normalize.__main__.create_parser > completions/ffmpeg-normalize-shtab.zsh\n\nbut these are not properly working yet.\n-->\n\n#### Bash\n\nIf you have [`bash-completion`](https://github.com/scop/bash-completion) installed, you can just copy your new completion script to the `/usr/local/etc/bash_completion.d` directory.\n\n```bash\ncurl -L https://raw.githubusercontent.com/slhck/ffmpeg-normalize/master/completions/ffmpeg-normalize-completion.bash \\\n  -o /usr/local/etc/bash_completion.d/ffmpeg-normalize\n```\n\nWithout bash-completion, you can manually install the completion script:\n\n```bash\n# create completions directory if it doesn't exist\nmkdir -p ~/.bash_completions.d\n\n# download and install completion script\ncurl -L https://raw.githubusercontent.com/slhck/ffmpeg-normalize/master/completions/ffmpeg-normalize-completion.bash \\\n  -o ~/.bash_completions.d/ffmpeg-normalize\n\n# source it in your ~/.bashrc\necho 'source ~/.bash_completions.d/ffmpeg-normalize' >> ~/.bashrc\n```\n\n#### Zsh\n\nDownload the completion script and place it in the default `site-functions` directory:\n\n```bash\ncurl -L https://raw.githubusercontent.com/slhck/ffmpeg-normalize/master/completions/ffmpeg-normalize.zsh \\\n  -o /usr/local/share/zsh/site-functions/\n```\n\nYou may choose any other directory that is in your `$FPATH` variable.\nMake sure your `.zshrc` file contains `autoload -Uz compinit && compinit`.\n\n## Usage with Docker\n\nYou can use the pre-built image from Docker Hub:\n\n```bash\ndocker run -v \"$(pwd):/tmp\" -it slhck/ffmpeg-normalize\n```\n\nAlternatively, download this repository and run\n\n```bash\ndocker build -t ffmpeg-normalize .\n```\n\nThen run the container with:\n\n```bash\ndocker run  -v \"$(pwd):/tmp\" -it ffmpeg-normalize\n```\n\nThis will mount your current directory to the `/tmp` directory inside the container. Everything else works the same way as if you had installed the program locally. For example, to normalize a file:\n\n```bash\ndocker run  -v \"$(pwd):/tmp\" -it ffmpeg-normalize /tmp/yourfile.mp4 -o /tmp/yourfile-normalized.wav\n```\n\nYou will then find the normalized file in your current directory.\n\n## High LeveL Introduction\n\nPlease read this section for a high level introduction.\n\n**What does the program do?**\n\nThe program takes one or more input files and, by default, writes them to a folder called `normalized`, using an `.mkv` container. All audio streams will be normalized so that they have the same (perceived) volume according to the EBU R128 standard. This is done by analyzing the audio streams and applying a filter to bring them to a target level. Under the hood, the program uses ffmpeg's `loudnorm` filter to do this.\n\n**How do I specify the input?**\n\nJust give the program one or more input files as arguments. It works with most media files, including video files.\n\n**How do I specify the output?**\n\nYou don't have to specify an output file name (the default is `normalized/<input>.mkv`), but if you want to override it, you can specify one output file name for each input file with the `-o` option. In this case, the container format (e.g. `.wav`) will be inferred from the file name extension that you've given.\n\nExample:\n\n```bash\nffmpeg-normalize 1.wav 2.wav -o 1-normalized.wav 2-normalized.wav\n```\n\nNote that if you don't specify the output file name for an input file, the container format will be MKV, and the output will be written to `normalized/<input>.mkv`. The reason for choosing the MKV container is that it can handle almost any codec combination.\n\nUsing the `-ext` option, you can supply a different output extension common to all output files, e.g. `-ext m4a`. However, you need to make sure that the container supports the codecs used for the output (see below).\n\n**What will get normalized?**\n\nBy default, all streams from the input file will be written to the output file. For example, if your input is a video with two language tracks and a subtitle track, both audio tracks will be normalized independently. The video and subtitle tracks will be copied over to the output file.\n\n**How will the normalization be done?**\n\nThe normalization will be performed according to the EBU R128 algorithm with the [`loudnorm` filter](https://ffmpeg.org/ffmpeg-filters.html#loudnorm) from FFmpeg, which was [originally written by Kyle Swanson](https://k.ylo.ph/2016/04/04/loudnorm.html). It will bring the audio to a specified target level. This ensures that multiple files normalized with this filter will have the same perceived loudness.\n\n**What codec is chosen?**\n\nThe default audio encoding method is uncompressed PCM (`pcm_s16le`) to avoid introducing compression artifacts. This will result in a much higher bitrate than you might want, for example if your input files are MP3s.\n\nSome containers (like MP4) also cannot handle PCM audio. If you want to use such containers and/or keep the file size down, use `-c:a` and specify an audio codec (e.g., `-c:a aac` for ffmpeg's built-in AAC encoder).\n\n## Basic Usage\n\nSupply one or more input files, and optionally, output file names:\n\n```bash\nffmpeg-normalize input [input ...][-h][-o OUTPUT [OUTPUT ...]] [options]\n```\n\nExample:\n\n```bash\nffmpeg-normalize 1.wav 2.wav -o 1-normalized.m4a 2-normalized.m4a -c:a aac -b:a 192k\n```\n\nFor more information on the options (`[options]`) available, run `ffmpeg-normalize -h`, or read on.\n\n## Examples\n\n[Read the examples on the wiki.](https://github.com/slhck/ffmpeg-normalize/wiki/examples)\n\n## Detailed Options\n\n### File Input/Output\n\n- `input`: Input media file(s)\n\n- `-o OUTPUT [OUTPUT ...], --output OUTPUT [OUTPUT ...]`: Output file names.\n\n    Will be applied per input file.\n\n    If no output file name is specified for an input file, the output files\n    will be written to the default output folder with the name `<input>.<ext>`, where `<ext>` is the output extension (see `-ext` option).\n\n    Example: `ffmpeg-normalize 1.wav 2.wav -o 1n.wav 2n.wav`\n\n- `-of OUTPUT_FOLDER, --output-folder OUTPUT_FOLDER`: Output folder (default: `normalized`)\n\n    This folder will be used for input files that have no explicit output name specified.\n\n### General\n\n- `-f, --force`: Force overwrite existing files\n\n- `-d, --debug`: Print debugging output\n\n- `-v, --verbose`: Print verbose output\n\n- `-q, --quiet`: Only print errors\n\n- `-n, --dry-run`: Do not run normalization, only print what would be done\n\n- `-pr`, `--progress`: Show progress bar for files and streams\n\n- `--version`: Print version and exit\n\n### Normalization\n\n- `-nt {ebu,rms,peak}, --normalization-type {ebu,rms,peak}`: Normalization type (default: `ebu`).\n\n    EBU normalization performs two passes and normalizes according to EBU R128.\n\n    RMS-based normalization brings the input file to the specified RMS level.\n\n    Peak normalization brings the signal to the specified peak level.\n\n- `-t TARGET_LEVEL, --target-level TARGET_LEVEL`: Normalization target level in dB/LUFS (default: -23).\n\n    For EBU normalization, it corresponds to Integrated Loudness Target in LUFS. The range is -70.0 - -5.0.\n\n    Otherwise, the range is -99 to 0.\n\n- `-p, --print-stats`: Print loudness statistics for both passes formatted as JSON to stdout.\n\n### EBU R128 Normalization\n\n- `-lrt LOUDNESS_RANGE_TARGET, --loudness-range-target LOUDNESS_RANGE_TARGET`: EBU Loudness Range Target in LUFS (default: 7.0).\n\n    Range is 1.0 - 50.0.\n\n- `--keep-loudness-range-target`: Keep the input loudness range target to allow for linear normalization.\n\n- `--keep-lra-above-loudness-range-target`: Keep input loudness range above loudness range target.\n\n    Can be used as an alternative to `--keep-loudness-range-target` to allow for linear normalization.\n\n- `-tp TRUE_PEAK, --true-peak TRUE_PEAK`: EBU Maximum True Peak in dBTP (default: -2.0).\n\n    Range is -9.0 - +0.0.\n\n- `--offset OFFSET`: EBU Offset Gain (default: 0.0).\n\n    The gain is applied before the true-peak limiter in the first pass only. The offset for the second pass will be automatically determined based on the first pass statistics.\n\n    Range is -99.0 - +99.0.\n\n- `--lower-only`: Whether the audio should not increase in loudness.\n\n    If the measured loudness from the first pass is lower than the target loudness then normalization pass will be skipped for the measured audio source.\n\n- `--auto-lower-loudness-target`: Automatically lower EBU Integrated Loudness Target.\n\n    Automatically lower EBU Integrated Loudness Target to prevent falling back to dynamic filtering.\n\n    Makes sure target loudness is lower than measured loudness minus peak loudness (input_i - input_tp) by a small amount.\n\n- `--dual-mono`: Treat mono input files as \"dual-mono\".\n\n    If a mono file is intended for playback on a stereo system, its EBU R128 measurement will be perceptually incorrect. If set, this option will compensate for this effect. Multi-channel input files are not affected by this option.\n\n- `--dynamic`: Force dynamic normalization mode.\n\n    Instead of applying linear EBU R128 normalization, choose a dynamic normalization. This is not usually recommended.\n\n    Dynamic mode will automatically change the sample rate to 192 kHz. Use `-ar`/`--sample-rate` to specify a different output sample rate.\n\n### Audio Encoding\n\n- `-c:a AUDIO_CODEC, --audio-codec AUDIO_CODEC`: Audio codec to use for output files.\n\n    See `ffmpeg -encoders` for a list.\n\n    Will use PCM audio with input stream bit depth by default.\n\n- `-b:a AUDIO_BITRATE, --audio-bitrate AUDIO_BITRATE`: Audio bitrate in bits/s, or with K suffix.\n\n    If not specified, will use codec default.\n\n- `-ar SAMPLE_RATE, --sample-rate SAMPLE_RATE`: Audio sample rate to use for output files in Hz.\n\n    Will use input sample rate by default, except for EBU normalization, which will change the input sample rate to 192 kHz.\n\n- `-ac`, `--audio-channels`: Set the number of audio channels. If not specified, the input channel layout will be used. This is equivalent to `-ac` in ffmpeg.\n\n- `-koa, --keep-original-audio`: Copy original, non-normalized audio streams to output file\n\n- `-prf PRE_FILTER, --pre-filter PRE_FILTER`: Add an audio filter chain before applying normalization.\n\n    Multiple filters can be specified by comma-separating them.\n\n- `-pof POST_FILTER, --post-filter POST_FILTER`: Add an audio filter chain after applying normalization.\n\n    Multiple filters can be specified by comma-separating them.\n\n    For EBU, the filter will be applied during the second pass.\n\n### Other Encoding Options\n\n- `-vn, --video-disable`: Do not write video streams to output\n\n- `-c:v VIDEO_CODEC, --video-codec VIDEO_CODEC`: Video codec to use for output files (default: 'copy').\n\n    See `ffmpeg -encoders` for a list.\n\n    Will attempt to copy video codec by default.\n\n- `-sn, --subtitle-disable`: Do not write subtitle streams to output\n\n- `-mn, --metadata-disable`: Do not write metadata to output\n\n- `-cn, --chapters-disable`: Do not write chapters to output\n\n\n### Input/Output Format\n\n- `-ei EXTRA_INPUT_OPTIONS, --extra-input-options EXTRA_INPUT_OPTIONS`: Extra input options list.\n\n    A list of extra ffmpeg command line arguments valid for the input, applied before ffmpeg's `-i`.\n\n    You can either use a JSON-formatted list (i.e., a list of comma-separated, quoted elements within square brackets), or a simple string of space-separated arguments.\n\n    If JSON is used, you need to wrap the whole argument in quotes to prevent shell expansion and to preserve literal quotes inside the string. If a simple string is used, you need to specify the argument with `-e=`.\n\n    Examples: `-ei '[ \"-f\", \"mpegts\", \"-r\", \"24\" ]'` or `-ei=\"-f mpegts -r 24\"`\n\n- `-e EXTRA_OUTPUT_OPTIONS, --extra-output-options EXTRA_OUTPUT_OPTIONS`: Extra output options list.\n\n    A list of extra ffmpeg command line arguments valid for the output.\n\n    You can either use a JSON-formatted list (i.e., a list of comma-separated, quoted elements within square brackets), or a simple string of space-separated arguments.\n\n    If JSON is used, you need to wrap the whole argument in quotes to prevent shell expansion and to preserve literal quotes inside the string. If a simple string is used, you need to specify the argument with `-e=`.\n\n    Examples: `-e '[ \"-vbr\", \"3\", \"-preset:v\", \"ultrafast\" ]'` or `-e=\"-vbr 3 -preset:v ultrafast\"`\n\n- `-ofmt OUTPUT_FORMAT, --output-format OUTPUT_FORMAT`: Media format to use for output file(s).\n\n    See `ffmpeg -formats` for a list.\n\n    If not specified, the format will be inferred by ffmpeg from the output file name. If the output file name is not explicitly specified, the extension will govern the format (see '--extension' option).\n\n- `-ext EXTENSION, --extension EXTENSION`: Output file extension to use for output files that were not explicitly specified. (Default: `mkv`)\n\n### Environment Variables\n\nThe program additionally respects environment variables:\n\n- `TMP` / `TEMP` / `TMPDIR`\n\n    Sets the path to the temporary directory in which files are\n    stored before being moved to the final output directory.\n    Note: You need to use full paths.\n\n- `FFMPEG_PATH`\n\n    Sets the full path to an `ffmpeg` executable other than\n    the system default or you can provide a file name available on $PATH\n\n## API\n\nThis program has a simple API that can be used to integrate it into other Python programs.\n\nFor more information see the [API documentation](https://htmlpreview.github.io/?https://github.com/slhck/ffmpeg-normalize/blob/master/docs/ffmpeg_normalize.html).\n\n## FAQ\n\n### My output file is too large?\n\nThis is because the default output codec is PCM, which is uncompressed. If you want to reduce the file size, you can specify an audio codec with `-c:a` (e.g., `-c:a aac` for ffmpeg's built-in AAC encoder), and optionally a bitrate with `-b:a`.\n\nFor example:\n\n```bash\nffmpeg-normalize input.wav -o output.m4a -c:a aac -b:a 192k\n```\n\n### What options should I choose for the EBU R128 filter? What is linear and dynamic mode?\n\nEBU R128 is a method for normalizing audio loudness across different tracks or programs. It works by analyzing the audio content and adjusting it to meet specific loudness targets. The main components are:\n\n* Integrated Loudness (I): The overall loudness of the entire audio.\n* Loudness Range (LRA): The variation in loudness over time.\n* True Peak (TP): The maximum level of the audio signal.\n\nThe normalization process involves measuring these values (input) and then applying gain adjustments to meet target levels (output), typically -23 LUFS for integrated loudness. You can also specify a target loudness range (LRA) and true peak level (TP).\n\n**Linear mode** applies a constant gain adjustment across the entire audio file. This is generally preferred because:\n\n* It preserves the original dynamic range of the audio.\n* It maintains the relative loudness between different parts of the audio.\n* It avoids potential artifacts or pumping effects that can occur with dynamic processing.\n\n**Dynamic mode**, on the other hand, can change the volume dynamically throughout the file. While this can achieve more consistent loudness, it may alter the original artistic intent and potentially introduce audible artifacts (possibly due to some bugs in the ffmpeg filter).\n\nFor most cases, linear mode is recommended. Dynamic mode should only be used when linear mode is not suitable or when a specific effect is desired. In some cases, `loudnorm` will still fall back to dynamic mode, and a warning will be printed to the console. Here's when this can happen:\n\n* When the input loudness range (LRA) is larger than the target loudness range: If the input file has a loudness range that exceeds the specified loudness range target, the loudnorm filter will automatically switch to dynamic mode. This is because linear normalization alone cannot reduce the loudness range without dynamic processing (limiting). The `--keep-loudness-range-target` option can be used to keep the input loudness range target above the specified target.\n\n* When the required gain adjustment to meet the integrated loudness target would result in the true peak exceeding the specified true peak limit. This is because linear processing alone cannot reduce peaks without affecting the entire signal. For example, if a file needs to be amplified by 6 dB to reach the target integrated loudness, but doing so would push the true peak above the specified limit, the filter might switch to dynamic mode to handle this situation. If your content allows for it, you can increase the true peak target to give more headroom for linear processing. If you're consistently running into true peak issues, you might also consider lowering your target integrated loudness level.\n\nAt this time, the `loudnorm` filter in ffmpeg does not provide a way to force linear mode when the input loudness range exceeds the target or when the true peak would be exceeded. There are some options to mitigate this:\n\n- The `--keep-lra-above-loudness-range-target` option can be used to keep the input loudness range above the specified target, but it will not force linear mode in all cases.\n- Similarly, the `--keep-loudness-range-target` option can be used to keep the input loudness range target.\n- The `--lower-only` option can be used to skip the normalization pass completely if the measured loudness is lower than the target loudness.\n\n### The program doesn't work because the \"loudnorm\" filter can't be found\n\nMake sure you run a recent ffmpeg version and that `loudnorm` is part of the output when you run `ffmpeg -filters`. Many distributions package outdated ffmpeg versions, or (even worse), Libav's `ffmpeg` disguising as a real `ffmpeg` from the FFmpeg project.\n\nSome ffmpeg builds also do not have the `loudnorm` filter enabled.\n\nYou can always download a static build from [their website](http://ffmpeg.org/download.html) and use that.\n\nIf you have to use an outdated ffmpeg version, you can only use `rms` or `peak` as normalization types, but I can't promise that the program will work correctly.\n\n### Should I use this to normalize my music collection?\n\nGenerally, no.\n\nWhen you run `ffmpeg-normalize` and re-encode files with MP3 or AAC, you will inevitably introduce [generation loss](https://en.wikipedia.org/wiki/Generation_loss). Therefore, I do not recommend running this on your precious music collection, unless you have a backup of the originals or accept potential quality reduction. If you just want to normalize the subjective volume of the files without changing the actual content, consider using [MP3Gain](http://mp3gain.sourceforge.net/) and [aacgain](http://aacgain.altosdesign.com/).\n\n### Why are my output files MKV?\n\nI chose MKV as a default output container since it handles almost every possible combination of audio, video, and subtitle codecs. If you know which audio/video codec you want, and which container is supported, use the output options to specify the encoder and output file name manually.\n\n### I get a \"Could not write header for output file\" error\n\nSee the [next section](#the-conversion-does-not-work-and-i-get-a-cryptic-ffmpeg-error).\n\n### The conversion does not work and I get a cryptic ffmpeg error!\n\nMaybe ffmpeg says something like:\n\n> Could not write header for output file #0 (incorrect codec parameters ?): Invalid argument\n\nOr the program says:\n\n> \u2026 Please choose a suitable audio codec with the `-c:a` option.\n\nOne possible reason is that the input file contains some streams that cannot be mapped to the output file, or that you are using a codec that does not work for the output file. Examples:\n\n- You are trying to normalize a movie file, writing to a `.wav` or `.mp3` file. WAV/MP3 files only support audio, not video. Disable video and subtitles with `-vn` and `-sn`, or choose a container that supports video (e.g. `.mkv`).\n\n- You are trying to normalize a file, writing to an `.mp4` container. This program defaults to PCM audio, but MP4 does not support PCM audio. Make sure that your audio codec is set to something MP4 containers support (e.g. `-c:a aac`).\n\nThe default output container is `.mkv` as it will support most input stream types. If you want a different output container, [make sure that it supports](https://en.wikipedia.org/wiki/Comparison_of_container_file_formats) your input file's video, audio, and subtitle streams (if any).\n\nAlso, if there is some other broken metadata, you can try to disable copying over of metadata with `-mn`.\n\nFinally, make sure you use a recent version of ffmpeg. The [static builds](https://ffmpeg.org/download.html) are usually the best option.\n\n### What are the different normalization algorithms?\n\n- **EBU R128** is an EBU standard that is commonly used in the broadcasting world. The normalization is performed using a psychoacoustic model that targets a subjective loudness level measured in LUFS (Loudness Unit Full Scale). R128 is subjectively more accurate than any peak or RMS-based normalization. More info on R128 can be found in the [official document](https://tech.ebu.ch/docs/r/r128.pdf) and [the `loudnorm` filter description](http://k.ylo.ph/2016/04/04/loudnorm.html) by its original author.\n\n- **Peak Normalization** analyzes the peak signal level in dBFS and increases the volume of the input signal such that the maximum in the output is 0 dB (or any other chosen threshold). Since spikes in the signal can cause high volume peaks, peak normalization might still result in files that are subjectively quieter than other, non-peak-normalized files.\n\n- **RMS-based Normalization** analyzes the [RMS power](https://en.wikipedia.org/wiki/Root_mean_square#Average_power) of the signal and changes the volume such that a new RMS target is reached. Otherwise it works similar to peak normalization.\n\n### Couldn't I just run `loudnorm` with ffmpeg?\n\nYou absolutely can. However, you can get better accuracy and linear normalization with two passes of the filter. Since ffmpeg does not allow you to automatically run these two passes, you have to do it yourself and parse the output values from the first run.\n\nIf ffmpeg-normalize is too over-engineered for you, you could also use an approach such as featured [in this Ruby script](https://gist.github.com/kylophone/84ba07f6205895e65c9634a956bf6d54) that performs the two `loudnorm` passes.\n\nIf you want dynamic normalization (the loudnorm default), simply use ffmpeg with one pass, e.g.:\n\n```bash\nffmpeg -i input.mp3 -af loudnorm -c:a aac -b:a 192k output.m4a\n```\n\n### What about speech?\n\nYou should check out the `speechnorm` filter that is part of ffmpeg. It is a designed to be used in one pass, so you don't need this script at all.\n\nSee [the documentation](https://ffmpeg.org/ffmpeg-all.html#speechnorm) for more information.\n\n### After updating, this program does not work as expected anymore!\n\nYou are probably using a 0.x version of this program. There are significant changes to the command line arguments and inner workings of this program, so please  adapt your scripts to the new one. Those changes were necessary to address a few issues that kept piling up; leaving the program as-is would have made it hard to extend it. You can continue using the old version (find it under *Releases* on GitHub or request the specific version from PyPi), but it will not be supported anymore.\n\n### Can I buy you a beer / coffee / random drink?\n\nIf you found this program useful and feel like giving back, feel free to send a donation [via PayPal](https://paypal.me/WernerRobitza).\n\n## Related Tools and Articles\n\n- [Create an AppleScript application to drop or open a folder of files in ffmpeg-normalize](https://prehensileblog.wordpress.com/2022/04/15/create-an-applescript-application-to-drop-or-open-a-folder-of-files-in-ffmpeg-normalize/)\n\n*(Have a link? Please propose an edit to this section via a pull request!)*\n\n## Contributors\n\n<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->\n<!-- prettier-ignore-start -->\n<!-- markdownlint-disable -->\n<table>\n  <tbody>\n    <tr>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://overtag.dk/\"><img src=\"https://avatars.githubusercontent.com/u/374612?v=4?s=100\" width=\"100px;\" alt=\"Benjamin Balder Bach\"/><br /><sub><b>Benjamin Balder Bach</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=benjaoming\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://chaos.social/@eleni\"><img src=\"https://avatars.githubusercontent.com/u/511547?v=4?s=100\" width=\"100px;\" alt=\"Eleni Lixourioti\"/><br /><sub><b>Eleni Lixourioti</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=Geekfish\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/thenewguy\"><img src=\"https://avatars.githubusercontent.com/u/77731?v=4?s=100\" width=\"100px;\" alt=\"thenewguy\"/><br /><sub><b>thenewguy</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=thenewguy\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/aviolo\"><img src=\"https://avatars.githubusercontent.com/u/560229?v=4?s=100\" width=\"100px;\" alt=\"Anthony Violo\"/><br /><sub><b>Anthony Violo</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=aviolo\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://jacobs.af/\"><img src=\"https://avatars.githubusercontent.com/u/952830?v=4?s=100\" width=\"100px;\" alt=\"Eric Jacobs\"/><br /><sub><b>Eric Jacobs</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=jetpks\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/kostalski\"><img src=\"https://avatars.githubusercontent.com/u/34033008?v=4?s=100\" width=\"100px;\" alt=\"kostalski\"/><br /><sub><b>kostalski</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=kostalski\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"http://justinppearson.com/\"><img src=\"https://avatars.githubusercontent.com/u/8844823?v=4?s=100\" width=\"100px;\" alt=\"Justin Pearson\"/><br /><sub><b>Justin Pearson</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=justinpearson\" title=\"Code\">\ud83d\udcbb</a></td>\n    </tr>\n    <tr>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/Nottt\"><img src=\"https://avatars.githubusercontent.com/u/13532436?v=4?s=100\" width=\"100px;\" alt=\"ad90xa0-aa\"/><br /><sub><b>ad90xa0-aa</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=Nottt\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/Mathijsz\"><img src=\"https://avatars.githubusercontent.com/u/1891187?v=4?s=100\" width=\"100px;\" alt=\"Mathijs\"/><br /><sub><b>Mathijs</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=Mathijsz\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/mpuels\"><img src=\"https://avatars.githubusercontent.com/u/2924816?v=4?s=100\" width=\"100px;\" alt=\"Marc P\u00fcls\"/><br /><sub><b>Marc P\u00fcls</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=mpuels\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"http://www.mvbattista.com/\"><img src=\"https://avatars.githubusercontent.com/u/158287?v=4?s=100\" width=\"100px;\" alt=\"Michael V. Battista\"/><br /><sub><b>Michael V. Battista</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=mvbattista\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"http://auto-editor.com\"><img src=\"https://avatars.githubusercontent.com/u/57511737?v=4?s=100\" width=\"100px;\" alt=\"WyattBlue\"/><br /><sub><b>WyattBlue</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=WyattBlue\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/g3n35i5\"><img src=\"https://avatars.githubusercontent.com/u/17593457?v=4?s=100\" width=\"100px;\" alt=\"Jan-Frederik Schmidt\"/><br /><sub><b>Jan-Frederik Schmidt</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=g3n35i5\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/mjhalwa\"><img src=\"https://avatars.githubusercontent.com/u/8994014?v=4?s=100\" width=\"100px;\" alt=\"mjhalwa\"/><br /><sub><b>mjhalwa</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=mjhalwa\" title=\"Code\">\ud83d\udcbb</a></td>\n    </tr>\n    <tr>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/07416\"><img src=\"https://avatars.githubusercontent.com/u/14923168?v=4?s=100\" width=\"100px;\" alt=\"07416\"/><br /><sub><b>07416</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=07416\" title=\"Documentation\">\ud83d\udcd6</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/sian1468\"><img src=\"https://avatars.githubusercontent.com/u/58017832?v=4?s=100\" width=\"100px;\" alt=\"sian1468\"/><br /><sub><b>sian1468</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=sian1468\" title=\"Tests\">\u26a0\ufe0f</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/psavva\"><img src=\"https://avatars.githubusercontent.com/u/1454758?v=4?s=100\" width=\"100px;\" alt=\"Panayiotis Savva\"/><br /><sub><b>Panayiotis Savva</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=psavva\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/HighMans\"><img src=\"https://avatars.githubusercontent.com/u/42877729?v=4?s=100\" width=\"100px;\" alt=\"HighMans\"/><br /><sub><b>HighMans</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=HighMans\" title=\"Code\">\ud83d\udcbb</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://github.com/kanjieater\"><img src=\"https://avatars.githubusercontent.com/u/32607317?v=4?s=100\" width=\"100px;\" alt=\"kanjieater\"/><br /><sub><b>kanjieater</b></sub></a><br /><a href=\"#ideas-kanjieater\" title=\"Ideas, Planning, & Feedback\">\ud83e\udd14</a></td>\n      <td align=\"center\" valign=\"top\" width=\"14.28%\"><a href=\"https://ahmetsait.com/\"><img src=\"https://avatars.githubusercontent.com/u/8372246?v=4?s=100\" width=\"100px;\" alt=\"Ahmet Sait\"/><br /><sub><b>Ahmet Sait</b></sub></a><br /><a href=\"https://github.com/slhck/ffmpeg-normalize/commits?author=ahmetsait\" title=\"Code\">\ud83d\udcbb</a></td>\n    </tr>\n  </tbody>\n  <tfoot>\n    <tr>\n      <td align=\"center\" size=\"13px\" colspan=\"7\">\n        <img src=\"https://raw.githubusercontent.com/all-contributors/all-contributors-cli/1b8533af435da9854653492b1327a23a4dbd0a10/assets/logo-small.svg\">\n          <a href=\"https://all-contributors.js.org/docs/en/bot/usage\">Add your contributions</a>\n        </img>\n      </td>\n    </tr>\n  </tfoot>\n</table>\n\n<!-- markdownlint-restore -->\n<!-- prettier-ignore-end -->\n\n<!-- ALL-CONTRIBUTORS-LIST:END -->\n\n## License\n\nThe MIT License (MIT)\n\nCopyright (c) 2015-2022 Werner Robitza\n\nPermission is hereby granted, free of charge, to any person obtaining a\ncopy of this software and associated documentation files (the\n\"Software\"), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be included\nin all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS\nOR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n\n\n# Changelog\n\n\n## v1.31.0 (2024-12-15)\n\n* Update docs and completions.\n\n* Implement `--auto-lower-loudness-target`\n\n* Fix deprecations and mypy --strict errors.\n\n* Feat: add completions.\n\n* Docs: update explainer.\n\n* Docs: update docs to include lower-only.\n\n\n## v1.30.0 (2024-11-22)\n\n* Change lower-only message to warning.\n\n* Make setup name PEP 625 compliant.\n\n* Docs: add @ahmetsait as a contributor.\n\n* Implement `--lower-only`\n\n* Fix: `--print-stats` only outputs the last stream.\n\n* More robust `loudnorm` output parsing.\n\n* Remove unnecessary conversions.\n\n* Update .editorconfig.\n\n* Remove python 3.8, add python 3.12, 3.13.\n\n* Add README on file size.\n\n\n## v1.29.2 (2024-11-18)\n\n* Fix: show percentage with two decimal digits in progress.\n\n* Chore: add python 12.\n\n\n## v1.29.1 (2024-10-22)\n\n* Fix: override argparse usage.\n\n\n## v1.29.0 (2024-10-14)\n\n* Add option to set audio channels directly.\n\n\n## v1.28.3 (2024-08-16)\n\n* Make colorama dependency windows-only.\n\n* Update.\n\n* Update badges.\n\n* Docs: add @kanjieater as a contributor.\n\n\n## v1.28.2 (2024-06-22)\n\n* Provide an entrypoint for the Docker image.\n\n* Add docker push (#261)\n\n\n## v1.28.1 (2024-05-15)\n\n* Fix assignment of audio statistics, fixes #257.\n\n\n## v1.28.0 (2024-05-13)\n\n* Warn if dynamic mode is used but linear specified (#256)\n\n* Print debug commands in shell-escaped form.\n\n\n## v1.27.7 (2023-09-26)\n\n* Allow cover art in MP3.\n\n\n## v1.27.6 (2023-07-22)\n\n* Remove warning for short files (#87)\n\n\n## v1.27.5 (2023-07-12)\n\n* Building on Windows failed due to character conversion (#244)\n\n\n## v1.27.4 (2023-06-29)\n\n* Docs: add HighMans as a contributor for code (#242)\n\n  * docs: update README.md\n\n  * docs: update .all-contributorsrc\n\n  ---------\n\n* Properly handle non-existing input files and skipping.\n\n* Fix: Dockerfile used wrong path.\n\n* Use `https` rather than `http` (#238)\n\n* Docs: add psavva as a contributor for code (#236)\n\n  * docs: update README.md\n\n  * docs: update .all-contributorsrc\n\n  ---------\n\n* Add Docker support (#235)\n\n* Docs: add @sian1468 as a contributor.\n\n* Round percentage in test (#234)\n\n\n## v1.27.3 (2023-05-06)\n\n* Bump ffmpeg-progress-yield, round percentage output.\n\n* Fix test for progress.\n\n  seems flaky, worked locally?\n\n\n## v1.27.2 (2023-05-05)\n\n* Bump ffmpeg-progress-yield, output progress in percent.\n\n* Docs: add @07416 as a contributor.\n\n* A typo in README (#231)\n\n\n## v1.27.1 (2023-04-25)\n\n* Fix capping to [1, 50] instead of [1, 7] (#230)\n\n* Clarify usage, output/input order.\n\n* Add note on ffmpeg 6.0.\n\n* Docs: add @mjhalwa as a contributor.\n\n\n## v1.27.0 (2023-04-24)\n\n* Constrain input LRA for second pass, addresses #227.\n\n* Feat: add linear loudnorm option to set lra up to target, then keep input lra.\n\n* Update question template.\n\n\n## v1.26.6 (2023-03-16)\n\n* Production status stable.\n\n* Make install_requires more abstract.\n\n\n## v1.26.5 (2023-03-15)\n\n* Add \"-hide_banner\" remove \"-nostdin\" (#222)\n\n  The `-nostdin` option is unnessary because of the `-y` option.\n  Adding `-hide_banner` makes DEBUG statements shorter.\n\n* Homebrew works on linux too.\n\n* Explain ffmpeg installation steps.\n\n\n## v1.26.4 (2023-02-08)\n\n* Re-add requirements.txt to (maybe) fix conda-forge builds.\n\n\n## v1.26.3 (2023-02-08)\n\n* Fix requirements (#218)\n\n* Improve types.\n\n* Docs: add @g3n35i5 as a contributor.\n\n\n## v1.26.2 (2023-02-06)\n\n* Add ignore-revs file.\n\n* Formatting and import sorting.\n\n* Refactor: Improved logging behavior (#216)\n\n* Add \"apt update\" (#215)\n\n* Remove stalebot.\n\n* Update README.\n\n* Move to_ms and make CommandRunner more ergonomic (#212)\n\n* Upgrade workflow, get ffmpeg from apt (#213)\n\n* Turn FFmpegNormalizeError into a normal Exception (#211)\n\n* Remove manifest.in (#210)\n\n* Simplify logging (#209)\n\n* Use pep585 type hints (#207)\n\n* Don't use tempfile's private module function (#206)\n\n* Fix smaller type errors.\n\n* Reduce mypy errors 12 -> 4 (#204)\n\n* Make input validation more efficient.\n\n  Make input validation more efficient\n\n  Re-separate formats and exts\n\n\n## v1.26.1 (2022-12-18)\n\n* Bump requirements.\n\n* Add py.typed support.\n\n* General refactoring + type hints (#202)\n\n* Re-write to f-strings when possible (#201)\n\n* Remove unnecessary utf-8 declarations (#200)\n\n  \"-*- coding: utf-8 -*-\" is a Python 2 construct and can be safely\n  removed. Other utf-8 declarations are also unnecessary.\n\n\n## v1.26.0 (2022-12-14)\n\n* Add .editorconfig.\n\n* Link to API docs.\n\n* Add docs.\n\n* Add type hints, document everything, refactor some code.\n\n* Add more audio formats (#199)\n\n* Add python 3.11 to CI.\n\n* Docs: add WyattBlue as a contributor for code (#198)\n\n  * docs: update README.md\n\n  * docs: update .all-contributorsrc\n\n* Upgrade to Python 3.8 syntax (#197)\n\n* Fix python version in github tests.\n\n* Bump requirements to latest versions.\n\n* Add python 3.11 to list of languages.\n\n* Bump required python version to 3.8.\n\n* Various minor code cleanups and type hints.\n\n* Harmonize logger code.\n\n* Update python version in tests.\n\n* Docs: add @mvbattista as a contributor.\n\n* Docs: add @mpuels as a contributor.\n\n* Docs: add @Mathijsz as a contributor.\n\n* Docs: add @Nottt as a contributor.\n\n* Docs: add @justinpearson as a contributor.\n\n* Docs: add @kostalski as a contributor.\n\n* Docs: add @jetpks as a contributor.\n\n* Docs: add @aviolo as a contributor.\n\n* Docs: add @thenewguy as a contributor.\n\n* Docs: add @Geekfish as a contributor.\n\n* Docs: add @benjaoming as a contributor.\n\n* Reference speechnorm.\n\n\n## v1.25.3 (2022-11-09)\n\n* Update README.\n\n* Update list of pcm-incompatible extensions.\n\n\n## v1.25.2 (2022-09-14)\n\n* Constrain parsed ranges to avoid out of bounds, fixes #189.\n\n* Fix readme for extra-input-options.\n\n* Warn about dynamic mode only if not already set, fixes #187.\n\n\n## v1.25.1 (2022-08-21)\n\n* Add warning in case user specifies both --lrt and --keep-loudness-range-target.\n\n\n## v1.25.0 (2022-08-20)\n\n* Add option to keep loudness range target, fixes #181.\n\n* Only show warning about disabling video if not yet disabled, addresses #184.\n\n\n## v1.24.1 (2022-08-20)\n\n* Code formatting.\n\n* Extend warning for audio-only format to opus, fixes #184.\n\n\n## v1.24.0 (2022-08-02)\n\n* Update python requirements.\n\n* Prevent race condition in output dir creation.\n\n\n## v1.23.1 (2022-07-12)\n\n* Increase possible loudness range target to 50.\n\n\n## v1.23.0 (2022-05-01)\n\n* Add way to force dynamic mode, clarify usage, fixes #176.\n\n\n## v1.22.10 (2022-04-25)\n\n* Add warning for cover art, addresses #174 and #175.\n\n* Update README.\n\n\n## v1.22.9 (2022-04-17)\n\n* Improve issue templates.\n\n* Do not print ffmpeg progress in debug logs.\n\n* Remove unused import.\n\n* Replace which() function with shlex version.\n\n* Add python 3.10 in setup.py.\n\n* Clarify minimum ffmpeg version.\n\n\n## v1.22.8 (2022-03-07)\n\n* Properly detect -inf dB input.\n\n\n## v1.22.7 (2022-02-25)\n\n* Debug command output for ffmpeg commands.\n\n* Remove unneeded warning message.\n\n\n## v1.22.6 (2022-02-20)\n\n* Use astats instead of volumedetect filter, fixes #163.\n\n  Allows floating point calculation.\n\n\n## v1.22.5 (2022-01-25)\n\n* Print warning for bit depths > 16, addresses #163.\n\n\n## v1.22.4 (2021-10-18)\n\n* Re-raise error on ffmpeg command failure.\n\n  This prevents incorrectly telling the user that a normalized file was written when it wasn't.\n\n\n## v1.22.3 (2021-08-31)\n\n* Set tqdm lock for logging only when multiprocessing is available.\n\n  Multiprocessing is not available in all environments, for example\n  on AWS lambda python run time lacks /dev/shm, so trying to acquire\n  a multiprocessing Lock throws an OSError. The module could also be\n  missing in some cases (ex. Jython, although this library doesn't support\n  Jython anyway).\n\n  The solution to this is to only try to set the lock when multiprocessing\n  is available. The tqdm library solves this in the same manner.\n\n  For more details: https://github.com/slhck/ffmpeg-normalize/issues/156\n\n* Add instructions on how to run tests.\n\n\n## v1.22.2 (2021-08-14)\n\n* Bump requirements, should fix #155.\n\n* Move all examples to Wiki.\n\n* Update badge link.\n\n\n## v1.22.1 (2021-03-10)\n\n* Add python_requires to setup.py.\n\n\n## v1.22.0 (2021-03-09)\n\n* Improve README.\n\n* Add GitHub actions badge.\n\n* Add GitHub actions tests.\n\n* Properly convert EBU JSON values to float.\n\n* Switch to f strings, remove Python 3.5 support.\n\n* Format code with black.\n\n* Fix flake8 errors.\n\n* Factor out method.\n\n* WIP: new tests.\n\n* Log to stderr by default to enable JSON parsing.\n\n* Remove release script.\n\n\n## v1.21.2 (2021-03-06)\n\n* Format setup.py.\n\n* Switch progress to external lib.\n\n* Remove support for older versions.\n\n\n## v1.21.1 (2021-03-05)\n\n* Adjusted handling of FFMPEG_PATH for binaries available via $PATH (#149)\n\n  * adjusted handling of FFMPEG_PATH for binaries available via $PATH\n\n  fixes #147\n\n  * adjusted use of %s to {} to match style\n\n  * documented the feature\n\n  * condensed error message as other lines are longer\n\n\n## v1.21.0 (2021-02-27)\n\n* Fix JSON output for multiple files.\n\n* Update badge URL.\n\n* Update README.md (#142)\n\n  * Update README.md\n\n  Added example of verifying levels\n\n  Fixes #141\n\n  * shorten example, add link to wiki page\n\n* Error if no ffmpeg exec exists.\n\n* Add stalebot.\n\n\n## v1.20.2 (2020-11-06)\n\n* Fixing stdin corruption caused by new subprocess (#138)\n\n* Update issue template.\n\n* Create FUNDING.yml.\n\n* Fix usage, addresses #132.\n\n\n## v1.20.1 (2020-07-22)\n\n* Manually specify usage string, fixes #132.\n\n* Fix local import for tests.\n\n\n## v1.20.0 (2020-07-04)\n\n* Add extra input options.\n\n\n## v1.19.1 (2020-06-25)\n\n* Add colorama to requirements, fixes #131.\n\n* Fix warning that is printed with default options.\n\n\n## v1.19.0 (2020-05-02)\n\n* Fix issue with output folder, fixes #126.\n\n* Fix typo in README's table of contents link to \"File Input/Output\". (#124)\n\n* Clarify readme, fixes #122.\n\n\n## v1.18.2 (2020-04-19)\n\n* Add warning for automatic sample rate conversion, addresses #122.\n\n* Ignore vscode folder.\n\n* Fix printing of errors in conversion.\n\n\n## v1.18.1 (2020-04-16)\n\n* Fix unit tests.\n\n* Improve handling of output file folder and errors.\n\n* Clarify usage of output options, add warning.\n\n* Improve documentation, fixes #120.\n\n* Do not include bump messages in changelog.\n\n\n## v1.18.0 (2020-04-13)\n\n* Use measured offset in second pass, fixes #119.\n\n* Update release instructions.\n\n* Remove author names from changelog.\n\n\n## v1.17.0 (2020-04-10)\n\n* Update release script and changelog template.\n\n* Apply pre-filters in all first passes, fixes #118.\n\n  This allows properly reading the level for any kind of normalization, even if\n  filters affect the loudness in the first pass.\n\n\n## v1.16.0 (2020-04-07)\n\n* Add all commits to changelog.\n\n* Remove python 2 support.\n\n* Add quiet option, fixes #116.\n\n  - Add a new quiet option\n  - Promote some warnings to actual errors that need to be shown\n  - Add a very basic test case\n\n\n## v1.15.8 (2020-03-15)\n\n* Improve release script.\n\n* Python 3.8.\n\n\n## v1.15.7 (2020-03-14)\n\n* Only print length warning for non-EBU type normalization.\n\n\n## v1.15.6 (2019-12-04)\n\n* Remove build and dist folder on release.\n\n* Do not exit on error in batch processing.\n\n  Simply process the next file if one has errors, addresses #110.\n\n\n## v1.15.5 (2019-11-19)\n\n* Use minimal dependency for tqdm.\n\n* Remove specific python version requirement.\n\n\n## v1.15.4 (2019-11-19)\n\n* Freeze tqdm version.\n\n* Update python to 3.7.\n\n* Improve release documentation.\n\n\n## v1.15.3 (2019-10-15)\n\n* Do not print stream warning when there is only one stream.\n\n* Remove previous dist versions before release.\n\n\n## v1.15.2 (2019-07-12)\n\n* Warn when duration cannot be read, fixes #105.\n\n* Update README.\n\n  minor improvements in the description\n\n\n## v1.15.1 (2019-06-17)\n\n* Add output to unit test failures.\n\n* Fix input label for audio stream.\n\n\n## v1.15.0 (2019-06-17)\n\n* Add pre-and post-filter hooks, fixes #67.\n\n  This allows users to specify filters to be run before or after the actual\n  normalization call, using regular ffmpeg syntax.\n  Only applies to audio.\n\n* Document audiostream class.\n\n* Warn when file is too short, fixes #87.\n\n* Update release method to twine.\n\n\n## v1.14.1 (2019-06-14)\n\n* Handle progress output from ffmpeg, fixes #10.\n\n* Merge pull request #99 from Nottt/patch-1.\n\n  fix -cn description\n\n* Fix -cn description.\n\n* Add nicer headers for options in README.\n\n\n## v1.14.0 (2019-04-24)\n\n* Add version file in release script before committing.\n\n* Add option to keep original audio, fixes #83.\n\n* Add pypi badge.\n\n* Allow release script to add changelog for future version; upload to pypi.\n\n\n## v1.13.11 (2019-04-16)\n\n* Add release script.\n\n* Add small developer guide on releasing.\n\n* Move HISTORY.md to CHANGELOG.md.\n\n* Fix ffmpeg static build download location.\n\n\n## v1.3.10 (2019-02-22)\n\n* Bump version.\n\n* Cap measured input loudness, fixes #92.\n\n\n## v1.3.9 (2019-01-10)\n\n* Bump version.\n\n* Fix handling of errors with tqdm.\n\n* Improve readme.\n\n* Delete issue template.\n\n* Bump version.\n\n* Clarify extra argument options, move to main entry point.\n\n* Update issue templates.\n\n\n## v1.3.8 (2018-11-28)\n\n* Bump version.\n\n* Clarify extra argument options, move to main entry point.\n\n\n## v1.3.7 (2018-10-28)\n\n* Bump version.\n\n* Copy metadata from individual streams, fixes #86.\n\n* Add python version for pyenv.\n\n\n## v1.3.6 (2018-07-09)\n\n* Bump version.\n\n* Update README, fixes #79 and addresses #80.\n\n\n## v1.3.5 (2018-06-12)\n\n* Bump version.\n\n* Minor README updates.\n\n* Fix documentation of TMPDIR parameter.\n\n\n## v1.3.4 (2018-05-05)\n\n* Bump version.\n\n* New way to specify extra options.\n\n\n## v1.3.3 (2018-05-05)\n\n* Update README.\n\n* Decode strings in extra options.\n\n\n## v1.3.2 (2018-04-25)\n\n* Bump version.\n\n* Merge pull request #69 from UbiCastTeam/master.\n\n  Stderror decoding ignoring utf8 encoding errors\n\n* Stderror decoding ignoring utf8 encoding errors.\n\n\n## v1.3.1 (2018-04-24)\n\n* Bump version.\n\n* Do not require main module in setup.py, fixes #68.\n\n\n## v1.3.0 (2018-04-15)\n\n* Bump version.\n\n* Remove dead code.\n\n* Fix for python2 division.\n\n* Update documentation.\n\n* Progress bar.\n\n* Remove imports from test file.\n\n* Fix travis file.\n\n* WIP: progress bar.\n\n* Minor typo in option group.\n\n* Add simple unit test for disabling chapters.\n\n\n## v1.2.3 (2018-04-11)\n\n* Fix unit test.\n\n* Bump version.\n\n* Add option to disable chapters, fixes #65, also fix issue with metadata.\n\n\n## v1.2.2 (2018-04-10)\n\n* Bump version.\n\n* Set default loudness target to -23, fixes #48.\n\n\n## v1.2.1 (2018-04-04)\n\n* Bump version.\n\n* Merge pull request #64 from UbiCastTeam/encoding-issue.\n\n  Stdout and stderror decoding ignoring utf8 encoding errors\n\n* Stdout and stderror decoding ignoring utf8 encoding errors.\n\n\n## v1.2.0 (2018-03-22)\n\n* Bump version.\n\n* Add errors for impossible format combinations, fixes #60.\n\n* Fix ordering of output maps, fixes #63.\n\n* Improve documentation.\n\n\n## v1.1.0 (2018-03-06)\n\n* Add option to print first pass statistics.\n\n\n## v1.0.10 (2018-03-04)\n\n* Bump version.\n\n* Restrict parsing to valid JSON part only, fixes #61.\n\n* Add an example for MP3 encoding.\n\n* Update paypal link.\n\n\n## v1.0.9 (2018-02-08)\n\n* Bump version.\n\n* Add normalized folder to gitignore.\n\n* Do not print escape sequences on Windows.\n\n* Do not check for file existence, fixes #57.\n\n* Add github issue template.\n\n\n## v1.0.8 (2018-02-01)\n\n* Bump version.\n\n* Do not check for ffmpeg upon module import.\n\n\n## v1.0.7 (2018-02-01)\n\n* Bump version.\n\n* Rename function test.\n\n* Fix issue with wrong adjustment parameters, fixes #54.\n\n\n## v1.0.6 (2018-01-30)\n\n* Allow setting FFMPEG_PATH and document TMP.\n\n\n## v1.0.5 (2018-01-26)\n\n* Handle edge case for short input clips.\n\n\n## v1.0.4 (2018-01-26)\n\n* Bump version.\n\n* Do not try to remove nonexisting file in case of error in command.\n\n\n## v1.0.3 (2018-01-26)\n\n* Bump version.\n\n* Always streamcopy when detecting streams to avoid initializing encoder.\n\n* Fix handling of temporary file.\n\n* Add build status.\n\n* Travis tests.\n\n\n## v1.0.2 (2018-01-25)\n\n* Fix bug with target level for peak/RMS.\n\n* Update documentation formatting.\n\n* Update history.\n\n\n## v1.0.1 (2018-01-24)\n\n* Bump version.\n\n* Set default target to -23.\n\n\n## v1.0.0 (2018-01-23)\n\n* Add version info and test case for dry run.\n\n* New feature detection, add documentation, contributors guide etc.\n\n* WIP: v1.0 rewrite.\n\n\n## v0.7.3 (2017-10-09)\n\n* Use shutil.move instead of os.rename.\n\n\n## v0.7.2 (2017-09-17)\n\n* Allow setting threshold to 0.\n\n\n## v0.7.1 (2017-09-14)\n\n* Bump version.\n\n* Update HISTORY.md.\n\n* Merge pull request #37 from Mathijsz/fix-which-path-expansion.\n\n  expand tilde and environment variables, fixes #36\n\n* Expand tilde and environment variables, fixes #36.\n\n* Update HISTORY.md.\n\n* Update README w.r.t. loudnorm filter.\n\n* Update README and indentation.\n\n\n## v0.7.0 (2017-08-02)\n\n* Bump version.\n\n* Fix handling of extra options with spaces.\n\n* Include test script.\n\n* Logging and other improvements.\n\n* Add test files.\n\n* Autopep8 that thing.\n\n* Logger improvements.\n\n* Add example for overwriting.\n\n\n## v0.6.0 (2017-07-31)\n\n* Allow overwriting input file, fixes #22.\n\n* Version bump.\n\n* Better handle cmd arguments.\n\n* Update README.md.\n\n  add another example\n\n\n## v0.5.1 (2017-04-04)\n\n* Fix for problem introduced in 304e8df.\n\n\n## v0.5 (2017-04-02)\n\n* Fix pypi topics.\n\n* Bump version and README.\n\n* Fix issue where output was wrong format.\n\n* Add EBU R128 filter.\n\n* Use Markdown instead of RST for README/HISTORY.\n\n* Define file encode for python3, fixes #24.\n\n* Fix history.\n\n* Fix option -np.\n\n* Clarify merge option.\n\n* Minor documentation improvements.\n\n  - change README from CRLF to LF\n  - add \"attenuated\" in description\n  - extend LICENSE year\n  - add license to main README\n\n\n## v0.4.1 (2017-02-13)\n\n* Update for release.\n\n* Merge pull request #21 from mpuels/patch-1.\n\n  Fix for #13\n\n* Fix for #13.\n\n* Mention Python 3.\n\n  mention that Python 3 may work, just didn't have time to test\n\n* Fix README's code blocks.\n\n\n## v0.4 (2017-01-24)\n\n* Code cleanup, add option to set format and audio codec.\n\n\n## v0.3 (2017-01-19)\n\n* Add option for no prefix, fixes #20.\n\n* Handle multiple spaces in path; fixes issue #18.\n\n* Handle spaces in path, fixes #12.\n\n* Update README.rst.\n\n* Change default level back  to -26.\n\n* Typo in README example.\n\n* Update documentation.\n\n* Bump to v0.2.0.\n\n  * Support for multiple files and output directories.\n  * Support merging of audio with input file\n  * Set audio codec and additional options\n  * User-definable threshold\n  * Better error handling and logging\n  * Deprecates avconv\n\n* Change default level back to -28.\n\n* Merge pull request #15 from auricgoldfinger/master.\n\n  Add extended normalisation options\n\n* Add extended normalisation options.\n\n  - add program option to write output in a separate directory in stead of\n     prefixing it\n\n  - add program option to merge the normalized audio in the original\n     (video) file rather than creating a separate WAV file\n\n  - change the maximum setting: will now normalize so that max\n     volume is set to 0, adjusted with the given level.\n     e.g. : -m -l -5 will increase the audio level to max = -5.0dB\n\n  - improve verbose logging: number of files are written to the\n     info log\n\n  - improve performance: check first whether the output file\n     exists before calculating the volume levels + not modifying\n     the file if the adjustment < 0.5dB (level is never exactly 0)\n\n* Update README, fixes #11.\n\n\n## v0.1.3 (2015-12-15)\n\n* Check for Windows .exe, fixes #10.\n\n* Check path and fix #9.\n\n* Merge pull request #8 from benjaoming/master.\n\n  Add MANIFEST.in\n\n* Bump version.\n\n* Add manifest to include missing files in sdist.\n\n* Merge pull request #6 from jetpks/master.\n\n  Fixed ffmpeg v2.6.3 compatibility and docopt config\n\n* Updated to work with ffmpeg v2.6.3, and fixed broken docopt config.\n\n  ffmpeg update:\n\n  ffmpeg v2.6.3 puts mean_volume on stderr instead of stdout, causing\n  `output` in `ffmpeg_get_mean` to be completely empty, and no match for\n  mean_volume or max_volume to be found.\n\n  Fixed by adding `stderr=subprocess.PIPE` in both Popen calls in\n  `run_command`, and combining stdout and stderr on return. We already\n  exit with non-zero return, so combining stderr/stdout shouldn't cause\n  any poor side-effects.\n\n  docopt config:\n\n  - args['--level'] was not recognizing its default because there was\n    an errant comma between -l and --level, and it needed <level> after\n    the arguments.\n  - Fixed spacing for --max\n  - Removed quotes around 'normalized' so single quote characters don't\n    end up in the output file names.\n\n* Removed Windows carraige returns from __main__.py.\n\n* Merge pull request #5 from mvbattista/master.\n\n  Installation update to ffmpeg\n\n* Installation update to ffmpeg.\n\n* Update to ffmpeg.\n\n* Update HISTORY.rst.\n\n* Update to ffmpeg.\n\n* Merge pull request #4 from benjaoming/rename.\n\n  Rename project\n\n* Make at least one file mandatory.\n\n* Rename project and remove pyc file.\n\n* Merge pull request #2 from benjaoming/docopt-setuptools-avconv.\n\n  Docopt, Setuptools, avconv compatibility\n\n* Use docopt.\n\n* Use normalize-audio when using avconv because it doesn't have a way to measure volume.\n\n* Functional setup.py, communicate with avconv/ffmpeg about overwriting.\n\n* Also detect avconv.\n\n* Use a main function instead.\n\n* Add a history for the project.\n\n* Move to more unique module name.\n\n* Update README.rst.\n\n* Change the README to rst (PyPi)\n\n* Delete .gitignore.\n\n* Update README.md.\n\n* Various improvements, fixes #1.\n\n* License.\n\n* Livense.\n\n* Update README.md.\n\n* Merge branch 'master' of https://github.com/slhck/audio-normalize.\n\n* Initial commit.\n\n* Initial commit.\n\n\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Normalize audio via ffmpeg",
    "version": "1.31.0",
    "project_urls": {
        "Homepage": "https://github.com/slhck/ffmpeg-normalize"
    },
    "split_keywords": [
        "ffmpeg",
        " normalize",
        " audio"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "685d5681d488635f4db61d5c87d97e23f75e137852252a9e7d3eae6c5f8c8442",
                "md5": "7221fa1e771e545c964c37b0587fbd22",
                "sha256": "88f556b468ac43d4eb5707ac3c0ab382a7a05c1f0048ca5d6477e8ccb6809810"
            },
            "downloads": -1,
            "filename": "ffmpeg_normalize-1.31.0-py2.py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "7221fa1e771e545c964c37b0587fbd22",
            "packagetype": "bdist_wheel",
            "python_version": "py2.py3",
            "requires_python": ">=3.9",
            "size": 42529,
            "upload_time": "2024-12-15T19:51:40",
            "upload_time_iso_8601": "2024-12-15T19:51:40.300223Z",
            "url": "https://files.pythonhosted.org/packages/68/5d/5681d488635f4db61d5c87d97e23f75e137852252a9e7d3eae6c5f8c8442/ffmpeg_normalize-1.31.0-py2.py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "f67fba9076fd5b1090631a6d8ee7e701ef5f480646c933bcfe045c64b7c21162",
                "md5": "d6d3a88bbb18d59785d7b47dccc78c99",
                "sha256": "7d1b1ddc0cf2febe558663b613ceb365b27fbe89f41be35d39894b6267041539"
            },
            "downloads": -1,
            "filename": "ffmpeg_normalize-1.31.0.tar.gz",
            "has_sig": false,
            "md5_digest": "d6d3a88bbb18d59785d7b47dccc78c99",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.9",
            "size": 1991748,
            "upload_time": "2024-12-15T19:51:43",
            "upload_time_iso_8601": "2024-12-15T19:51:43.730947Z",
            "url": "https://files.pythonhosted.org/packages/f6/7f/ba9076fd5b1090631a6d8ee7e701ef5f480646c933bcfe045c64b7c21162/ffmpeg_normalize-1.31.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-12-15 19:51:43",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "slhck",
    "github_project": "ffmpeg-normalize",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "tqdm",
            "specs": [
                [
                    ">=",
                    "4.64.1"
                ]
            ]
        },
        {
            "name": "colorama",
            "specs": [
                [
                    ">=",
                    "0.4.6"
                ]
            ]
        },
        {
            "name": "ffmpeg-progress-yield",
            "specs": [
                [
                    ">=",
                    "0.11.0"
                ]
            ]
        },
        {
            "name": "colorlog",
            "specs": [
                [
                    "==",
                    "6.7.0"
                ]
            ]
        }
    ],
    "lcname": "ffmpeg-normalize"
}
        
Elapsed time: 4.58818s