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

Ok gents, I've been staring at this for a long time now and I have to be missing something critical. I'm wanting to make the push button turn on the camera in one instance.
In another instance I would like to publish "ON" from MQTT and have the camera start streaming.
I've been looking in this thread for some clues/examples/templates to mirror.

I'm wondering if I should use a function and use an "if" statement or some other node.

Here's what I'm working with so far.

[
    {
        "id": "1bedceafe187983d",
        "type": "tab",
        "label": "Flow 1",
        "disabled": false,
        "info": "",
        "env": []
    },
    {
        "id": "5a101c146af37907",
        "type": "inject",
        "z": "1bedceafe187983d",
        "name": "start",
        "props": [
            {
                "p": "action",
                "v": "{\"command\":\"start\"}",
                "vt": "json"
            }
        ],
        "repeat": "",
        "crontab": "",
        "once": false,
        "onceDelay": "5",
        "topic": "",
        "x": 410,
        "y": 300,
        "wires": [
            [
                "252ccca709d4bf8e",
                "26ed0f803e0cfac9"
            ]
        ]
    },
    {
        "id": "03dd7c763e6f08d8",
        "type": "inject",
        "z": "1bedceafe187983d",
        "name": "restart",
        "props": [
            {
                "p": "action",
                "v": "{\"command\":\"restart\"}",
                "vt": "json"
            }
        ],
        "repeat": "",
        "crontab": "",
        "once": false,
        "onceDelay": "1",
        "topic": "",
        "x": 650,
        "y": 400,
        "wires": [
            [
                "26ed0f803e0cfac9"
            ]
        ]
    },
    {
        "id": "ead7234e3dd533f4",
        "type": "inject",
        "z": "1bedceafe187983d",
        "name": "stop",
        "props": [
            {
                "p": "action",
                "v": "{\"command\":\"stop\"}",
                "vt": "json"
            }
        ],
        "repeat": "",
        "crontab": "",
        "once": false,
        "onceDelay": 0.1,
        "topic": "",
        "x": 810,
        "y": 520,
        "wires": [
            [
                "26ed0f803e0cfac9"
            ]
        ]
    },
    {
        "id": "26ed0f803e0cfac9",
        "type": "ffmpeg-spawn",
        "z": "1bedceafe187983d",
        "name": "",
        "outputs": 2,
        "cmdPath": "",
        "cmdArgs": "[\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://admin:1qaz2wsx@192.168.1.120:554\",\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]",
        "cmdOutputs": 1,
        "killSignal": "SIGTERM",
        "x": 1160,
        "y": 380,
        "wires": [
            [
                "39c72fee3454e01c"
            ],
            [
                "39c72fee3454e01c"
            ]
        ]
    },
    {
        "id": "39c72fee3454e01c",
        "type": "mp4frag",
        "z": "1bedceafe187983d",
        "name": "",
        "outputs": 2,
        "hlsPlaylistSize": "10",
        "hlsPlaylistExtra": "5",
        "basePath": "id",
        "repeated": "false",
        "timeLimit": "100000",
        "preBuffer": "1",
        "autoStart": "false",
        "statusLocation": "displayed",
        "x": 1520,
        "y": 380,
        "wires": [
            [
                "89141bcb54b77612"
            ],
            []
        ]
    },
    {
        "id": "89141bcb54b77612",
        "type": "ui_mp4frag",
        "z": "1bedceafe187983d",
        "name": "",
        "group": "da0ca243d2d46654",
        "order": 2,
        "width": "16",
        "height": "16",
        "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": 1870,
        "y": 380,
        "wires": [
            [
                "2ffdcbb09e9e5bc4"
            ]
        ]
    },
    {
        "id": "71b87a3fc8db300a",
        "type": "mqtt in",
        "z": "1bedceafe187983d",
        "name": "Cam2",
        "topic": "bbanzai/kit/1/cam/2",
        "qos": "2",
        "datatype": "auto-detect",
        "broker": "1dd00dfda20dc411",
        "nl": false,
        "rap": true,
        "rh": 0,
        "inputs": 0,
        "x": 290,
        "y": 160,
        "wires": [
            [
                "252ccca709d4bf8e",
                "29bec927f9fd40a4",
                "3b4a39ca6f921df7"
            ]
        ]
    },
    {
        "id": "2ffdcbb09e9e5bc4",
        "type": "mqtt out",
        "z": "1bedceafe187983d",
        "name": "ON",
        "topic": "bbanzai/kit/1/cam/2",
        "qos": "2",
        "retain": "false",
        "respTopic": "",
        "contentType": "",
        "userProps": "",
        "correl": "",
        "expiry": "",
        "broker": "1dd00dfda20dc411",
        "x": 1970,
        "y": 200,
        "wires": []
    },
    {
        "id": "2f15bf0f73a45385",
        "type": "change",
        "z": "1bedceafe187983d",
        "name": "",
        "rules": [
            {
                "t": "set",
                "p": "payload",
                "pt": "msg",
                "to": "action",
                "tot": "msg"
            }
        ],
        "action": "",
        "property": "",
        "from": "",
        "to": "",
        "reg": false,
        "x": 960,
        "y": 200,
        "wires": [
            [
                "2ffdcbb09e9e5bc4",
                "26ed0f803e0cfac9"
            ]
        ]
    },
    {
        "id": "252ccca709d4bf8e",
        "type": "ui_button",
        "z": "1bedceafe187983d",
        "name": "",
        "group": "da0ca243d2d46654",
        "order": 1,
        "width": "5",
        "height": "2",
        "passthru": true,
        "label": "ON",
        "tooltip": "",
        "color": "",
        "bgcolor": "",
        "className": "",
        "icon": "",
        "payload": "{\"ON\"}",
        "payloadType": "str",
        "topic": "action",
        "topicType": "msg",
        "x": 630,
        "y": 180,
        "wires": [
            [
                "2f15bf0f73a45385"
            ]
        ]
    },
    {
        "id": "29bec927f9fd40a4",
        "type": "debug",
        "z": "1bedceafe187983d",
        "name": "debug 1",
        "active": true,
        "tosidebar": true,
        "console": false,
        "tostatus": false,
        "complete": "true",
        "targetType": "full",
        "statusVal": "",
        "statusType": "auto",
        "x": 520,
        "y": 100,
        "wires": []
    },
    {
        "id": "3b4a39ca6f921df7",
        "type": "function",
        "z": "1bedceafe187983d",
        "name": "function 1",
        "func": "if(msg.payload == \"ON\") msg.payload = 1\n\nif (msg.payload == \"OFF\") msg.payload = 0\n\nreturn msg;\nreturn null;\n",
        "outputs": 1,
        "noerr": 0,
        "initialize": "",
        "finalize": "",
        "libs": [],
        "x": 660,
        "y": 260,
        "wires": [
            [
                "2f15bf0f73a45385"
            ]
        ]
    },
    {
        "id": "da0ca243d2d46654",
        "type": "ui_group",
        "name": "Group 1",
        "tab": "f51158cf17eb8e52",
        "order": 1,
        "disp": true,
        "width": "16",
        "collapse": false,
        "className": ""
    },
    {
        "id": "1dd00dfda20dc411",
        "type": "mqtt-broker",
        "name": "",
        "broker": "broker.emqx.io",
        "port": "1883",
        "clientid": "",
        "autoConnect": true,
        "usetls": false,
        "protocolVersion": "4",
        "keepalive": "60",
        "cleansession": true,
        "birthTopic": "",
        "birthQos": "0",
        "birthPayload": "",
        "birthMsg": {},
        "closeTopic": "",
        "closeQos": "0",
        "closePayload": "",
        "closeMsg": {},
        "willTopic": "",
        "willQos": "0",
        "willPayload": "",
        "willMsg": {},
        "userProps": "",
        "sessionExpiry": ""
    },
    {
        "id": "f51158cf17eb8e52",
        "type": "ui_tab",
        "name": "Test",
        "icon": "dashboard",
        "order": 3,
        "disabled": false,
        "hidden": false
    }
]

