Проблемы с частотой кадров преобразования RTMP в HLS при использовании ffmpeg/nginx
В настоящее время я отправляю поток RTMP на nginx 1.15.3, работающий на Ubuntu 18.04 LTS, с камеры Raspberry Pi 3 (которая, кажется, работает хорошо сама по себе, я могу получить поток без проблем на VLC и убедиться, что он работает) с пи с помощью этой команды:
raspivid -w 640 -h 480 -fps 5 -t 0 -b 1800000 -o - | ffmpeg -y -framerate 5 -f h264 -i - -c:v copy -map 0:0 -f flv -rtmp_buffer 100 -rtmp_live live rtmp://10.2.10.149/cam01/test
Если бы я должен был установить аргумент fps для raspivid равным 25, а затем избавиться от аргумента ffmpeg framerate, то это было бы просто отлично преобразовать его в поток HLS, выполнив эту команду на сервере nginx:
ffmpeg -i rtmp://10.2.10.149/cam01/test -vcodec libx264 -vprofile baseline -acodec h264 -strict -2 -f flv rtmp://10.2.10.149/show1/stream1
Тем не менее, он мне нужен на более низкой частоте кадров, и кажется, что попытка сделать это каким-либо образом действительно расстраивает ffmpeg, потому что это заставит поток много зависать, как будто он ожидает загрузки. Я подозреваю, что это как-то связано с тем, что ffmpeg имеет частоту кадров по умолчанию 25 или что-то еще, потому что в потоке RTMP, если бы я попытался установить только аргумент fps в raspivid равным 5 и не включить аргумент framerate в ffmpeg, то это сделайте видео на несколько секунд, а затем запустите на 5 раз больше скорости видео на секунду, так как ffmpeg все еще пытается запустить видео с частотой 25 кадров в секунду.
Использование аргумента framerate в команде преобразования ffmpeg, кажется, делает ситуацию еще хуже, и я не понимаю, почему это не работает, так как кажется, что это отлично работает для многих других людей. В основном я использовал это руководство, чтобы попытаться помочь мне в настройке nginx и еще чего: https://docs.peer5.com/guides/setting-up-hls-live-streaming-server-using-nginx/
Я также собираюсь разместить свой nginx.conf здесь, чтобы все могли его увидеть, и заметьте, что у меня возникает эта проблема всякий раз, когда я пытаюсь вытащить видео через веб-страницу или с VLC, напрямую получающим сетевой поток:
worker_processes auto;
events {
worker_connections 1024;
}
# RTMP configuration
rtmp {
server {
listen 1935; # Listen on standard RTMP port
chunk_size 4096;
application show1 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls1/;
hls_fragment 3;
hls_playlist_length 60;
}
application show2 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls2/;
hls_fragment 3;
hls_playlist_length 60;
}
application show3 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls3/;
hls_fragment 3;
hls_playlist_length 60;
}
application show4 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls4/;
hls_fragment 3;
hls_playlist_length 60;
}
application show5 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls5/;
hls_fragment 3;
hls_playlist_length 60;
}
application show6 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls6/;
hls_fragment 3;
hls_playlist_length 60;
}
application show7 {
live on;
# Turn on HLS
hls on;
hls_path /mnt/hls7/;
hls_fragment 3;
hls_playlist_length 60;
}
application cam01 {
live on;
record off;
}
application cam02 {
live on;
record off;
}
application cam03 {
live on;
record off;
}
application cam04 {
live on;
record off;
}
application cam05 {
live on;
record off;
}
application cam06 {
live on;
record off;
}
application cam07 {
live on;
record off;
}
}
}
http {
sendfile off;
tcp_nopush on;
default_type application/octet-stream;
server {
listen 80;
server_name localhost;
location / {
# Disable cache
add_header 'Cache-Control' 'no-cache';
# CORS setup
add_header 'Access-Control-Allow-Origin' '*' always;
add_header 'Access-Control-Expose-Headers' 'Content-Length';
# allow CORS preflight requests
if ($request_method = 'OPTIONS') {
add_header 'Access-Control-Allow-Origin' '*';
add_header 'Access-Control-Max-Age' 1728000;
add_header 'Content-Type' 'text/plain charset=UTF-8';
add_header 'Content-Length' 0;
return 204;
}
types {
application/dash+xml mpd;
application/vnd.apple.mpegurl m3u8;
video/mp2t ts;
}
root /mnt/;
location /index.html {
default_type "text/html";
}
location /testing.html {
default_type "text/html";
}
location /test.html {
default_type "text/html";
}
location /cam01.html {
default_type "text/html";
}
location /cam02.html {
default_type "text/html";
}
location /cam03.html {
default_type "text/html";
}
location /cam04.html {
default_type "text/html";
}
location /cam05.html {
default_type "text/html";
}
location /cam06.html {
default_type "text/html";
}
location /cam07.html {
default_type "text/html";
}
}
}
}
Если мне нужно будет включить больше деталей, не стесняйтесь, дайте мне знать, мне просто нужно, чтобы это работало должным образом вместо всех вопросов, которые я только что объяснил, любая помощь будет оценена, спасибо.
Редактировать 1: Я также замечаю, что поверх зацепок и вырезов из потока, используя вторую команду, как описано ранее, иногда кажется, что она "застревает" в определенном кадре, не уверенный, является ли это более полезным или нет Я просто хотел убедиться, что у меня есть вся информация, которую я могу найти здесь.
Редактировать 2: Я начинаю верить, что это больше проблема с nginx, потому что, когда я указываю первую команду на местоположение / show1 / stream1, которое преобразует rtmp в hls, не выполняя вторую команду, которую я счел абсолютно необходимой, я Я получаю точно такой же вопрос точно так же. Я не очень разбираюсь в nginx, и ничто из того, что я нахожу в Интернете, тоже не сильно помогает. Буду очень признателен за понимание, я знаю, что hls всегда будет иметь большую задержку, и это не то, что я прошу исправить, я прошу исправить вырезы и сцепление.
Редактировать 3: Я, возможно, на самом деле несколько взломал это, я заметил по команде raspivid, что установка битрейта, казалось, действительно смягчает это, однако, даже если я установил битрейт на абсурдный или даже не настолько абсурдный, но все же избыточный объем иногда каждые несколько минут случайно вырезают несколько кадров, и я не совсем уверен, почему. Был бы признателен за любое понимание.
1 ответ
Извините всех, кого я запутал, оказалось, что этот код верен (на самом деле, мне даже не нужна вторая команда, я могу просто передать ее напрямую в hls в nginx, без проблем с pi). Что происходит, так это то, что ffmpeg ДЕЙСТВИТЕЛЬНО не нравится принимать менее 7 FPS при выполнении такого рода потоковой передачи, которая по-прежнему достаточна для моих целей. Это должно быть честно задокументировано где-то (ну, я думаю, что это сейчас, но все же).
Изменить: камера, которую я использую, также может повлиять на это, так как я только что понял, что некоторые из моих камер имеют минимальную частоту кадров 7.