Sending rtsp camera streaming over internet

Fantastic!!!! It works!!

Yeah, she is good!

This is pretty cool, the video is on my ssd disk here in Stockholm, Sweden, I publish it to a mqtt broker I have no idea where it is, and you view from somewhere in US, that's cool

1 Like

If you modify your flow like the picture below, you can restart the server stream yourself if needed

1 Like

Now you can host a private viewing party with your friends and watch a movie together!

But, seriously, the only issue i can foresee is storing the initialization fragment (ftyp atom) on the receiving side, so that it can be used later when connecting to a live feed that is already in progress.

Cool, perfect during pandemic situations with an isolated social lifestyle :wink:

Yeah, maybe that could be one of the properties that always should be included in the "constructed message" we discussed above. If it is relativly small, it would not cause too much overhead

I was considering the exact same idea. But still, some have metered bandwidth and that little bit might add up. Is there a way to detect when a new subscriber is attached so that we can ensure that the ftyp is sent first?

Or the alternative, use 2 topics, 1 for the initialization, 1 for the live segments. If the initialization piece is set to retain, that would be ok since it can be old and does not receive updates after it is created.

2 Likes

Regarding QOS as 2, that will be applied to all messages, so to all the video streaming, und it could be an overload if it is not important losing some streaming. Maybe it could be used only for the start messages with a dedicated topic. I'm not that expert as you so honestly it's hard for me to be sure to completely understand your points, but I tryed several times and I didn't have problem with the configuration I signed as solution.
Instead, what you think about a counter for start and stop the streaming? Do you have a better idea?

Yep, that would work fine I think!

Yes, @kevinGodell using retained flag for the init is the solution

Somehow I think I remember that there is a way to find out how many clients that are subscribing to a specific topic but needs to be investigated "how to". However, nothing prevents you from sending a notification/client id to the server (on a dedicated topic) when a client connects and subscribes. In this way the server could keep a list of connected clients and would know when it would be ok to stop the server from streaming

I suddenly thought I saw the light in the tunnel, what if the ffmpeg-spawn node had a dedicated output with the needed message structure fitted for mqtt communication??

(Well, realizing, obviously the mp4frag node would have to able to decode it but it could eventually also be solved by using some nodes like the switch and json nodes)

I would say you are onto a good (more extensible) solution myself.

If the mp4frag node was modified first to handle the payload being a buffer (Buffer.isBuffer(...)) or an object with a buffer property you could then later adjust the output of ffmpeg-spawn to be an object with a buffer property (plus others props as required) negating the need for any function/json/switch nodes etc

If there is concern about backwards compatibility, have a legacy mode check box to revert to old output style (not ideal but doable)

Hi Walter,
I didn't realise you live in Sweden. I visited Stockholm and Gothenburg many times (years ago) to run EDA courses at various Ericsson sites. Used to stay at the Globe Hotel in Stockholm. Great times, great people.

Hello Dave!! That must have been great & interesting!! I don't know where you are living in UK but I spent many, many hours during the eighties in Tewkesbury. Lovely area, green hills, love thos houses built with yellow limbstones, nice views towards Wales with the mountains in the background. I used to stay at a golf club, that was very luxury. Nice memories!!

From the mqtt protocol there is no way to know how many clients that currently subscribes to a specific topic. So you may end up having to design a kind of counter in NR keeping track for you if you want a multi-user solution where each user independently can decide if he wants the video or not. That being said, to make a client disconnect from receiving video would require you to unsubscribe from the topic. Just disabling the view would not help preventing video data sent to the clients mqtt in node as long as the server is still streaming. There is no command to send to the mqtt in node to unsubscribe

For myself I instead decided to make a solution where any client is able to turn on/off the video server. So what one client does affects all the others. I'm happy with that, I won't forget to leave a client "hangin in" viewing video nobody is looking at. When I turn it off I know it's off for all

So here is my latest:


