I have been experimenting with a different way of recording video from my rtsp ip cams lately. The new technique I have settled on is using a single file byte-range HLS playlist. When the video is recorded and written to disc, there will be a single mp4 file along with it's companion HLS playlist.
So far, video playback has been working great using hls.js on modern browsers and native HLS on mac/ios safari browsers. The benefits, as I see them, is that the recording can be played in the browser while being written to disc and allows for easy scrubbing along the video timeline due to the nature of the byte-range requests.
I was previously just recording as mp4 files using FFmpeg's segment muxer, but found that there was no good way to play the video before the segment was complete. Unfortunately, I could not find a way to combine FFmpeg's segment muxer and the byte-range HLS muxer to do what I wanted, so I had to involve the function node to programmably write the HLS playlist.
For my particular setup with a raspberry pi 4, I am using an externally powered usb connected case holding a 6TB WD Purple meant for 24/7 video recording. I have configured node-red's settings.js file to include:
httpStaticRoot: '/static/',
httpStatic: [
{ path: '/run/media/system/surveillance1/cctv/recordings/', root: '/recordings/' }
],
The recordings are placed into the folder /run/media/system/surveillance1/cctv/recordings/
and served via http at /static/recordings/
.
The files created are automatically named based on the current timestamp every 15 minutes and placed into folders created for the year, month, day.
The HLS playlist will look a bit like this:
#EXTM3U
#EXT-X-VERSION:7
#EXT-X-PLAYLIST-TYPE:EVENT
#EXT-X-TARGETDURATION:1
#EXT-X-MEDIA-SEQUENCE:0
#EXT-X-MAP:URI="02h.15m.00s.mp4",BYTERANGE="1242@0"
#EXTINF:1.0002555555555555
#EXT-X-BYTERANGE:707851@1242
02h.15m.00s.mp4
#EXTINF:0.9962444444444445
#EXT-X-BYTERANGE:679209@709093
02h.15m.00s.mp4
flow for a 24/7 video recording setup: (Do not use this without configuring the recording node)
[{"id":"f3874cf4a963908d","type":"subflow","name":"recording","info":"","category":"","in":[{"x":80,"y":160,"wires":[{"id":"d0615cb16ccf58fb"}]}],"out":[{"x":320,"y":120,"wires":[{"id":"d0615cb16ccf58fb","port":0}]}],"env":[{"name":"STATIC_PATH","type":"str","value":"/run/media/system/surveillance1/cctv/recordings"},{"name":"STATIC_ROOT","type":"str","value":"/static/recordings"}],"meta":{},"color":"#DDAA99","status":{"x":320,"y":200,"wires":[{"id":"d0615cb16ccf58fb","port":1}]}},{"id":"d0615cb16ccf58fb","type":"function","z":"f3874cf4a963908d","name":"recording","func":"const { payload, payload: { length }, duration, topic } = msg;\n\nconst [ , id, , type ] = topic.split('/');\n\nconst recording = context.get('recording');\n\nif (type === 'seg' || type === 'pre') {\n\n const { mp4WriteStream = {}, m3u8WriteStream = {}, mp4Name, bytesWritten, segments, playlist } = recording;\n\n if (mp4WriteStream.writable && m3u8WriteStream.writable) {\n\n // write live (seg) or previous (pre) segments to existing writeStreams\n\n mp4WriteStream.write(payload);\n\n m3u8WriteStream.write(`#EXTINF:${duration}\\n#EXT-X-BYTERANGE:${length}@${bytesWritten}\\n${mp4Name}\\n`);\n\n recording.bytesWritten += length;\n\n recording.segments++;\n\n // only send playlist after the first segment is written so that hls.js does not throw Hls.Events.ERROR, Hls.ErrorTypes.NETWORK_ERROR, Hls.ErrorDetails.LEVEL_EMPTY_ERROR\n\n if (segments === 0) {\n\n return { payload: playlist };\n\n }\n\n } else {\n\n node.error(`mp4WriteStream and m3u8WriteStream must be writable`);\n\n }\n\n return;\n\n}\n\nif (type === 'init') {\n\n // create new or end existing write streams based on buffer length\n\n if (length) {\n\n const date = new Date();\n\n const year = date.getFullYear();\n\n const month = (date.getMonth() + 1).toString().padStart(2, '0');\n\n const day = date.getDate().toString().padStart(2, '0');\n\n const hours = date.getHours().toString().padStart(2, '0');\n\n const minutes = date.getMinutes().toString().padStart(2, '0');\n\n const seconds = date.getSeconds().toString().padStart(2, '0');\n\n // create name based on timestamp\n\n const baseName = `${hours}h.${minutes}m.${seconds}s`; // 00h.00m.00s\n\n const mp4Name = `${baseName}.mp4`; // 00h.00m.00s.mp4\n\n const m3u8Name = `${baseName}.m3u8`; // 00h.00m.00s.m3u8\n\n const baseRoot = `${env.get('STATIC_ROOT')}/${id}`;\n\n const basePath = `${env.get('STATIC_PATH')}/${id}`;\n\n // create dir base on timestamp\n\n const fullDir = `${basePath}/${year}/${month}/${day}`; // basePath/yyyy/mm/dd\n\n const { mkdirSync, createWriteStream, writeFile } = fs;\n\n const newDir = mkdirSync(fullDir, { recursive: true });\n\n newDir && writeFile(`${basePath}/listing.txt`, `${year}/${month}/${day}\\n`, { encoding: 'utf8', flag: 'a' }, (err) => { err && node.error(err) });\n\n writeFile(`${fullDir}/listing.txt`, `${m3u8Name}\\n`, { encoding: 'utf8', flag: 'a' }, (err) => { err && node.error(err) });\n\n // create new write streams for mp4 and playlist\n \n const mp4WriteStream = createWriteStream(`${fullDir}/${mp4Name}`)\n .on('error', (err) => {\n node.error(err);\n });\n\n const m3u8WriteStream = createWriteStream(`${fullDir}/${m3u8Name}`)\n .on('error', (err) => {\n node.error(err);\n });\n\n mp4WriteStream.write(payload);\n\n m3u8WriteStream.write(`#EXTM3U\\n#EXT-X-VERSION:7\\n#EXT-X-PLAYLIST-TYPE:EVENT\\n#EXT-X-TARGETDURATION:${Math.round(duration) || 1}\\n#EXT-X-MEDIA-SEQUENCE:0\\n#EXT-X-MAP:URI=\"${mp4Name}\",BYTERANGE=\"${length}@0\"\\n`);\n\n recording.mp4WriteStream = mp4WriteStream;\n\n recording.m3u8WriteStream = m3u8WriteStream;\n\n recording.mp4Name = mp4Name;\n\n recording.bytesWritten = length;\n\n recording.segments = 0;\n\n // save playlist\n\n recording.playlist = `${baseRoot}/${year}/${month}/${day}/${m3u8Name}`;\n\n // send status\n\n return [ null, { payload: { fill: 'green', shape: \"dot\", text: `${id}/${year}/${month}/${day}/${baseName}` } }];\n\n } else {\n\n // end existing writeStreams\n\n const { mp4WriteStream = {}, m3u8WriteStream = {} } = recording;\n\n if (mp4WriteStream.writable) {\n\n mp4WriteStream.end();\n\n }\n\n if (m3u8WriteStream.writable) {\n\n m3u8WriteStream.end('#EXT-X-ENDLIST\\n');\n\n }\n\n return;\n\n }\n\n}","outputs":2,"noerr":0,"initialize":"// Code added here will be run once\n// whenever the node is deployed.\n\ncontext.set('recording', {});","finalize":"// Code added here will be run when the\n// node is being stopped or re-deployed.\n\nconst { mp4WriteStream = {}, m3u8WriteStream = {} } = context.get('recording');\n\nif (mp4WriteStream.writable) {\n\n mp4WriteStream.end();\n\n}\n\nif (m3u8WriteStream.writable) {\n\n m3u8WriteStream.end('#EXT-X-ENDLIST\\n');\n\n}\n\ncontext.set('recording', undefined);","libs":[{"var":"fs","module":"fs"}],"x":200,"y":160,"wires":[[],[]]},{"id":"32d61e0f.0490b2","type":"subflow","name":"stderr","info":"","category":"","in":[{"x":60,"y":80,"wires":[{"id":"211d46bf.a79c7a"}]}],"out":[],"env":[],"color":"#DDAA99","status":{"x":280,"y":120,"wires":[{"id":"211d46bf.a79c7a","port":1}]}},{"id":"211d46bf.a79c7a","type":"function","z":"32d61e0f.0490b2","name":"stderr","func":"// convert payload for Buffer to String and split by newlines\n\nconst stderr = msg.payload.toString().split('\\n');\n\nnode.send([ { stderr }, { payload: { fill: 'red', text: `${new Date().toString()}` } } ]);","outputs":2,"noerr":0,"initialize":"","finalize":"","libs":[],"x":170,"y":80,"wires":[["c6e5408.f19c4c"],[]]},{"id":"c6e5408.f19c4c","type":"debug","z":"32d61e0f.0490b2","name":"","active":true,"tosidebar":true,"console":false,"tostatus":false,"complete":"stderr","targetType":"msg","statusVal":"","statusType":"auto","x":330,"y":40,"wires":[]},{"id":"9b59c1a2222210b8","type":"inject","z":"0528fb314ac39109","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"5","topic":"","x":130,"y":220,"wires":[["2281776c992410dd"]]},{"id":"2281776c992410dd","type":"ffmpeg-spawn","z":"0528fb314ac39109","name":"","outputs":3,"cmdPath":"","cmdArgs":"[\"-loglevel\",\"+level+fatal\",\"-nostats\",\"-re\",\"-f\",\"lavfi\",\"-i\",\"testsrc=size=qcif:rate=5[out0];anoisesrc=c=brown:r=44100:a=0.5[out1]\",\"-g\",\"5\",\"-c:a\",\"aac\",\"-c:v\",\"libx264\",\"-profile:v\",\"high\",\"-pix_fmt\",\"yuv420p\",\"-level\",\"5.0\",\"-f\",\"mp4\",\"-movflags\",\"+faststart+frag_keyframe+empty_moov+default_base_moof\",\"-metadata\",\"title=test source\",\"pipe:1\"]","cmdOutputs":2,"killSignal":"SIGTERM","x":300,"y":260,"wires":[["58b77ee9a2e29092"],["58b77ee9a2e29092"],["a8741b7c88d88102"]]},{"id":"b4a8c0bc44499078","type":"inject","z":"0528fb314ac39109","name":"restart","props":[{"p":"action","v":"{\"command\":\"restart\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","x":130,"y":260,"wires":[["2281776c992410dd"]]},{"id":"7dde9f15d2aa5933","type":"inject","z":"0528fb314ac39109","name":"stop","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":130,"y":300,"wires":[["2281776c992410dd"]]},{"id":"58b77ee9a2e29092","type":"mp4frag","z":"0528fb314ac39109","name":"","outputs":2,"hlsPlaylistSize":4,"hlsPlaylistExtra":0,"basePath":"id","repeated":"false","timeLimit":"-1","preBuffer":"0","autoStart":"true","statusLocation":"displayed","x":550,"y":220,"wires":[[],["571c8af49dfc40ae"]]},{"id":"a8741b7c88d88102","type":"subflow:32d61e0f.0490b2","z":"0528fb314ac39109","name":"","x":470,"y":300,"wires":[]},{"id":"ef6008f4d7425c22","type":"inject","z":"0528fb314ac39109","name":"write restart","props":[{"p":"action","v":"{\"command\":\"restart\",\"subject\":\"write\",\"preBuffer\":1,\"timeLimit\":-1,\"repeated\":false}","vt":"json"}],"repeat":"","crontab":"*/15 0-23 * * *","once":false,"onceDelay":"5","topic":"","x":310,"y":140,"wires":[["58b77ee9a2e29092"]]},{"id":"e23111864612f2c5","type":"inject","z":"0528fb314ac39109","name":"write stop","props":[{"p":"action","v":"{\"command\":\"stop\",\"subject\":\"write\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":320,"y":180,"wires":[["58b77ee9a2e29092"]]},{"id":"571c8af49dfc40ae","type":"subflow:f3874cf4a963908d","z":"0528fb314ac39109","name":"","x":780,"y":260,"wires":[["e2da3c3cee315665"]]},{"id":"e2da3c3cee315665","type":"ui_mp4frag","z":"0528fb314ac39109","name":"","group":"9a076144e0b8c687","order":1,"width":"20","height":"14","readyPoster":"","errorPoster":"","hlsJsConfig":"{\"startPosition\":0,\"liveDurationInfinity\":false,\"progressive\":false}","autoplay":"true","unload":"false","threshold":0.1,"controls":"true","muted":"true","players":["hls.js","hls","socket.io","mp4"],"x":950,"y":220,"wires":[["9ec4f3ad80a3ec74"]]},{"id":"9ec4f3ad80a3ec74","type":"ui_template","z":"0528fb314ac39109","group":"9a076144e0b8c687","name":"video controls","order":15,"width":"20","height":"1","format":"<div layout-fill layout=\"row\">\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"toggleControls()\">\n <md-icon class=\"material-icons\">smart_button</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"playPause()\">\n <md-icon class=\"material-icons\">play_arrow</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"forwardFive()\">\n <md-icon class=\"material-icons\">forward_5</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"replayFive()\">\n <md-icon class=\"material-icons\">replay_5</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"decreaseRate()\">\n <md-icon class=\"material-icons\">slow_motion_video</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"increaseRate()\">\n <md-icon class=\"material-icons\">speed</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"toggleMuted()\">\n <md-icon class=\"material-icons\">volume_mute</md-icon>\n </md-button>\n </div>\n\n <div layout-fill>\n <md-button layout-fill class=\"md-raised\" ng-click=\"videoFullscreen()\">\n <md-icon class=\"material-icons\">fullscreen</md-icon>\n </md-button>\n </div>\n\n</div>\n\n<script>\n ((scope) => {\n\n const unwatchMsg = scope.$watch('msg', (msg) => {\n\n if (msg && msg.videoId) {\n\n const video = document.getElementById(msg.videoId);\n \n if (video) {\n\n const rates = [ 0.1, 0.2, 0.5, 1, 2, 5, 10];\n\n let rateIndex = 3;\n\n const requestFullscreen = video.webkitRequestFullscreen || video.webkitEnterFullScreen || video.mozRequestFullScreen || video.requestFullscreen;\n\n scope.toggleControls = () => {\n\n video.controls = !video.controls;\n\n }\n\n scope.playPause = () => {\n\n video.paused ? video.play() : video.pause();\n\n }\n\n scope.toggleMuted = () => {\n\n video.muted = !video.muted;\n\n }\n\n scope.videoFullscreen = () => {\n\n requestFullscreen.call(video);\n\n }\n\n scope.videoDisplay = () => {\n\n video.style.display = video.style.display === 'none' ? 'initial' : 'none';\n\n }\n\n scope.increaseRate = () => {\n\n rateIndex++;\n\n if (rateIndex > 6) {\n\n rateIndex = 6;\n\n }\n\n video.playbackRate = rates[rateIndex];\n\n }\n\n scope.decreaseRate = () => {\n \n rateIndex--;\n \n if (rateIndex < 0) {\n\n rateIndex = 0;\n \n }\n \n video.playbackRate = rates[rateIndex];\n \n }\n\n scope.forwardFive = () => {\n \n video.currentTime += 5;\n \n }\n\n scope.replayFive = () => {\n \n video.currentTime -= 5;\n \n }\n\n unwatchMsg();\n\n }\n\n }\n\n });\n\n})(scope);\n</script>","storeOutMessages":true,"fwdInMessages":true,"resendOnRefresh":true,"templateScope":"local","className":"","x":1140,"y":260,"wires":[[]]},{"id":"9a076144e0b8c687","type":"ui_group","name":"shared","tab":"e671ec4ca9656cab","order":1,"disp":true,"width":"20","collapse":false,"className":""},{"id":"e671ec4ca9656cab","type":"ui_tab","name":"shared","icon":"dashboard","disabled":false,"hidden":false}]
This requires 3 of my unpublished experimental nodes that are available from github by using the commands:
npm install kevinGodell/node-red-contrib-ffmpeg-spawn
npm install kevinGodell/node-red-contrib-mp4frag
npm install kevinGodell/node-red-contrib-ui-mp4frag
For those of you already using these nodes, you will have to update them because of some small changes I made recently to be compatible with this flow.
*note: FFmpeg is not included with these nodes.