Rtsp live stream gets delayed overtime

I'm using @kevinGodell his nodes for my cctv cameras now for a couple of years with great joy.
Bought myself an Reolink doorbel last week and integrated (rtsp stream) this also with Kevin his nodes in my dashboard and Bart his Onvif node to capture bell presses. Nice.
Works all great, however during the day the live stream gets delay with 20sec or something like that, delay will increase overtime.
20 sec delay is to much to see "live" who is in front of the door.
Restarting the ffmpeg node with the restart command will solve this issue and the live feed is back at his "normal" 5 sec delay.
Is there a better / neater way to get the everything running on time?

[{"id":"5654e6d653ad2b72","type":"inject","z":"44908087.da8e9","name":"stop","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":390,"y":4680,"wires":[["7a8c3f95b486b8f9"]]},{"id":"3db9c9f03c8f6cae","type":"inject","z":"44908087.da8e9","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":390,"y":4560,"wires":[["7a8c3f95b486b8f9"]]},{"id":"30bbbddd32d43bde","type":"debug","z":"44908087.da8e9","name":"stderr","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":820,"y":4780,"wires":[]},{"id":"4cc7afbbcc802243","type":"mp4frag","z":"44908087.da8e9","name":"","outputs":2,"basePath":"ui_mp4frag_01","serveHttp":"true","serveIo":"true","hlsPlaylistSize":"6","hlsPlaylistExtra":"2","autoStart":"false","preBuffer":1,"timeLimit":10000,"repeated":"false","statusData":"playlist","x":890,"y":4620,"wires":[["91780c25f9e1edae"],[]]},{"id":"91780c25f9e1edae","type":"ui_mp4frag","z":"44908087.da8e9","name":"Camera Deur","group":"7ea407c1b15941c7","order":1,"width":"21","height":"11","readyPoster":"","errorPoster":"","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":5,\"maxBufferLength\":10,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","autoplay":"true","unload":"true","threshold":"0.1","controls":"false","muted":"false","players":["socket.io","hls.js","hls","mp4"],"x":920,"y":4500,"wires":[[]]},{"id":"7a8c3f95b486b8f9","type":"ffmpeg","z":"44908087.da8e9","name":"","outputs":3,"cmdPath":"ffmpeg","cmdArgs":"[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://nvr:12345@192.168.1.95:554/h264Preview_01_sub\",\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]","cmdOutputs":2,"killSignal":"SIGTERM","credentials":{},"x":630,"y":4640,"wires":[["4cc7afbbcc802243"],["4cc7afbbcc802243"],["30bbbddd32d43bde"]]},{"id":"71f89f0fcd720dc8","type":"inject","z":"44908087.da8e9","name":"restart","props":[{"p":"action","v":"{\"command\":\"restart\"}","vt":"json"}],"repeat":"","crontab":"00 06 * * *","once":false,"onceDelay":0.1,"topic":"","x":380,"y":4600,"wires":[["7a8c3f95b486b8f9"]]},{"id":"7ea407c1b15941c7","type":"ui_group","name":"Camera Deur","tab":"d9c0e801b43deec2","order":1,"disp":false,"width":"21","collapse":false,"className":""},{"id":"d9c0e801b43deec2","type":"ui_tab","name":"Camera Deur","icon":"fa-video-camera","order":27,"disabled":false,"hidden":false}]
1 Like

It could possibly be your input camera or your browser/device used to play the video.

So, the first thing to figure out is how long of durations the media segments are coming from the camera. Based on your settings, you are copying the video with ffmpeg, which puts you at the mercy of whatever the camera feed gives you. For example, I have 1 camera that cannot output media segments less than 4 seconds duration. And I have another cam that changes the durations of media segments based on if it is in day or night mode. Neither of the cams allow me to change those settings. This causes my playback to be a little erratic at times. Maybe yours does something similar. There can be a fix for that, but first we need to know what it is doing.

I have updated the flow you shared and then added a function node that can show some details that we need to get.
Screen Shot 2023-04-03 at 4.59.39 PM

