Problem with video feed

Hi to all, thanks again for the great job you're doing and for sharing it.
Using the ffmpeg-spawn node I even need to have a base64 string. I tryed to convert the output with the relative base64 node, but it's not recognized. I even test on some online tools, but nothing. At this point I checked with the base64 images I already have and they're ok. So the toll works well. Even getting the image from IP camera --> exec node --> base64 it's ok.
There is something that I miss on the ffmpeg-spawn or its buffer output cannot be converted in base64 string?

@Lupin_III I've moved your post to its own thread, as it was off-topic where you originally posted it.

1 Like

Ok thanks. I thought it was better not to open a new topic and post directly on a one related to the node' owner. Sorry :rofl: :rofl: :rofl:

1 Like

i can guess that the exec node was in exec mode, allowing the single jpeg to be buffered and then output when the process exited. ffmpeg-spawn only runs in spawn mode and not exec mode, hence you will have to handle the output differently. I deliberately named it ffmpeg-spawn because I only plan on supporting spawn mode at this time, which is ideal for streaming. Although, it may evolve in time and support exec mode since I am now doing things with it that were not originally planned. Gonna need a name change on the node.

Of course, that is just a guess. If you give a screenshot or flow, then I can give a solution. I have no trouble outputting jpegs. I am also assuming that you want to output jpegs, but not sure.

Also, are you sure that you need to convert to base64? That is not actually necessary for a jpeg to be shown in the browser. It can handle an arraybuffer just fine with a little magic. Converting to base64 will make it a larger size to transmit.

[{"id":"f553cf93.25142","type":"ffmpeg-spawn","z":"846423c8.7aedd","name":"","outputs":3,"cmdPath":"ffmpeg","cmdArgs":"[\"-f\",\"mp4\",\"-i\",\"pipe:0\",\"-vf\",\"drawtext=text='%{localtime\\\\:%a %b %d %Y %H.%M.%S}':x=(w-text_w)/2:y=(h-text_h)/2:fontsize=120:fontcolor=black:box=1:boxborderw=10:boxcolor=white@0.5\",\"-c\",\"mjpeg\",\"-f\",\"image2pipe\",\"-vframes\",\"1\",\"pipe:1\"]","cmdOutputs":2,"killSignal":"SIGTERM","x":1140,"y":320,"wires":[[],["41373f64.26e0a"],["a53c221b.783aa"]],"info":"[ffmpeg drawtext](https://ffmpeg.org/ffmpeg-filters.html#drawtext)\n\n[drawtext tutorial](https://ottverse.com/ffmpeg-drawtext-filter-dynamic-overlays-timecode-scrolling-text-credits/)\n\n\n[\n    \"-f\",\n    \"mp4\",\n    \"-i\",\n    \"pipe:0\",\n    \"-vf\",\n    \"drawtext=text='video playback ready':x=(w-text_w)/2:y=(h-text_h)/2:fontsize=120:fontcolor=black:box=1:boxborderw=10:boxcolor=white@0.5\",\n    \"-c\",\n    \"mjpeg\",\n    \"-f\",\n    \"image2pipe\",\n    \"-vframes\",\n    \"1\",\n    \"pipe:1\"\n]"},{"id":"41373f64.26e0a","type":"pipe2jpeg","z":"846423c8.7aedd","name":"","x":1440,"y":260,"wires":[["7cb97f28.aa27"]]},{"id":"7cb97f28.aa27","type":"ui_template","z":"846423c8.7aedd","group":"aeb54dbc.59ea6","name":"","order":5,"width":"4","height":"3","format":"<img ng-src=\"{{src}}\" ng-on-load=\"onLoad()\"/>\n\n<script>\n\n((scope) => {\n\n    scope.$watch('msg', (msg) => {\n\n        if (msg && msg.payload instanceof ArrayBuffer) {\n\n            const arrayBufferView = new Uint8Array(msg.payload);\n    \n            const blob = new window.Blob([arrayBufferView], { type: 'image/jpeg' });\n    \n            const urlCreator = window.URL || window.webkitURL;\n    \n            const objectURL = urlCreator.createObjectURL(blob);\n\n            scope.src = objectURL;\n\n            scope.onLoad = () => {\n\n                urlCreator.revokeObjectURL(objectURL);\n\n            }\n\n        }\n\n    });\n\n})(scope);\n</script>\n","storeOutMessages":true,"fwdInMessages":true,"resendOnRefresh":true,"templateScope":"local","x":1620,"y":260,"wires":[[]]},{"id":"aeb54dbc.59ea6","type":"ui_group","z":"","name":"image","tab":"582231a0.48924","order":25,"disp":true,"width":"4","collapse":false},{"id":"582231a0.48924","type":"ui_tab","z":"","name":"health","icon":"dashboard","disabled":false,"hidden":false}]

