[beta testing] nodes for live streaming mp4

I thought about it but my CSS skills are limited to "Beginner"


This is the DASHBOARD from the node-red-dashboard node. Node-Red can works without DASHBOARD


This is the Node-Red EDITOR where you create your flows

That would be good :+1:

1 Like

That’s puts you 1 level above me.

3 Likes

Hi @kevinGodell

I'm trying to run the example I shared here @krambriw to stream my ip camera with rtsp through ffmpeg. But in the last node nothing appears in the players section. How could I solve this? .

Thank you.

My guess is that you may have used a shared flow that has data not compatible with the current dev version from github. Also, the payload shows playlists with undefined values in it. The best thing to do is create a new mp4frag and ui_mp4frag and manually set its properties in the admin editor.

p.s. If you share your flow, I can try to clean it up and post it back to you.

Hi @kevinGodell ,

I attached the flow code, have two examples with ui_mp4frag and mp4player, but i cant make work.
If u helpme i aprreciace it

rtsp_stream_ip_camera_examples.json (6 KB)

I changed a couple things to atleast make the players show up in the list. Of course, I cant fully test it since I dont have access to your ip cams. Also, I didnt have up_mp4_player available for testing compatibility. If you are just trying to get your ip cam rtsp stream to play in the browser as mp4 video, we can make this work.

[{"id":"f2ce3d9a.df172","type":"inject","z":"e1dd03a3.208c7","name":"Start stream","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payload":"true","payloadType":"str","x":110,"y":40,"wires":[["d7f9bb8b.3067e8"]]},{"id":"e0b3bfc6.f7928","type":"inject","z":"e1dd03a3.208c7","name":"Stop stream","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payload":"false","payloadType":"str","x":110,"y":86,"wires":[["d7f9bb8b.3067e8"]]},{"id":"d7f9bb8b.3067e8","type":"switch","z":"e1dd03a3.208c7","name":"","property":"payload","propertyType":"msg","rules":[{"t":"eq","v":"true","vt":"str"},{"t":"eq","v":"false","vt":"str"}],"checkall":"true","repair":false,"outputs":2,"x":261,"y":40,"wires":[["a03c4e97.b45d4"],["3c6fcc17.0f37c4"]]},{"id":"3c6fcc17.0f37c4","type":"function","z":"e1dd03a3.208c7","name":"stop","func":"msg= [\n    {\n    kill:'SIGHUP',\n    payload : 'SIGHUP'\n    }\n    \n    \n    ];     // set a new payload & the counter\nreturn msg;","outputs":1,"noerr":0,"initialize":"","finalize":"","x":281,"y":89,"wires":[["a03c4e97.b45d4"]]},{"id":"a03c4e97.b45d4","type":"exec","z":"e1dd03a3.208c7","command":"ffmpeg -loglevel quiet -hwaccel rpi -c:v h264_mmal -rtsp_transport tcp -i  rtsp://192.168.100.57:554/user=admin_password=ZeCWU42p_channel=0_stream=0.sdp?real_stream -q 2 -vf fps=fps=2,scale=-1:-1 -c:v mjpeg -f image2pipe pipe:2 -an -c:v copy -f mp4 -movflags +frag_keyframe+empty_moov+default_base_moof pipe:1","addpay":false,"append":"","useSpawn":"true","timer":"","oldrc":false,"name":"front porch ip cam main","x":474,"y":49,"wires":[["a923fa6b.3749d8"],[],["a923fa6b.3749d8"]]},{"id":"a923fa6b.3749d8","type":"mp4frag","z":"e1dd03a3.208c7","name":"","migrate":2e-9,"hlsPlaylistSize":"4","hlsPlaylistExtra":"0","basePath":"420","repeated":"false","timeLimit":"10000","preBuffer":"1","x":750,"y":40,"wires":[["1f792a47.c16136"],["3ad18361.c9ccdc"]]},{"id":"1f792a47.c16136","type":"debug","z":"e1dd03a3.208c7","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1050,"y":20,"wires":[]},{"id":"3ad18361.c9ccdc","type":"debug","z":"e1dd03a3.208c7","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1070,"y":80,"wires":[]},{"id":"4e9f96d7.1fbab8","type":"inject","z":"e1dd03a3.208c7","name":"Start stream","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","payload":"true","payloadType":"bool","x":90,"y":340,"wires":[["f43ad028.b897d"]]},{"id":"aef35fb5.cbed","type":"inject","z":"e1dd03a3.208c7","name":"Stop stream","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payload":"false","payloadType":"bool","x":90,"y":386,"wires":[["f43ad028.b897d"]]},{"id":"f43ad028.b897d","type":"switch","z":"e1dd03a3.208c7","name":"","property":"payload","propertyType":"msg","rules":[{"t":"true"},{"t":"false"}],"checkall":"true","repair":false,"outputs":2,"x":241,"y":340,"wires":[["38ac3fa.6b6f0c"],["b0710a4.925daf8"]]},{"id":"b0710a4.925daf8","type":"function","z":"e1dd03a3.208c7","name":"stop","func":"msg = {\n    kill:'SIGHUP',\n    payload : 'SIGHUP'  \n}\n\nreturn msg;","outputs":1,"noerr":0,"initialize":"","finalize":"","x":261,"y":389,"wires":[["38ac3fa.6b6f0c"]]},{"id":"38ac3fa.6b6f0c","type":"exec","z":"e1dd03a3.208c7","command":"ffmpeg -loglevel quiet -rtsp_transport tcp -i rtsp://192.168.100.57:554/user=admin_password=ZeCWU42p_channel=0_stream=0.sdp?real_stream -an -c:v copy -f mp4 -movflags +frag_keyframe+empty_moov+default_base_moof pipe:1","addpay":false,"append":"","useSpawn":"true","timer":"","oldrc":false,"name":"Camera in Norway","x":460,"y":360,"wires":[["dc20e77b.0984d8"],[],["dc20e77b.0984d8"]]},{"id":"dc20e77b.0984d8","type":"mp4frag","z":"e1dd03a3.208c7","name":"Camera in Norway mp4frag","migrate":2e-9,"hlsPlaylistSize":"10","hlsPlaylistExtra":"5","basePath":"dc20e77b.0984d8","repeated":"false","timeLimit":"10000","preBuffer":"1","x":740,"y":360,"wires":[["5b96f008.fb2af","dfcbfaac.c21c28"],[]]},{"id":"5b96f008.fb2af","type":"debug","z":"e1dd03a3.208c7","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":990,"y":300,"wires":[]},{"id":"dfcbfaac.c21c28","type":"ui_mp4frag","z":"e1dd03a3.208c7","name":"Camera in Norway ui_mp4frag","group":"69d9ac35.efceb4","order":1,"width":"10","height":"6","readyPoster":"","errorPoster":"","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":5,\"maxBufferLength\":10,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","play":"true","unload":"true","threshold":0.1,"players":["socket.io","hls.js","hls","mp4"],"x":1070,"y":360,"wires":[[]]},{"id":"69d9ac35.efceb4","type":"ui_group","name":"Tablero","tab":"f111a0ad.0d4d4","order":1,"disp":false,"width":"28","collapse":false},{"id":"f111a0ad.0d4d4","type":"ui_tab","name":"Dashboard","icon":"dashboard","order":1,"disabled":false,"hidden":false}]