[{"id":"372df3f2323e3a3f","type":"inject","z":"9f566d31b9e867ee","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":270,"y":260,"wires":[["933322a61f8b7c41"]]},{"id":"933322a61f8b7c41","type":"ffmpeg","z":"9f566d31b9e867ee","name":"","outputs":3,"cmdPath":"ffmpeg","cmdArgs":"[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://nvr:12345@192.168.1.95:554/h264Preview_01_sub\",\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]","cmdOutputs":2,"killSignal":"SIGTERM","x":510,"y":340,"wires":[["4a2a1286f0a6d95a"],["4a2a1286f0a6d95a"],["a8a6b62e943dad43"]]},{"id":"d49b9764533a74e3","type":"inject","z":"9f566d31b9e867ee","name":"stop","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":270,"y":380,"wires":[["933322a61f8b7c41"]]},{"id":"71f89f0fcd720dc8","type":"inject","z":"9f566d31b9e867ee","name":"restart","props":[{"p":"action","v":"{\"command\":\"restart\"}","vt":"json"}],"repeat":"","crontab":"00 06 * * *","once":false,"onceDelay":0.1,"topic":"","x":260,"y":300,"wires":[["933322a61f8b7c41"]]},{"id":"4a2a1286f0a6d95a","type":"mp4frag","z":"9f566d31b9e867ee","name":"","outputs":2,"basePath":"ui_mp4frag_01","serveHttp":"true","serveIo":"true","hlsPlaylistSize":"6","hlsPlaylistExtra":"2","autoStart":"false","preBuffer":1,"timeLimit":10000,"repeated":"false","statusData":"all","x":770,"y":320,"wires":[["dc113a743b091d48"],[]]},{"id":"a8a6b62e943dad43","type":"debug","z":"9f566d31b9e867ee","name":"stderr","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":700,"y":480,"wires":[]},{"id":"dc113a743b091d48","type":"function","z":"9f566d31b9e867ee","name":"status","func":"const { status, payload } = msg;\n\n// most frequent, handle segment status first\nif (status === 'segment') {\n\n    let { sequence, duration, totalDuration, byteLength, totalByteLength, keyframe } = payload;\n\n    // duration of current segment\n    duration = duration.toFixed(1);\n\n    // total duration of all segments\n    totalDuration = totalDuration.toFixed(1);\n\n    // byte length of current segment\n    byteLength = (byteLength / 1024).toFixed(0);\n\n    // total byte length of all segments\n    totalByteLength = (totalByteLength / 1024).toFixed(0);\n\n    // indicate if current segment contains keyframe\n    keyframe = keyframe ? 1 : 0;\n\n    const text = `${duration}/${totalDuration}(s), ${byteLength}/${totalByteLength}(KB), ${keyframe}, ${sequence}`;\n\n    node.status({ fill: 'green', shape: 'dot', text });\n\n    return null;\n\n}\n\n// 2nd most frequent, pass this msg along to ui_mp4frag\nif (status === 'playlist' || status === 'reset') {\n\n    return msg;\n\n}\n\n// least frequent, initialization fragment is parsed\nif (status === 'initialized') {\n\n    const { mime/*, videoCodec, audioCodec*/ } = payload;\n\n    // const text = `vc: ${videoCodec}, ac: ${audioCodec}`;\n\n    node.status({ fill: 'green', shape: 'dot', text: mime });\n\n    return null;\n\n}","outputs":1,"noerr":0,"initialize":"","finalize":"","libs":[],"x":1010,"y":300,"wires":[["91780c25f9e1edae"]]},{"id":"91780c25f9e1edae","type":"ui_mp4frag","z":"9f566d31b9e867ee","name":"Camera Deur","group":"7ea407c1b15941c7","order":1,"width":"21","height":"11","readyPoster":"","errorPoster":"","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":5,\"maxBufferLength\":10,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","autoplay":"true","unload":"true","threshold":"0.1","controls":"false","muted":"false","players":["socket.io","hls.js","hls","mp4"],"x":1210,"y":300,"wires":[[]]},{"id":"7ea407c1b15941c7","type":"ui_group","name":"Camera Deur","tab":"d9c0e801b43deec2","order":1,"disp":false,"width":"21","collapse":false,"className":""},{"id":"d9c0e801b43deec2","type":"ui_tab","name":"Camera Deur","icon":"fa-video-camera","order":27,"disabled":false,"hidden":false}]

