Node-red-contrib-ffmpeg-spawn

EDIT: and of course I have the same error in my settings, +frag_keyframe versus +frag_every_frame
Embarrassing

I believe I have the same setting for ffmpeg-spawn and the exec, this is my simple test flow taking the stream from one of my cameras as mentioned. I connect to them using http. Please do modify the url in the commands to fit any of your own sources

BTW How can we check so that we know we have the same most recent version as you mention?

[{"id":"dd40be61.4e29f","type":"inject","z":"4e2296ba.635af8","name":"Start stream","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","payload":"true","payloadType":"bool","x":140,"y":130,"wires":[["96fdc212.96085"]]},{"id":"9665a078.26fe7","type":"inject","z":"4e2296ba.635af8","name":"Stop stream","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payload":"false","payloadType":"bool","x":140,"y":176,"wires":[["5d4a09e.dd61cf8"]]},{"id":"5d4a09e.dd61cf8","type":"switch","z":"4e2296ba.635af8","name":"","property":"payload","propertyType":"msg","rules":[{"t":"true"},{"t":"false"}],"checkall":"true","repair":false,"outputs":2,"x":470,"y":130,"wires":[["bf1296a0.fb6008"],["ea9b78f2.69c808"]]},{"id":"ea9b78f2.69c808","type":"function","z":"4e2296ba.635af8","name":"stop","func":"msg = {\n    kill:'SIGHUP',\n    payload : 'SIGHUP'  \n}\n\nreturn msg;","outputs":1,"noerr":0,"initialize":"","finalize":"","x":490,"y":179,"wires":[["bf1296a0.fb6008"]]},{"id":"bf1296a0.fb6008","type":"exec","z":"4e2296ba.635af8","command":"ffmpeg -i http://192.168.0.237:8889/?action=stream -c:v h264_omx -c:a aac -f mp4 -movflags +frag_every_frame+empty_moov+default_base_moof -min_frag_duration 500000 pipe:1","addpay":false,"append":"","useSpawn":"true","timer":"","oldrc":false,"name":"Camera testing","x":679,"y":150,"wires":[["8c43fe42.67586"],[],["8c43fe42.67586"]]},{"id":"13a8f08d.2ebf6f","type":"ui_mp4frag","z":"4e2296ba.635af8","name":"exec","group":"f850b4c5.fff4d8","order":3,"width":"9","height":"5","readyPoster":"https://raw.githubusercontent.com/kevinGodell/node-red-contrib-ui-mp4frag/master/video_playback_ready.png","errorPoster":"https://raw.githubusercontent.com/kevinGodell/node-red-contrib-ui-mp4frag/master/video_playback_error.png","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":0,\"maxBufferLength\":5,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","retry":true,"play":true,"unload":true,"threshold":0.5,"players":["hls.js","mp4","hls","socket.io"],"x":1080,"y":150,"wires":[[]]},{"id":"8c43fe42.67586","type":"mp4frag","z":"4e2296ba.635af8","name":"exec","migrate":1e-9,"hlsPlaylistSize":"10","hlsPlaylistExtra":"5","basePath":"kalle1","x":901,"y":147,"wires":[["13a8f08d.2ebf6f"],[]]},{"id":"96fdc212.96085","type":"delay","z":"4e2296ba.635af8","name":"","pauseType":"rate","timeout":"5","timeoutUnits":"seconds","rate":"1","nbRateUnits":"1","rateUnits":"second","randomFirst":"1","randomLast":"5","randomUnits":"seconds","drop":true,"x":310,"y":130,"wires":[["5d4a09e.dd61cf8"]]},{"id":"842cfeac.32eaf","type":"ffmpeg-spawn","z":"4e2296ba.635af8","name":"","outputs":5,"migrate":1e-9,"cmdPath":"ffmpeg","cmdArgs":"[\"-i\",\"http://192.168.0.237:8889/?action=stream\",\"-c:v\",\"h264_omx\",\"-c:a\",\"aac\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"-min_frag_duration\",\"500000\",\"pipe:1\"]","cmdOutputs":4,"killSignal":"SIGTERM","x":360,"y":350,"wires":[["81d2ebcd.11f048"],["81d2ebcd.11f048"],["d60b30b7.66176"],[],[]]},{"id":"c2c98b01.1707d8","type":"inject","z":"4e2296ba.635af8","name":"start default","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","payloadType":"str","x":140,"y":290,"wires":[["842cfeac.32eaf"]]},{"id":"b033e9c.c1cda18","type":"inject","z":"4e2296ba.635af8","name":"stop default","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payloadType":"str","x":140,"y":391,"wires":[["842cfeac.32eaf"]]},{"id":"6958e941.bb1788","type":"comment","z":"4e2296ba.635af8","name":"start ffmpeg","info":"","x":140,"y":254,"wires":[]},{"id":"ec3d0da.21a88f","type":"comment","z":"4e2296ba.635af8","name":"stop ffmpeg","info":"","x":140,"y":355,"wires":[]},{"id":"81d2ebcd.11f048","type":"mp4frag","z":"4e2296ba.635af8","name":"","migrate":1e-9,"hlsPlaylistSize":"20","hlsPlaylistExtra":"10","basePath":"kalle2","x":650,"y":360,"wires":[["17225931.4f28b7"],[]]},{"id":"17225931.4f28b7","type":"ui_mp4frag","z":"4e2296ba.635af8","name":"","group":"f850b4c5.fff4d8","order":1,"width":"10","height":"6","readyPoster":"","errorPoster":"","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":5,\"maxBufferLength\":10,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","retry":true,"play":true,"unload":true,"threshold":0.5,"players":["hls.js","mp4","hls","socket.io"],"x":870,"y":360,"wires":[[]]},{"id":"e6b316e0.a12ba8","type":"comment","z":"4e2296ba.635af8","name":"start ffmpeg","info":"","x":140,"y":464,"wires":[]},{"id":"d60b30b7.66176","type":"debug","z":"4e2296ba.635af8","name":"","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"false","statusVal":"","statusType":"auto","x":640,"y":460,"wires":[]},{"id":"f850b4c5.fff4d8","type":"ui_group","name":"Video compare","tab":"5cc557bf.a009c8","order":1,"disp":false,"width":"12","collapse":false},{"id":"5cc557bf.a009c8","type":"ui_tab","name":"Video compare","icon":"dashboard","disabled":false,"hidden":false}]

yes of course :angel: lol

[
    "-loglevel",
    "quiet",
    "-rtsp_transport",
    "tcp",
    "-i",
    "rtsp://192.168.1.9:554/user=****_password=****_channel=0_stream=1.sdp?real_stream",
    "-c:a",
    "aac",
    "-c:v",
    "copy",
    "-f",
    "mp4",
    "-movflags",
    "+frag_keyframe+empty_moov+default_base_moof",
    "pipe:1",
    "-progress",
    "pipe:2"
]

This is the ffmpeg-spawn

ffmpeg -loglevel quiet -rtsp_transport tcp -i rtsp://192.168.1.9:554/user=****_password=****_channel=0_stream=1.sdp?real_stream -c:a aac -c:v copy -f mp4 -movflags +frag_every_frame+empty_moov+default_base_moof -min_frag_duration 500000 pipe:1

This is the EXE
I just saw that -min_frag_duration 500000is missing in the ffmpeg-spawn, Could this be the source of the problem?

[EDIT]

Ooohhhh, yes the error, frag_keyframe instead of frag_every_frame. Like what you always have to check in the smallest details. They looked so alike that I hadn't seen the difference !!!
Obviously it works, I no longer have any delay compared to EXE !! Great :star_struck: ! Thanks all .

2 Likes

Thanks for confirming. And frankly, thanks for making me have to prove it. This helps for the future when another user may have a similar issue. Its definitely good to work out the small details now.

On another note, I merged the node-red-contrib-mp4frag #recorder branch with #master. The internal dependency, mp4frag, has been updated to v0.4.0 and published to npm. Using branches with yarn has caused me a bit of issues.

The new way to get the latest version of node-red-contrib-mp4frag is:
npm install kevinGodell/node-red-contrib-mp4frag
or
yarn add kevinGodell/node-red-contrib-mp4frag

I will be in the process of cleaning it up so in the near future after dashboard is at 3.0 and its dependency of socket io is at 3.0, I will publish to npm to make these nodes easier to install. Maintaining versions to support dashboard 2.0 and 3.0 would have consumed too much of my time that is already in short supply.

As for ffmpeg-spawn, I will probably be able to publish to npm sooner than the mp4frags since it has no dependencies. Just have to get the readme and help text written after finalizing features.

2 Likes

An update to ffmpeg-spawn. Added feature to split or combine the output messages, in the case that you want to combine output and filter by topic. Currently, output topics are fixed and derived from array: ['status', 'stdout', 'stderr', 'stdio3', 'stdio4', 'stdio5']. The index of the topic corresponds to what would be the node output when split.

A simple flow example showing a side by side comparison of split vs combined output:

[{"id":"48e02cfe.7cb014","type":"inject","z":"5967c855.68b978","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":269,"y":783,"wires":[["ffc6a303.54b73"]]},{"id":"ffc6a303.54b73","type":"ffmpeg-spawn","z":"5967c855.68b978","name":"","outputs":2,"migrate":2e-9,"cmdPath":"","cmdArgs":"[\"-version\"]","cmdOutputs":1,"killSignal":"SIGTERM","msgOutput":"split","x":435,"y":783,"wires":[[],["23ddb71.08b0548"]]},{"id":"abb559f9.d13f48","type":"debug","z":"5967c855.68b978","name":"","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":762,"y":783,"wires":[]},{"id":"23ddb71.08b0548","type":"function","z":"5967c855.68b978","name":"","func":"const version = msg.payload.toString().split(' ')[2];\n\nconst message = `version: ${version}`;\n\nnode.status({ fill: 'green', shape: 'dot', text: message });\n\nreturn { payload: version };","outputs":1,"noerr":0,"initialize":"","finalize":"","x":618,"y":783,"wires":[["abb559f9.d13f48"]]},{"id":"9f6f436d.b10aa","type":"inject","z":"5967c855.68b978","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":270,"y":860,"wires":[["4ee683b9.ccf13c"]]},{"id":"4ee683b9.ccf13c","type":"ffmpeg-spawn","z":"5967c855.68b978","name":"","outputs":1,"migrate":2e-9,"cmdPath":"","cmdArgs":"[\"-version\"]","cmdOutputs":1,"killSignal":"SIGTERM","msgOutput":"combined","x":436,"y":860,"wires":[["4a38c204.c85b4c"]]},{"id":"6d72faf.b638204","type":"debug","z":"5967c855.68b978","name":"","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":763,"y":860,"wires":[]},{"id":"4a38c204.c85b4c","type":"function","z":"5967c855.68b978","name":"","func":"const { topic } = msg;\n\nif (topic === 'stdout') {\n    const version = msg.payload.toString().split(' ')[2];\n\n    const message = `version: ${version}`;\n\n    node.status({ fill: 'green', shape: 'dot', text: message });\n\n    return { payload: version };\n}\n\nreturn;","outputs":1,"noerr":0,"initialize":"","finalize":"","x":619,"y":860,"wires":[["6d72faf.b638204"]]}]

Screen Shot 2021-04-24 at 6.49.03 PM

1 Like

For Docker users (Linux Alpine or Debian), It's mandatory to have FFMPEG installed,
Since I'm getting the following error.

Error: spawn ffmpeg ENOENT

image

I tried installing ffmpeg-for-homebridge which worked before, but no luck.
Any Idea how to overcome this FFMPEG limitation? Is there a way to make it standalone?

BTW, thanks for the node-red-contrib-ffmpeg-spawn node;-)

1 Like
  • i juste update all "package" and see that always v0.1.0
    versions.
cd ~/.node-red
npm install kevinGodell/node-red-contrib-ffmpeg-spawn
npm install kevinGodell/node-red-contrib-mp4frag
npm install kevinGodell/node-red-contrib-ui-mp4frag
npm install node-red-contrib-pipe2jpeg

result after install  : 25 avril 2021
-------------------
+ node-red-contrib-ffmpeg-spawn@0.1.0
updated 1 package in 35.443s
+ node-red-contrib-mp4frag@0.1.0
updated 2 packages in 35.219s
+ node-red-contrib-ui-mp4frag@0.1.0
updated 1 package in 33.968s
+ node-red-contrib-pipe2jpeg@0.1.0
updated 1 package in 29.371s
  • Not see v0.4.0 ? :thinking:

One option is that you can install ffmpeg from npm:

  1. npm install @ffmpeg-installer/ffmpeg
  2. look in your node_modules folder to try and deduce its path to the ffmpeg binary
  3. add the path to ffmpeg-spawn (for me its /home/pi/.node-red/node_modules/@ffmpeg-installer/linux-arm/ffmpeg) but it should be different for you based on your system
  4. run the flow to see the ffmpeg version output (mine is now showing different)

EDIT I just tried installing ffmpeg-for-homebridge and found its ffmpeg binary installed to /home/pi/.node-red/node_modules/ffmpeg-for-homebridge/ffmpeg
I tired that path and it also worked. That claims to be ffmpeg built for pi?

sample flow adding local ffmpeg path:

[{"id":"91d42cf4.68948","type":"ffmpeg-spawn","z":"5967c855.68b978","name":"","outputs":2,"migrate":2e-9,"cmdPath":"/home/pi/.node-red/node_modules/@ffmpeg-installer/linux-arm/ffmpeg","cmdArgs":"[\"-version\"]","cmdOutputs":1,"killSignal":"SIGTERM","msgOutput":"split","x":444,"y":980,"wires":[[],["584da5c6.da389c"]]},{"id":"4cbb3789.6910c8","type":"inject","z":"5967c855.68b978","name":"","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":271,"y":980,"wires":[["91d42cf4.68948"]]},{"id":"584da5c6.da389c","type":"function","z":"5967c855.68b978","name":"","func":"const version = msg.payload.toString().split(' ')[2];\n\nconst message = `version: ${version}`;\n\nnode.status({ fill: 'green', shape: 'dot', text: message });\n\nreturn { payload: version };","outputs":1,"noerr":0,"initialize":"","finalize":"","x":629,"y":980,"wires":[["bd1a13cf.ea256"]]},{"id":"bd1a13cf.ea256","type":"debug","z":"5967c855.68b978","name":"","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":790,"y":980,"wires":[]}]

ffmpeg version when using @ffmpeg-installer/ffmpeg

ffmpeg version when using ffmpeg-for-homebridge

Thanks for checking that. mp4frag@0.4.0 is a dependency used by node-red-contrib-mp4frag. I currently was using an experimental branch to add features. You wouldnt see it in your package.json file, but it would be a dependency in the node_modules folder.

@thejollypop

Added a new feature where you can define the ffmpeg path in settings.js

Firstly, install the dependency that will have the ffmpeg that you desire.

At the top of settings.js, require the selected lib to get the path.

Add the value to ffmpegSpawn.cmdPath.

excerpt from settings.js showing where I am requiring 3 ffmpeg libs ( I tested all 3 for compatibility )

const ffmpegPath1 = require('@ffmpeg-installer/ffmpeg').path;

const ffmpegPath2 = require('ffmpeg-for-homebridge');

const ffmpegPath3 = require('ffmpeg-static');

Inside the modules.exports object i added the path ( trying all 3 separately ):

ffmpegSpawn: {
    //cmdPath: ffmpegPath1
    cmdPath: ffmpegPath2
    //cmdPath: ffmpegPath3
  }

This will now show up as the default value when adding a new ffmpeg-spawn to your editor or if you leave the path value blank in the editor.

2 Likes

Accidentally posted this to wrong thread a minute ago. oops.

When tweaking your ffmpeg command, its always good to know what type of cpu load it is using.

Uses top in batch mode to get the stdout. top -b -p <pid>

[{"id":"fc058328.2af29","type":"exec","z":"ac290215.9a244","command":"top -b -p","addpay":true,"append":"","useSpawn":"true","timer":"","oldrc":false,"name":"","x":718,"y":153,"wires":[["b37719e3.1423b8"],[],[]]},{"id":"b37719e3.1423b8","type":"function","z":"ac290215.9a244","name":"%CPU, %MEM","func":"const { payload } = msg;\n\nif (typeof payload === 'string') {\n    \n    const lines = payload.toString().trim().split(/[\\n]+/);\n    \n    if (lines.length > 2) {\n        \n        const labels = lines[lines.length - 2].trim().split(/[\\s]+/);\n\n        const values = lines[lines.length - 1].trim().split(/[\\s]+/);\n\n        const message = `${labels[8]} ${values[8]}, ${labels[9]} ${values[9]}`;\n\n        node.status({ fill: 'green', shape: 'dot', text: message });\n\n        return { payload: message };\n    }\n\n}\n\nnode.status({});\n\nreturn null;","outputs":1,"noerr":0,"initialize":"","finalize":"","x":887,"y":140,"wires":[["8bbdeeb9.3f4d2"]]},{"id":"8bbdeeb9.3f4d2","type":"debug","z":"ac290215.9a244","name":"","active":false,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1043,"y":140,"wires":[]},{"id":"a1410491.d289a8","type":"function","z":"ac290215.9a244","name":"status pid","func":"const { payload = {} } = msg;\n\nconst { status, pid } = payload;\n\nif (status === 'spawn') {\n    \n    return { payload: pid };\n\n    \n} else if (status === 'close') {\n    \n    return { kill: 'SIGTERM' };\n    \n}\n\nreturn null;","outputs":1,"noerr":0,"initialize":"","finalize":"","x":573,"y":153,"wires":[["fc058328.2af29"]]}]

There were probably a bunch of ways to do this better, but this was the best I could come up with.

2 Likes

A bit of advertisment for my nodes can never be harmful: node-red-contrib-cpu

I was actually looking at your node with intentions to use it, but didnt see an option to get an individual PID's data. Would be a nice feature, but I could not find any node.js way of getting the value and had to resort to the system command.

Thanks @kevinGodell, now it's compatible with docker as well!

I'm running instances also on AWS, and now I get the following node-red-contrib-mp4frag error on t4g.micro Ubuntu and/or Debian servers.
Both node-red-contrib-ffmpeg-spawn and node-red-contrib-ui-mp4frag work fine.

It's particularly strange, because the first run went well, and only then the flows stopped working. Is there anything we can do about it? 10x

It’s hard to say the trouble could be. Those nodes are still experimental and have extra debugging info. Not sure what that image is showing the unhelpful error. Can you post a screenshot of the flow so I can see how things are connected.

I have an opposite problem in that I can’t make mine break. So, I am sure that we can make yours better. Personally, I have 24 live streams working through these nodes without fail.

Totally agree. I was able to fix it in Ubuntu 18.04 Version 43.0, so now it's working perfect