I GOT IT TO WORK!!
Nice!
And with an Amcrest. Interesting because I also have two of those (based on Kevin's feedback of his one).
When you start streaming, does it appear immediately on your dashboard?
Because in my case it takes about 20 seconds before it starts. Very inconvenient, but had no time to look at it yet...
Mine appears nearly instant! Maybe one second.
I have your example loaded in another flow but I keep getting a playback error.
It shows it's running but when I go to the dashboard I get the error.
I'll dive into it a lot more tomorrow. It's nearly 2am, I'm getting a few winks and I'll be back at this in a few hours.
Cheers!

I believe the playback error I'm getting has to do with my other camera. It's not an Amcrest, but a knockoff version. I'm going to try and update the firmware and see if it will play then. I was able to get my Amcrest to play on your flow.
If you have time to test my rtsp node further that would be nice. It works for my Amcrest cams, but need some other folks to test it with their cams...
I'll gladly test it with my other camera and let you know what tweaks I have to make to get it to work. That's the least I can do for you and Kevin for all the knowledge y'all have shared with this community!
I figured it out! My Encode mode on my "knock off" Amcrest was set to H.265 by default. Once I selected H.264 the feeds came right in! I actually used @kevinGodell flow for this one and now I'm going to test yours @BartButenaers
H265 would have only played in safari browser. The other browsers refuse to support it.
That's good to know!
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
}
]
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.
Could you post working flow please ?
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:
-
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. -
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
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
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.
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.
This topic was automatically closed 14 days after the last reply. New replies are no longer allowed.