Как определить частоту дискретизации аудио с помощью avprobe / ffprobe?
Я использую libav 9.6, установленную через Homebrew.
$ avprobe -version
avprobe version 9.6, Copyright (c) 2007-2013 the Libav developers
built on Jun 8 2013 02:44:19 with Apple LLVM version 4.2 (clang-425.0.24) (based on LLVM 3.2svn)
avprobe 9.6
libavutil 52. 3. 0 / 52. 3. 0
libavcodec 54. 35. 0 / 54. 35. 0
libavformat 54. 20. 3 / 54. 20. 3
libavdevice 53. 2. 0 / 53. 2. 0
libavfilter 3. 3. 0 / 3. 3. 0
libavresample 1. 0. 1 / 1. 0. 1
libswscale 2. 1. 1 / 2. 1. 1
Даже если частота дискретизации отображается в stdout в выходных данных командной строки, -show_format
опция вообще не отображает информацию о частоте дискретизации для аудиофайла.
Вот вывод терминала BASH:
$ avprobe -v verbose -show_format -of json sample.gsm
avprobe version 9.6, Copyright (c) 2007-2013 the Libav developers
built on Jun 8 2013 02:44:19 with Apple LLVM version 4.2 (clang-425.0.24)
(based on LLVM 3.2svn)
configuration: --prefix=/usr/local/Cellar/libav/9.6 --enable-shared
--enable-pthreads --enable-gpl --enable-version3 --enable-nonfree
--enable-hardcoded-tables --enable-avresample --enable-vda --enable-gnutls
--enable-runtime-cpudetect --disable-indev=jack --cc=cc --host-cflags=
--host-ldflags= --enable-libx264 --enable-libfaac --enable-libmp3lame
--enable-libxvid --enable-avplay
libavutil 52. 3. 0 / 52. 3. 0
libavcodec 54. 35. 0 / 54. 35. 0
libavformat 54. 20. 3 / 54. 20. 3
libavdevice 53. 2. 0 / 53. 2. 0
libavfilter 3. 3. 0 / 3. 3. 0
libavresample 1. 0. 1 / 1. 0. 1
libswscale 2. 1. 1 / 2. 1. 1
[gsm @ 0x7f8012806600] Estimating duration from bitrate, this may be inaccurate
Input #0, gsm, from 'sample.gsm':
Duration: 00:03:52.32, start: 0.000000, bitrate: 13 kb/s
Stream #0.0: Audio: gsm, 8000 Hz, mono, s16, 13 kb/s
{ "format" : {
"filename" : "sample.gsm",
"nb_streams" : 1,
"format_name" : "gsm",
"format_long_name" : "raw GSM",
"start_time" : "0.000000",
"duration" : "232.320000",
"size" : "383328.000000",
"bit_rate" : "13200.000000"
}}
И пример кода Python:
>>> filename = 'sample.gsm'
>>> result = subprocess.check_output(['avprobe', '-show_format', '-of',
'json', filename])
avprobe version 9.6, Copyright (c) 2007-2013 the Libav developers
built on Jun 8 2013 02:44:19 with Apple LLVM version 4.2
(clang-425.0.24) (based on LLVM 3.2svn)
[gsm @ 0x7fe0b1806600] Estimating duration from bitrate, this may be
inaccurate
Input #0, gsm, from 'sample.gsm':
Duration: 00:03:52.32, start: 0.000000, bitrate: 13 kb/s
Stream #0.0: Audio: gsm, 8000 Hz, mono, s16, 13 kb/s
>>> print result
{ "format" : {
"filename" : "sample.gsm",
"nb_streams" : 1,
"format_name" : "gsm",
"format_long_name" : "raw GSM",
"start_time" : "0.000000",
"duration" : "232.320000",
"size" : "383328.000000",
"bit_rate" : "13200.000000"
}}
Таким образом, я знаю, что частота дискретизации может быть отображением конкретного потока, который будет показан в -show_format
Вариант результатов. Но других вариантов определения частоты дискретизации в конкретном аудиопотоке нет, хотя можно установить ее с помощью -ar
при перекодировании.
Я подал заявку в libav, но мне просто любопытно, есть ли какой-нибудь другой способ извлечь частоту дискретизации из утилит libav probing. Я ценю ответ заранее.
PS: это был бы тот же вопрос для исходного проекта ffmpeg (ffprobe) в этом случае.
4 ответа
-show_format
показывает информацию уровня контейнера - то есть материал, который применяется ко всем потокам. Частота дискретизации является свойством отдельного потока, поэтому вполне нормально, что -show_format
не отображать это. Вам нужно использовать -show_streams
,
Я обнаружил, что библиотека json может анализировать вывод ffprobe в словарь, и разработал ваш код для хранения информации внутри Python. Вот функция, которая делает это и при желании печатает информацию о носителе:
from json import loads
from subprocess import check_output
def get_media_info(filename, print_result=True):
"""
Returns:
result = dict with audio info where:
result['format'] contains dict of tags, bit rate etc.
result['streams'] contains a dict per stream with sample rate, channels etc.
"""
result = check_output(['ffprobe',
'-hide_banner', '-loglevel', 'panic',
'-show_format',
'-show_streams',
'-of',
'json', filename])
result = json.loads(result)
if print_result:
print('\nFormat')
for key, value in result['format'].items():
print(' ', key, ':', value)
print('\nStreams')
for stream in result['streams']:
for key, value in stream.items():
print(' ', key, ':', value)
print('\n')
return result
Следуя user3275163 , мы можем сделать то же самое в оболочке, используя
jq
(который анализирует json).
ffprobe -hide_banner -loglevel panic -show_format -show_streams -of json input.wav | \
jq '.streams[0].sample_rate'
"44100"
Это должно работать даже для видео:
ffprobe -v error -select_streams a -of default=noprint_wrappers=1:nokey=1 -show_entries stream=sample_rate input.wav