[{"id":"17d3345a.8e43fc","type":"comment","z":"e193914.078e67","name":"Server side (machine one)","info":"","x":160,"y":50,"wires":[]},{"id":"d353460.3cd53b8","type":"mqtt in","z":"e193914.078e67","name":"","topic":"mp4_345_test","qos":"2","datatype":"auto","broker":"ed237c6b.40d1b","nl":false,"rap":true,"rh":0,"x":120,"y":690,"wires":[["d8444f28.0523"]]},{"id":"96a55dd1.544a7","type":"ui_mp4frag","z":"e193914.078e67","name":"","group":"9c7f211e.e2d06","order":1,"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":730,"y":720,"wires":[[]]},{"id":"5c03658f.ca806c","type":"mp4frag","z":"e193914.078e67","name":"","migrate":2e-9,"hlsPlaylistSize":4,"hlsPlaylistExtra":0,"basePath":"123client","repeated":"false","timeLimit":"10000","preBuffer":"1","x":510,"y":720,"wires":[["96a55dd1.544a7","61205bd6.757aa4"],[]]},{"id":"d8444f28.0523","type":"switch","z":"e193914.078e67","name":"","property":"payload","propertyType":"msg","rules":[{"t":"cont","v":"close","vt":"str"},{"t":"istype","v":"buffer","vt":"buffer"}],"checkall":"true","repair":false,"outputs":2,"x":310,"y":690,"wires":[["f64b8385.04e23"],["5c03658f.ca806c"]]},{"id":"f64b8385.04e23","type":"json","z":"e193914.078e67","name":"","property":"payload","action":"","pretty":false,"x":470,"y":660,"wires":[["5c03658f.ca806c"]]},{"id":"e5574b65.3a9308","type":"mqtt in","z":"e193914.078e67","name":"","topic":"mp4_345_server_on_off","qos":"2","datatype":"auto","broker":"ed237c6b.40d1b","nl":false,"rap":true,"rh":0,"x":170,"y":520,"wires":[["ac466242.598ef"]]},{"id":"ac466242.598ef","type":"json","z":"e193914.078e67","name":"","property":"payload","action":"","pretty":false,"x":180,"y":460,"wires":[["9ca7c9c3.171a78","d46a1000.7ed04"]]},{"id":"ccb74ab3.5b1278","type":"mqtt out","z":"e193914.078e67","name":"","topic":"mp4_345_server_on_off","qos":"","retain":"","respTopic":"","contentType":"","userProps":"","correl":"","expiry":"","broker":"ed237c6b.40d1b","x":530,"y":790,"wires":[]},{"id":"371dc912.1723b6","type":"ui_button","z":"e193914.078e67","name":"","group":"9c7f211e.e2d06","order":2,"width":"6","height":"1","passthru":false,"label":"{{msg.txt}}","tooltip":"","color":"","bgcolor":"{{msg.background}}","icon":"","payload":"true","payloadType":"bool","topic":"","topicType":"str","x":100,"y":790,"wires":[["84b6c29d.4f7d1"]]},{"id":"84b6c29d.4f7d1","type":"function","z":"e193914.078e67","name":"","func":"var state = context.get('state_buffer')||'off';\n//node.warn(state);\n\nswitch (state) {\n    case 'off':\n        msg.payload = {\"subject\":\"write\",\"command\":\"start\",\"timeLimit\":-1};\n        context.set('state_buffer', 'on');\n        break;\n    case 'on':\n        msg.payload = {\"subject\":\"write\",\"command\":\"stop\"};\n        context.set('state_buffer', 'off');\n        break;\n    default:\n        msg.payload = {\"subject\":\"write\",\"command\":\"start\",\"timeLimit\":-1};\n        context.set('state_buffer', 'on');\n        break;\n}\n\nnode.send(msg);","outputs":"1","noerr":0,"initialize":"","finalize":"","libs":[],"x":270,"y":790,"wires":[["ccb74ab3.5b1278","12ae2d42.f85043"]]},{"id":"12ae2d42.f85043","type":"change","z":"e193914.078e67","name":"","rules":[{"t":"set","p":"background","pt":"msg","to":"orange","tot":"str"},{"t":"set","p":"txt","pt":"msg","to":"Please wait...","tot":"str"}],"action":"","property":"","from":"","to":"","reg":false,"x":270,"y":840,"wires":[["371dc912.1723b6"]]},{"id":"ea0f87a0.a9a9c8","type":"change","z":"e193914.078e67","name":"","rules":[{"t":"set","p":"background","pt":"msg","to":"red","tot":"str"},{"t":"set","p":"txt","pt":"msg","to":"Streaming OFF","tot":"str"}],"action":"","property":"","from":"","to":"","reg":false,"x":270,"y":940,"wires":[["371dc912.1723b6"]]},{"id":"9ca7c9c3.171a78","type":"change","z":"e193914.078e67","name":"","rules":[{"t":"move","p":"payload","pt":"msg","to":"action","tot":"msg"}],"action":"","property":"","from":"","to":"","reg":false,"x":180,"y":400,"wires":[["9cc6812c.92bde"]]},{"id":"9cc6812c.92bde","type":"switch","z":"e193914.078e67","name":"","property":"action.command","propertyType":"msg","rules":[{"t":"cont","v":"start","vt":"str"},{"t":"cont","v":"stop","vt":"str"}],"checkall":"true","repair":false,"outputs":2,"x":180,"y":310,"wires":[["f1b369f9.af9dd8"],["ac1c1ece.98579"]]},{"id":"1d252606.19165a","type":"ffmpeg-spawn","z":"e193914.078e67","name":"","outputs":5,"migrate":1e-9,"cmdPath":"ffmpeg","cmdArgs":"[\"-re\",\"-stream_loop\",\"-1\",\"-i\",\"/home/pi/Qvideos/AMA_by_Julie_Gautier.mp4\",\"-bsf:a\",\"aac_adtstoasc\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]","cmdOutputs":4,"killSignal":"SIGTERM","x":340,"y":180,"wires":[["6fa778b2.984748"],["6fa778b2.984748"],[],[],[]]},{"id":"fefa83f.91c2e8","type":"inject","z":"e193914.078e67","name":"start default","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","payloadType":"str","x":120,"y":140,"wires":[["1d252606.19165a"]]},{"id":"dd4ea19f.2afee","type":"inject","z":"e193914.078e67","name":"stop default","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","payloadType":"str","x":120,"y":220,"wires":[["1d252606.19165a"]]},{"id":"a40b3aed.b0a918","type":"comment","z":"e193914.078e67","name":"start ffmpeg","info":"","x":120,"y":104,"wires":[]},{"id":"6e73c22e.fe57dc","type":"comment","z":"e193914.078e67","name":"stop ffmpeg","info":"","x":120,"y":180,"wires":[]},{"id":"6fa778b2.984748","type":"mqtt out","z":"e193914.078e67","name":"","topic":"mp4_345_test","qos":"","retain":"","respTopic":"","contentType":"","userProps":"","correl":"","expiry":"","broker":"ed237c6b.40d1b","x":570,"y":160,"wires":[]},{"id":"4617d743.771aa8","type":"comment","z":"e193914.078e67","name":"Client side (machine xy)","info":"","x":150,"y":640,"wires":[]},{"id":"5d4a1672.e90538","type":"change","z":"e193914.078e67","name":"","rules":[{"t":"set","p":"background","pt":"msg","to":"green","tot":"str"},{"t":"set","p":"txt","pt":"msg","to":"Streaming ON","tot":"str"}],"action":"","property":"","from":"","to":"","reg":false,"x":270,"y":890,"wires":[["371dc912.1723b6"]]},{"id":"61205bd6.757aa4","type":"switch","z":"e193914.078e67","name":"","property":"payload","propertyType":"msg","rules":[{"t":"nempty"},{"t":"empty"}],"checkall":"true","repair":false,"outputs":2,"x":750,"y":790,"wires":[["5d4a1672.e90538"],["ea0f87a0.a9a9c8"]]},{"id":"f1b369f9.af9dd8","type":"change","z":"e193914.078e67","name":"","rules":[{"t":"set","p":"payload","pt":"msg","to":"{\"command\":\"start\"}","tot":"json"}],"action":"","property":"","from":"","to":"","reg":false,"x":340,"y":290,"wires":[["1d252606.19165a"]]},{"id":"ac1c1ece.98579","type":"change","z":"e193914.078e67","name":"","rules":[{"t":"set","p":"payload","pt":"msg","to":"{\"command\":\"stop\"}","tot":"json"}],"action":"","property":"","from":"","to":"","reg":false,"x":340,"y":330,"wires":[["1d252606.19165a"]]},{"id":"d46a1000.7ed04","type":"debug","z":"e193914.078e67","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"false","statusVal":"","statusType":"auto","x":440,"y":460,"wires":[]},{"id":"ed237c6b.40d1b","type":"mqtt-broker","name":"test.mosquitto.org","broker":"test.mosquitto.org","port":"1883","tls":"ce0f0716.0b63b8","clientid":"","usetls":false,"protocolVersion":"4","keepalive":"60","cleansession":true,"birthTopic":"","birthQos":"0","birthPayload":"","birthMsg":{},"closeTopic":"","closeQos":"0","closePayload":"","closeMsg":{},"willTopic":"","willQos":"0","willPayload":"","willMsg":{},"sessionExpiry":""},{"id":"9c7f211e.e2d06","type":"ui_group","name":"Video over MQTT","tab":"4ea63eac.dccaf","order":1,"disp":false,"width":"6","collapse":false},{"id":"ce0f0716.0b63b8","type":"tls-config","name":"","cert":"","key":"","ca":"","certname":"","keyname":"","caname":"","servername":"","verifyservercert":true},{"id":"4ea63eac.dccaf","type":"ui_tab","name":"Video over MQTT","icon":"dashboard","disabled":false,"hidden":false}]
2 Likes

Yes, good alternative.
I'm still thinking on something else, in my particular case, and making test.
Since I have a server, which is the only place where user login, I'm trying to use this configuration, even because a node-red instance is running on the server as well keeping contact with the others local instances:
When a user login, a msg is sent every x time via mqtt that is read by the local instance. Locally, with this message is easy enough to implement the counter I posted here Sending rtsp camera streaming over internet - #30 by Lupin_III , substituting the ui_control with a function node with some kind of filters code

Depends what & why you want to achieve. I thought you wanted to save costs or bandwith to each client and what I wrote was that nothing will help unless the client unsubscribes from the topic. As long as the cllient is subscribed, buffer data will flow to him and consume bandwith both from the server and to the client

The wanting of saving unused bandwidth is mainly for the server (that can have several cameras streaming) and for the instance running locally, which send the streaming. For this I would like to avoid manually to turn on/off the video.

Not really sure about your use case. I have understood you would like to

  • stream to external clients and automatically shutoff streaming to them when no external client is watching anymore
  • stream to local clients as well
    Is this correct?

If so, you could build further on your (external) client connected counter idea. If you got that part working I could imagine to use gate nodes to turn video distribution on/off

First of all you could publish the video to two mqtt topics; one for local usage in a local machine running a local mqtt broker and the second topic in the remote mqtt broker. Then publish the video to the external mqtt topic via a gate node (video out channel). Then as long as you have at least one external client connected, keep the gate open, when no external client is connected, close the gate and no video will be published to the topic. Repeat the same typ of setup for each camera. This would close all channels automatically that are not currently used for viewing. You also have to have a mechanism so that a client can notify interest in watching a specific camera, i.e. being able to send a message via mqtt to tell the server to open the specific gate/gates

Yes, thanks a lot Krambriw.
For local clients (I mean the ones within the IP camera Lan) the client can see the straming directly with the ui_mp4frag, without needing of a local mqtt brocker.

Yes correct, that is true if you want. I don't know what would be most efficient though. Take the rtsp once and create video buffers distributed via mqtt locally or each client having to grab the rtsp stream by itself

Another thing I thought of when you are counting number of connected external clients. I think you actually have to count per camera how many are watching it to be able to know when you can turn the stream for the specific camera off or on. So when a client connects or is connected it will have to update the server for each camera it is viewing

Yes, some test is needed

You're right, correct

I did some playing around with this a bit over a year ago. Short answer is its going to depend on now many active camera streams, the camera resolution, and the power of the clients.

For example, the Pi4B is not particularly good at decoding 4K UHD RTSP streams, one is about it (can do 4-6 1080p depending on how low a frame rate you can accept, I consider ~3 fps per camera about right to feed into my AI sub-system).

I expected feeding it MQTT image buffers instead would help, but turns out the Pi4B network layer is not up to the task. Feeding an i3 class machine with real Gbit Ethernet using MQTT buffers could handle more cameras.

Feeding mp4fragments via MQTT I would guess would maybe be somewhere between sending jpeg image buffers (network limitations) and rtsp stream decoding (CPU/GPU limitations), on the assumption that the mp4fragments somewhat split the RTSP workload between server and client, but I could be mistaken here -- I've been following these threads but haven't had any time to actually try the nodes.

My bias for security cameras is that resolution is far more important than frame rate. There have been a fair number of high profile US court cases where "standard" D1 type resolution was not of good enough quality for the "beyond a reasonable doubt" legal standard, which is not helped by the fact the perps generally get a shave, haircut, and new suit for their court appearances.