Hi @kevinGodell

First of all thanks for your help. Although now the player options appear, I could not make it work.

from what I read in this conversation. the node does not work for ip cameras that use the h.265 codec, right?
I have the IP camera with the following configuration:

FFMPEG consumes a lot of cpu from the RPi3 B +, the only solution I found was to generate a request loop from @BartButenaers 's Onvif Media node ... and I refresh the image every X seconds through a template node

Do you think there is a solution to be able to see the streaming fluid from the rtsp?

If you can set your ip cam to use h264, then we should be able to re-stream that with minimal cpu load. Anytime you have ffmpeg encode video, such as converting h265 to h264 or h264 to jpegs, etc., there will be a high cpu load if there is not a hardware accelerated codec available.

When I made the underlying mp4frag library (few years back) that is used inside of node-red-contrib-mp4frag, I did not have access to ip cams containing h265 encoded rtsp, and never explored that possibility. And I am not sure about browsers support of h265 encoded mp4. https://caniuse.com/?search=h.265

Hi @kevinGodell ,
I noticed that since I put the last versions of spawn + mp4frag + mp4frag_ui, the video is stretched vertically, to occupy the dimensions of the widget on the dashboard. Is there a setting somewhere to keep the original dimensions of the video?
image
on the left the elongated rtsp stream. On the right an image captured by a movement of original size

try this :
you have to install in 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

image

