2023-04-26 15:25:26 +02:00
""" Maintain recording segments in cache. """
2020-11-30 04:31:02 +01:00
import datetime
import logging
2021-12-11 05:56:29 +01:00
import multiprocessing as mp
2020-11-30 04:31:02 +01:00
import os
2021-12-11 05:56:29 +01:00
import queue
2021-06-07 03:24:36 +02:00
import random
import string
2020-11-30 04:31:02 +01:00
import subprocess as sp
import threading
2023-04-26 15:25:26 +02:00
import psutil
2021-10-23 23:18:13 +02:00
from collections import defaultdict
2023-04-26 15:25:26 +02:00
from multiprocessing . synchronize import Event as MpEvent
2020-11-30 04:31:02 +01:00
from pathlib import Path
2023-04-26 15:25:26 +02:00
from typing import Any , Tuple
2021-07-09 22:14:16 +02:00
2021-12-11 20:11:39 +01:00
from frigate . config import RetainModeEnum , FrigateConfig
2022-12-18 00:53:34 +01:00
from frigate . const import CACHE_DIR , MAX_SEGMENT_DURATION , RECORD_DIR
2021-07-09 22:14:16 +02:00
from frigate . models import Event , Recordings
2023-04-26 15:25:26 +02:00
from frigate . types import RecordMetricsTypes
2021-12-11 20:11:39 +01:00
from frigate . util import area
2020-11-30 04:31:02 +01:00
logger = logging . getLogger ( __name__ )
2020-12-01 04:08:47 +01:00
2020-11-30 04:31:02 +01:00
class RecordingMaintainer ( threading . Thread ) :
2021-12-11 05:56:29 +01:00
def __init__ (
2023-04-26 15:25:26 +02:00
self ,
config : FrigateConfig ,
recordings_info_queue : mp . Queue ,
process_info : dict [ str , RecordMetricsTypes ] ,
stop_event : MpEvent ,
2021-12-11 05:56:29 +01:00
) :
2020-11-30 04:31:02 +01:00
threading . Thread . __init__ ( self )
2023-04-26 15:25:26 +02:00
self . name = " recording_maintainer "
2020-11-30 04:31:02 +01:00
self . config = config
2021-12-11 05:56:29 +01:00
self . recordings_info_queue = recordings_info_queue
2023-04-26 15:25:26 +02:00
self . process_info = process_info
2020-11-30 04:31:02 +01:00
self . stop_event = stop_event
2023-04-26 15:25:26 +02:00
self . recordings_info : dict [ str , Any ] = defaultdict ( list )
self . end_time_cache : dict [ str , Tuple [ datetime . datetime , float ] ] = { }
2020-11-30 04:31:02 +01:00
2023-04-26 15:25:26 +02:00
def move_files ( self ) - > None :
2021-12-11 05:56:29 +01:00
cache_files = sorted (
[
d
for d in os . listdir ( CACHE_DIR )
if os . path . isfile ( os . path . join ( CACHE_DIR , d ) )
and d . endswith ( " .mp4 " )
and not d . startswith ( " clip_ " )
]
)
2020-11-30 04:31:02 +01:00
files_in_use = [ ]
for process in psutil . process_iter ( ) :
try :
2021-02-17 14:23:32 +01:00
if process . name ( ) != " ffmpeg " :
2020-12-24 21:23:59 +01:00
continue
2020-11-30 04:31:02 +01:00
flist = process . open_files ( )
if flist :
for nt in flist :
2021-07-09 22:14:16 +02:00
if nt . path . startswith ( CACHE_DIR ) :
2021-02-17 14:23:32 +01:00
files_in_use . append ( nt . path . split ( " / " ) [ - 1 ] )
2020-11-30 04:31:02 +01:00
except :
continue
2021-10-23 23:18:13 +02:00
# group recordings by camera
2023-04-26 15:25:26 +02:00
grouped_recordings : defaultdict [ str , list [ dict [ str , Any ] ] ] = defaultdict ( list )
for cache in cache_files :
2021-07-11 21:34:48 +02:00
# Skip files currently in use
2023-04-26 15:25:26 +02:00
if cache in files_in_use :
2020-11-30 04:31:02 +01:00
continue
2023-04-26 15:25:26 +02:00
cache_path = os . path . join ( CACHE_DIR , cache )
basename = os . path . splitext ( cache ) [ 0 ]
2021-05-22 05:35:25 +02:00
camera , date = basename . rsplit ( " - " , maxsplit = 1 )
start_time = datetime . datetime . strptime ( date , " % Y % m %d % H % M % S " )
2021-10-23 23:18:13 +02:00
grouped_recordings [ camera ] . append (
{
" cache_path " : cache_path ,
" start_time " : start_time ,
}
2021-02-17 14:23:32 +01:00
)
2020-11-30 04:31:02 +01:00
2021-11-19 14:16:29 +01:00
# delete all cached files past the most recent 5
2021-11-19 14:19:45 +01:00
keep_count = 5
2021-11-17 15:57:57 +01:00
for camera in grouped_recordings . keys ( ) :
2022-07-19 14:24:44 +02:00
segment_count = len ( grouped_recordings [ camera ] )
if segment_count > keep_count :
2023-01-31 00:42:53 +01:00
logger . warning (
f " Unable to keep up with recording segments in cache for { camera } . Keeping the { keep_count } most recent segments out of { segment_count } and discarding the rest... "
)
2021-12-11 05:56:29 +01:00
to_remove = grouped_recordings [ camera ] [ : - keep_count ]
2023-04-26 15:25:26 +02:00
for rec in to_remove :
cache_path = rec [ " cache_path " ]
2022-07-19 14:24:44 +02:00
Path ( cache_path ) . unlink ( missing_ok = True )
self . end_time_cache . pop ( cache_path , None )
2021-12-11 05:56:29 +01:00
grouped_recordings [ camera ] = grouped_recordings [ camera ] [ - keep_count : ]
2021-11-11 04:12:41 +01:00
2021-10-23 23:18:13 +02:00
for camera , recordings in grouped_recordings . items ( ) :
2021-12-11 05:56:29 +01:00
# clear out all the recording info for old frames
while (
len ( self . recordings_info [ camera ] ) > 0
and self . recordings_info [ camera ] [ 0 ] [ 0 ]
< recordings [ 0 ] [ " start_time " ] . timestamp ( )
) :
self . recordings_info [ camera ] . pop ( 0 )
2021-10-23 23:18:13 +02:00
# get all events with the end time after the start of the oldest cache file
# or with end_time None
events : Event = (
Event . select ( )
. where (
Event . camera == camera ,
( Event . end_time == None )
2021-11-21 16:43:37 +01:00
| ( Event . end_time > = recordings [ 0 ] [ " start_time " ] . timestamp ( ) ) ,
2021-10-23 23:18:13 +02:00
Event . has_clip ,
)
. order_by ( Event . start_time )
)
for r in recordings :
cache_path = r [ " cache_path " ]
start_time = r [ " start_time " ]
# Just delete files if recordings are turned off
if (
not camera in self . config . cameras
2023-04-26 15:25:26 +02:00
or not self . process_info [ camera ] [ " record_enabled " ] . value
2021-10-23 23:18:13 +02:00
) :
Path ( cache_path ) . unlink ( missing_ok = True )
2021-11-19 14:19:45 +01:00
self . end_time_cache . pop ( cache_path , None )
2021-10-23 23:18:13 +02:00
continue
2020-11-30 04:31:02 +01:00
2021-11-19 14:19:45 +01:00
if cache_path in self . end_time_cache :
2021-11-19 23:56:00 +01:00
end_time , duration = self . end_time_cache [ cache_path ]
2021-10-23 23:18:13 +02:00
else :
2021-11-19 14:19:45 +01:00
ffprobe_cmd = [
" ffprobe " ,
" -v " ,
" error " ,
" -show_entries " ,
" format=duration " ,
" -of " ,
" default=noprint_wrappers=1:nokey=1 " ,
f " { cache_path } " ,
]
p = sp . run ( ffprobe_cmd , capture_output = True )
2022-10-05 02:56:04 +02:00
if p . returncode == 0 and p . stdout . decode ( ) :
2021-11-19 14:19:45 +01:00
duration = float ( p . stdout . decode ( ) . strip ( ) )
2022-11-02 12:37:27 +01:00
else :
duration = - 1
# ensure duration is within expected length
2022-12-18 00:53:34 +01:00
if 0 < duration < MAX_SEGMENT_DURATION :
2021-11-19 23:56:00 +01:00
end_time = start_time + datetime . timedelta ( seconds = duration )
self . end_time_cache [ cache_path ] = ( end_time , duration )
2021-11-19 14:19:45 +01:00
else :
2022-11-02 12:37:27 +01:00
if duration == - 1 :
logger . warning (
2023-04-26 15:25:26 +02:00
f " Failed to probe corrupt segment { cache_path } : { p . returncode } - { str ( p . stderr ) } "
2022-11-02 12:37:27 +01:00
)
2023-02-25 02:13:33 +01:00
logger . warning (
f " Discarding a corrupt recording segment: { cache_path } "
)
2021-11-19 14:19:45 +01:00
Path ( cache_path ) . unlink ( missing_ok = True )
continue
2021-10-22 14:23:18 +02:00
2021-12-11 16:22:44 +01:00
# if cached file's start_time is earlier than the retain days for the camera
2021-10-23 23:18:13 +02:00
if start_time < = (
(
datetime . datetime . now ( )
- datetime . timedelta (
2021-12-11 16:22:44 +01:00
days = self . config . cameras [ camera ] . record . retain . days
2021-10-23 23:18:13 +02:00
)
)
) :
# if the cached segment overlaps with the events:
overlaps = False
for event in events :
# if the event starts in the future, stop checking events
2021-11-17 14:44:58 +01:00
# and remove this segment
2021-10-23 23:18:13 +02:00
if event . start_time > end_time . timestamp ( ) :
overlaps = False
2021-12-11 05:56:29 +01:00
Path ( cache_path ) . unlink ( missing_ok = True )
self . end_time_cache . pop ( cache_path , None )
2021-10-23 23:18:13 +02:00
break
# if the event is in progress or ends after the recording starts, keep it
# and stop looking at events
2021-11-21 16:43:37 +01:00
if (
event . end_time is None
or event . end_time > = start_time . timestamp ( )
) :
2021-10-23 23:18:13 +02:00
overlaps = True
break
if overlaps :
2021-12-11 20:11:39 +01:00
record_mode = self . config . cameras [
camera
] . record . events . retain . mode
2021-10-23 23:18:13 +02:00
# move from cache to recordings immediately
self . store_segment (
camera ,
start_time ,
end_time ,
duration ,
cache_path ,
2021-12-11 20:11:39 +01:00
record_mode ,
2021-10-23 23:18:13 +02:00
)
2023-01-31 00:42:53 +01:00
# if it doesn't overlap with an event, go ahead and drop the segment
# if it ends more than the configured pre_capture for the camera
else :
pre_capture = self . config . cameras [
camera
] . record . events . pre_capture
most_recently_processed_frame_time = self . recordings_info [
camera
] [ - 1 ] [ 0 ]
retain_cutoff = most_recently_processed_frame_time - pre_capture
if end_time . timestamp ( ) < retain_cutoff :
Path ( cache_path ) . unlink ( missing_ok = True )
self . end_time_cache . pop ( cache_path , None )
2021-12-11 16:22:44 +01:00
# else retain days includes this segment
2021-10-23 23:18:13 +02:00
else :
2021-12-11 20:11:39 +01:00
record_mode = self . config . cameras [ camera ] . record . retain . mode
2021-10-23 23:18:13 +02:00
self . store_segment (
2021-12-11 20:11:39 +01:00
camera , start_time , end_time , duration , cache_path , record_mode
2021-10-23 23:18:13 +02:00
)
2023-04-26 15:25:26 +02:00
def segment_stats (
self , camera : str , start_time : datetime . datetime , end_time : datetime . datetime
) - > Tuple [ int , int ] :
2021-12-11 20:11:39 +01:00
active_count = 0
motion_count = 0
for frame in self . recordings_info [ camera ] :
# frame is after end time of segment
if frame [ 0 ] > end_time . timestamp ( ) :
break
# frame is before start time of segment
if frame [ 0 ] < start_time . timestamp ( ) :
continue
active_count + = len (
[
o
for o in frame [ 1 ]
2022-02-06 16:56:06 +01:00
if not o [ " false_positive " ] and o [ " motionless_count " ] == 0
2021-12-11 20:11:39 +01:00
]
)
motion_count + = sum ( [ area ( box ) for box in frame [ 2 ] ] )
return ( motion_count , active_count )
def store_segment (
self ,
2023-04-26 15:25:26 +02:00
camera : str ,
2022-12-11 14:45:32 +01:00
start_time : datetime . datetime ,
end_time : datetime . datetime ,
2023-04-26 15:25:26 +02:00
duration : float ,
cache_path : str ,
2021-12-11 20:11:39 +01:00
store_mode : RetainModeEnum ,
2023-04-26 15:25:26 +02:00
) - > None :
2021-12-11 20:11:39 +01:00
motion_count , active_count = self . segment_stats ( camera , start_time , end_time )
# check if the segment shouldn't be stored
if ( store_mode == RetainModeEnum . motion and motion_count == 0 ) or (
store_mode == RetainModeEnum . active_objects and active_count == 0
) :
Path ( cache_path ) . unlink ( missing_ok = True )
self . end_time_cache . pop ( cache_path , None )
return
2022-12-11 14:45:32 +01:00
directory = os . path . join (
RECORD_DIR ,
2023-01-12 12:53:38 +01:00
start_time . astimezone ( tz = datetime . timezone . utc ) . strftime ( " % Y- % m- %d / % H " ) ,
2022-12-11 14:45:32 +01:00
camera ,
)
2021-10-23 23:18:13 +02:00
if not os . path . exists ( directory ) :
os . makedirs ( directory )
2022-12-11 14:45:32 +01:00
file_name = (
f " { start_time . replace ( tzinfo = datetime . timezone . utc ) . strftime ( ' % M. % S.mp4 ' ) } "
)
2021-10-23 23:18:13 +02:00
file_path = os . path . join ( directory , file_name )
2021-11-09 14:05:21 +01:00
try :
2022-10-02 01:11:29 +02:00
if not os . path . exists ( file_path ) :
start_frame = datetime . datetime . now ( ) . timestamp ( )
2022-12-18 00:53:34 +01:00
# add faststart to kept segments to improve metadata reading
ffmpeg_cmd = [
" ffmpeg " ,
" -y " ,
" -i " ,
cache_path ,
" -c " ,
" copy " ,
" -movflags " ,
" +faststart " ,
file_path ,
]
p = sp . run (
ffmpeg_cmd ,
encoding = " ascii " ,
capture_output = True ,
2022-10-02 01:11:29 +02:00
)
2021-11-09 14:05:21 +01:00
2022-12-18 00:53:34 +01:00
if p . returncode != 0 :
logger . error ( f " Unable to convert { cache_path } to { file_path } " )
logger . error ( p . stderr )
return
else :
logger . debug (
f " Copied { file_path } in { datetime . datetime . now ( ) . timestamp ( ) - start_frame } seconds. "
)
2022-10-09 13:28:26 +02:00
try :
2022-12-18 00:53:34 +01:00
# get the segment size of the cache file
# file without faststart is same size
2022-10-09 13:28:26 +02:00
segment_size = round (
float ( os . path . getsize ( cache_path ) ) / 1000000 , 1
)
except OSError :
segment_size = 0
os . remove ( cache_path )
2022-10-02 01:11:29 +02:00
rand_id = " " . join (
random . choices ( string . ascii_lowercase + string . digits , k = 6 )
)
Recordings . create (
id = f " { start_time . timestamp ( ) } - { rand_id } " ,
camera = camera ,
path = file_path ,
start_time = start_time . timestamp ( ) ,
end_time = end_time . timestamp ( ) ,
duration = duration ,
motion = motion_count ,
# TODO: update this to store list of active objects at some point
objects = active_count ,
2022-10-09 13:28:26 +02:00
segment_size = segment_size ,
2022-10-02 01:11:29 +02:00
)
2021-11-09 14:05:21 +01:00
except Exception as e :
logger . error ( f " Unable to store recording segment { cache_path } " )
Path ( cache_path ) . unlink ( missing_ok = True )
logger . error ( e )
2020-11-30 04:31:02 +01:00
2021-11-19 14:19:45 +01:00
# clear end_time cache
self . end_time_cache . pop ( cache_path , None )
2023-04-26 15:25:26 +02:00
def run ( self ) - > None :
2021-07-11 21:34:48 +02:00
# Check for new files every 5 seconds
2023-04-26 15:25:26 +02:00
wait_time = 5.0
2021-10-22 14:23:18 +02:00
while not self . stop_event . wait ( wait_time ) :
run_start = datetime . datetime . now ( ) . timestamp ( )
2021-12-11 05:56:29 +01:00
# empty the recordings info queue
while True :
try :
(
camera ,
frame_time ,
current_tracked_objects ,
motion_boxes ,
regions ,
) = self . recordings_info_queue . get ( False )
2023-04-26 15:25:26 +02:00
if self . process_info [ camera ] [ " record_enabled " ] . value :
2021-12-11 05:56:29 +01:00
self . recordings_info [ camera ] . append (
(
frame_time ,
current_tracked_objects ,
motion_boxes ,
regions ,
)
)
except queue . Empty :
break
2021-11-09 14:05:21 +01:00
try :
self . move_files ( )
except Exception as e :
logger . error (
" Error occurred when attempting to maintain recording cache "
)
logger . error ( e )
2021-11-17 15:57:57 +01:00
duration = datetime . datetime . now ( ) . timestamp ( ) - run_start
wait_time = max ( 0 , 5 - duration )
2021-07-11 21:34:48 +02:00
logger . info ( f " Exiting recording maintenance... " )