Thanks Kevin.


No, everything work great and perfectly with node-red, camera, dashboard. Really great.
Now what I would like to do is to sent the image over mqtt so that I can use them in a simpe html with java script and I was making some test. If I have a simple base64 string I'm able to do it with paho-mqtt js. I will just convert the base64 into jpeg for example and show it. And since I already use all your nodes I wold like to try a solution with them. At the moment I'm able to test my idea with the exec node in spawn mode.

Maybe my approach is not that good, so every help, suggestion is more then welcome.
Practically I have node-red where everything works great and I just use your nodes. Now I want to show the image as streaming outside in a simple html and I can do it over mqtt. I've seen it's possible using some library in Javascript to directly receive messages from mqtt and show them by simply converting in jppeg. In this way, using a simple base64 string, I've seen even the messages' size remain low for the mqtt transfer.

One more detail that can help to understand better what I'm trying. On node-red I have a continuous ip camera streaming flow. From outside, when I need, trough mqtt I would like to get a copy of this streaming on the fly usable within an html (no node-red instance running on this html host)
@Kevin, I remember once you explained that it's mandatory to get the first piece of code in order to utilize the upcoming buffers (the initialization data). To make it works as I am trying, do you think it could help If I store this initialization data so that the buffer streaming is immediately usable when received? I'm trying it but with no result :frowning:

Sorry @kevinGodell, any newson how I can gate base64 output data from your node?

Are you trying to occasionally create a single jpeg from the mp4 buffer output of node-red-contrib-mp4frag?

And then you want to base64 encode the jpeg buffer before sending it via mqtt?

If yes to the first question, then you can trigger the mp4 buffer output by sending a command msg to node-red-contrib-mp4frag and then receive that in a node-red-contrib-ffmpeg-spawn node which will process the mp4 buffer and transcode a jpeg. see this post and check out the shared flow showing ffmpeg-spawn --> mp4frag --> ffmpeg-spawn --> pipe2jpeg.

If yes to the second question, I am not sure how you convert buffer to base64 string. There is probably a node for that.

Thanks @kevinGodell, yes the second I need to send a base64 string. Yes there is node to convert into base64, but it seems that your buffer from the spawn node is not what it expects. I am able to do it with exec node in spawn mode, as you can see from the previous post. https://aws1.discourse-cdn.com/business6/uploads/nodered/original/3X/8/5/85307a28cb21ce1f5e3de6b28632e40a0243f375.png
For a quick test you can copy paste the converted base64 string here Best Online Base64 to Image Decoder / Converter. At the moment I'm fine with the flow I shared using exec node, but I would like to avoid another process running on the pi and just use your nodes. I'll try with all the output, but from no one buffer I can return an image

Please post the flow with the working exec node and the flow with the non-working ffmpeg-spawn node. Don't forget to hide any passwords that may be used in the video's url.

Thanks Kevin.
This is what is working. The base64 payload output can be converted into an image with no problem