Let me know if you can get that segment duration info or make a screenshot of the function node while the camera feed is running. Also, please let me know if you are using mobile phone or desktop and which browsers.

2 Likes

Hi Kevin,
Thanks again for your prompt and quick reply.
My other 3 camera's (Annke) are working fine and don't have a delay that will increase overtime.
Most likely the problem is indeed coming from this Reolink doorbel.
Here is the requested screenshot:

The problem of the increasing delay occurs on all my devices, however testing is always be done on my win10 desktop in Mozilla firefox 111.0.1

Let's will break down what each of the numbers mean for 4.0/32.0(s), 95/834(KB), 1, 21

4 - the most recent segment was of 4 seconds durations.
32 - there is a total of 32 seconds of video buffered (in memory).
95 - the most recent was 95 kilobytes
834 - total of 834 kilobytes of video buffered (in memory).
1 - current segment contains a keyframe/iframe
21 - 21st segment to be received through mp4frag from ffmpeg

Those aforementioned values totals are based on adding up the numbers in mp4frag settings under buffered segments size/extra. From your shared flow, I will conclude that your are still using 6 + 2.

(6+2) x 4 = 32. It looks like all of you segments were approximately 4 seconds duration, which is good because it means they are pretty consistent. I wonder if your segment duration changes with the lighting of color vs black&white modes. Further testing would be needed to verify if it fluctuates.

If the duration is 4 seconds for a segment, then the real-time viewing will be +4 seconds behind realtime due to the nature of when ffmpeg spits out the newest segment and mp4frag can parse it.

In order to keep the latency low, there is a newer feature of ffmpeg that allows you to still benefit from the low cpu load option of doing stream copying, while packaging the segments into shorter durations.

If you change the movflags from +frag_keyframe+empty_moov+default_base_moof to +frag_every_frame+empty_moov+default_base_moof, it will cause ffmpeg to fragment every single frame into its own segment. That would be overkill and not function well, so we must add an extra parameter to limit the size of the segment so that it is not so small.

"-movflags",
"+frag_every_frame+empty_moov+default_base_moof",
"-min_frag_duration",
"1000000",

This will give you segments that are close to a duration of 1 second. If you do this, you will notice that the value on that function node will change from 1 to 0 to indicate if that segment had a keyframe or not. If you are doing recordings with this data, then more things will have to be considered. Let me know if this helps.

2 Likes

Hi Kevin,
Thanks again for you extensive explanation.
I have changed the movflags settings and the numbers are looking now different as expected.

The initial delay dropped from 5 to 4 seconds showing the video in the dashboard.
Cpu load (pi-3B+) increased from 1.6 to 2% for ffmpeg so that's nothing.
To test if the delay will increase overtime needs sometime to test and also the night mode of the camera takes some time to test. Night mode is also in color, but I will test.

Will keep you posted.

1 Like

Just a question related; If you have, in this case 8,8 seconds of video buffered in memory, is that history or new video waiting to be presented/viewed? If new video, doesn't that mean you are 8.8 seconds late in relation to live view?

It is history AND video waiting to be viewed. I guess that as soon as we receive a single segment, it is already in the history, even if it is just 1 second in the past.

That also depends on the front end / client side video player. hls.js or native HLS usually need to consume more than a single segment before it can start playing video, as opposed to the hackey socket .io streamer that can play it right away and does not need a list of segments. We are always late, but try to be less late when the conditions are just right, such as small video size, fast internet, good browser with plenty of ram, etc.

If using the buffer for initiating a recording of recently past video, then the size/extra can be altered to keep a reasonable amount of video available for that. Since i am doing 24/7 recording vs motion detection recording, I have kept my size at 4 and extra at 1 to not keep as much old video. This still allows the native hls on my old iphone 6s safari browser to play the video smoothly.

I guess this stuff could be called recent-past vs live, but then it wouldn't sound as nice.

3 Likes

This topic was automatically closed 60 days after the last reply. New replies are no longer allowed.