[{"id":"f622d954.ab6448","type":"mp4frag","z":"31914bf3.119024","name":"","migrate":2e-9,"hlsPlaylistSize":4,"hlsPlaylistExtra":0,"basePath":"camtest","repeated":"false","timeLimit":"10000","preBuffer":"1","x":620,"y":1810,"wires":[["d4e965c3.f74de8"],[]]},{"id":"d4e965c3.f74de8","type":"ui_mp4frag","z":"31914bf3.119024","name":"","group":"a90d6a68.8dc3b8","order":9,"width":6,"height":4,"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":["socket.io","hls.js","hls","mp4"],"x":790,"y":1800,"wires":[[]]},{"id":"28d2bc75.6fa6a4","type":"inject","z":"31914bf3.119024","name":"start default","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","x":120,"y":1810,"wires":[["c9cdc64f.eea768"]]},{"id":"2d88b2b2.82aade","type":"comment","z":"31914bf3.119024","name":"start ffmpeg","info":"","x":130,"y":1780,"wires":[]},{"id":"c9cdc64f.eea768","type":"ffmpeg-spawn","z":"31914bf3.119024","name":"marche ok avec args ICI","outputs":3,"migrate":2e-9,"cmdPath":"ffmpeg","cmdArgs":"[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://192.168.1.10:554/user=XXXX_password=XXXX_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\"]","cmdOutputs":2,"killSignal":"SIGTERM","msgOutput":"split","x":320,"y":1820,"wires":[["5ea25c5d.23d1b4"],["5ea25c5d.23d1b4"],[]],"info":"ORIGINAL Kevin ffmpeg:\n------------------------\n[\n    \"-loglevel\",\n    \"error\",\n    \"-nostats\",\n    \"-f\",\n    \"hls\",\n    \"-http_multiple\",\n    \"1\",\n    \"-re\",\n    \"-i\",\n    \"https://weather-lh.akamaihd.net/i/twc_1@92006/index_1200_av-p.m3u8?sd=10&rebase=on\",\n    \"-c:v\",\n    \"copy\",\n    \"-c:a\",\n    \"aac\",\n    \"-f\",\n    \"mp4\",\n    \"-movflags\",\n    \"+frag_keyframe+empty_moov+default_base_moof\",\n    \"pipe:1\",\n    \"-progress\",\n    \"pipe:3\",\n    \"-f\",\n    \"image2pipe\",\n    \"-vf\",\n    \"select='eq(pict_type,PICT_TYPE_I)',scale=trunc(iw/4):-2\",\n    \"-vsync\",\n    \"vfr\",\n    \"pipe:4\"\n]"},{"id":"5ea25c5d.23d1b4","type":"change","z":"31914bf3.119024","name":"","rules":[],"action":"","property":"","from":"","to":"","reg":false,"x":485,"y":1810,"wires":[["f622d954.ab6448"]],"l":false},{"id":"7d2954e.97be9ac","type":"inject","z":"31914bf3.119024","name":"stop default","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payloadType":"str","x":120,"y":1840,"wires":[["c9cdc64f.eea768"]]},{"id":"a90d6a68.8dc3b8","type":"ui_group","name":"videoTest","tab":"250de740.57ee78","order":16,"disp":true,"width":"6","collapse":true},{"id":"250de740.57ee78","type":"ui_tab","name":"Dashboard","icon":"dashboard","order":1,"disabled":false,"hidden":false}]

you have to put the rtsp arguments of your camera in the ffmpeg-spawn node

Yeah sorry, didn’t update the docs for that yet. I recently moved to a class based styling so that the user has more control over the appearance. Look in the head of the html and you will see the style section with video.ui_mp4frag or something like that. You can add a ui node set to global and override the object fit value and add !important after the value. When I get to my computer later I can give a better response.

1 Like

template node adding css to override value:

[{"id":"57adb1f0.9ae51","type":"ui_template","z":"26b5a4eb.db1d34","group":"5c433dd6.328804","name":"object-fit","order":2,"width":0,"height":0,"format":"<style>\n  .ui-mp4frag-video {\n    object-fit: contain !important;\n    /*object-fit: scale-down !important;*/\n  }\n</style>","storeOutMessages":true,"fwdInMessages":true,"resendOnRefresh":true,"templateScope":"global","x":444,"y":893,"wires":[[]]},{"id":"5c433dd6.328804","type":"ui_group","z":"","name":"Default","tab":"eeb0f4e7.54d118","order":1,"disp":true,"width":"6","collapse":false},{"id":"eeb0f4e7.54d118","type":"ui_tab","z":"","name":"test","icon":"dashboard","disabled":false,"hidden":false}]

template type: head
css property: object-fit
css value: contain or scale-down both seem to keep the aspect ratio

<style>
  .ui-mp4frag-video {
    object-fit: contain !important;
    /*object-fit: scale-down !important;*/
  }
</style>
1 Like

:ok_hand: that's good

1 Like

@kevinGodell
In your node ffmpeg-spawn can you, in a next update, remove the kbits/s bitrate: "153.3kbits/s"
so that we can format it to the unit we want as ko/s or mo/s ?
you can say in the notice that this measurement is in kbits/s

Unfortunately, that value is directly generated from ffmpeg. I have found no way to configure it differently. The alternative would be to place a node between ffmpeg and whatever node you normally have set to receive its buffer. Just check the buffer length to calculate the size on your timescale. There might already be a counter node that does that.

Thank you Kevin, I'll figure it out :wink:

Hi Kevin , did you test ffmpeg-spawn with node-red in WIN10
I test quickly and I have an error:


I imagine I need to install FFMPEG, do you have any other tips to make it work?

No, I have only tested on pi 4. You can try to install 1 of the ffmpeg libraries via npm, such as @ffmpeg-installer/ffmpeg, ffmpeg-for-homebridge, or ffmpeg-static, etc (there is probably more).

In the setting.js file, require the dependency in the ffmpeg-spawn config (you might have to test each one for compatibility):

ffmpegSpawn: {
  cmdPath: require('ffmpeg-for-homebridge'),
},

or

ffmpegSpawn: {
  cmdPath: require('@ffmpeg-installer/ffmpeg').path,
},

or

ffmpegSpawn: {
  cmdPath: require('ffmpeg-static'),
},

This will set the path to the node_modules folder and should be seen in the editor as in the text input placeholder:


Leave the value empty so that it uses the default value set in the settings.js file.

:star_struck: nice it's working as you describe :+1: