How to display live video to the node red dashboard and have the ability to simply turn it on and turn it off with either a slide switch or "On" "Off" push buttons

@kevinGodell

Hey Kevin, first off I'm going preface and say that I am a complete NEWB when it comes using node-red and ip cameras. I am able to stream my ip cameras with VLC using rtsp and see the live streams. I spent the better part of 5 hours yesterday combing through the threads on node-red (primarily the one with you and @BartButenaers) and google looking for solutions to my problem. Any assistance that you could help walk me through on this would be extremely appreciated and I would be most grateful.

Respectfully,
Jory

1 Like

I am sure that we can make it work, but I will need a little extra info:

  • what have you already tried
  • os type and how did you install node-red
  • ffmpeg installed and available on the command line
  • which nodes are you trying to use
  • whatever other details that you think are relevant
1 Like

@kevinGodell
Thank you so much for replying and willing help me give this a crack.
So, I was running node-red on my host system with Windows 11 and tried to use the RTSP Grab Frame flow as a template to work off of but I've had zero success with it. So then when I went to the discussion boards to see if other Node-Red and Git members had tried something similar and that when I ran across the posts from you and Mr. Bart. I couldn't initially find any Ffmpeg nodes and after installing the Ffmpeg Streamer node (the green one), it caused issues with my flows not starting, therefore I went back and deleted the node and thought it to be best if I just build an Ubuntu VM and get this working in Linux.
I installed node-red via the terminal in Ubuntu and spent the better part of 3 hours trying to properly download your Ffmpeg spawn node from Git which I got it downloaded successfully (so I think). It doesn't show up in my nodes to select from though, so I may be doing something wrong.
I've gotten so dang frustrated and flustered the last week that I've deleted my flows at least 5 times.
To me it seems like a simple task to have a camera stream to the dashboard and have two buttons "On" "Off" be present on the dashboard and when "On" is clicked the stream shows up and when "Off" is pushed then the stream goes away. But for some reason this is kicking my butt and draining my patience. LOL

fyi, i have tested my nodes to work on windows, linux, and mac, but should be easiest to setup on your ubuntu.

My preferred way of streaming video from rtsp cams to the dashboard (assuming you are referring to node-red-dashboard) is to connect to the cam using ffmpeg and then remuxing the h264 encoded video into a fragmented mp4 container and relaying that to the front end client side.

Firstly, before worrying about the on/off buttons, let's get a video stream working.
Currently, my nodes are not published to npm.js, but are available directly from github.
Assuming you have git installed, cd to the .node-red directory in the users home. There should be a package.json there. Run the following commands and then open the package.json to see if the modules were installed.

npm install kevinGodell/node-red-contrib-ffmpeg-spawn kevinGodell/node-red-contrib-mp4frag kevinGodell/node-red-contrib-ui-mp4frag

After installing and verifying the package.json file if you have easy access to it, then you must restart the node-red process for it to detect the new nodes in the pallette.

Import this flow and add the rtsp url to the SECRET input.

[{"id":"5273a6de.b59318","type":"tab","label":"mp4 video test","disabled":false,"info":""},{"id":"b80cada9.8f784","type":"inject","z":"5273a6de.b59318","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"5","topic":"","x":150,"y":160,"wires":[["856b3636.3e0ab8"]]},{"id":"53023373.a797ec","type":"inject","z":"5273a6de.b59318","name":"restart","props":[{"p":"action","v":"{\"command\":\"restart\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","x":150,"y":220,"wires":[["856b3636.3e0ab8"]]},{"id":"cc88f04f.3fad1","type":"inject","z":"5273a6de.b59318","name":"stop","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":150,"y":280,"wires":[["856b3636.3e0ab8"]]},{"id":"856b3636.3e0ab8","type":"ffmpeg-spawn","z":"5273a6de.b59318","name":"","outputs":2,"cmdPath":"","cmdArgs":"[\"-rtsp_transport\",\"tcp\",\"-i\",\"SECRET\",\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]","cmdOutputs":1,"killSignal":"SIGTERM","x":380,"y":220,"wires":[["5ba95f6b.9d606"],["5ba95f6b.9d606"]]},{"id":"5ba95f6b.9d606","type":"mp4frag","z":"5273a6de.b59318","name":"","outputs":2,"hlsPlaylistSize":"10","hlsPlaylistExtra":"5","basePath":"id","repeated":"false","timeLimit":"100000","preBuffer":"1","autoStart":"false","statusLocation":"displayed","x":680,"y":220,"wires":[["ade19c3c.45e87"],[]]},{"id":"ade19c3c.45e87","type":"ui_mp4frag","z":"5273a6de.b59318","name":"","group":"d6044a4e.91ee38","order":1,"width":"12","height":"10","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":"true","players":["socket.io","hls.js","hls","mp4"],"x":950,"y":220,"wires":[[]]},{"id":"d6044a4e.91ee38","type":"ui_group","z":"","name":"video 1","tab":"8666e04a.dbd34","order":1,"disp":true,"width":"12","collapse":true},{"id":"8666e04a.dbd34","type":"ui_tab","z":"","name":"mp4 video test","icon":"dashboard","order":34,"disabled":false,"hidden":false}]

Let me know if you get stuck anywhere.

1 Like

@jorymathis13,
Really?
From what I have read about streaming, this is a difficult task to accomplish.
But if you have found an easy way to do it, don't hesitate to share your solution with us!!!

I will explain shortly following flow that does something like you need:

[{"id":"0da35190b74701bf","type":"inject","z":"47b91ceb.38a754","name":"start rtsp stream","props":[{"p":"topic","vt":"str"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"start","x":220,"y":1080,"wires":[["8ed1d84559c36844"]]},{"id":"3be67ccd3f81cd49","type":"inject","z":"47b91ceb.38a754","name":"stop rtsp stream","props":[{"p":"topic","vt":"str"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"stop","x":220,"y":1140,"wires":[["8ed1d84559c36844"]]},{"id":"8ed1d84559c36844","type":"rtsp-client","z":"47b91ceb.38a754","name":"","ffmpegPath":"ffmpeg","rtspUrl":"rtsp://my_cam_rtsp_stream_url","statisticsInterval":"","restartPeriod":"4","autoStart":"disable","videoCodec":"copy","videoFrameRate":"","videoWidth":"320","videoHeight":"240","videoQuality":"","minFragDuration":"","audioCodec":"copy","audioSampleRate":"","audioBitRate":"","transportProtocol":"tcp","imageSource":"none","imageFrameRate":"","imageWidth":"","imageHeight":"","socketTimeout":"","maximumDelay":"","socketBufferSize":"","reorderQueueSize":"","credentials":{"userName":"","password":""},"x":430,"y":1080,"wires":[["fd04afc4ce8e354e"],[],[],[]]},{"id":"fd04afc4ce8e354e","type":"mp4frag","z":"47b91ceb.38a754","name":"","outputs":2,"hlsPlaylistSize":"4","hlsPlaylistExtra":"0","basePath":"front_door","repeated":"false","timeLimit":"60000","preBuffer":"1","autoStart":"false","statusLocation":"displayed","x":630,"y":1060,"wires":[["9be5b5030df84d59"],[]]},{"id":"9be5b5030df84d59","type":"ui_mp4frag","z":"47b91ceb.38a754","name":"","group":"c18d5dc32a9ceb50","order":0,"width":"10","height":"7","readyPoster":"","errorPoster":"","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":5,\"maxBufferLength\":10,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","autoplay":"true","unload":"true","threshold":0.1,"controls":"true","muted":"true","players":["socket.io","hls.js","hls","mp4"],"x":870,"y":1060,"wires":[[]]},{"id":"ebc8e7b28b5f9ce8","type":"ui_multistate_switch","z":"47b91ceb.38a754","name":"","group":"c18d5dc32a9ceb50","order":1,"width":0,"height":1,"label":"Camera","stateField":"payload","enableField":"enable","passthroughField":"passthrough","inputMsgField":"inputmsg","rounded":false,"useThemeColors":false,"hideSelectedLabel":false,"multilineLabel":false,"passThrough":"never","inputMsg":"all","userInput":"enabled_show","options":[{"label":"Stop","value":"stop","valueType":"str","color":"#009933"},{"label":"Play","value":"start","valueType":"str","color":"#999999"}],"topic":"","x":180,"y":1200,"wires":[["2e5d9b2d9d8f8f8c"]]},{"id":"2e5d9b2d9d8f8f8c","type":"change","z":"47b91ceb.38a754","name":"payload->topic","rules":[{"t":"set","p":"topic","pt":"msg","to":"payload","tot":"msg"}],"action":"","property":"","from":"","to":"","reg":false,"x":360,"y":1200,"wires":[["8ed1d84559c36844"]]},{"id":"c18d5dc32a9ceb50","type":"ui_group","name":"Front door","tab":"f136a522.adc2a8","order":2,"disp":true,"width":"10","collapse":false,"className":""},{"id":"f136a522.adc2a8","type":"ui_tab","name":"Home","icon":"home","order":1}]
  1. Based on Kevin's ffmpeg-spawn node and all the feedback that he has given in numerous discussions, I had build my own node-red-contrib-rtsp-client node. While Kevin's node is able to do lot more with ffmpeg, I tried to capture all the knowledge from Kevin's discussions about rtsp into my node. So my node only supports rtsp streams, to avoid that everybody needs to read to the long discussions...

    Note that my node is also not on npm yet, so you also need to install it directly from Github (from within your .node-red folder!!!):

    npm install bartbutenaers/node-red-contrib-rtsp-client
    

    You need to add the rtsp url of your cam into this node, and the path to your ffmpeg executable.

    Note that the frame grabbing of this node consumes lot's of CPU (see details).

  2. The output of this node are rtsp segments which are feed into Kevin's node-red-contrib-mp4frag node, to create mp4 fragments. This node needs to be installed from his Github repo.

    npm install kevinGodell/node-red-contrib-mp4frag
    
  3. And you also need to install Kevin's node-red-contrib-ui-mp4frag node, which displays the stream of mp4 fragments in the Node-RED dashboard.

    npm install kevinGodell/node-red-contrib-ui-mp4frag
    
  4. The green node displays a switch in the dashboard, which allows you to start and stop the rtsp stream. For this you need to install the node-red-contrib-ui-multistate-switch node from the palette menu.

I created this flow very quickly, so the look and feel is very minimalistic. If you are able to make the ui part more user-friendly, please share your solution in this discussion so we can learn from it!

Bart

1 Like

Ah seems that Kevin already answered while I was typing...

2 answers are better than 1.

3 Likes

@BartButenaers @kevinGodell
I need to apologize, because my "simple task" comment was taken out of context. I know this is super difficult and me being new at this, makes it exponentially more difficult. Please don't think that I view this is easy by any means, nor discredit the work and time that you and all the other contributors have painstakingly accomplished. It was just the way I was processing it my head and trying to keep the solve process as simple as possible with out over analyzing and making the trouble shooting steps any more difficult that it already is.

Trying to install the nodes now and getting the following errors.
I'm sure it's something I'm overlooking on my end or my inexperience or both. :man_facepalming:

jd@jd-virtual-machine:~/.node-red$ npm install kevinGodell/node-red-contrib-ffmpeg-spawn kevinGodell/node-red-contrib-mp4frag kevinGodell/node-red-contrib-ui-mp4frag
npm ERR! code ENOENT
npm ERR! syscall spawn git
npm ERR! path git
npm ERR! errno -2
npm ERR! enoent An unknown git error occurred
npm ERR! enoent This is related to npm not being able to find a file.
npm ERR! enoent

npm ERR! A complete log of this run can be found in:
npm ERR! /home/jd/.npm/_logs/2022-08-29T02_24_43_771Z-debug-0.log

I'm not an idiot after all. I just had to read Kevin's directions more closely. We had a saying in the military "Attention to Detail" and I did not.

If I remember correctly the spawn git error means that you don't have git installed. Normally you don't need git, if you install packages directly from npm. However in this case, the npm install command needs to get our packages from Github, so that command will call in turn a git .. command.

https://git-scm.com/book/en/v2/Getting-Started-Installing-Git

1 Like

Ah ok, now you were responding while a was typing :wink:
Anyway then others with the same error can find it quicker...

1 Like

I GOT IT TO WORK!!


1 Like

Nice!
And with an Amcrest. Interesting because I also have two of those (based on Kevin's feedback of his one).
When you start streaming, does it appear immediately on your dashboard?
Because in my case it takes about 20 seconds before it starts. Very inconvenient, but had no time to look at it yet...

1 Like

Mine appears nearly instant! Maybe one second.

I have your example loaded in another flow but I keep getting a playback error.
It shows it's running but when I go to the dashboard I get the error.



I'll dive into it a lot more tomorrow. It's nearly 2am, I'm getting a few winks and I'll be back at this in a few hours.
Cheers! :beers:

1 Like

I believe the playback error I'm getting has to do with my other camera. It's not an Amcrest, but a knockoff version. I'm going to try and update the firmware and see if it will play then. I was able to get my Amcrest to play on your flow.

1 Like

If you have time to test my rtsp node further that would be nice. It works for my Amcrest cams, but need some other folks to test it with their cams...

1 Like

I'll gladly test it with my other camera and let you know what tweaks I have to make to get it to work. That's the least I can do for you and Kevin for all the knowledge y'all have shared with this community!

1 Like

I figured it out! My Encode mode on my "knock off" Amcrest was set to H.265 by default. Once I selected H.264 the feeds came right in! I actually used @kevinGodell flow for this one and now I'm going to test yours @BartButenaers

1 Like

H265 would have only played in safari browser. The other browsers refuse to support it.

2 Likes

That's good to know!