Cctv recording and playback (work in progress)

The node is unaware of what is happening on the client side. It simply receives the playlist and sends it to the client side where there is an attempt to play it.

If you wanted to handle the client side event on the video element, you can add the event listener to detect ended and send a message back using scope.send. If you added this little piece of code on the controls, you could detect the event and send a message and somehow load the next video from your list. Of course, this would make it change for all viewers since the dashboard is designed to be single user.

const video = document.getElementById(msg.videoId);

            if (video) {

                video.addEventListener('ended', () => {
                    scope.send('it ended');
                });

This successfully sent a message to my debug panel.

It would need more work so that to only detects the ended event when you want it to, maybe with some checkbox control. Just a proof of concept at this point...

1 Like

Works great, really useful. Understood, it is events from the HTMLMediaElement you have to watch. Many useful and interesting:

1 Like

In this situation, it might be better to set the onended property with the callback function vs adding the event listener. This will help to ensure that you have only added the callback a single time. And instead of removing the event listener, you can just delete the onended property (or maybe set to null or call removeAttribute, can't remember right now at the top of my head).

Well, both works fine, why is one better than the other?
Best regards, Walter

            if (video) {

                video.onended = function() {myEnd()};

                function myEnd() {
                    scope.send('the end');
                }

                video.addEventListener('ended', () => {
                scope.send('it ended');
                });

                video.addEventListener('playing', () => {
                scope.send('it plays');
                });

                video.addEventListener('pause', () => {
                scope.send('it paused');
                });

Assigning it as a property only allows 1 handler, whereas adding the event listener will allow more than 1 handler. If you are only need a single callback, which is most likely the case for this particular situation, then using the property might be simpler and you won't accidentally add too many event listeners. Other than that, I don't know if one is better or just a matter of preference.

1 Like

Is anybody here using nginx along with the cctv stuff? I have been deep diving nginx over the last month and found some interesting techniques to offload some burden from the node-red instance, in particular when playing back recordings and using nginx's slice module. Hopefully I can get a small change to node-red core to allow this to work. If anybody is interested in nginx integration with the cctv things, then maybe we can start a new thread for that. Let me know.

1 Like

If you found something interesting, please yes explain it a new thread :wink:

1 Like

How is this project that involves the recordings?

I'm going to fall into those possibilities now that I've managed to make it work. The RTSP camera on the system.

To my knowledge, this worked when I added & worked with a sample to my flow (using that RedBull stream, but it should work with your camera(s) as well in the same way

1 Like

Who is the user who creates these folders so I can give this mkdir permission?

I'm a little lost on this item of code ??

[{"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":"5273a6de.b59318","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"5","topic":"","x":450,"y":340,"wires":[["7a8c3f95b486b8f9"]]},{"id":"b4a8c0bc44499078","type":"inject","z":"5273a6de.b59318","name":"restart","props":[{"p":"action","v":"{\"command\":\"restart\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","x":450,"y":380,"wires":[["7a8c3f95b486b8f9"]]},{"id":"7dde9f15d2aa5933","type":"inject","z":"5273a6de.b59318","name":"stop","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":450,"y":420,"wires":[["7a8c3f95b486b8f9"]]},{"id":"58b77ee9a2e29092","type":"mp4frag","z":"5273a6de.b59318","name":"","outputs":2,"basePath":"telemetria","serveHttp":"false","serveIo":"false","hlsPlaylistSize":4,"hlsPlaylistExtra":0,"autoStart":"true","preBuffer":"0","timeLimit":"-1","repeated":"false","statusData":"playlist","x":1010,"y":100,"wires":[[],["571c8af49dfc40ae"]]},{"id":"a8741b7c88d88102","type":"subflow:32d61e0f.0490b2","z":"5273a6de.b59318","name":"","x":910,"y":340,"wires":[]},{"id":"ef6008f4d7425c22","type":"inject","z":"5273a6de.b59318","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":750,"y":60,"wires":[["58b77ee9a2e29092"]]},{"id":"e23111864612f2c5","type":"inject","z":"5273a6de.b59318","name":"write stop","props":[{"p":"action","v":"{\"command\":\"stop\",\"subject\":\"write\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":760,"y":100,"wires":[["58b77ee9a2e29092"]]},{"id":"571c8af49dfc40ae","type":"subflow:f3874cf4a963908d","z":"5273a6de.b59318","name":"","x":1320,"y":180,"wires":[["e2da3c3cee315665"]]},{"id":"e2da3c3cee315665","type":"ui_mp4frag","z":"5273a6de.b59318","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":1330,"y":240,"wires":[["9ec4f3ad80a3ec74"]]},{"id":"9ec4f3ad80a3ec74","type":"ui_template","z":"5273a6de.b59318","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":1340,"y":340,"wires":[[]]},{"id":"cfa37a1fd41f4570","type":"group","z":"5273a6de.b59318","name":"Camera IP - Protocolo RTSP","style":{"stroke":"#ffC000","fill":"#ffefbf","label":true,"color":"#6f2fa0"},"nodes":["7a8c3f95b486b8f9","4cc7afbbcc802243","652bde80360a1fe1","d1112dfcbfef615b","2858e68578995d73","379c223a6ef2f8ea","adffeae04f85fed5","96e79d1eccb9df9d"],"x":74,"y":139,"w":1132,"h":122},{"id":"7a8c3f95b486b8f9","type":"ffmpeg","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","outputs":3,"cmdPath":"ffmpeg","cmdArgs":"[]","cmdOutputs":2,"killSignal":"SIGTERM","x":750,"y":200,"wires":[["4cc7afbbcc802243","58b77ee9a2e29092"],["4cc7afbbcc802243","58b77ee9a2e29092"],["a8741b7c88d88102"]]},{"id":"4cc7afbbcc802243","type":"mp4frag","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","outputs":2,"basePath":"teste","serveHttp":"true","serveIo":"true","hlsPlaylistSize":"6","hlsPlaylistExtra":"2","autoStart":"false","preBuffer":1,"timeLimit":10000,"repeated":"false","statusData":"playlist","x":920,"y":200,"wires":[["652bde80360a1fe1"],[]]},{"id":"652bde80360a1fe1","type":"ui_mp4frag","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","group":"46d6d60673638810","order":1,"width":22,"height":13,"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":"true","players":["socket.io","hls.js","hls","mp4"],"x":1110,"y":200,"wires":[[]]},{"id":"d1112dfcbfef615b","type":"ui_button","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","group":"46d6d60673638810","order":1,"width":0,"height":0,"passthru":false,"label":"Start","tooltip":"","color":"","bgcolor":"","className":"","icon":"","payload":"","payloadType":"str","topic":"topic","topicType":"msg","x":150,"y":180,"wires":[["adffeae04f85fed5"]]},{"id":"2858e68578995d73","type":"ui_button","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","group":"46d6d60673638810","order":1,"width":0,"height":0,"passthru":false,"label":"Stop","tooltip":"","color":"","bgcolor":"","className":"","icon":"","payload":"","payloadType":"str","topic":"topic","topicType":"msg","x":150,"y":220,"wires":[["379c223a6ef2f8ea"]]},{"id":"379c223a6ef2f8ea","type":"change","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","rules":[{"t":"set","p":"action","pt":"msg","to":"{\"command\":\"stop\",\"args\":[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://belov:belov12345678@192.168.1.129:554/cam/realmonitor?channel=1&subtype=0\",\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_every_frame+empty_moov+default_base_moof\",\"-min_frag_duration\",\"1000000\",\"pipe:1\"]}","tot":"json"}],"action":"","property":"","from":"","to":"","reg":false,"x":320,"y":220,"wires":[["7a8c3f95b486b8f9"]]},{"id":"adffeae04f85fed5","type":"change","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","rules":[{"t":"set","p":"URLCAM","pt":"msg","to":"#:(persistent)::URLCAM_conf","tot":"global"}],"action":"","property":"","from":"","to":"","reg":false,"x":330,"y":180,"wires":[["96e79d1eccb9df9d"]]},{"id":"96e79d1eccb9df9d","type":"change","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","rules":[{"t":"set","p":"action","pt":"msg","to":"{\"command\":\"start\",\"args\":[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\", $.URLCAM,\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_every_frame+empty_moov+default_base_moof\",\"-min_frag_duration\",\"1000000\",\"pipe:1\"]}","tot":"jsonata"}],"action":"","property":"","from":"","to":"","reg":false,"x":540,"y":180,"wires":[["7a8c3f95b486b8f9"]]},{"id":"46d6d60673638810","type":"ui_group","name":"","tab":"9eb9a9fcd56859ba","order":1,"disp":true,"width":"22","collapse":false,"className":""},{"id":"9eb9a9fcd56859ba","type":"ui_tab","name":"Camera Mergulhador","icon":"dashboard","order":2,"disabled":false,"hidden":false},{"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}]

I solved with chmod 777 in dir.

Is possible send me link or here, with code the sample with updated ?

I try is not sucess

[{"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":"5273a6de.b59318","name":"start","props":[{"p":"action","v":"{\"command\":\"start\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"5","topic":"","x":150,"y":320,"wires":[["adffeae04f85fed5"]]},{"id":"b4a8c0bc44499078","type":"inject","z":"5273a6de.b59318","name":"restart","props":[{"p":"action","v":"{\"command\":\"restart\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":"1","topic":"","x":150,"y":360,"wires":[["7a8c3f95b486b8f9"]]},{"id":"7dde9f15d2aa5933","type":"inject","z":"5273a6de.b59318","name":"stop","props":[{"p":"action","v":"{\"command\":\"stop\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":150,"y":400,"wires":[["7a8c3f95b486b8f9"]]},{"id":"58b77ee9a2e29092","type":"mp4frag","z":"5273a6de.b59318","name":"","outputs":2,"basePath":"telemetria","serveHttp":"false","serveIo":"false","hlsPlaylistSize":4,"hlsPlaylistExtra":0,"autoStart":"true","preBuffer":"0","timeLimit":"-1","repeated":"false","statusData":"playlist","x":1010,"y":100,"wires":[[],["571c8af49dfc40ae"]]},{"id":"a8741b7c88d88102","type":"subflow:32d61e0f.0490b2","z":"5273a6de.b59318","name":"","x":910,"y":340,"wires":[]},{"id":"ef6008f4d7425c22","type":"inject","z":"5273a6de.b59318","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":750,"y":60,"wires":[["58b77ee9a2e29092"]]},{"id":"e23111864612f2c5","type":"inject","z":"5273a6de.b59318","name":"write stop","props":[{"p":"action","v":"{\"command\":\"stop\",\"subject\":\"write\"}","vt":"json"}],"repeat":"","crontab":"","once":false,"onceDelay":0.1,"topic":"","x":760,"y":100,"wires":[["58b77ee9a2e29092"]]},{"id":"571c8af49dfc40ae","type":"subflow:f3874cf4a963908d","z":"5273a6de.b59318","name":"","env":[{"name":"STATIC_PATH","value":"/var/www/arq","type":"str"},{"name":"STATIC_ROOT","value":"/var/www/arq","type":"str"}],"x":1320,"y":180,"wires":[["e2da3c3cee315665"]]},{"id":"e2da3c3cee315665","type":"ui_mp4frag","z":"5273a6de.b59318","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":1330,"y":340,"wires":[["9ec4f3ad80a3ec74"]]},{"id":"9ec4f3ad80a3ec74","type":"ui_template","z":"5273a6de.b59318","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":1340,"y":440,"wires":[[]]},{"id":"cfa37a1fd41f4570","type":"group","z":"5273a6de.b59318","name":"Camera IP - Protocolo RTSP","style":{"stroke":"#ffC000","fill":"#ffefbf","label":true,"color":"#6f2fa0"},"nodes":["7a8c3f95b486b8f9","4cc7afbbcc802243","652bde80360a1fe1","d1112dfcbfef615b","2858e68578995d73","379c223a6ef2f8ea","adffeae04f85fed5","96e79d1eccb9df9d"],"x":74,"y":139,"w":1132,"h":122},{"id":"7a8c3f95b486b8f9","type":"ffmpeg","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","outputs":3,"cmdPath":"ffmpeg","cmdArgs":"[]","cmdOutputs":2,"killSignal":"SIGTERM","x":750,"y":200,"wires":[["4cc7afbbcc802243","58b77ee9a2e29092"],["4cc7afbbcc802243","58b77ee9a2e29092"],["a8741b7c88d88102"]]},{"id":"4cc7afbbcc802243","type":"mp4frag","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","outputs":2,"basePath":"teste","serveHttp":"true","serveIo":"true","hlsPlaylistSize":"6","hlsPlaylistExtra":"2","autoStart":"false","preBuffer":1,"timeLimit":10000,"repeated":"false","statusData":"playlist","x":920,"y":200,"wires":[["652bde80360a1fe1"],[]]},{"id":"652bde80360a1fe1","type":"ui_mp4frag","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","group":"46d6d60673638810","order":1,"width":22,"height":13,"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":"true","players":["socket.io","hls.js","hls","mp4"],"x":1110,"y":200,"wires":[[]]},{"id":"d1112dfcbfef615b","type":"ui_button","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","group":"46d6d60673638810","order":1,"width":0,"height":0,"passthru":false,"label":"Start","tooltip":"","color":"","bgcolor":"","className":"","icon":"","payload":"","payloadType":"str","topic":"topic","topicType":"msg","x":150,"y":180,"wires":[["adffeae04f85fed5"]]},{"id":"2858e68578995d73","type":"ui_button","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","group":"46d6d60673638810","order":1,"width":0,"height":0,"passthru":false,"label":"Stop","tooltip":"","color":"","bgcolor":"","className":"","icon":"","payload":"","payloadType":"str","topic":"topic","topicType":"msg","x":150,"y":220,"wires":[["379c223a6ef2f8ea"]]},{"id":"379c223a6ef2f8ea","type":"change","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","rules":[{"t":"set","p":"action","pt":"msg","to":"{\"command\":\"stop\",\"args\":[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\",\"rtsp://belov:belov12345678@192.168.1.129:554/cam/realmonitor?channel=1&subtype=0\",\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_every_frame+empty_moov+default_base_moof\",\"-min_frag_duration\",\"1000000\",\"pipe:1\"]}","tot":"json"}],"action":"","property":"","from":"","to":"","reg":false,"x":320,"y":220,"wires":[["7a8c3f95b486b8f9"]]},{"id":"adffeae04f85fed5","type":"change","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","rules":[{"t":"set","p":"URLCAM","pt":"msg","to":"#:(persistent)::URLCAM_conf","tot":"global"}],"action":"","property":"","from":"","to":"","reg":false,"x":330,"y":180,"wires":[["96e79d1eccb9df9d"]]},{"id":"96e79d1eccb9df9d","type":"change","z":"5273a6de.b59318","g":"cfa37a1fd41f4570","name":"","rules":[{"t":"set","p":"action","pt":"msg","to":"{\"command\":\"start\",\"args\":[\"-loglevel\",\"quiet\",\"-rtsp_transport\",\"tcp\",\"-i\", $.URLCAM,\"-an\",\"-c:v\",\"copy\",\"-f\",\"mp4\",\"-movflags\",\"+frag_every_frame+empty_moov+default_base_moof\",\"-min_frag_duration\",\"1000000\",\"pipe:1\"]}","tot":"jsonata"}],"action":"","property":"","from":"","to":"","reg":false,"x":540,"y":180,"wires":[["7a8c3f95b486b8f9"]]},{"id":"46d6d60673638810","type":"ui_group","name":"","tab":"9eb9a9fcd56859ba","order":1,"disp":true,"width":"22","collapse":false,"className":""},{"id":"9eb9a9fcd56859ba","type":"ui_tab","name":"Camera Mergulhador","icon":"dashboard","order":2,"disabled":false,"hidden":false},{"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}]

I had to re-read some of the previous posts from last year to remember what we did.

  • Firstly, you have to setup static hosting from node-red by configuring httpStatic in the setting.js file.
  • Then you have to set the values of static path/root in the subflow to match those values.

The current problem is that your system is setup differently than mine and you probably do not have an external usb 6tb hard drive mounted at /run/media/system/surveillance1 as I do. The values are configurable so that you can make it match your specific setup of where you would want to write the data.

To clarify, I have a hard drive mounted at a location that I am using to write large quantities of video.

Filesystem      Size  Used Avail Use% Mounted on
/dev/root        30G  9.3G   19G  34% /
devtmpfs        3.5G     0  3.5G   0% /dev
tmpfs           3.7G  904K  3.7G   1% /dev/shm
tmpfs           1.5G  932K  1.5G   1% /run
tmpfs           5.0M  4.0K  5.0M   1% /run/lock
/dev/mmcblk0p1  253M   31M  222M  13% /boot
/dev/sda1       5.5T  4.2T 1015G  81% /run/media/system/surveillance1
tmpfs           739M     0  739M   0% /run/user/1000

So, in the settings.js file, I added some configuration to tell node-red to map a location there and host files from it in the httpStatic sections.

httpStaticRoot: '/static/',
httpStatic: [
        { path: '/run/media/system/surveillance1/cctv/recordings/', root: '/recordings/' }
    ],

Using those values, I added that to the subflow node so that I can write to the correct location.

Currently, I am writing video 24/7 for 11 main streams and it seems to work ok.

2 Likes

@allacmc
What is not yet working??

Hey Kevin,

Is this a turn key solution now or you have provided just the (excellent) building blocks ?

We are just deploying a number of security cameras around our place and would like to integrate them into Node Red.

I run my Node Red in a Virtual machine under ESXI so could easily spawn a dedicated machine for this if required

All of my storage is centralised and will be made available to the Node Red host through NFS shares

Craig

Sorry jumpin in

Turn-Key and Turn-Key, well you have to follow the instructions given and import the flow sample (one per camera) and make the modifications to match their url's etc. (In addition I did on my own some experiments how to handle and play the recordings in a convenient way from the browser. So far that worked excellent bur unfortunately all my cameras are usb types so I just archived my setup for future, next house or camera upgrading, whatever happens first)

Myself I do have a house-keeping script that erases older recordings. Since your drive is reaching the upper limit, do you have something similar as well?

It's definitely closer to building blocks than it is to turn-key. Although, I have ensured that any library I have published is designed to be cross platform for linux, macos, and windows.

Dealing with the local police department yesterday reminded of how turn-key it is NOT when I was trying to extract video footage that caught the getaway driver's vehicle of someone that stole a car from the dealership on the other side of the apartment complex. And that reminds me, I only setup the 24/7 recording after the previous car theft occurred at the same location and I had no recordings available. I guess necessity is the mother of all invention. In fact, I only ever set up cameras many years ago due to an ex girlfriend who was threatening my property while I had to be away at work. And then I wasn't satisfied with the "off the shelf" camera systems and started working on my own stuff. It's funny life works.

Thanks for jumping in.

I do, but it's not perfect. When I updated it a while back I broke something and my empty folders no longer get deleted. I still have time to fix it since they don't take up much space.


The use percent gets run once per hour, while the removal script run once per night to delete anything older than 2 weeks. I used the standard fs module but my hunch is that your cleanup script is made in python?

(Cctv recording and playback (work in progress) - #39 by allacmc)

Is it possible for you to add a sample code? of the nodes with the recording feature?

The last ones I got don't have the node updated, it's missing.

That's correct, :smile: but the idea and concept is more or less the same, I keep the last 10 days

1 Like