[{"id":"8d4ad352.eb215","type":"function","z":"1bb0d808.274788","name":"start video","func":"var msg_cam={}, msg_stop={};\nif (msg.payload == \"start\"){\n    var ff_input = 'ffmpeg -re -rtsp_transport tcp -f rtsp -i ';\n    var ff_camera = '\"rtsp://xxxxx:xxxx@192.168.1.100:8081/12\" ';\n    var ff_output = '-r 2 -f image2pipe pipe:1';\n    msg_cam.payload = ff_input+ff_camera+ff_output;\n    msg_stop.payload=\"wait\";\n} else {\n    msg_cam.kill=\"SIGTERM\";\n    msg_cam.payload=\"stop\";\n    msg_stop.payload=\"stop\";\n}\nreturn [msg_cam,msg_stop];\n","outputs":1,"noerr":0,"initialize":"","finalize":"","libs":[],"x":970,"y":180,"wires":[["ec5f02ae.5d614"]]},{"id":"ec5f02ae.5d614","type":"exec","z":"1bb0d808.274788","command":"","addpay":true,"append":"","useSpawn":"true","timer":"","oldrc":false,"name":"RTSP stream","x":1180,"y":180,"wires":[["89122fcb.84517"],[],[]]},{"id":"60e1cc89.e00524","type":"debug","z":"1bb0d808.274788","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1550,"y":120,"wires":[]},{"id":"89122fcb.84517","type":"base64","z":"1bb0d808.274788","name":"","action":"","property":"payload","x":1380,"y":180,"wires":[["60e1cc89.e00524","c3e08394.ede65"]]},{"id":"c1e55fb7.315c1","type":"inject","z":"1bb0d808.274788","name":"start default","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payload":"start","payloadType":"str","x":770,"y":160,"wires":[["8d4ad352.eb215"]]},{"id":"4ad55ffe.54272","type":"inject","z":"1bb0d808.274788","name":"stop default","props":[{"p":"payload"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payload":"stop","payloadType":"str","x":770,"y":220,"wires":[["8d4ad352.eb215"]]},{"id":"c3e08394.ede65","type":"mqtt out","z":"1bb0d808.274788","name":"","topic":"","qos":"","retain":"","respTopic":"","contentType":"","userProps":"","correl":"","expiry":"","x":1550,"y":180,"wires":[]}]

I tryed the same with the ffmpeg-spawn node, but I'm not able to converter any buffer back into an image