GUYS!!! @kevinGodell @BartButenaers
I got it to work!!


2 Likes

So I finally got this working to using Kevin Ā“s nodes.
Only sound is not working. While in VLC player its working fine. Any idea on that ?
( ui_mp4frag has the mute option set to false. That did not change anything compared to setting it to true)

EDIT: Ok, I had to remove the "-an" argument from the ffmpeg spawn node. Now it works like charm.
Tested BartButenaers node to which causes a much bigger delay than the one of Kevin.

1 Like

Could you post working flow please ?

1 Like

That's great information on the sound!

To get this work you need to install nodes of Kevin and ARP node:

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

then install ffempeg on your system. (google it!)

Import this flow and edit it at:

  1. function node "setCamId XX:XX......":
    change the mac address to the one of your cam.
    change user login to the one of your RTSP login.
    change password to the one of your RTSP password.

  2. function node "getIpFormMac":
    change the port and address at line 38 to the RTSP url of your cam.

[{
        "id": "9e0055b2467fee29",
        "type": "mp4frag",
        "z": "a6c955fb5f273da8",
        "name": "",
        "outputs": 2,
        "hlsPlaylistSize": "10",
        "hlsPlaylistExtra": "5",
        "basePath": "id",
        "repeated": "false",
        "timeLimit": "100000",
        "preBuffer": "1",
        "autoStart": "false",
        "statusLocation": "displayed",
        "x": 1520,
        "y": 1260,
        "wires": [["2f15f30e7d6375cf"], []]
    }, {
        "id": "2f15f30e7d6375cf",
        "type": "ui_mp4frag",
        "z": "a6c955fb5f273da8",
        "name": "",
        "group": "f4f1a14b.1adf4",
        "order": 12,
        "width": "14",
        "height": "8",
        "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": "false",
        "players": ["mp4", "socket.io", "hls.js", "hls"],
        "x": 1810,
        "y": 1260,
        "wires": [[]]
    }, {
        "id": "649be26c2832834a",
        "type": "ffmpeg-spawn",
        "z": "a6c955fb5f273da8",
        "name": "",
        "outputs": 2,
        "cmdPath": "/usr/bin/ffmpeg",
        "cmdArgs": "[\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://doesnotmatter\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]",
        "cmdOutputs": 1,
        "killSignal": "SIGTERM",
        "x": 1200,
        "y": 1260,
        "wires": [["9e0055b2467fee29", "c031afee4b51c66c"], ["9e0055b2467fee29", "8f3202c05d1832e6"]]
    }, {
        "id": "a34c078afaaefa7b",
        "type": "ui_button",
        "z": "a6c955fb5f273da8",
        "name": "",
        "group": "f4f1a14b.1adf4",
        "order": 4,
        "width": "2",
        "height": "1",
        "passthru": true,
        "label": "Start",
        "tooltip": "",
        "color": "",
        "bgcolor": "",
        "className": "",
        "icon": "",
        "payload": "{\"command\":\"start\",\"path\":\"/usr/bin/ffmpeg\",\"args\":[\"-rtsp_transport\",\"tcp\",\"-i\",\"RTSPCAM\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_keyframe+empty_moov+default_base_moof\",\"pipe:1\"]}",
        "payloadType": "str",
        "topic": "",
        "topicType": "str",
        "x": 570,
        "y": 1220,
        "wires": [["42b2b9eb4dcfb5e2", "877c18df3f7dc770"]]
    }, {
        "id": "42b2b9eb4dcfb5e2",
        "type": "function",
        "z": "a6c955fb5f273da8",
        "name": "payload2action",
        "func": "let camRtsp = flow.get(\"CamRtsp\") || \"empty\";\nif(camRtsp === \"empty\") return null;\nmsg.action = msg.payload.replace('RTSPCAM', camRtsp);\nreturn msg;\n",
        "outputs": 1,
        "noerr": 0,
        "initialize": "",
        "finalize": "",
        "libs": [],
        "x": 800,
        "y": 1260,
        "wires": [["e5e1557e92959e17", "3c25376d85e16edf"]]
    }, {
        "id": "46e2c18ffb442d7f",
        "type": "ui_button",
        "z": "a6c955fb5f273da8",
        "name": "",
        "group": "f4f1a14b.1adf4",
        "order": 6,
        "width": "2",
        "height": "1",
        "passthru": false,
        "label": "Stop",
        "tooltip": "",
        "color": "",
        "bgcolor": "",
        "className": "",
        "icon": "",
        "payload": "{\"command\":\"stop\"}",
        "payloadType": "json",
        "topic": "",
        "topicType": "str",
        "x": 570,
        "y": 1280,
        "wires": [["42b2b9eb4dcfb5e2"]]
    }, {
        "id": "5228fbe809204399",
        "type": "ui_button",
        "z": "a6c955fb5f273da8",
        "name": "",
        "group": "f4f1a14b.1adf4",
        "order": 8,
        "width": "2",
        "height": "1",
        "passthru": false,
        "label": "ReStart",
        "tooltip": "",
        "color": "",
        "bgcolor": "",
        "className": "",
        "icon": "",
        "payload": "{\"command\":\"restart\"}",
        "payloadType": "json",
        "topic": "",
        "topicType": "str",
        "x": 580,
        "y": 1340,
        "wires": [["42b2b9eb4dcfb5e2"]]
    }, {
        "id": "3c25376d85e16edf",
        "type": "json",
        "z": "a6c955fb5f273da8",
        "name": "",
        "property": "action",
        "action": "",
        "pretty": false,
        "x": 990,
        "y": 1260,
        "wires": [["649be26c2832834a", "6a895998298d435c"]]
    }, {
        "id": "a3dee14acca104a3",
        "type": "function",
        "z": "a6c955fb5f273da8",
        "name": "getIpFormMac",
        "func": "//msg.mac = \"18:01:f1:94:85:d4\";\nlet mac = flow.get(\"CamId\")||\"empty\";\nlet port = flow.get(\"CamPort\") || 80;;\nlet name = flow.get('CamName') || \"empty\";\nlet user = flow.get('CamUser') || \"empty\";\nlet pass = flow.get('CamPass') || \"empty\"\n\nif (mac === \"empty\") return null;\nmac = mac.toLowerCase();\nlet arp = msg.payload;\nif(arp === \"empty\") return null;\n\nlet macFound = false;\nlet arryCounter = 0;\nfor(let entry in arp)\n{\n    if (arp[arryCounter].mac === mac) \n    {\n        macFound = true;\n        break;\n    } \n    arryCounter+=1;\n}\n\nif(macFound === true)\n{\n    msg.ip = arp[arryCounter].ip;\n    msg.mac = mac;\n    //flow.set(\"SensorIp\", msg.ip)\n    //msg.payload = 1;\n    msg.linkId = mac;\n    msg.payload = \"http://\" + msg.ip;\n    msg.port = port;\n    flow.set(\"CamIp\", msg.ip);\n    flow.set(\"CamPort\", port);\n    msg.discription = msg.ip + \":\" + port;\n    //msg.status = \"online\";\n    flow.set(\"CamRtsp\", \"rtsp://\" + user + \":\" + pass + \"@\" + msg.ip + \":554/h264Preview_01_sub\");\n\n    return [msg, {payload:true}];\n}\nmsg.arryCounter = arryCounter;\nmsg.mac = mac;\nmsg.arp = arp;\nmsg.payload = false;\n//msg.status = \"offline\";\nreturn [null, msg];\n//return null;\n",
        "outputs": 2,
        "noerr": 0,
        "initialize": "",
        "finalize": "",
        "libs": [],
        "x": 600,
        "y": 220,
        "wires": [["e355ba462ff50d00"], ["e355ba462ff50d00"]]
    }, {
        "id": "c62861f7bf68bb1b",
        "type": "arp",
        "z": "a6c955fb5f273da8",
        "name": "ARP",
        "macs": "",
        "x": 570,
        "y": 180,
        "wires": [["a3dee14acca104a3", "c6e4377db32c97a1"]]
    }, {
        "id": "e1d840dee3bc8871",
        "type": "function",
        "z": "a6c955fb5f273da8",
        "name": "getMacs",
        "func": "msg.payload.macs = \"\";\n//msg.topic = \"CamFull\";\nreturn msg;",
        "outputs": 1,
        "noerr": 0,
        "initialize": "",
        "finalize": "",
        "libs": [],
        "x": 580,
        "y": 140,
        "wires": [["c62861f7bf68bb1b", "3abe49b128f5688f"]]
    }, {
        "id": "b840cae5765174f2",
        "type": "function",
        "z": "a6c955fb5f273da8",
        "name": "setCamId 68:39:43:AA:61:65",
        "func": "// Edit here:\nflow.set('CamId',\"68:39:43:AA:61:65\");\nflow.set('CamName', \"cam3\");\nflow.set('CamUser', \"CAM_USER_LOGIN\");\nflow.set('CamPass', \"CAM_USER_PASS\");\nreturn {payload: \"empty\"};",
        "outputs": 1,
        "noerr": 0,
        "initialize": "",
        "finalize": "",
        "libs": [],
        "x": 640,
        "y": 100,
        "wires": [["e1d840dee3bc8871"]]
    }, {
        "id": "7149d35a79696583",
        "type": "inject",
        "z": "a6c955fb5f273da8",
        "name": "",
        "props": [{
                "p": "payload"
            }, {
                "p": "topic",
                "vt": "str"
            }
        ],
        "repeat": "",
        "crontab": "",
        "once": true,
        "onceDelay": "0.1",
        "topic": "ExecuteOnStart",
        "payload": "",
        "payloadType": "str",
        "x": 610,
        "y": 60,
        "wires": [["b840cae5765174f2", "290a37e429b3e118"]]
    }, {
        "id": "f4f1a14b.1adf4",
        "type": "ui_group",
        "name": "Cam 3 - Parkplatz Video (Reolink)",
        "tab": "9f7f7c23.5626d8",
        "order": 4,
        "disp": true,
        "width": "14",
        "collapse": true,
        "className": ""
    }, {
        "id": "9f7f7c23.5626d8",
        "type": "ui_tab",
        "name": "Webcams",
        "icon": "videocam",
        "order": 14,
        "disabled": false,
        "hidden": false
    }
]

EDIT: DonĀ“t forget to create a tab and group at the dashboard and set it in the ui_mp4frag - node :slight_smile:

1 Like

Sorry this was directed @jorymathis13

I have a flow that used to work but now (after many things have been updated) the UI_mp4frag node doesn't stop when ffmpg is stopped. I wanted to check how the stop start is working in jorymathis13 flow :wink:

1 Like

no problem. I think its same as mine since I used his / kevinĀ“s flow in the same way.
You send an JSON object:

{"command":"stop"}

to the ffmpeg (spawn) node. this node will kill the ffmpeg process the way its defined in the node.
(default SIGTERM)

EDIT: Of course you have to use this node to start the cam / process. It will then only remember the process id which is needed to stop the process. Dont use this node to spawn multiple instances. need an extra copy of it for every cam.

2 Likes

The flow I posted above is the working flow. @WhiteLion is correct, I initially got one camera working and then just copied the nodes and changed the rtsp address to match the second camera and it worked.

1 Like

Thanks @jorymathis13 @WhiteLion I will study it now :wink:

2 Likes

This topic was automatically closed 14 days after the last reply. New replies are no longer allowed.