[{"id":"b5a47e.3107bb8","type":"inject","z":"1bb0d808.274788","name":"start default","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","payloadType":"str","x":830,"y":340,"wires":[["5d3ab0ff.381e"]]},{"id":"22c5a9e9.4217a6","type":"inject","z":"1bb0d808.274788","name":"stop default","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payloadType":"str","x":830,"y":420,"wires":[["5d3ab0ff.381e"]]},{"id":"5d3ab0ff.381e","type":"ffmpeg-spawn","z":"1bb0d808.274788","name":"Video","outputs":5,"cmdPath":"ffmpeg","cmdArgs":"[\"-loglevel\",\"error\",\"-nostats\",\"-f\",\"rtsp\",\"-re\",\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://xxxxx:xxxx@192.168.1.100:8081/12\",\"-c:v\",\"copy\",\"-c:a\",\"aac\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\",\"-progress\",\"pipe:3\"]","cmdOutputs":4,"killSignal":"SIGTERM","x":1030,"y":350,"wires":[["5753ef4a.78bce"],["5753ef4a.78bce"],["9c63051c.54c1a8"],["1c9aa6a1.6bf169"],["8afa808e.300d2"]]},{"id":"5753ef4a.78bce","type":"mp4frag","z":"1bb0d808.274788","name":"","migrate":2e-9,"hlsPlaylistSize":4,"hlsPlaylistExtra":0,"basePath":"Testcam","repeated":"false","timeLimit":"-1","preBuffer":"1","x":1310,"y":330,"wires":[["3fd20a1d.777cb6"],[]]},{"id":"3fd20a1d.777cb6","type":"ui_mp4frag","z":"1bb0d808.274788","name":"ui_mp4frag_Testcam","group":"a90d6a68.8dc3b8","order":1,"width":6,"height":4,"readyPoster":"","errorPoster":"","hlsJsConfig":"{\"liveDurationInfinity\":true,\"liveBackBufferLength\":5,\"maxBufferLength\":10,\"manifestLoadingTimeOut\":1000,\"manifestLoadingMaxRetry\":10,\"manifestLoadingRetryDelay\":500}","autoplay":"true","unload":"true","threshold":0.5,"controls":"true","muted":"true","players":["socket.io","hls.js","hls","mp4"],"x":1600,"y":330,"wires":[[]]},{"id":"f026b7f.33fec48","type":"comment","z":"1bb0d808.274788","name":"start ffmpeg","info":"","x":830,"y":300,"wires":[]},{"id":"22440546.4ceb0a","type":"comment","z":"1bb0d808.274788","name":"stop ffmpeg","info":"","x":830,"y":380,"wires":[]},{"id":"60e1cc89.e00524","type":"debug","z":"1bb0d808.274788","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1450,"y":380,"wires":[]},{"id":"2d8cb0fb.38b24","type":"debug","z":"1bb0d808.274788","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1450,"y":420,"wires":[]},{"id":"422e4e31.4261","type":"debug","z":"1bb0d808.274788","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"true","targetType":"full","statusVal":"","statusType":"auto","x":1450,"y":460,"wires":[]},{"id":"9c63051c.54c1a8","type":"base64","z":"1bb0d808.274788","name":"","action":"","property":"payload","x":1280,"y":380,"wires":[["60e1cc89.e00524"]]},{"id":"1c9aa6a1.6bf169","type":"base64","z":"1bb0d808.274788","name":"","action":"","property":"payload","x":1280,"y":420,"wires":[["2d8cb0fb.38b24"]]},{"id":"8afa808e.300d2","type":"base64","z":"1bb0d808.274788","name":"","action":"","property":"payload","x":1280,"y":460,"wires":[["422e4e31.4261"]]},{"id":"a90d6a68.8dc3b8","type":"ui_group","name":"videoTest","tab":"250de740.57ee78","order":14,"disp":false,"width":24,"collapse":false},{"id":"250de740.57ee78","type":"ui_tab","name":"Dashboard","icon":"dashboard","order":1,"disabled":false,"hidden":false}]

Well, you are not piping the image data to pipe 4.

Try this code in the ffmppeg-spawn node

[
    "-loglevel",
    "error",
    "-nostats",
    "-f",
    "rtsp",
    "-re",
    "-rtsp_transport",
    "tcp",
    "-i",
    "rtsp://xxxxx:xxxx@192.168.1.100:8081/12",
    "-c:v",
    "copy",
    "-c:a",
    "aac",
    "-f",
    "mp4",
    "-movflags",
    "+frag_keyframe+empty_moov+default_base_moof",
    "pipe:1",
    "-progress",
    "pipe:3",
    "-f",
    "image2pipe",
    "-vf",
    "select='eq(pict_type,PICT_TYPE_I)',scale=trunc(iw/4):-2",
    "-vsync",
    "vfr",
    "pipe:4"
]
2 Likes

Maybe you posted the wrong flow? The exec node and ffmpeg-spawn are not being given the same command, which explains why you are not getting the same output.

@krambriw's recommendation will add jpeg output in addition to the existing mp4 video output. It also includes a filter that only uses input iframes and is scaled down to 1/4 the size of the input to reduce some cpu load.

1 Like

Ok thanks, I understand. No I wanted to use the same process already used for mp4. I wanted to avoid to start another process, and just used the same stream.
Thanks so much

Last thing. Always in order to have just one process running per camera, it is possible to use just the main camera stream as ffmpeg-spawn node input and then pass it as it is (as high quality I mean) to the mp4frag, and reduce its quality for the image2pipe? Because for the image2pipe the substream video from the camera could be enough